mirror of
https://github.com/apache/superset.git
synced 2026-04-07 10:31:50 +00:00
436 lines
16 KiB
Python
436 lines
16 KiB
Python
# Licensed to the Apache Software Foundation (ASF) under one
|
|
# or more contributor license agreements. See the NOTICE file
|
|
# distributed with this work for additional information
|
|
# regarding copyright ownership. The ASF licenses this file
|
|
# to you under the Apache License, Version 2.0 (the
|
|
# "License"); you may not use this file except in compliance
|
|
# with the License. You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing,
|
|
# software distributed under the License is distributed on an
|
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
# KIND, either express or implied. See the License for the
|
|
# specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
"""
|
|
FastMCP app factory and initialization for the MCP service.
|
|
This file provides a configurable factory function to create FastMCP instances
|
|
following the Flask application factory pattern. All tool modules should import
|
|
mcp from here and use @mcp.tool decorators.
|
|
"""
|
|
|
|
import logging
|
|
from typing import Any, Callable, Dict, List, Sequence, Set
|
|
|
|
from fastmcp import FastMCP
|
|
from fastmcp.server.middleware import Middleware
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def get_default_instructions(branding: str = "Apache Superset") -> str:
|
|
"""Get default instructions with configurable branding.
|
|
|
|
Args:
|
|
branding: Product name to use in instructions
|
|
(e.g., "ACME Analytics", "Apache Superset")
|
|
|
|
Returns:
|
|
Formatted instructions string with branding applied
|
|
"""
|
|
return f"""
|
|
You are connected to the {branding} MCP (Model Context Protocol) service.
|
|
This service provides programmatic access to {branding} dashboards, charts, datasets,
|
|
SQL Lab, and instance metadata via a comprehensive set of tools.
|
|
|
|
Available tools:
|
|
|
|
Dashboard Management:
|
|
- list_dashboards: List dashboards with advanced filters (1-based pagination)
|
|
- get_dashboard_info: Get detailed dashboard information by ID
|
|
- generate_dashboard: Create a dashboard from chart IDs
|
|
- add_chart_to_existing_dashboard: Add a chart to an existing dashboard
|
|
|
|
Dataset Management:
|
|
- list_datasets: List datasets with advanced filters (1-based pagination)
|
|
- get_dataset_info: Get detailed dataset information by ID (includes columns/metrics)
|
|
|
|
Chart Management:
|
|
- list_charts: List charts with advanced filters (1-based pagination)
|
|
- get_chart_info: Get detailed chart information by ID
|
|
- get_chart_preview: Get a visual preview of a chart with image URL
|
|
- get_chart_data: Get underlying chart data in text-friendly format
|
|
- generate_chart: Create and save a new chart permanently
|
|
- generate_explore_link: Create an interactive explore URL (preferred for exploration)
|
|
- update_chart: Update existing saved chart configuration
|
|
- update_chart_preview: Update cached chart preview without saving
|
|
|
|
SQL Lab Integration:
|
|
- execute_sql: Execute SQL queries and get results (requires database_id)
|
|
- open_sql_lab_with_context: Generate SQL Lab URL with pre-filled query
|
|
|
|
Schema Discovery:
|
|
- get_schema: Get schema metadata for chart/dataset/dashboard (columns, filters)
|
|
|
|
System Information:
|
|
- get_instance_info: Get instance-wide statistics, metadata, and current user identity
|
|
- health_check: Simple health check tool (takes NO parameters, call without arguments)
|
|
|
|
Available Resources:
|
|
- instance://metadata: Instance configuration, stats, and available dataset IDs
|
|
- chart://configs: Valid chart configuration examples and best practices
|
|
|
|
Available Prompts:
|
|
- quickstart: Interactive guide for getting started with the MCP service
|
|
- create_chart_guided: Step-by-step chart creation wizard
|
|
|
|
Recommended Workflows:
|
|
|
|
To create a chart:
|
|
1. list_datasets -> find a dataset
|
|
2. get_dataset_info(id) -> examine columns and metrics
|
|
3. generate_explore_link(dataset_id, config) -> preview interactively
|
|
4. generate_chart(dataset_id, config, save_chart=True) -> save permanently
|
|
|
|
To find your own charts/dashboards:
|
|
1. get_instance_info -> get current_user.id
|
|
2. list_charts(filters=[{{"col": "created_by_fk",
|
|
"opr": "eq", "value": current_user.id}}])
|
|
3. Or: list_dashboards(filters=[{{"col": "created_by_fk",
|
|
"opr": "eq", "value": current_user.id}}])
|
|
|
|
To explore data with SQL:
|
|
1. get_instance_info -> find database_id
|
|
2. execute_sql(database_id, sql) -> run query
|
|
3. open_sql_lab_with_context(database_id) -> open SQL Lab UI
|
|
|
|
generate_explore_link vs generate_chart:
|
|
- Use generate_explore_link for exploration (no permanent chart created)
|
|
- Use generate_chart with save_chart=True only when user wants to save permanently
|
|
|
|
Chart Types You Can CREATE with generate_chart/generate_explore_link:
|
|
- chart_type="xy", kind="line": Line chart for time series and trends
|
|
- chart_type="xy", kind="bar": Bar chart for category comparison
|
|
- chart_type="xy", kind="area": Area chart for volume visualization
|
|
- chart_type="xy", kind="scatter": Scatter plot for correlation analysis
|
|
- chart_type="table": Data table for detailed views
|
|
- chart_type="table", viz_type="ag-grid-table": Interactive AG Grid table
|
|
|
|
Time grain for temporal x-axis (time_grain parameter):
|
|
- PT1H (hourly), P1D (daily), P1W (weekly), P1M (monthly), P1Y (yearly)
|
|
|
|
Chart Types in Existing Charts (viewable via list_charts/get_chart_info):
|
|
- pie, big_number, big_number_total, funnel, gauge_chart
|
|
- echarts_timeseries_line, echarts_timeseries_bar, echarts_timeseries_area
|
|
- pivot_table_v2, heatmap_v2, sankey_v2, sunburst_v2, treemap_v2
|
|
- word_cloud, world_map, box_plot, bubble, mixed_timeseries
|
|
|
|
Query Examples:
|
|
- List all tables:
|
|
filters=[{{"col": "viz_type", "opr": "in", "value": ["table", "pivot_table_v2"]}}]
|
|
- List time series charts:
|
|
filters=[{{"col": "viz_type", "opr": "sw", "value": "echarts_timeseries"}}]
|
|
- Search by name: search="sales"
|
|
- My charts (use current_user.id from get_instance_info):
|
|
filters=[{{"col": "created_by_fk", "opr": "eq", "value": <user_id>}}]
|
|
- My dashboards:
|
|
filters=[{{"col": "created_by_fk", "opr": "eq", "value": <user_id>}}]
|
|
|
|
General usage tips:
|
|
- All listing tools use 1-based pagination (first page is 1)
|
|
- Use get_schema to discover filterable columns, sortable columns, and default columns
|
|
- Use 'filters' parameter for advanced queries with filter columns from get_schema
|
|
- IDs can be integer or UUID format where supported
|
|
- All tools return structured, Pydantic-typed responses
|
|
- Chart previews are served as PNG images via custom screenshot endpoints
|
|
|
|
Input format:
|
|
- Tool request parameters accept structured objects (dicts/JSON)
|
|
- When MCP_PARSE_REQUEST_ENABLED is True (default), string-serialized JSON is also
|
|
accepted as input, which works around double-serialization bugs in some MCP clients
|
|
|
|
Feature Availability:
|
|
- Call get_instance_info to discover accessible menus for the current user.
|
|
- Do NOT assume features exist; always check get_instance_info first.
|
|
|
|
If you are unsure which tool to use, start with get_instance_info
|
|
or use the quickstart prompt for an interactive guide.
|
|
|
|
When you first connect, call get_instance_info to learn the user's identity.
|
|
Greet them by their first name (from current_user) and offer to help.
|
|
"""
|
|
|
|
|
|
# For backwards compatibility, keep DEFAULT_INSTRUCTIONS pointing to default branding
|
|
DEFAULT_INSTRUCTIONS = get_default_instructions()
|
|
|
|
|
|
def _build_mcp_kwargs(
|
|
name: str,
|
|
instructions: str,
|
|
auth: Any | None,
|
|
lifespan: Callable[..., Any] | None,
|
|
tools: List[Any] | None,
|
|
include_tags: Set[str] | None,
|
|
exclude_tags: Set[str] | None,
|
|
middleware: Sequence[Middleware] | None = None,
|
|
**kwargs: Any,
|
|
) -> Dict[str, Any]:
|
|
"""Build FastMCP constructor arguments."""
|
|
mcp_kwargs: Dict[str, Any] = {
|
|
"name": name,
|
|
"instructions": instructions,
|
|
}
|
|
|
|
# Add optional parameters if provided
|
|
if auth is not None:
|
|
mcp_kwargs["auth"] = auth
|
|
if lifespan is not None:
|
|
mcp_kwargs["lifespan"] = lifespan
|
|
if tools is not None:
|
|
mcp_kwargs["tools"] = tools
|
|
if include_tags is not None:
|
|
mcp_kwargs["include_tags"] = include_tags
|
|
if exclude_tags is not None:
|
|
mcp_kwargs["exclude_tags"] = exclude_tags
|
|
if middleware is not None:
|
|
mcp_kwargs["middleware"] = middleware
|
|
|
|
# Add any additional kwargs
|
|
mcp_kwargs.update(kwargs)
|
|
return mcp_kwargs
|
|
|
|
|
|
def _apply_config(mcp_instance: FastMCP, config: Dict[str, Any] | None) -> None:
|
|
"""Apply additional configuration to FastMCP instance."""
|
|
if config:
|
|
for key, value in config.items():
|
|
setattr(mcp_instance, key, value)
|
|
|
|
|
|
def _log_instance_creation(
|
|
name: str,
|
|
auth: Any | None,
|
|
include_tags: Set[str] | None,
|
|
exclude_tags: Set[str] | None,
|
|
) -> None:
|
|
"""Log FastMCP instance creation details."""
|
|
logger.info("Created FastMCP instance: %s", name)
|
|
if auth:
|
|
logger.info("Authentication enabled")
|
|
if include_tags or exclude_tags:
|
|
logger.info(
|
|
"Tag filtering enabled - include: %s, exclude: %s",
|
|
include_tags,
|
|
exclude_tags,
|
|
)
|
|
|
|
|
|
def create_mcp_app(
|
|
name: str | None = None,
|
|
instructions: str | None = None,
|
|
branding: str | None = None,
|
|
auth: Any | None = None,
|
|
lifespan: Callable[..., Any] | None = None,
|
|
tools: List[Any] | None = None,
|
|
include_tags: Set[str] | None = None,
|
|
exclude_tags: Set[str] | None = None,
|
|
config: Dict[str, Any] | None = None,
|
|
middleware: Sequence[Middleware] | None = None,
|
|
**kwargs: Any,
|
|
) -> FastMCP:
|
|
"""
|
|
Application factory for creating FastMCP instances.
|
|
|
|
This follows the Flask application factory pattern, allowing users to
|
|
configure the FastMCP instance with custom authentication, middleware,
|
|
and other settings.
|
|
|
|
Args:
|
|
name: Human-readable server name
|
|
instructions: Server description and usage instructions
|
|
branding: Product name for instructions (e.g., "ACME Analytics")
|
|
auth: Authentication provider for securing HTTP transports
|
|
lifespan: Async context manager for startup/shutdown logic
|
|
tools: List of tools or functions to add to the server
|
|
include_tags: Set of tags to include (whitelist)
|
|
exclude_tags: Set of tags to exclude (blacklist)
|
|
config: Additional configuration dictionary
|
|
middleware: Sequence of middleware to apply to the server
|
|
**kwargs: Additional FastMCP constructor arguments
|
|
|
|
Returns:
|
|
Configured FastMCP instance
|
|
"""
|
|
# Default name if not provided
|
|
if name is None:
|
|
name = "MCP Server"
|
|
|
|
# Use default instructions if none provided
|
|
if instructions is None:
|
|
# If branding is provided, use it to generate instructions
|
|
if branding is not None:
|
|
instructions = get_default_instructions(branding)
|
|
else:
|
|
instructions = DEFAULT_INSTRUCTIONS
|
|
|
|
# Build FastMCP constructor arguments
|
|
mcp_kwargs = _build_mcp_kwargs(
|
|
name,
|
|
instructions,
|
|
auth,
|
|
lifespan,
|
|
tools,
|
|
include_tags,
|
|
exclude_tags,
|
|
middleware,
|
|
**kwargs,
|
|
)
|
|
|
|
# Create the FastMCP instance
|
|
mcp_instance = FastMCP(**mcp_kwargs)
|
|
|
|
# Apply any additional configuration
|
|
_apply_config(mcp_instance, config)
|
|
|
|
# Log instance creation
|
|
_log_instance_creation(name, auth, include_tags, exclude_tags)
|
|
|
|
return mcp_instance
|
|
|
|
|
|
# Create default MCP instance for backward compatibility
|
|
mcp = create_mcp_app()
|
|
|
|
# Initialize MCP dependency injection BEFORE importing tools/prompts
|
|
# This replaces the abstract @tool and @prompt decorators in superset_core.api.mcp
|
|
# with concrete implementations that can register with the mcp instance
|
|
from superset.core.mcp.core_mcp_injection import ( # noqa: E402
|
|
initialize_core_mcp_dependencies,
|
|
)
|
|
|
|
initialize_core_mcp_dependencies()
|
|
|
|
# Import all MCP tools to register them with the mcp instance
|
|
# NOTE: Always add new tool imports here when creating new MCP tools.
|
|
# Tools use the @tool decorator from `superset-core` and register automatically
|
|
# on import. Import prompts and resources to register them with the mcp instance
|
|
# NOTE: Always add new prompt/resource imports here when creating new prompts/resources.
|
|
# Prompts use @mcp.prompt decorators and resources use @mcp.resource decorators.
|
|
# They register automatically on import, similar to tools.
|
|
from superset.mcp_service.chart import ( # noqa: F401, E402
|
|
prompts as chart_prompts,
|
|
resources as chart_resources,
|
|
)
|
|
from superset.mcp_service.chart.tool import ( # noqa: F401, E402
|
|
generate_chart,
|
|
get_chart_data,
|
|
get_chart_info,
|
|
get_chart_preview,
|
|
list_charts,
|
|
update_chart,
|
|
update_chart_preview,
|
|
)
|
|
from superset.mcp_service.dashboard.tool import ( # noqa: F401, E402
|
|
add_chart_to_existing_dashboard,
|
|
generate_dashboard,
|
|
get_dashboard_info,
|
|
list_dashboards,
|
|
)
|
|
from superset.mcp_service.dataset.tool import ( # noqa: F401, E402
|
|
get_dataset_info,
|
|
list_datasets,
|
|
)
|
|
from superset.mcp_service.explore.tool import ( # noqa: F401, E402
|
|
generate_explore_link,
|
|
)
|
|
from superset.mcp_service.sql_lab.tool import ( # noqa: F401, E402
|
|
execute_sql,
|
|
open_sql_lab_with_context,
|
|
)
|
|
from superset.mcp_service.system import ( # noqa: F401, E402
|
|
prompts as system_prompts,
|
|
resources as system_resources,
|
|
)
|
|
from superset.mcp_service.system.tool import ( # noqa: F401, E402
|
|
get_instance_info,
|
|
get_schema,
|
|
health_check,
|
|
)
|
|
|
|
|
|
def init_fastmcp_server(
|
|
name: str | None = None,
|
|
instructions: str | None = None,
|
|
auth: Any | None = None,
|
|
lifespan: Callable[..., Any] | None = None,
|
|
tools: List[Any] | None = None,
|
|
include_tags: Set[str] | None = None,
|
|
exclude_tags: Set[str] | None = None,
|
|
config: Dict[str, Any] | None = None,
|
|
middleware: Sequence[Middleware] | None = None,
|
|
**kwargs: Any,
|
|
) -> FastMCP:
|
|
"""
|
|
Initialize and configure the FastMCP server.
|
|
|
|
This function configures the global MCP instance (which has all tools
|
|
already registered) with auth, middleware, and other settings.
|
|
|
|
Args:
|
|
name: Server name (defaults to "{APP_NAME} MCP Server")
|
|
instructions: Custom instructions (defaults to branded with APP_NAME)
|
|
auth, lifespan, tools, include_tags, exclude_tags, config: FastMCP configuration
|
|
middleware: Sequence of middleware to apply to the server
|
|
**kwargs: Additional FastMCP configuration
|
|
|
|
Returns:
|
|
The global FastMCP instance configured with the provided settings
|
|
"""
|
|
# Read branding from Flask config's APP_NAME
|
|
from superset.mcp_service.flask_singleton import app as flask_app
|
|
|
|
# Derive branding from Superset's APP_NAME config (defaults to "Superset")
|
|
app_name = flask_app.config.get("APP_NAME", "Superset")
|
|
branding = app_name
|
|
default_name = f"{app_name} MCP Server"
|
|
|
|
# Apply branding defaults if not explicitly provided
|
|
if name is None:
|
|
name = default_name
|
|
if instructions is None:
|
|
instructions = get_default_instructions(branding)
|
|
|
|
# Configure the global mcp instance with provided settings.
|
|
# Tools are already registered on this instance via @tool decorator imports above.
|
|
# name and instructions are read-only properties that delegate to _mcp_server
|
|
mcp._mcp_server.name = name
|
|
mcp._mcp_server.instructions = instructions
|
|
|
|
if auth is not None:
|
|
mcp.auth = auth
|
|
logger.info("Authentication configured on MCP instance")
|
|
|
|
if middleware is not None:
|
|
for mw in middleware:
|
|
mcp.add_middleware(mw)
|
|
logger.info("Added %d middleware(s) to MCP instance", len(middleware))
|
|
|
|
if lifespan is not None:
|
|
mcp.lifespan = lifespan
|
|
|
|
if include_tags is not None:
|
|
mcp.include_tags = include_tags
|
|
|
|
if exclude_tags is not None:
|
|
mcp.exclude_tags = exclude_tags
|
|
|
|
# Apply any additional configuration
|
|
_apply_config(mcp, config)
|
|
|
|
logger.info("Configured FastMCP instance: %s (auth=%s)", name, auth is not None)
|
|
return mcp
|