MCP server for DuckDB with interactive table viewer
Project description
sqlnow-mcp
An MCP server for DuckDB with an interactive table viewer. Point it at a data directory, attach files or external databases, and explore data through an LLM host such as Cursor or Claude Desktop. Query results from run_query, sample_table, and value_counts render as a scrollable grid inside the host via MCP Apps. Use run_query_json for small SELECT results the model needs as JSON (aggregations, counts, spot checks) — not for large exports.
Requirements
- Python 3.11+
- uv (recommended) or pip
- Node.js 18+ (only if you need to rebuild the table UI)
Install
From a clone of this repo:
uv sync
Or run without installing globally:
uvx sqlnow-mcp --mode local --data-dir ~/mydata
MCP server
Two modes are available:
- local — read/write; point at a data directory, switch databases, attach files and external connections.
- publish — read-only; single database with curated
metadata.yaml(and optionalstats.json) for sharing a branded dataset.
Transport (stdio, HTTP) works the same in both modes.
Local mode
sqlnow-mcp --mode local --data-dir ~/mydata
--data-dir is the working directory for .db files and attachable data. Session state (active database, attached sources) persists for the lifetime of the server process.
Publish mode
sqlnow-mcp --mode publish \
--db ~/datasets/gemlive.db \
--metadata ~/datasets/metadata.yaml \
--stats ~/datasets/stats.json
--stats is optional. The server name and instructions come from metadata (short_name, welcome, example_questions, etc.). Queries are SELECT-only with a default 10s timeout, 50% memory limit, and 1 DuckDB thread (all overridable).
stdio (Claude Desktop):
{
"mcpServers": {
"gem-data": {
"command": "uvx",
"args": [
"sqlnow-mcp", "--mode", "publish",
"--db", "/path/to/gemlive.db",
"--metadata", "/path/to/metadata.yaml"
]
}
}
}
HTTP:
sqlnow-mcp --mode publish \
--db ~/datasets/gemlive.db \
--metadata ~/datasets/metadata.yaml \
--transport streamable-http \
--host 127.0.0.1 \
--port 8000
{
"mcpServers": {
"gem-data": {
"url": "http://127.0.0.1:8000/mcp"
}
}
}
Publish workflow: call database_info() first, then list_tables / describe_table (with metadata descriptions), then run_query_json or run_query. Publish mode exposes only native tables in the database file — attached databases (sidecar) are not supported.
Generate a starter metadata file from a local database:
sqlnow-mcp --data-dir ./data -d demo generate-metadata -o metadata.yaml
sqlnow-mcp --data-dir ./data -d demo generate-stats -o stats.json
By default both commands include only native tables, not tables from attached databases.
Options
Shared (local and publish server startup):
| Option | Default | Description |
|---|---|---|
--transport |
stdio |
MCP transport: stdio, http, streamable-http, or sse |
--host |
127.0.0.1 |
Bind address when using an HTTP transport |
--port |
8000 |
Port when using an HTTP transport |
Local mode only:
| Option | Default | Description |
|---|---|---|
--data-dir |
. |
Directory containing DuckDB .db files |
--allow-path |
$HOME |
Extra directories allowed for attach_file (repeatable) |
--allow-external / --no-allow-external |
on | Allow attaching Postgres, SQLite, and MySQL databases |
Publish mode only:
| Option | Default | Description |
|---|---|---|
--db |
— | DuckDB file (required) |
--metadata |
— | metadata.yaml (required) |
--stats |
— | Precomputed stats.json (optional) |
--strict-metadata |
off | Fail startup on metadata/DB mismatches |
--query-timeout |
10 |
Per-query timeout in seconds |
--memory-limit |
50% |
DuckDB memory_limit (percentage or absolute, e.g. 2GiB) |
--threads |
1 |
DuckDB worker threads |
--max-temp-directory-size |
— | Cap disk spill for large queries (e.g. 10GiB) |
For security, --allow-path / is rejected (it would allow reading any file on disk).
stdio (default)
Best for editor integration. The host spawns the process and talks over stdin/stdout. Works for both --mode local and --mode publish.
Cursor — add to .cursor/mcp.json (project) or ~/.cursor/mcp.json (global):
{
"mcpServers": {
"sqlnow": {
"command": "uv",
"args": [
"run",
"--directory",
"/absolute/path/to/sqlnow-mcp",
"sqlnow-mcp",
"--mode",
"local",
"--data-dir",
"/home/you/mydata"
]
}
}
}
If the package is installed or you use uvx, you can omit --directory:
{
"mcpServers": {
"sqlnow": {
"command": "uvx",
"args": [
"sqlnow-mcp",
"--mode",
"local",
"--data-dir",
"/home/you/mydata"
]
}
}
}
Claude Desktop — add to claude_desktop_config.json:
{
"mcpServers": {
"sqlnow": {
"command": "uvx",
"args": ["sqlnow-mcp", "--mode", "local", "--data-dir", "/home/you/mydata"]
}
}
}
Restart the host after changing MCP config.
HTTP
Run the server as a local HTTP service, then point the host at a URL instead of spawning a command. Works for both --mode local and --mode publish.
sqlnow-mcp --mode local \
--data-dir ~/mydata \
--transport streamable-http \
--host 127.0.0.1 \
--port 8000
Cursor / Claude Desktop:
{
"mcpServers": {
"sqlnow": {
"url": "http://127.0.0.1:8000/mcp"
}
}
}
Use http or sse instead of streamable-http if your MCP client expects a different transport.
Typical workflow
The server exposes tools for database discovery, schema inspection, profiling, and SQL. A sensible order for the LLM (or you, when calling tools manually) is:
list_databases— find.dbfiles in the data directorycreate_database— create a new empty database (optional)use_databaseoruse_memory— open a session (required before most other tools)attach_file/attach_database— load CSV, Parquet, JSON, XLSX, or external DBslist_tables— every table includes column names and types; calldescribe_tablefor one tablesample_table,profile_table,value_counts— build context before writing SQL (no ad-hocSELECT ... LIMIT 2for schema)run_query_json— small SELECT for model-side analysis (aggregations, counts, low limits); full JSON in context — not for large result setsrun_query— larger or user-visible SELECT results in the table viewer
Schema vs. results: After list_tables, column names and types are already available — do not use run_query with a small LIMIT just to inspect structure. Use describe_table, profile_table, or run_query_json with a low limit instead.
Tools that open the interactive table viewer (run_query, sample_table, value_counts) are wired to the MCP Apps UI resource ui://sqlnow/table.html. Large run_query results are paginated internally; the viewer calls fetch_table_page to load additional rows. run_query_json inlines all returned rows as JSON in the tool response — keep result sets small and aggregate in SQL when possible.
Dev CLI
The same entry point also provides subcommands for testing without an MCP host:
# Create and use a database
sqlnow-mcp --data-dir ./data create-database demo
sqlnow-mcp --data-dir ./data -d demo attach-file ./samples/events.csv
sqlnow-mcp --data-dir ./data -d demo list-tables
sqlnow-mcp --data-dir ./data -d demo generate-metadata -o metadata.yaml
sqlnow-mcp --data-dir ./data -d demo generate-stats -o stats.json
sqlnow-mcp --data-dir ./data -d demo query "SELECT * FROM events LIMIT 10"
Run sqlnow-mcp --help for the full command list.
Building the table UI
The shipped artifact is a single self-contained HTML file at sqlnow_mcp/ui/table.html. The MCP server serves it as the ui://sqlnow/table.html resource (see sqlnow_mcp/ui.py). You normally use the committed copy as-is; rebuild only after changing the React source under ui/.
Source layout
ui/
├── index.html # Vite entry HTML
├── vite.config.ts # React + single-file bundle
├── package.json
└── src/
├── main.tsx
├── TableApp.tsx # glide-data-grid + @modelcontextprotocol/ext-apps
└── types.ts
Vite bundles React, glide-data-grid, and the MCP Apps client into one inline HTML file via vite-plugin-singlefile.
Build steps
cd ui
npm install
npm run build
The build script runs vite build and copies the output to the Python package:
ui/dist/index.html → sqlnow_mcp/ui/table.html
Commit sqlnow_mcp/ui/table.html if you changed the UI and want others to get the update without running Node.
Local UI preview
There is no dev script in package.json, but you can run Vite's dev server from ui/:
cd ui
npm install
npx vite
This serves ui/index.html with hot reload. MCP Apps integration (host postMessage, fetch_table_page) only works inside a real MCP host; use the built table.html with the running MCP server for end-to-end testing.
Development
uv sync --group dev
uv run pytest
License
See repository metadata and author information in pyproject.toml.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file sqlnow_mcp-0.1.0.tar.gz.
File metadata
- Download URL: sqlnow_mcp-0.1.0.tar.gz
- Upload date:
- Size: 279.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e0cbe80bbf77203f338eeeb1f5730da100e75094197bfebb5ff32e20f002c868
|
|
| MD5 |
fd949ec779793e4428572e53a07d9b57
|
|
| BLAKE2b-256 |
dddc5d12e9e44d9470d54661de8f92a3ac939dc8debc95e484e16da52f9bdeb2
|
File details
Details for the file sqlnow_mcp-0.1.0-py3-none-any.whl.
File metadata
- Download URL: sqlnow_mcp-0.1.0-py3-none-any.whl
- Upload date:
- Size: 284.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6bdcdb9b888555fd613cfcb9ae75027508dfd9c404ac8693a39f002d9d94fe7d
|
|
| MD5 |
da2fc050e68c36d2940248365992257b
|
|
| BLAKE2b-256 |
59c8a0feaf691b37377b84ea6b50d6eadb4524f683627c92e8d790a18abedd90
|