MCP server that lets AI agents run read/write queries against PostgreSQL, MySQL, MongoDB and Oracle through alias-based connections, without exposing credentials.
Project description
Custom MCP Database
mcp-name: io.github.renanlido/custom-mcp-database
An MCP server that lets AI agents run alias-based queries against PostgreSQL, MySQL, MongoDB and Oracle — without ever exposing credentials to the model. Connections are configured once and stored locally; the agent only ever references them by alias.
Works with Claude Code, Claude Desktop, Cursor, VS Code, Windsurf, Gemini CLI, and any other MCP client (all use the same stdio launch command).
Install
The server runs over stdio. The universal launch command is uvx custom-mcp-database run
(requires uv; the package is fetched from PyPI on first run).
Claude Code
# Direct (published package)
claude mcp add custom-mcp-database -- uvx custom-mcp-database run
# Or install the full plugin from this repo's marketplace
/plugin marketplace add renanlido/custom-mcp-database
/plugin install custom-mcp-database@renanlido-mcp
Claude Desktop
Two options:
- One-click bundle — build the
.mcpb(mcpb pack) and open it in Claude Desktop. See Distribution. - Manual config — add the snippet from
examples/mcp-clients/claude-desktop.jsontoclaude_desktop_config.json.
Other clients
Copy the matching snippet — all use the same command/args, only the file and key differ:
| Client | Config file | Key | Snippet |
|---|---|---|---|
| Cursor | ~/.cursor/mcp.json |
mcpServers |
cursor.json |
| VS Code | .vscode/mcp.json |
servers |
vscode.json |
| Windsurf | ~/.codeium/windsurf/mcp_config.json |
mcpServers |
windsurf.json |
| Gemini CLI | ~/.gemini/settings.json |
mcpServers |
gemini-cli.json |
Full client matrix and a local-checkout variant: examples/mcp-clients/README.md.
Configure connections
The server starts empty. Add connections via the CLI (or the db_add_database tool).
Credentials are written to local SQLite and never re-sent to the model.
# PostgreSQL
uvx custom-mcp-database add-db --alias pg --type postgres \
--host localhost --port 5432 --user me --password secret --dbname app
# MySQL
uvx custom-mcp-database add-db --alias my --type mysql \
--host localhost --port 3306 --user root --password secret --dbname app
# Oracle
uvx custom-mcp-database add-db --alias ora --type oracle \
--host db.example.com --port 1521 --user system --password pw --dbname ORCLPDB1
# MongoDB
uvx custom-mcp-database add-db --alias mongo --type mongo \
--uri "mongodb+srv://user:pw@cluster.mongodb.net/" --dbname app
uvx custom-mcp-database list-aliases
uvx custom-mcp-database remove-db --alias pg
Config location (override with MCP_DB_CONFIG):
$XDG_CONFIG_HOME/custom-mcp-database/mcp_config.sqlite3
(default ~/.config/custom-mcp-database/mcp_config.sqlite3).
Credentials are stored as plaintext JSON in that SQLite file. Keep it secret; it is not committed and not encrypted.
MCP tools
| Tool | Purpose |
|---|---|
db_list_aliases |
List configured aliases and types |
db_add_database |
Add/replace a connection |
db_remove_database |
Remove a connection |
db_execute_query |
Run SQL or a MongoDB JSON filter |
db_list_collections |
List MongoDB collections |
db_security_status |
Report the active security policy |
db_execute_query notes: SQL runs as given with parameterized binds (add your own
LIMIT); MongoDB takes a JSON filter + collection, caps results at 10 (--limit),
rejects empty filters, and coerces 24-char hex strings to ObjectId.
Security
This server handles real credentials and production data, so it ships deny-by-default:
- Read-only by default. Only SELECT-class SQL runs. Writes/DDL require explicit opt-in.
- No stacked statements (
;-injection blocked), single statement per call. - MongoDB server-side JavaScript blocked (
$where,$function,$accumulator, mapReduce, …). - Identifiers validated (
oracle_schemacan't be used for injection). - Results capped at
MCP_DB_MAX_ROWS(default 1000); secrets redacted from errors. - Credential store is
0600plaintext SQLite — keep the host disk encrypted.
Check the live posture: custom-mcp-database security-status (or the db_security_status tool).
Enable writes for a specific task (then turn it back off):
export MCP_DB_READONLY=0
export MCP_DB_ALLOW_WRITES=1 # INSERT/UPDATE/DELETE
# export MCP_DB_ALLOW_DDL=1 # only if you really need CREATE/DROP/ALTER/...
Read the full protocol — least-privilege DB roles, TLS, prompt-injection handling, vulnerability reporting — in SECURITY.md. The app-layer guards are defense-in-depth; the authoritative control is a least-privilege database account.
Develop
uv sync # create .venv and install deps
make run # run the server (stdio)
make lint # ruff
make build # sdist + wheel into dist/
Inspect tools interactively:
uv run mcp dev src/custom_mcp_database/server.py
Distribution
This repo ships ready-to-publish metadata for every major channel. All of it is
published automatically on push to main (see below):
| Channel | File | Published by |
|---|---|---|
| PyPI | pyproject.toml |
release.yml (push to main) |
| MCP Registry | server.json |
release.yml (push to main) |
| Claude Code plugin | .claude-plugin/plugin.json, .mcp.json |
available on GitHub push |
| Claude Code marketplace | .claude-plugin/marketplace.json |
available on GitHub push |
| Claude Desktop bundle | manifest.json |
release.yml attaches .mcpb to the Release |
Automated release — just push to main
Releases are fully automated. On every push to main,
.github/workflows/release.yml:
- Picks the next semantic version from your commits since the last tag
(
feat:→ minor,BREAKING CHANGE/type!:→ major, anything else → patch; add[skip release]to a commit message to skip). - Writes that version into
pyproject.tomland syncs it into every artifact (server.json,manifest.json, plugin + marketplace) viascripts/sync_version.py— version lives in one place, no hand-bumping. - Builds, commits
chore(release): vX [skip ci], tagsvX, pushes. - Publishes to PyPI (Trusted Publishing/OIDC), then the MCP Registry (GitHub OIDC).
- Packs the
.mcpband cuts a GitHub Release with the wheel + bundle attached.
The release commit carries [skip ci], so it does not re-trigger the workflow.
One-time setup (can't be automated — needs your accounts):
- Create a PyPI Trusted Publisher for
renanlido/custom-mcp-database, workflowrelease.yml. - Allow GitHub Actions to push to
main(repo → Settings → Actions → Read and write permissions; ifmainis a protected branch, allow the actions bot to bypass or use a PAT).
The MCP Registry namespace is io.github.renanlido/custom-mcp-database (GitHub-validated).
Local manual escape hatch: make build (syncs version + builds) then uv publish.
License
MIT
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 custom_mcp_database-0.4.0.tar.gz.
File metadata
- Download URL: custom_mcp_database-0.4.0.tar.gz
- Upload date:
- Size: 111.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
49ed81857b3a1f648e60f01f5f84e2a853b641507d7c6874d0a56089d3846d6c
|
|
| MD5 |
9c003220de3cd94c5c9adec2080deb83
|
|
| BLAKE2b-256 |
d130ba8d9fe536da5ebafe7f557968a15479521f71daf22909687b112d3dea01
|
Provenance
The following attestation bundles were made for custom_mcp_database-0.4.0.tar.gz:
Publisher:
release.yml on renanlido/custom-mcp-database
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
custom_mcp_database-0.4.0.tar.gz -
Subject digest:
49ed81857b3a1f648e60f01f5f84e2a853b641507d7c6874d0a56089d3846d6c - Sigstore transparency entry: 1887007438
- Sigstore integration time:
-
Permalink:
renanlido/custom-mcp-database@d5e59f27c2c96daceb398f351c10e6390e4126ed -
Branch / Tag:
refs/heads/main - Owner: https://github.com/renanlido
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@d5e59f27c2c96daceb398f351c10e6390e4126ed -
Trigger Event:
push
-
Statement type:
File details
Details for the file custom_mcp_database-0.4.0-py3-none-any.whl.
File metadata
- Download URL: custom_mcp_database-0.4.0-py3-none-any.whl
- Upload date:
- Size: 16.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
356fcbddfb6df9f58bfb35698788d3b888fc54dca09188bbc2a584ecb42a007a
|
|
| MD5 |
c4835092d3d438fb2ee809477d69c22d
|
|
| BLAKE2b-256 |
0540ca479ab186f21a4b5f7d35015b82663627395526d223f6af494196f6f1cc
|
Provenance
The following attestation bundles were made for custom_mcp_database-0.4.0-py3-none-any.whl:
Publisher:
release.yml on renanlido/custom-mcp-database
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
custom_mcp_database-0.4.0-py3-none-any.whl -
Subject digest:
356fcbddfb6df9f58bfb35698788d3b888fc54dca09188bbc2a584ecb42a007a - Sigstore transparency entry: 1887007607
- Sigstore integration time:
-
Permalink:
renanlido/custom-mcp-database@d5e59f27c2c96daceb398f351c10e6390e4126ed -
Branch / Tag:
refs/heads/main - Owner: https://github.com/renanlido
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@d5e59f27c2c96daceb398f351c10e6390e4126ed -
Trigger Event:
push
-
Statement type: