Backstage
    Preparing search index...

    Module @backstage/backend-dynamic-feature-service

    @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.

    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).

    • @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.

    The service consists of several key components:

    1. Plugin Scanner: Scans a configured directory (dynamicPlugins.rootDirectory) for plugin packages
    2. Module Loader: Loads plugin modules
    3. Feature Loader: Integrates loaded plugins into the Backstage backend system
    4. Frontend Asset Server: Serves frontend plugin assets (JavaScript, CSS, manifests) over HTTP for dynamic frontend plugins
    1. Discovery: The scanner identifies valid plugin packages in the dynamic plugins root directory
    2. Validation: Each package is validated for required fields (main, backstage.role)
    3. Loading: The module loader loads the plugin code
    4. Integration: Plugins are registered as Backstage features and made available to the backend

    Add the service to your backend application:

    1. Install the package:
    yarn add @backstage/backend-dynamic-feature-service
    
    1. Add one line to your backend:
    import { createBackend } from '@backstage/backend-defaults';
    import { dynamicPluginsFeatureLoader } from '@backstage/backend-dynamic-feature-service';

    const backend = createBackend();
    // Add this line to enable dynamic plugin support:
    backend.add(dynamicPluginsFeatureLoader);
    // ... your other plugins
    backend.start();

    Configure the dynamic plugins directory in your app-config.yaml:

    dynamicPlugins:
    rootDirectory: dynamic-plugins-root

    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"

    Since this service only handles loading, you would choose a packaging approach based on your plugin's dependencies:

    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
    # Results in: package.tgz

    # Extract to dynamic plugins directory
    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.

    When to use: Plugin has private dependencies not available in the main Backstage application.

    How to use:

    # Package the plugin
    cd my-backstage-plugin
    yarn pack

    # Extract and install dependencies
    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 # Installs all the plugin's dependencies

    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.

    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
    # Creates a self-contained bundle in the /path/to/dynamic-plugins-root/my-backstage-plugin/ sub-folder

    Batch bundling: When bundling many plugins from the same monorepo, use --pre-packed-dir to avoid redundant work:

    # First, build a shared dist workspace
    backstage-cli build-workspace dist-workspace --alwaysPack ...plugin-packages

    # Then bundle each plugin using the pre-packed output
    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:

    1. Builds the plugin — Produces CJS output for the backend plugin and its transitively-required monorepo packages (*-node or *-common)
    2. Packs local packages — Resolves workspace:^ and backstage:^ dependencies on both the main plugin package and its transitively-required monorepo packages (*-node or *-common)
    3. Installs private dependencies — Seeds a lockfile from the plugin source monorepo and prunes it, then installs a private node_modules containing all required dependencies
    4. 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

    Classes

    CommonJSModuleLoader
    DynamicPluginManager

    Interfaces

    BackendDynamicPlugin
    BackendPluginProvider
    BaseDynamicPlugin
    DynamicPluginManagerOptions
    DynamicPluginProvider
    DynamicPluginsFactoryOptions
    DynamicPluginsFrontendRemotesService
    DynamicPluginsSchemasOptions
    DynamicPluginsSchemasService
    FrontendDynamicPlugin
    FrontendPluginProvider
    ModuleLoader
    NewBackendPluginInstaller
    RemoteInfo
    ScannedPluginPackage

    Type Aliases

    AdditionalRemoteInfo
    BackendDynamicPluginInstaller
    CommonJSModuleLoaderOptions
    DynamicPlugin
    DynamicPluginsFeatureLoaderOptions
    DynamicPluginsRootLoggerFactoryOptions
    FrontendRemoteResolver
    FrontendRemoteResolverProvider
    RemoteInfoTypeEnum
    ScannedPluginManifest

    Variables

    dynamicPluginsFeatureDiscoveryLoader
    dynamicPluginsFeatureLoader
    dynamicPluginsFrontendSchemas
    dynamicPluginsFrontendServiceRef
    dynamicPluginsRootLoggerServiceFactory
    dynamicPluginsSchemasServiceFactory
    dynamicPluginsServiceFactory
    dynamicPluginsServiceRef

    Functions

    dynamicPluginsServiceFactoryWithOptions
    isBackendDynamicPluginInstaller