Product Registration (Experimental)
Available from Rancher 2.15 and onwards
Note: This API is experimental and may change in future releases.
Overview
The product registration API provides a simplified way to register products and pages in Rancher Dashboard Extensions. It replaces the need to manually configure routes and DSL functions (product, virtualType, configureType, basicType) — all of which are handled automatically under the hood.
When to use this API
Use this API when you want to:
- Quickly scaffold a new Extension product without dealing with routing boilerplate
- Add custom pages or resource pages to a product with minimal configuration
- Extend an existing Rancher Dashboard product (Explorer, Cluster Management, etc.) with new pages
For full control over routing and DSL configuration, continue using the existing approach.
Note: For a set of copy-paste-ready examples covering common use cases, see Product Registration Examples.
Terminology
Before diving in, here are the key terms used throughout this API. If you're not familiar with the concepts of top-level and cluster-level products, check their definitions here.
| Term | Description |
|---|---|
| Product | A top-level view in Rancher Dashboard that adds a navigation entry into the top-level slide-in menu (e.g. Fleet, Cluster Management) |
| Custom page | A page inside a product that renders a Vue component you provide. Equivalent to defining a virtualType in the DSL approach |
| Resource page | A page inside a product that displays a Kubernetes resource type using Rancher's built-in list/detail/edit views. Equivalent to defining a configureType in the DSL approach |
| Group | A collapsible folder/group in the product's side-menu that contains pages or other groups |
| Single page product | A product with no side-menu — just one full-page Vue component |
API Reference
All methods are called on the extension object in your Extension's index.ts.
addProduct(productName)
The simplest way to register a product. Pass a string name and the product will be registered with a default empty page.
extension.addProduct('my-first-product');
This is a convenience/bridge method useful for getting started. You can expand to the full API signatures below once you are ready to add custom pages.
addProduct(product, config)
Register a product with pages and side-menu navigation. Can only be called once per product name — calling it again with the same name will throw an error.
Parameters:
| Parameter | Type | Description |
|---|---|---|
product | ProductMetadata | Product name, label, icon, and other metadata |
config | ProductChild[] | Array of pages or groups that form the product's side-menu |
addProduct(product)
Register a single page product (no side-menu).
Parameters:
| Parameter | Type | Description |
|---|---|---|
product | ProductSinglePage | Product metadata plus a component to render |
extendProduct(product, config)
Add pages to an existing Rancher Dashboard product.
Parameters:
| Parameter | Type | Description |
|---|---|---|
product | StandardProductName | One of: 'explorer', 'manager', 'settings', 'auth' |
config | ProductChild[] | Array of pages or groups to add to the product |
Usage
Note: We recommend building up configuration objects as separate
constvariables before passing them toaddProductorextendProduct. This gives you better TypeScript intellisense and autocomplete, and avoids some of the confusion when creating the objects directly inaddProductorextendProduct.
Quick start
The fastest way to get a product registered. This registers a product with a default empty page:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
extension.addProduct('my-first-product');
}
Single page product
A product that renders a single Vue component with no side-menu:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
import { ProductSinglePage } from '@shell/core/plugin-types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
const product: ProductSinglePage = {
name: 'my-dashboard',
label: 'My Dashboard',
icon: 'globe',
component: () => import('./pages/DashboardPage.vue')
};
extension.addProduct(product);
}
Product with custom pages
A product with multiple custom pages listed as side-menu entries:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
import { ProductMetadata, ProductChildCustomPage } from '@shell/core/plugin-types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
const overviewPage: ProductChildCustomPage = {
name: 'overview',
label: 'Overview',
component: () => import('./pages/OverviewPage.vue'),
weight: 2, // side-menu ordering (bigger number on top)
};
const settingsPage: ProductChildCustomPage = {
name: 'settings',
label: 'Settings',
component: () => import('./pages/SettingsPage.vue'),
weight: 1,
};
const product: ProductMetadata = {
name: 'my-app',
label: 'My App',
icon: 'gear',
};
extension.addProduct(product, [overviewPage, settingsPage]);
}
Product with resource pages
A product that displays Kubernetes resources using Rancher's built-in list/detail/edit views:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
import { ProductMetadata, ProductChildResourcePage } from '@shell/core/plugin-types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
weight: 2,
config: {
displayName: 'Clusters',
isCreatable: true,
isEditable: true,
isRemovable: true,
canYaml: true,
},
};
const nodePage: ProductChildResourcePage = {
type: 'management.cattle.io.node',
weight: 1,
};
const product: ProductMetadata = {
name: 'my-resources',
label: 'My Resources',
};
extension.addProduct(product, [clusterPage, nodePage]);
}
Product with groups
Organize pages into collapsible folder/groups in the side-menu:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
import {
ProductMetadata,
ProductChildCustomPage,
ProductChildGroup
} from '@shell/core/plugin-types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
// Pages for the "Monitoring" group
const alertsPage: ProductChildCustomPage = {
name: 'alerts',
label: 'Alerts',
component: () => import('./pages/AlertsPage.vue'),
};
const metricsPage: ProductChildCustomPage = {
name: 'metrics',
label: 'Metrics',
component: () => import('./pages/MetricsPage.vue'),
};
const monitoringGroup: ProductChildGroup = {
name: 'monitoring',
label: 'Monitoring',
weight: 2,
children: [alertsPage, metricsPage],
};
// A standalone page outside any group
const overviewPage: ProductChildCustomPage = {
name: 'overview',
label: 'Overview',
component: () => import('./pages/OverviewPage.vue'),
weight: 3, // side-menu ordering (bigger number on top)
};
const product: ProductMetadata = {
name: 'my-platform',
label: 'My Platform',
};
extension.addProduct(product, [overviewPage, monitoringGroup]);
}
Extending an existing product
Add new pages to one of Rancher Dashboard's built-in products:
// ./index.ts
import { importTypes } from '@rancher/auto-import';
import { IPlugin } from '@shell/core/types';
import { ProductChildCustomPage } from '@shell/core/plugin-types';
export default function(extension: IPlugin) {
importTypes(extension);
extension.metadata = require('./package.json');
const customPage: ProductChildCustomPage = {
name: 'my-custom-view',
label: 'My Custom View',
component: () => import('./pages/MyCustomView.vue'),
};
// Add a page to the Cluster Explorer product
extension.extendProduct('explorer', [customPage]);
}
The products available for extension are:
| Name | Description |
|---|---|
'explorer' | Cluster Explorer |
'manager' | Cluster Management |
'settings' | Global Settings |
'auth' | Authentication & API Keys |
Type Reference
All types are importable from @shell/core/plugin-types:
import {
ProductMetadata,
ProductSinglePage,
ProductChildCustomPage,
ProductChildResourcePage,
ProductChildPage,
ProductChildGroup,
ProductChild,
StandardProductName,
} from '@shell/core/plugin-types';
ProductMetadata
The metadata that defines a product — its identity, icon, and product-level settings. This is passed as the first argument to addProduct.
| Property | Type | Required | Description |
|---|---|---|---|
name | string | Yes | Unique product identifier |
label | string | Yes* | Display label (* either label or labelKey is required) |
labelKey | string | Yes* | Translation key for the label |
icon | string | No | Icon name (based on rancher icons). Defaults to 'extension' |
weight | number | No | Side-menu ordering (bigger number on top) |
showClusterSwitcher | boolean | No | Show the cluster switcher in navigation |
showNamespaceFilter | boolean | No | Show the namespace filter in the header |
ProductSinglePage
A product that renders a single full-page Vue component with no side-menu. Extends ProductMetadata with:
| Property | Type | Required | Description |
|---|---|---|---|
component | RouteComponent | Yes | Vue component to render for the entire product |
ProductChildCustomPage
A page inside a product that renders a Vue component you provide. Equivalent to a virtualType in the legacy DSL approach.
| Property | Type | Required | Description |
|---|---|---|---|
name | string | Yes | Unique page identifier |
label | string | Yes* | Display label (* either label or labelKey is required) |
labelKey | string | Yes* | Translation key for the label |
component | RouteComponent | Yes | Vue component to render |
weight | number | No | Side-menu ordering (bigger number on top) |
ProductChildResourcePage
A page that displays a Kubernetes resource type using Rancher Dashboard's built-in list/detail/edit views. Equivalent to a configureType in the DSL approach.
| Property | Type | Required | Description |
|---|---|---|---|
type | string | Yes | Kubernetes resource type (e.g. 'provisioning.cattle.io.cluster') |
weight | number | No | Side-menu ordering (bigger number on top) |
config | ConfigureTypeConfiguration | No | Resource page options (creatable, editable, removable, etc.) |
headers | HeaderOptions[] | No | Custom table column headers for the list view (client-side pagination). See Table headers |
sspHeaders | PaginationHeaderOptions[] | No | Custom table column headers for the list view (server-side pagination). See Server-side pagination headers |
overrideListResourceName | string | No | Override the display name for this resource type in the list view. See Renaming types |
hideFromNav | boolean | No | Hide this resource type from the side-menu entirely. See Hiding types |
hideBulkActions | boolean | No | Hide bulk action buttons (e.g. delete) for this resource type in the list view. See Hiding bulk actions |
ProductChildGroup
A collapsible folder/group in the product's side-menu that contains pages or other nested groups. Optionally, a group can have its own Vue component that renders when the group header is clicked (only for new products, not when extending).
| Property | Type | Required | Description |
|---|---|---|---|
name | string | Yes | Unique group identifier |
label | string | Yes* | Display label (* either label or labelKey is required) |
labelKey | string | Yes* | Translation key for the label |
children | ProductChild[] | Yes | Array of pages or nested groups |
component | RouteComponent | No | Optional component for the group's own page |
weight | number | No | Side-menu ordering (bigger number on top) |
ProductChildPage
A union type representing any page item in a product configuration. It can be one of:
ProductChildCustomPage— a custom page with a Vue componentProductChildResourcePage— a resource page for a Kubernetes type
type ProductChildPage = ProductChildCustomPage | ProductChildResourcePage;
ProductChild
A union type representing any item that can appear in a product configuration — either a page or a group. This is the type used for arrays passed to addProduct and extendProduct, and for the children property of ProductChildGroup.
type ProductChild = ProductChildGroup | ProductChildPage;
Advanced Configuration
Table headers (headers)
Resource lists use Headers to control which columns appear in the resource list view, what data they display, and how they behave (sorting, searching, formatting).
By default these mostly come from the resource's definition and it's additionalPrinterColumns field.
When you set headers on a resource page, these columns replace the default columns.
Using built-in headers
Rancher Dashboard provides a set of commonly used header definitions in @shell/config/table-headers. These are pre-configured with the correct value paths, formatters, and sorting for standard resource fields. Some of the most commonly used built-in headers:
| Header | Description |
|---|---|
NAME | Resource name with a clickable link to the detail view (uses LinkDetail formatter) |
STATE | Resource state with a colored badge (uses BadgeStateFormatter) |
AGE | Resource age since creation (uses LiveDate formatter) |
NAMESPACE | Resource namespace |
import { NAME, STATE, AGE, NAMESPACE } from '@shell/config/table-headers';
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
headers: [STATE, NAME, NAMESPACE, AGE],
};
Note: The order of headers in the array determines the column order in the table (left to right).
Defining custom headers
You can define your own headers by providing objects that follow the HeaderOptions format. Custom headers can be mixed with built-in ones:
import { NAME, AGE } from '@shell/config/table-headers';
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
headers: [
NAME,
{
name: 'provider',
label: 'Provider',
value: 'status.provider',
sort: 'status.provider',
},
{
name: 'nodeCount',
label: 'Nodes',
value: 'status.nodeCount',
sort: 'status.nodeCount',
search: false, // exclude from search filtering
width: 80, // fixed column width
},
AGE,
],
};
Using getValue for computed columns
When the data you want to display doesn't exist at a simple path on the row object, use the getValue function to compute the display value:
{
name: 'fullName',
label: 'Full Name',
getValue: (row) => `${ row.metadata?.labels?.['first-name'] } ${ row.metadata?.labels?.['last-name'] }`,
sort: 'metadata.labels.first-name',
}
Using formatters
Formatters transform raw values into formatted display output. Rancher Dashboard includes several built-in formatters in @shell/components/formatter/. Some commonly used ones:
| Formatter | Description |
|---|---|
LinkDetail | Renders the value as a clickable link to the resource detail page |
BadgeStateFormatter | Renders a colored state badge |
LiveDate | Renders a relative timestamp that updates in real-time (e.g. "5 minutes ago") |
Link | Renders a generic clickable link |
{
name: 'status',
label: 'Status',
value: 'stateDisplay',
formatter: 'BadgeStateFormatter',
formatterOpts: { row: true }, // pass row data to the formatter
width: 120,
}
Full header property reference
| Header Property | Type | Description |
|---|---|---|
name | string | Unique column identifier. Must be unique within the headers array |
label | string | Column header text displayed at the top of the column |
labelKey | string | Translation key for the column header (alternative to label) |
value | string | Dot-notation path to the value on the row object (e.g. 'metadata.name', 'status.provider') |
sort | string | string[] | boolean | Path(s) used for sorting. Use an array for multi-field sort (e.g. ['stateSort', 'nameSort']). Set to false to disable sorting |
search | string | boolean | Path used for search/filtering. Set to false to exclude this column from search |
formatter | string | Name of a built-in formatter component from @shell/components/formatter/ or a custom formatter defined in your extension |
formatterOpts | any | Configuration options passed to the formatter component |
width | number | Fixed column width in pixels. If not set, the column auto-sizes |
getValue | (row: any) => string | number | null | Custom function to extract or compute the display value from a row. Takes precedence over value for display, but value is still used for sorting/searching |
Server-side pagination headers (sspHeaders)
When server-side pagination is enabled for a resource type, Rancher Dashboard uses a separate set of headers optimized for paginated data. These headers follow the same format as headers but do not support getValue — since rows are fetched page-by-page from the server, computed client-side values are not available.
Important: Property paths used to display, sort and search on must exist natively on the resource and not in the Dashboard's model for the resource type. For more information please see Choosing fields to display, sort and filter on.
Before using sspHeaders, server-side pagination must be enabled for the resource type via plugin.enableServerSidePagination. See Update Global Configuration for details on how to set this up.
Important: Server-side pagination via
sspHeadersis currently supported for global-level resources (e.g.provisioning.cattle.io.cluster) in the Rancherlocalcluster. Resources in downstream clusters are not yet supported byaddProduct.
Use sspHeaders alongside headers to define columns for both pagination modes:
import { NAME, STATE, AGE, NAMESPACE } from '@shell/config/table-headers';
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
headers: [STATE, NAME, NAMESPACE, AGE],
sspHeaders: [
{ name: 'name', label: 'Name', value: 'metadata.name' },
{ name: 'namespace', label: 'Namespace', value: 'metadata.namespace' },
],
};
You can also set sspHeaders without headers if you only need to customize the server-side pagination view:
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
sspHeaders: [
{ name: 'name', label: 'Name', value: 'metadata.name' },
{ name: 'status', label: 'Status', value: 'status.display' },
],
};
Note:
PaginationHeaderOptionsis the same asHeaderOptionsbut without thegetValueproperty. All other header properties (name,label,value,sort,search,formatter,formatterOpts,width) are supported.
Renaming types (overrideListResourceName)
Use overrideListResourceName to override the display name for a resource type in the list view. This maps the resource's internal type to a custom label via the DSL mapType method.
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
overrideListResourceName: 'Provisioned Clusters',
};
Hiding types from navigation (hideFromNav)
Set hideFromNav: true on a resource page to hide it from the side-menu entirely. The resource page is still accessible via its route — it just won't appear in the navigation. This is useful for resources that should be reachable from other pages but don't need a dedicated side-menu entry.
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
hideFromNav: true,
};
Hiding bulk actions (hideBulkActions)
Set hideBulkActions: true on a resource page to hide the bulk action buttons (e.g. "Delete") from the list view toolbar. Users can still perform actions on individual resources.
const clusterPage: ProductChildResourcePage = {
type: 'provisioning.cattle.io.cluster',
hideBulkActions: true,
};
Known limitation: Some built-in resource types in Rancher Dashboard have custom list views that explicitly control bulk action visibility. For those types,
hideBulkActionsmay have no effect. This may apply to a small number of resource types. If you notice that bulk actions are still showing after settinghideBulkActions: true, the resource type likely has a custom list view that overrides this setting. See #17426 for more details.
Rules & Constraints
When defining pages and groups in your product configuration, there are some important rules to keep in mind:
addProduct can only be called once per product
Each product can only be registered once via addProduct. Calling addProduct a second time with the same product name will throw an error. If you need to add pages to an existing product, use extendProduct instead.
// ✅ Correct
extension.addProduct(product, [pageA, pageB]);
// ❌ Error — same product registered twice
extension.addProduct(product, [pageA]);
extension.addProduct(product, [pageB]); // throws
Page types are mutually exclusive
A page item is either a custom page (has component) or a resource page (has type). Do not set both type and component on the same item. The API uses these properties to determine what kind of page to register:
componentdefined → registers a custom page (virtualType)typedefined → registers a resource page (configureType)
Resource pages cannot have children
Only custom pages and groups can have children. A resource page (an item with type) cannot be turned into a group — it uses Rancher Dashboard's built-in list/detail/edit views and does not support child navigation.
Groups cannot have a type property
A group (an item with children) cannot also have a type property. Groups are for organizing navigation — they do not display Kubernetes resources directly. Place resource pages as children of the group instead.
Naming requirements
- Custom pages and groups require a
nameproperty (unique identifier used in routes) - Resource pages require a
typeproperty (the Kubernetes resource type) - Custom pages and groups require either
labelorlabelKey(for display in the side-menu) - Resource pages do not need
labelorlabelKey— Rancher Dashboard resolves the display name from the resource schema
Page names must be unique within a product
Each custom page and group must have a unique name within the same product configuration. Registering two pages with the same name at the same level will throw an error at registration time.
Note: Pages with the same
namein different groups are allowed, because the group prefix makes their resolved names different (e.g.myapp-overviewandmyapp-admin-overvieware distinct).
Resource types must be unique within a product
Each resource page type must appear only once in a product configuration. Registering the same Kubernetes resource type twice will throw an error at registration time.
Ordering and default routes
- The first item in the config array (after ordering by
weight) becomes the product's default landing route - For groups without a
component, the default route falls through to the group's first child - Higher
weightvalues place items higher in the side-menu (bigger number on top) - When no weights are specified, the array order determines the side-menu order