@idle-sync/db-conn-mcp
v0.3.0
Published
A dead-simple, self-hosted MCP server for securely querying databases via AI agents.
Downloads
64
Maintainers
Readme
db-conn-mcp (Node)
A dead-simple, self-hosted Model Context Protocol (MCP) server for querying your databases with AI agents (Claude, Cursor, Windsurf, VS Code, Zed, and more).
It does one thing well: let an agent safely explore and query a database you point it at — with security delegated to the simplest possible primitives (a static JSON file and your database's own read-only transactions), not custom auth servers or fragile SQL parsing.
This is the Node/TypeScript port of db-conn-mcp (originally Python). Same configuration, same tools, same security model — published to npm so you can run it with
npx/npminstead ofpipx/pip.
v1 ships PostgreSQL only. All database-specific code lives behind a
Dialectseam, so adding MySQL/SQLite later is a single new file.
Why
- Read stays read. A
readdatabase runs every query in a native read-only transaction, and the read tool only accepts a single read-only statement (SELECT/WITH/VALUES/TABLE/SHOW/EXPLAIN) — so an agent can't slip in a write or aSET … READ WRITEto flip the session. For a hard, privilege-level guarantee that holds no matter what, point the DSN at a read-only database role (see Use a read-only role). - No secret leaks. DSNs/passwords are never logged or returned by any tool. Connection failures come back as sanitized diagnostics (a category + fix), never a raw traceback with your host and credentials in it.
- Tiered write safety. Writes are gated server-side:
mode(hard, native) →yolo(per-database trust) →user_consent(explicit per-operation approval). - Zero-friction setup. An interactive wizard registers your database and injects the server into your AI client's config for you — across 8 popular clients, each in its own format.
Install
Requires Node.js 18+.
# Run without installing (recommended for trying it out):
npx @idle-sync/db-conn-mcp setup
# or install globally so `db-conn-mcp` is on your PATH:
npm install -g @idle-sync/db-conn-mcpThis provides the db-conn-mcp command.
Quick start
db-conn-mcp setupThe wizard asks for:
- Scope — global (
~/.db-conn-mcp/connections.json) or repo (./connections.json). - Connection name — e.g.
prod. - DSN — e.g.
postgresql://user:pass@host:5432/dbname. - Mode —
read(recommended) orwrite. - Client injection — pick which detected MCP clients to wire up (e.g.
1,3orall).
It then writes your config and (optionally) registers the server in your chosen AI clients. Restart/reconnect the client and the tools are available.
Cancelling is safe. Press Ctrl+C at any prompt and nothing is written.
Configuration
The single source of truth is connections.json, resolved in this order (first match wins):
--config /path/to/connections.json./connections.json(repo-scoped)~/.db-conn-mcp/connections.json(global-scoped)
{
"connections": [
{ "name": "prod", "dsn": "postgresql://…", "mode": "read" },
{ "name": "dev", "dsn": "postgresql://…", "mode": "write", "yolo": false }
]
}| Field | Required | Meaning |
| ------ | -------------------- | --------------------------------------------------------------- |
| name | yes | Unique identifier the agent uses to pick a database. |
| dsn | yes | Connection string. Secret — never shown by any tool. |
| mode | yes | read or write. An absolute, native security boundary. |
| yolo | no (default false) | If true, skip the per-write consent prompt for this database. |
connections.jsonis git-ignored by this project's.gitignore— never commit real DSNs.
The security model
Writes pass through three gates, in order:
mode(hard, native). If the database isn't"mode": "write", the write is rejected — and the connection is opened read-only at the PostgreSQL session level regardless, so it's blocked twice over.yoloanduser_consentcan never make areaddatabase writable.yolo(persisted trust). On awritedatabase withyolo: true, writes proceed without prompting.user_consent(per-operation). Otherwise the agent must first read the schema, show you the exact SQL, get your "yes", and re-call withuser_consent=true.
Reads always run inside a native read-only transaction, and execute_read_query accepts only a single read-only statement (SELECT/WITH/VALUES/TABLE/SHOW/EXPLAIN). That allowlist is what stops an agent from sending SET SESSION CHARACTERISTICS AS TRANSACTION READ WRITE to flip the session, or piggy-backing a ; DELETE … onto a read — there's no SQL parsing involved, just a leading-keyword check plus the driver's single-command protocol.
Note (Node port): node-postgres only runs a query through the single-command extended protocol when a prepared-statement name is set — an empty parameter list still uses the multi-statement simple protocol. The read tool therefore issues its query as a named prepared statement, so a piggy-backed
; DELETE …is rejected by PostgreSQL at parse time (cannot insert multiple commands into a prepared statement), matching asyncpg's behavior in the original.
Use a read-only role (strongest guarantee)
The application-level checks above are defense-in-depth. The hardest boundary is a privilege one: connect with a PostgreSQL role that simply cannot write, so a write fails even if every layer above were bypassed. Create one per database and use its DSN for read connections:
CREATE ROLE agent_ro LOGIN PASSWORD '…';
GRANT CONNECT ON DATABASE mydb TO agent_ro;
GRANT USAGE ON SCHEMA public TO agent_ro;
GRANT SELECT ON ALL TABLES IN SCHEMA public TO agent_ro;
ALTER DEFAULT PRIVILEGES IN SCHEMA public GRANT SELECT ON TABLES TO agent_ro;This is the recommended setup for any database that holds data you care about.
MCP tools
The server exposes 10 tools and 1 prompt:
| Tool | Kind | Description |
| --------------------- | ------- | ------------------------------------------------------------------------------------------------------------------------- |
| list_databases | explore | Configured databases (name, mode, yolo — no DSN). |
| list_tables | explore | Tables and views in a database. |
| get_table_schema | explore | Columns, types, primary/foreign keys for a table. |
| sample_table_rows | explore | First N rows of a table (default 10). |
| find_columns | search | Find columns by name across all tables (fuzzy, case-insensitive). |
| search_value | search | Find where a value appears across tables (fuzzy); returns table/column hits + samples. Pass tables=[…] to scope it. |
| execute_read_query | execute | Run a single read-only statement (SELECT/WITH/…) inside a read-only transaction. |
| execute_write_query | execute | Run a mutation — gated by the safety model above. |
| set_yolo_mode | config | Enable/disable yolo for one database (persisted). |
| check_database | doctor | Test one database (or all) → OK or a sanitized diagnostic. |
Prompt: troubleshoot_connection — a discoverable, full connection-gotchas checklist (host/port, firewall, sslmode, Docker localhost, db-name case, pool limits, …).
CLI reference
db-conn-mcp is both the server and a management tool.
| Command | What it does |
| --------------------------------- | --------------------------------------------------------------------- |
| db-conn-mcp | Run the server over stdio (the default an MCP client uses). |
| db-conn-mcp --transport http | Run over Streamable HTTP instead (http://127.0.0.1:8000/mcp). |
| db-conn-mcp setup | Guided setup; shows status + an action menu if already configured. |
| db-conn-mcp status | List configured databases and which clients have the server injected. |
| db-conn-mcp add | Add another database connection. |
| db-conn-mcp clients | Inject the server into detected MCP clients. |
| db-conn-mcp clients --remove | Uninject the server from chosen clients. |
| db-conn-mcp check [name] | Probe connectivity (exit 0 all-OK, 2 if any unreachable). |
| db-conn-mcp remove <name> | Remove one connection. |
| db-conn-mcp reset | Remove all connections (delete connections.json) — fresh slate. |
| db-conn-mcp yolo <name> on\|off | Toggle yolo for one database. |
| db-conn-mcp -v / --version | Print the installed version and exit. |
--config <path> works before or after any subcommand.
The original Python server's
--transport httpused SSE (deprecated in the MCP SDKs). This port uses Streamable HTTP, the current standard, at the same role.stdiois what MCP clients use and is unchanged.
Connecting an AI client
db-conn-mcp setup (or db-conn-mcp clients) auto-detects and writes the right config for:
Claude Desktop · Cursor · Windsurf · Agy (Antigravity) · Claude Code · Cline · VS Code · Zed
Prefer to wire it manually? Use the absolute path the wizard would (so the client can find it regardless of PATH). For an mcpServers-style client (Claude Desktop, Cursor, Windsurf, …):
{
"mcpServers": {
"db-conn-mcp": {
"command": "db-conn-mcp",
"args": ["--config", "/absolute/path/to/connections.json"]
}
}
}If
db-conn-mcpisn't on the client's PATH, use the interpreter form instead:"command": "/abs/path/to/node", "args": ["/abs/path/to/dist/cli.js", "--config", "…"]. Thesetup/clientscommands figure this out for you automatically.
VS Code (servers key, "type": "stdio") and Zed (context_servers, nested command) use different shapes — the wizard handles those too.
Provider notes
- Railway / managed Postgres over a public proxy: use the public connection URL (e.g. Railway's
DATABASE_PUBLIC_URL, not the internal*.railway.internalone) and append?sslmode=require— these proxies require SSL with a self-signed cert, whichsslmode=requireaccepts without verification.
Development
git clone https://github.com/Idle-Sync/db-conn-mcp-node
cd db-conn-mcp-node
npm install
npm run typecheck # tsc, no emit
npm run format:check # prettier
npm test # vitest (154 tests)
npm run build # compile to dist/The codebase is split into single-purpose layers (config, models, dialects/, safety, diagnostics, handlers, server, cli); only the dialect layer knows a specific database exists.
License
MIT — see LICENSE.
