@backstage/backend-dynamic-feature-service
This package provides experimental support for dynamic backend features (plugins and modules) in Backstage, allowing you to load plugins at runtime from a separate directory without including them in your main application's package.json.
Purpose
This enables:
- Plugin distribution: Distributing plugins as standalone artifacts
- Runtime flexibility: Adding or updating plugins without rebuilding the entire application
- Isolation: Keeping plugin-specific dependencies separate from core application dependencies
- Modular deployments: Different environments can load different sets of plugins
Important: This service handles loading only - packaging and distribution are separate concerns with multiple approaches available (see Packaging Approaches section).
Related Packages
@backstage/frontend-dynamic-feature-loader: Companion package for loading dynamic frontend plugins. This is only supported in the New Frontend System. For more details, checkout the README.
How it works
Core Architecture
The service consists of several key components:
- Plugin Scanner: Scans a configured directory (
dynamicPlugins.rootDirectory) for plugin packages
- Module Loader: Loads plugin modules
- Feature Loader: Integrates loaded plugins into the Backstage backend system
- Frontend Asset Server: Serves frontend plugin assets (JavaScript, CSS, manifests) over HTTP for dynamic frontend plugins
Loading Process
- Discovery: The scanner identifies valid plugin packages in the dynamic plugins root directory
- Validation: Each package is validated for required fields (
main, backstage.role)
- Loading: The module loader loads the plugin code
- Integration: Plugins are registered as Backstage features and made available to the backend
Usage
Basic Setup
Add the service to your backend application:
yarn add @backstage/backend-dynamic-feature-service
- Add one line to your backend:
import { createBackend } from '@backstage/backend-defaults';
import { dynamicPluginsFeatureLoader } from '@backstage/backend-dynamic-feature-service';
const backend = createBackend();
backend.add(dynamicPluginsFeatureLoader);
backend.start();
Configuration
Configure the dynamic plugins directory in your app-config.yaml:
dynamicPlugins:
rootDirectory: dynamic-plugins-root
Dynamic Plugin Packaging
Package Structure Requirements
Before exploring specific packaging approaches, it's important to understand what constitutes a valid dynamic plugin package. Dynamic plugins follow the basic rules of Node.js packages, including:
Package contents:
- package.json: Package metadata and configuration
- JavaScript files: Either loadable by the Node.js CommonJS module loaders for backend plugins, or RSPack/webpack assets for frontend plugins
- Configuration schema (optional): JSON schema file (typically
dist/.config-schema.json) for plugin configuration validation
- Private dependencies (optional): Embedded
node_modules folder containing private dependencies for backend plugins
Required package.json fields:
name: Package identifier
version: Package version
main: Entry point to the plugin's JavaScript code
backstage.role: Must be set to "backend-plugin" or "backend-plugin-module"
Packaging Approaches
Since this service only handles loading, you would choose a packaging approach based on your plugin's dependencies:
1. Simple npm pack
When to use: Plugin only uses dependencies that are already provided by the main Backstage application.
How to use:
cd my-backstage-plugin
yarn pack
mkdir -p /path/to/dynamic-plugins-root/my-backstage-plugin
tar -xzf package.tgz -C /path/to/dynamic-plugins-root/my-backstage-plugin --strip-components=1
Why this works: The plugin can resolve all its dependencies from the main application's node_modules.
Reality: Most plugins have private dependencies not available in the main application, so this approach has limited applicability.
2. Manual packaging with dependency installation
When to use: Plugin has private dependencies not available in the main Backstage application.
How to use:
cd my-backstage-plugin
yarn pack
mkdir -p /path/to/dynamic-plugins-root/my-backstage-plugin
tar -xzf package.tgz -C /path/to/dynamic-plugin-root/my-backstage-plugin --strip-components=1
cp yarn.lock /path/to/dynamic-plugin-root/my-backstage-plugin
cd /path/to/dynamic-plugins-root/my-backstage-plugin
yarn install
Why this works: Each plugin gets its own node_modules directory with all its dependencies.
Example scenario: Plugin needs axios@1.4.0 which isn't available in the main application.
3. Dedicated bundling CLI command
When to use:
- When you want to produce self-contained dynamic plugin packages that can be directly extracted and loaded without any post-action,
- especially when your plugin depends on other packages in the same monorepo.
How to use:
The backstage-cli package bundle command automates the required steps. Run it from within a plugin directory:
cd my-backstage-plugin
yarn backstage-cli package bundle --output-destination /path/to/dynamic-plugins-root
Batch bundling: When bundling many plugins from the same monorepo, use --pre-packed-dir to avoid redundant work:
backstage-cli build-workspace dist-workspace --alwaysPack ...plugin-packages
cd plugins/my-backstage-plugin
backstage-cli package bundle --pre-packed-dir ../../dist-workspace
See the full list of options in the CLI reference.
What the command does:
- Builds the plugin — Produces CJS output for the backend plugin and its transitively-required monorepo packages (
*-node or *-common)
- Packs local packages — Resolves
workspace:^ and backstage:^ dependencies on both the main plugin package and its transitively-required monorepo packages (*-node or *-common)
- Installs private dependencies — Seeds a lockfile from the plugin source monorepo and prunes it, then installs a private
node_modules containing all required dependencies
- Collects configuration schemas — Gathers plugin config schemas and writes them to
dist/.config-schema.json so they are available for validation at load time
Benefits:
- Self-contained packages with all necessary dependencies
- No post-installation steps required (extract and run)
- Consistent dependency structure across all dynamic plugins
- Automatic configuration schema collection
- Production-ready distribution format