Architecture
Apache Superset's extension system is designed to enable powerful customization while maintaining stability, security, and performance. This page explains the architectural principles, system design, and technical mechanisms that make the extension ecosystem possible.
Architectural Principles
The extension architecture is built on six core principles that guide all technical decisions and ensure extensions can be developed safely and predictably:
1. Lean Core
Superset's core should remain minimal, with many features delegated to extensions. Built-in features use the same APIs and extension mechanisms available to external developers. This approach:
- Reduces maintenance burden and complexity
- Encourages modularity
- Allows the community to innovate independently of the main codebase
2. Explicit Contribution Points
All extension points are clearly defined and documented. Extension authors know exactly where and how they can interact with the host system. Each extension declares its capabilities in a metadata file, enabling the host to:
- Manage the extension lifecycle
- Provide a consistent user experience
- Validate extension compatibility
3. Versioned and Stable APIs
Public interfaces for extensions follow semantic versioning, allowing for:
- Safe evolution of the platform
- Backward compatibility
- Clear upgrade paths for extension authors
4. Lazy Loading and Activation
Extensions are loaded and activated only when needed, which:
- Minimizes performance overhead
- Reduces resource consumption
- Improves startup time
5. Composability and Reuse
The architecture encourages reusing extension points and patterns across different modules, promoting:
- Consistency across extensions
- Reduced duplication
- Shared best practices
6. Community-Driven Evolution
The system evolves based on real-world feedback and contributions. New extension points and capabilities are added as needs emerge, ensuring the platform remains relevant and flexible.
System Overview
The extension architecture is built around three main components that work together to create a flexible, maintainable ecosystem:
Core Packages
Two core packages provide the foundation for extension development:
Frontend: @apache-superset/core
This package provides essential building blocks for frontend extensions and the host application:
- Shared UI components
- Utility functions
- APIs and hooks
- Type definitions
By centralizing these resources, both extensions and built-in features use the same APIs, ensuring consistency, type safety, and a seamless user experience. The package is versioned to support safe platform evolution while maintaining compatibility.
Backend: apache-superset-core
This package exposes key classes and APIs for backend extensions:
- Database connectors
- API extensions
- Security manager customization
- Core utilities and models
It includes dependencies on critical libraries like Flask-AppBuilder and SQLAlchemy, and follows semantic versioning for compatibility and stability.
Developer Tools
apache-superset-extensions-cli
The CLI provides comprehensive commands for extension development:
- Project scaffolding
- Code generation
- Building and bundling
- Packaging for distribution
By standardizing these processes, the CLI ensures extensions are built consistently, remain compatible with evolving versions of Superset, and follow best practices.
Host Application
The Superset host application serves as the runtime environment for extensions:
Extension Management
- Exposes
/api/v1/extensionsendpoint for registration and management - Provides a dedicated UI for managing extensions
- Stores extension metadata in the
extensionsdatabase table
Extension Storage
The extensions table contains:
- Extension name, version, and author
- Contributed features and exposed modules
- Metadata and configuration
- Built frontend and/or backend code
Architecture Diagram
The following diagram illustrates how these components work together:
The diagram shows:
- Extension projects depend on core packages for development
- Core packages provide APIs and type definitions
- The host application implements the APIs and manages extensions
- Extensions integrate seamlessly with the host through well-defined interfaces
Extension Dependencies
Extensions can depend on any combination of packages based on their needs. For example:
Frontend-only extension (e.g., a custom chart type):
- Depends on
@apache-superset/corefor UI components and React APIs
Full-stack extension (e.g., a custom SQL editor with new API endpoints):
- Depends on
@apache-superset/corefor frontend components - Depends on
apache-superset-corefor backend APIs and models
This modular approach allows extension authors to choose exactly what they need while promoting consistency and reusability.
Dynamic Module Loading
One of the most sophisticated aspects of the extension architecture is how frontend code is dynamically loaded at runtime using Webpack's Module Federation.
Module Federation
The architecture leverages Webpack's Module Federation to enable dynamic loading of frontend assets. This allows extensions to be built independently from Superset.
How It Works
Extension Configuration
Extensions configure Webpack to expose their entry points:
new ModuleFederationPlugin({
name: 'my_extension',
filename: 'remoteEntry.[contenthash].js',
exposes: {
'./index': './src/index.tsx',
},
externalsType: 'window',
externals: {
'@apache-superset/core': 'superset',
},
shared: {
react: { singleton: true },
'react-dom': { singleton: true },
'antd-v5': { singleton: true }
}
})
This configuration does several important things:
exposes - Declares which modules are available to the host application. The extension makes ./index available as its entry point.
externals and externalsType - Tell Webpack that when the extension imports @apache-superset/core, it should use window.superset at runtime instead of bundling its own copy. This ensures extensions use the host's implementation of shared packages.
shared - Prevents duplication of common libraries like React and Ant Design. The singleton: true setting ensures only one instance of each library exists, avoiding version conflicts and reducing bundle size.
Runtime Resolution
The following diagram illustrates the module loading process:
Here's what happens at runtime:
- Extension Registration: When an extension is registered, Superset stores its remote entry URL
- Dynamic Loading: When the extension is activated, the host fetches the remote entry file
- Module Resolution: The extension imports
@apache-superset/core, which resolves towindow.superset - Execution: The extension code runs with access to the host's APIs and shared dependencies
Host API Setup
On the Superset side, the APIs are mapped to window.superset during application bootstrap:
import * as supersetCore from '@apache-superset/core';
import {
authentication,
core,
commands,
environment,
extensions,
sqlLab,
} from 'src/extensions';
export default function setupExtensionsAPI() {
window.superset = {
...supersetCore,
authentication,
core,
commands,
environment,
extensions,
sqlLab,
};
}
This function runs before any extensions are loaded, ensuring the APIs are available when extensions import from @apache-superset/core.
Benefits
This architecture provides several key benefits:
- Independent development: Extensions can be built separately from Superset's codebase
- Version isolation: Each extension can be developed with its own release cycle
- Shared dependencies: Common libraries are shared, reducing memory usage and bundle size
- Type safety: TypeScript types flow from the core package to extensions
Next Steps
Now that you understand the architecture, explore:
- Quick Start - Build your first extension
- Contribution Types - What kinds of extensions you can build
- Development - Project structure, APIs, and development workflow