Skip to main content

Interactive UI Tools for MCP Servers

Project description

Gdansk: React Frontends for Python MCP Servers

[!WARNING] This project is currently in beta. The APIs are subject to change leading up to v1.0. The v1.0 release will coincide with the v2.0 release of the python mcp sdk

Installation

uv add gdansk

Skill for Coding Agents

If you use coding agents such as Claude Code or Cursor, add the gdansk skills to your repository:

npx skills add mplemay/gdansk

Then use:

  • $use-gdansk to bootstrap gdansk in a new repo or add another widget to an existing integration.
  • $debug-gdansk to diagnose widget path, bundling, render, and runtime failures in an existing gdansk setup.

Compatibility

  • Python: gdansk currently requires >=3.12,<3.15.
  • Frontend package: use an ESM package with @gdansk/vite, vite, @vitejs/plugin-react, react, react-dom, and @modelcontextprotocol/ext-apps. Inertia page mode also needs @inertiajs/react.
  • Runtime tooling: gdansk starts the frontend through uv run deno .... If you run frontend package scripts directly, the published @gdansk/vite package currently declares Node >=22.

Examples

  • FastAPI: Mounting the MCP app inside an existing FastAPI service.
  • inertia: Ship-backed Inertia pages for FastAPI with gdanskPages().
  • get-time: Small copyable widget example for first-time adoption in another repo.
  • production: Minimal production-rendered and hydrated widget example with a single tool.
  • shadcn: Multi-tool todo app with structured_output=True and shadcn/ui.

Inertia Pages

Ship can serve convention-driven Inertia pages directly: the first request returns an HTML shell, follow-up requests use the Inertia JSON protocol, and production assets still come from ship.assets.

Page mode is convention-driven. Put the root page at app/page.tsx, nested pages at app/**/page.tsx, and co-located layouts at app/**/layout.tsx. Render the root page with page.render("/"); nested folders map to slash-delimited component ids like page.render("dashboard/reports").

For FastAPI, inject the page with Depends(ship.page) and run the frontend with ship.lifespan(...). Call ship.inertia(...) only when you need non-default page settings such as a custom root id or explicit version.

type PageDependency = Annotated["InertiaPage", Depends(ship.page)]

Pair the backend with gdanskPages() in your frontend vite.config.ts:

import react from "@vitejs/plugin-react";
import { defineConfig } from "vite";
import { gdanskPages } from "@gdansk/vite";

export default defineConfig({
  plugins: [gdanskPages({ refresh: true }), react()],
});

For a full FastAPI example with validation errors, flash messages, deferred props, once props, merge helpers, scroll props, and fragment redirects, see examples/inertia.

The backend helper surface is now close to the official non-SSR Inertia protocol:

  • prop(value) creates a fluent prop builder.
  • optional(value), always(value), and defer(value, group=...) control eager vs partial/deferred loading.
  • once(value, key=...) and page.share_once(...) emit onceProps so the client can reuse previously loaded data.
  • merge(value) / deep_merge(value, match_on=...) and prop(...).append(...) / .prepend(...) emit merge metadata.
  • scroll(...) emits both merge metadata and scrollProps for infinite-scroll style payloads.
  • page.encrypt_history(...), page.clear_history(), and page.redirect(..., preserve_fragment=True) control history and redirect behavior.
from gdansk import deep_merge, merge, once, prop, scroll

page.share_once(sessionToken=load_session_token)

return await page.render(
    "/",
    {
        "announcements": merge(load_announcements()).append(match_on="id"),
        "conversation": deep_merge(load_conversation(), match_on="messages.id"),
        "feed": scroll(
            load_feed(),
            items_path="items",
            current_page_path="pagination.current",
            next_page_path="pagination.next",
            previous_page_path="pagination.previous",
            page_name="feed_page",
        ),
        "profile": once(load_profile, key="shared-profile"),
        "stats": prop(load_stats).optional(),
    },
)

Quick Start

Here's a complete example showing how to build a simple greeting tool with a React UI:

Project Structure:

my-mcp-server/
├── server.py
└── frontend/
    ├── package.json
    ├── vite.config.ts
    └── widgets/
        └── hello/
            └── widget.tsx

The frontend folder name is only an example. Pass any frontend package root to Vite(...). That frontend package owns its own vite.config.ts; import @gdansk/vite there alongside any framework plugins.

server.py:

from collections.abc import AsyncIterator
from contextlib import asynccontextmanager
from pathlib import Path

import uvicorn
from mcp.server import MCPServer
from mcp.types import TextContent
from starlette.middleware.cors import CORSMiddleware

from gdansk import Ship, Vite

frontend_path = Path(__file__).parent / "frontend"
ship = Ship(vite=Vite(frontend_path))


@ship.widget(path=Path("hello/widget.tsx"), name="greet")
def greet(name: str) -> list[TextContent]:
    """Greet someone by name."""
    return [TextContent(type="text", text=f"Hello, {name}!")]


@asynccontextmanager
async def lifespan(mcp: MCPServer) -> AsyncIterator[None]:
    async with ship.lifespan(mcp=mcp, watch=True):
        yield


mcp = MCPServer(name="Hello World Server", lifespan=lifespan)


def main() -> None:
    app = mcp.streamable_http_app()
    app.add_middleware(
        CORSMiddleware,
        allow_origins=["*"],
        allow_methods=["*"],
        allow_headers=["*"],
    )
    app.mount(path=ship.assets_path, app=ship.assets)
    uvicorn.run(app, port=3000)


if __name__ == "__main__":
    main()

frontend/widgets/hello/widget.tsx:

import { useApp } from "@modelcontextprotocol/ext-apps/react";
import { useState } from "react";

export default function App() {
  const [name, setName] = useState("");
  const [greeting, setGreeting] = useState("");

  const { app, error } = useApp({
    appInfo: { name: "Greeter", version: "1.0.0" },
    capabilities: {},
  });

  if (error) return <div>Error: {error.message}</div>;
  if (!app) return <div>Connecting...</div>;

  return (
    <main>
      <h2>Say Hello</h2>
      <input
        value={name}
        onChange={(e) => setName(e.target.value)}
        placeholder="Enter your name..."
      />
      <button
        onClick={async () => {
          const result = await app.callServerTool({
            name: "greet",
            arguments: { name },
          });
          const text = result.content?.find((c) => c.type === "text");
          if (text && "text" in text) setGreeting(text.text);
        }}
      >
        Greet Me
      </button>
      {greeting && <p>{greeting}</p>}
    </main>
  );
}

frontend/package.json:

{
  "name": "my-mcp-frontend",
  "private": true,
  "type": "module",
  "dependencies": {
    "@gdansk/vite": "^0.1.0",
    "@modelcontextprotocol/ext-apps": "^1.5.0",
    "@vitejs/plugin-react": "^6.0.1",
    "react": "^19.2.5",
    "react-dom": "^19.2.5",
    "vite": "^8.0.8"
  },
  "devDependencies": {
    "@types/react": "^19.2.14",
    "@types/react-dom": "^19.2.3"
  }
}

frontend/vite.config.ts:

import react from "@vitejs/plugin-react";
import { defineConfig } from "vite";
import gdansk from "@gdansk/vite";

export default defineConfig({
  plugins: [gdansk({ refresh: true }), react()],
});

@gdansk/vite now provides a default @ alias that points at the frontend package root, so you only need a manual alias when you want @ to resolve somewhere else. Use refresh: true to trigger full browser reloads when nearby Python or Jinja files change during development.

For widget-based MCP apps, ship.lifespan(..., watch=...) controls how the frontend is prepared:

  • watch=True — runs the Vite dev server in the background with React refresh; JS/CSS load from the Vite origin.
  • watch=False (default) — runs vite build on startup, then serves static hydration assets and the gdansk manifest from ship.assets.
  • watch=None — skips Vite/Deno entirely and loads an existing gdansk-manifest.json under the assets directory. Use this when assets are prebuilt (for example in CI) to avoid cold-start build cost.

If you need a non-default build output directory, keep the Vite plugin and Python runtime aligned. Widget sources always live under widgets/ at the frontend package root (Vite(root=...) / Vite root).

ship = Ship(
    vite=Vite(
        Path(__file__).parent / "frontend",
        build_directory="public/ui",
    ),
)
export default defineConfig({
  plugins: [
    gdansk({
      buildDirectory: "public/ui",
      refresh: true,
    }),
    react(),
  ],
});

Production widgets load their hydration assets from ship.assets_path. Mount ship.assets at that path on the public app; with the default settings this is /dist.

The default production output now mirrors Vite/Laravel conventions more closely:

  • standard Vite manifest: dist/manifest.json
  • gdansk runtime manifest: dist/gdansk-manifest.json
  • stable widget entries: dist/<widget>/client.js and dist/<widget>/client.css
  • shared hashed assets and chunks: dist/assets/*

If your MCP client renders widget HTML on a different origin, pass base_url to Ship so production asset URLs point back to your public app instead of the client host:

ship = Ship(vite=Vite(Path(__file__).parent / "frontend"), base_url="https://example.com")

If you want a different dev runtime host or port, configure both sides explicitly:

from gdansk import Ship, Vite

ship = Ship(vite=Vite(Path(__file__).parent / "frontend", host="127.0.0.1", port=14000))
export default defineConfig({
  plugins: [gdansk({ host: "127.0.0.1", port: 14000, refresh: true }), react()],
});

Install the frontend package dependencies from frontend/ after editing them:

cd frontend
uv run deno install

Gdansk mounts your default export into #root automatically and wraps it with React.StrictMode.

Run the server with uv run python server.py, configure it in your MCP client (like Claude Desktop), and you'll have an interactive greeting tool ready to use.

Why Use Gdansk?

  1. Python Backend, React Frontend — Use familiar technologies you already know. Write your logic in Python with type hints, build your UI in React/TypeScript. No need to learn a new framework-specific language.

  2. Built for MCP — Composes with MCPServer from the official Python SDK: register widget tools and HTML resources via Ship, wire them in with ship.lifespan(mcp=...), and integrate with Claude Desktop and other MCP clients.

  3. Fast bundling with Rolldown — The Rolldown bundler processes your TypeScript/JSX automatically. Hot-reload in development mode means you see changes instantly without manual rebuilds.

  4. Type-Safe — Full type safety across the stack. Python type hints on the backend, TypeScript on the frontend, with automatic type checking via ruff and TypeScript compiler.

  5. Developer-Friendly — Simple decorator API (@ship.widget()), automatic resource registration, dev mode on ship.lifespan(...), and comprehensive error messages. Get started in minutes, not hours.

  6. Production Ready — Comprehensive test suite covering Python 3.12+ across Linux, macOS, and Windows. Used in production MCP servers with proven reliability.

Credits

Gdansk builds on the shoulders of giants:

Special thanks to the Model Context Protocol team at Anthropic for creating the MCP standard and the @modelcontextprotocol/ext-apps package.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

gdansk-0.8.0.tar.gz (40.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

gdansk-0.8.0-py3-none-any.whl (49.4 kB view details)

Uploaded Python 3

File details

Details for the file gdansk-0.8.0.tar.gz.

File metadata

  • Download URL: gdansk-0.8.0.tar.gz
  • Upload date:
  • Size: 40.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.8 {"installer":{"name":"uv","version":"0.11.8","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for gdansk-0.8.0.tar.gz
Algorithm Hash digest
SHA256 da5c43075eacfc0f09f43847e4312b485a81257e278e8bb3969b6fbdfbfd01fa
MD5 acef58bcfb6649597e9ea2be7a099196
BLAKE2b-256 6192f79a1ac0282ac05cbe93d211f9ecde1105a967b4f4f598862a3d6bc63205

See more details on using hashes here.

File details

Details for the file gdansk-0.8.0-py3-none-any.whl.

File metadata

  • Download URL: gdansk-0.8.0-py3-none-any.whl
  • Upload date:
  • Size: 49.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.8 {"installer":{"name":"uv","version":"0.11.8","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for gdansk-0.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 08a6f83b54247bb2f563c70b8fc4e512cf528b56975b65c0176114486508bb76
MD5 d8f5f94a90f499f7121bef7f0aaf4bc3
BLAKE2b-256 045e2b4d8da418a52f5ae58d8c8192370b0fcdc41204bb9687fda6c6854af7f0

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page