Skip to main content
Version: Next

Frontend Plugins

NOTE: The new frontend system is in alpha and is only supported by a small number of plugins.

Introduction

Frontend plugins are a foundational building block in Backstage and the frontend system. They are used to encapsulate and provide functionality for a Backstage app, such as new pages, navigational elements, and APIs; as well as extensions and features for other plugins, such as entity page cards and content for the Software Catalog, or result list items for the search plugin.

Each plugin is typically shipped in a separate NPM package, whether that's a published package, or just in the local workspace. The plugin instance should always the default export of the package, either via the main entry-point or the /alpha sub-path export. Each plugin package is limited to exporting a single plugin instance. In a local workspace you could use a different structure if preferred, but this is considered a non-standard layout and should be avoided in published packages.

Creating a Plugin

Frontend plugin instances are created with the createFrontendPlugin function, which is provided by the @backstage/frontend-plugin-api package. It takes a single options object that provides all of the necessary configuration for the plugin. In particular you will want to provide extensions for your plugin, as that is the way that you can provide new functionality to the app.

// This creates a new extension, see "Extension Blueprints" documentation for more details
const myPage = PageBlueprint.make({
params: {
defaultPath: '/my-page',
loader: () => import('./MyPage').then(m => <m.MyPage />),
},
});

export default createFrontendPlugin({
id: 'my-plugin',
extensions: [myPage],
});

pluginId option

Each plugin needs an ID, which is used to uniquely identify the plugin within an entire Backstage system. The ID does not have to be globally unique across all of the NPM ecosystem, although you generally want to strive for that. It is not possible to install multiple plugins with the same ID in a single Backstage app.

The plugin ID should generally be part of the of the package name and use kebab-case. See both the frontend naming patterns section, as well as the package metadata section for more information.

extensions option

These are the extensions that the plugin provides to the app. Note that you should not export any of these extensions separately from the plugin package, as they can already by accessed via the getExtension method of the plugin instance using the extension ID.

The extensions that you provide to a plugin will have their namespace set to the plugin ID by default. For example, if you create an extension using the PageBlueprint without any particular naming options and install that via a plugin with the ID my-plugin, the final extension ID will be page:my-plugin. You can read more about how this works in the extension structure documentation.

routes and externalRoutes options

These are the routes that the plugin exposes to the app. The routes option declares all of the target routes that your plugin provides, i.e. routes that other plugins link to. The externalRoutes option instead declares all the outgoing routes, i.e. routes that your plugins links to, which you can bind to the routes of other plugins. See the routes documentation for more information how to set up cross-plugin navigation.

featureFlags option

This is a list of feature flag declarations that your plugin provides to the app. This makes sure that the feature flags are correctly registered and can be toggled in the app. To read a feature flag you can use the feature flags Utility API, accessible via featureFlagsApiRef.

Installing a Plugin in an App

A plugin instance is considered a frontend feature and can be installed directly in any Backstage frontend app. See the app documentation for more information about the different ways in which you can install new features in an app.

Overriding a Plugin

A plugin might not always behave exactly the way you want. It could be that you want to remove particular extensions, decorate them a bit, replace them with your own, or simply add new ones. Regardless of your exact use-case, you can use the plugin.withOverrides method to create a new copy of the plugin with the desired changes. When doing so you can also access the original extensions provided by the plugin, and use the extension override API to make changes to individual extensions:

import plugin from '@backstage/plugin-catalog';

export default plugin.withOverrides({
// These overrides are merged with the original extensions
extensions: [
// Override the catalog nav item to use a custom icon
plugin.getExtension('nav-item:catalog').override({
factory: origFactory => [
NavItemBlueprint.dataRefs.target({
...origFactory().get(NavItemBlueprint.dataRefs.target),
icon: CustomCatalogIcon,
}),
],
}),
// Override the catalog index page with a completely custom implementation
PageBlueprint.make({
params: {
defaultPath: '/catalog',
routeRef: plugin.routes.catalogIndex,
loader: () => import('./CustomCatalogIndexPage').then(m => <m.Page />),
},
}),
],
});

You can keep the plugin override in your app package, but it can often be a good idea to separate it out into its own package, especially if the overrides are complex or you want distinct ownership of the override. For example, if you are overriding the @backstage/plugin-catalog plugin, you might create a new package called @internal/plugin-catalog at plugins/catalog in your workspace, which exports the overridden plugin instance.