Skip to main content
Version: v3

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.

TermDescription
ProductA top-level view in Rancher Dashboard that adds a navigation entry into the top-level slide-in menu (e.g. Fleet, Cluster Management)
Custom pageA page inside a product that renders a Vue component you provide. Equivalent to defining a virtualType in the DSL approach
Resource pageA 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
GroupA collapsible folder/group in the product's side-menu that contains pages or other groups
Single page productA 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:

ParameterTypeDescription
productProductMetadataProduct name, label, icon, and other metadata
configProductChild[]Array of pages or groups that form the product's side-menu

addProduct(product)

Register a single page product (no side-menu).

Parameters:

ParameterTypeDescription
productProductSinglePageProduct metadata plus a component to render

extendProduct(product, config)

Add pages to an existing Rancher Dashboard product.

Parameters:

ParameterTypeDescription
productStandardProductNameOne of: 'explorer', 'manager', 'settings', 'auth'
configProductChild[]Array of pages or groups to add to the product

Usage

Note: We recommend building up configuration objects as separate const variables before passing them to addProduct or extendProduct. This gives you better TypeScript intellisense and autocomplete, and avoids some of the confusion when creating the objects directly in addProduct or extendProduct.

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:

NameDescription
'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.

PropertyTypeRequiredDescription
namestringYesUnique product identifier
labelstringYes*Display label (* either label or labelKey is required)
labelKeystringYes*Translation key for the label
iconstringNoIcon name (based on rancher icons). Defaults to 'extension'
weightnumberNoSide-menu ordering (bigger number on top)
showClusterSwitcherbooleanNoShow the cluster switcher in navigation
showNamespaceFilterbooleanNoShow the namespace filter in the header

ProductSinglePage

A product that renders a single full-page Vue component with no side-menu. Extends ProductMetadata with:

PropertyTypeRequiredDescription
componentRouteComponentYesVue 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.

PropertyTypeRequiredDescription
namestringYesUnique page identifier
labelstringYes*Display label (* either label or labelKey is required)
labelKeystringYes*Translation key for the label
componentRouteComponentYesVue component to render
weightnumberNoSide-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.

PropertyTypeRequiredDescription
typestringYesKubernetes resource type (e.g. 'provisioning.cattle.io.cluster')
weightnumberNoSide-menu ordering (bigger number on top)
configConfigureTypeConfigurationNoResource page options (creatable, editable, removable, etc.)
headersHeaderOptions[]NoCustom table column headers for the list view (client-side pagination). See Table headers
sspHeadersPaginationHeaderOptions[]NoCustom table column headers for the list view (server-side pagination). See Server-side pagination headers
overrideListResourceNamestringNoOverride the display name for this resource type in the list view. See Renaming types
hideFromNavbooleanNoHide this resource type from the side-menu entirely. See Hiding types
hideBulkActionsbooleanNoHide 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).

PropertyTypeRequiredDescription
namestringYesUnique group identifier
labelstringYes*Display label (* either label or labelKey is required)
labelKeystringYes*Translation key for the label
childrenProductChild[]YesArray of pages or nested groups
componentRouteComponentNoOptional component for the group's own page
weightnumberNoSide-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 component
  • ProductChildResourcePage — 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:

HeaderDescription
NAMEResource name with a clickable link to the detail view (uses LinkDetail formatter)
STATEResource state with a colored badge (uses BadgeStateFormatter)
AGEResource age since creation (uses LiveDate formatter)
NAMESPACEResource 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:

FormatterDescription
LinkDetailRenders the value as a clickable link to the resource detail page
BadgeStateFormatterRenders a colored state badge
LiveDateRenders a relative timestamp that updates in real-time (e.g. "5 minutes ago")
LinkRenders 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 PropertyTypeDescription
namestringUnique column identifier. Must be unique within the headers array
labelstringColumn header text displayed at the top of the column
labelKeystringTranslation key for the column header (alternative to label)
valuestringDot-notation path to the value on the row object (e.g. 'metadata.name', 'status.provider')
sortstring | string[] | booleanPath(s) used for sorting. Use an array for multi-field sort (e.g. ['stateSort', 'nameSort']). Set to false to disable sorting
searchstring | booleanPath used for search/filtering. Set to false to exclude this column from search
formatterstringName of a built-in formatter component from @shell/components/formatter/ or a custom formatter defined in your extension
formatterOptsanyConfiguration options passed to the formatter component
widthnumberFixed column width in pixels. If not set, the column auto-sizes
getValue(row: any) => string | number | nullCustom 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 sspHeaders is currently supported for global-level resources (e.g. provisioning.cattle.io.cluster) in the Rancher local cluster. Resources in downstream clusters are not yet supported by addProduct.

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: PaginationHeaderOptions is the same as HeaderOptions but without the getValue property. 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, hideBulkActions may have no effect. This may apply to a small number of resource types. If you notice that bulk actions are still showing after setting hideBulkActions: 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:

  • component defined → registers a custom page (virtualType)
  • type defined → 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 name property (unique identifier used in routes)
  • Resource pages require a type property (the Kubernetes resource type)
  • Custom pages and groups require either label or labelKey (for display in the side-menu)
  • Resource pages do not need label or labelKey — 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 name in different groups are allowed, because the group prefix makes their resolved names different (e.g. myapp-overview and myapp-admin-overview are 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 weight values place items higher in the side-menu (bigger number on top)
  • When no weights are specified, the array order determines the side-menu order