Server data from the Official MCP Registry
PostgreSQL MCP wrapper with .env credential mapping, tool selection, and safe read-only defaults.
PostgreSQL MCP wrapper with .env credential mapping, tool selection, and safe read-only defaults.
This PostgreSQL MCP server implements reasonable security controls with explicit read-only vs. write-capable tool separation and input validation via Zod schemas. However, there are several moderate concerns: (1) the .env file parser lacks proper escaping for quoted values, (2) sensitive credential data (passwords) is logged in error messages and connection strings, (3) database connection strings containing credentials are built at runtime without secure disposal, and (4) the write-operation guard uses regex patterns that could be bypassed via comment-heavy payloads. These issues are not critical but represent real security considerations users should understand. Supply chain analysis found 3 known vulnerabilities in dependencies (0 critical, 3 high severity). Package verification found 1 issue (1 critical, 0 high severity).
3 files analyzed Β· 13 issues found
Security scores are indicators to help you make informed decisions, not guarantees. Always review permissions before connecting any MCP server.
This plugin requests these system permissions. Most are normal for its category.
Unverified package source
We couldn't verify that the installable package matches the reviewed source code. Proceed with caution.
Set these up before or after installing:
Environment variable: MCP_KEY_HOST
Environment variable: MCP_KEY_PORT
Environment variable: MCP_KEY_NAME
Environment variable: MCP_KEY_USER
Environment variable: MCP_KEY_PASS
Environment variable: MCP_KEY_SSLMODE
Add this to your MCP configuration file:
{
"mcpServers": {
"io-github-edelciomolina-postgres-mcp": {
"env": {
"MCP_KEY_HOST": "your-mcp-key-host-here",
"MCP_KEY_NAME": "your-mcp-key-name-here",
"MCP_KEY_PASS": "your-mcp-key-pass-here",
"MCP_KEY_PORT": "your-mcp-key-port-here",
"MCP_KEY_USER": "your-mcp-key-user-here",
"MCP_KEY_SSLMODE": "your-mcp-key-sslmode-here"
},
"args": [
"-y",
"postgres-mcp"
],
"command": "npx"
}
}
}From the project's GitHub README.
English | π PortuguΓͺs
Most LLMs interact with databases by guessing - assuming table names, inventing column names, and writing queries that may fail or expose sensitive data. Postgres MCP solves this by giving the LLM a structured, safe interface to actually understand the database before touching it.
Built with @modelcontextprotocol/sdk and pg, it provides:
pg_classify_query_risk lets the LLM check a query's safety before running it.env at startup; nothing sensitive lives in mcp.jsontool=<name> args, so the LLM only sees what you choose to expose.env file with database credentials (anywhere in the project tree - see .env Discovery)There are two ways to use this package. Choose the one that best fits your workflow.
npx, recommended for quick start)No installation needed. npx downloads and runs the package on demand. Add -y as the first argument to skip the confirmation prompt.
{
"servers": {
"Postgres Tools": {
"type": "stdio",
"command": "npx",
"args": [
"-y",
"@edelciomolina/postgres-mcp"
],
"env": {
"MCP_KEY_HOST": "DB_HOST",
"MCP_KEY_PORT": "DB_PORT",
"MCP_KEY_NAME": "DB_NAME",
"MCP_KEY_SSLMODE": "DB_SSLMODE",
"MCP_KEY_USER": "DB_USER",
"MCP_KEY_PASS": "DB_PASS"
}
}
}
}
This starts the server with the default read-only tool set - no tool= arguments needed. To enable write-capable tools, see Write-capable tools.
π‘ Using Supabase, Neon, Railway or another platform that only provides a connection string? Use
MCP_KEY_URLpointing toDATABASE_URL(or whatever variable name the platform uses). The server will prioritize the URL and ignore the individual variables. See Connection via URL.
VS Code supports discovering and installing MCP servers directly in the editor, without using the terminal.
MCP: Add Serverpostgres-mcp or edelciomolinamcp.json automaticallyπ‘ You can also open the MCP Servers panel via Copilot chat icon β Manage MCP Servers to browse, enable, or disable servers at any time.
After installing, edit the generated entry in .vscode/mcp.json to add your tool= arguments and env key mappings as shown in the Usage section below.
mcp.json)Read-only (default - no tool= arguments needed):
{
"servers": {
"Postgres Tools": {
"type": "stdio",
"command": "npx",
"args": ["@edelciomolina/postgres-mcp"],
"env": {
"MCP_KEY_HOST": "DB_HOST",
"MCP_KEY_PORT": "DB_PORT",
"MCP_KEY_NAME": "DB_NAME",
"MCP_KEY_SSLMODE": "DB_SSLMODE",
"MCP_KEY_USER": "DB_USER",
"MCP_KEY_PASS": "DB_PASS"
}
}
}
}
With write tools (explicit opt-in required):
{
"servers": {
"Postgres Tools": {
"type": "stdio",
"command": "npx",
"args": [
"@edelciomolina/postgres-mcp",
"tool=pg_manage_schema",
"tool=pg_manage_indexes"
],
"env": {
"POSTGRES_MCP_ALLOW_WRITE": "true",
"MCP_KEY_HOST": "DB_HOST",
"MCP_KEY_PORT": "DB_PORT",
"MCP_KEY_NAME": "DB_NAME",
"MCP_KEY_SSLMODE": "DB_SSLMODE",
"MCP_KEY_USER": "DB_USER",
"MCP_KEY_PASS": "DB_PASS"
}
}
}
}
β οΈ Write-capable tools require
POSTGRES_MCP_ALLOW_WRITE=trueinenv. Without it, the server exits at startup.
The corresponding .env file at the root of your project:
DB_HOST=db.your-project.supabase.co
DB_PORT=5432
DB_NAME=postgres
DB_SSLMODE=require
DB_USER=readonly_user
DB_PASS=your_password
mcp.json configuration worksenv - credential key mappingThe env block does not contain the actual credentials. It maps each MCP_KEY_* to the variable name in your .env file.
Key in env | Points to .env variable | Example value |
|---|---|---|
MCP_KEY_URL | DATABASE_URL | postgresql://user:pass@host:5432/db?sslmode=require |
MCP_KEY_HOST | DB_HOST | db.example.supabase.co |
MCP_KEY_PORT | DB_PORT | 5432 |
MCP_KEY_NAME | DB_NAME | postgres |
MCP_KEY_SSLMODE | DB_SSLMODE | require |
MCP_KEY_USER | DB_USER | readonly_user |
MCP_KEY_PASS | DB_PASS | secret |
Priority: when
MCP_KEY_URL(orDATABASE_URL) is present, the server uses the URL directly and ignores the individual credential keys.
This indirection lets you use any variable name in your .env - useful when sharing a .env across multiple services with different naming conventions.
args - tool selection via tool= prefixEach enabled MCP tool is declared as a separate argument in the format tool=<name>:
"args": [
"-y",
"@edelciomolina/postgres-mcp",
"tool=pg_manage_schema",
"tool=pg_manage_indexes"
]
This makes the tool list explicit and auditable directly in mcp.json - no hidden configuration files. π
DATABASE_URL)In addition to individual credentials, you can provide a full connection string - the standard format on platforms like Supabase, Neon, and Railway.
.env:
DATABASE_URL=postgresql://user:password@host:5432/database?sslmode=require
mcp.json:
{
"servers": {
"Postgres Tools": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@edelciomolina/postgres-mcp"],
"env": {
"MCP_KEY_URL": "DATABASE_URL"
}
}
}
}
The variable mapped by MCP_KEY_URL has priority over the other keys (MCP_KEY_HOST, MCP_KEY_PORT, etc.). If the URL is present, the other variables are ignored.
If the platform uses a different name (e.g. DB_URL), just adjust the mapping:
"MCP_KEY_URL": "DB_URL"
If you omit all tool= arguments, the server starts with a curated read-only set - all tools that can retrieve, analyze, or explain data, but nothing that can modify it.
β Included in defaults (read-only):
pg_execute_query pg_manage_query pg_inspect_schema
pg_get_setup_instructions pg_analyze_database
pg_monitor_database pg_debug_database
pg_inspect_database_graph pg_describe_table_semantics
pg_find_related_tables pg_classify_query_risk
π‘
pg_execute_queryrejectsINSERT,UPDATE,DELETE, DDL,ANALYZE,VACUUM,EXPLAIN ANALYZEand other write/maintenance commands before the database is queried.
π‘
pg_inspect_schemaprovides read-only schema introspection (get_info,get_enums). For DDL operations, usepg_manage_schemawith explicit opt-in.
β οΈ Excluded from defaults - require tool= argument AND POSTGRES_MCP_ALLOW_WRITE=true:
| Tool | Operations |
|---|---|
pg_manage_schema | CREATE TABLE, ALTER TABLE, CREATE TYPE |
pg_manage_indexes | CREATE INDEX, DROP INDEX, REINDEX |
pg_manage_constraints | ADD CONSTRAINT, DROP CONSTRAINT |
pg_manage_functions | CREATE FUNCTION, DROP FUNCTION |
pg_manage_triggers | CREATE TRIGGER, DROP TRIGGER, enable/disable |
pg_manage_rls | ENABLE/DISABLE RLS, CREATE/ALTER/DROP POLICY |
pg_manage_users | CREATE/DROP/ALTER USER, GRANT, REVOKE |
pg_execute_mutation | INSERT / UPDATE / DELETE / UPSERT |
pg_execute_sql | Arbitrary SQL with transaction support |
.env file discoveryThe server resolves the .env file in this order:
env-file=<path> argument - explicit path relative to cwd; takes priority over everythingcwd, searches each parent directory until a .env is found or the filesystem root is reachedIf no .env is found, the server exits with a clear error message.
When VS Code starts the MCP process, cwd is typically the workspace root. If your .env is in a subfolder (e.g. functions/.env), use env-file= to point to it explicitly:
{
"servers": {
"Postgres Tools": {
"type": "stdio",
"command": "npx",
"args": [
"-y",
"@edelciomolina/postgres-mcp",
"env-file=functions/.env"
],
"env": {
"MCP_KEY_HOST": "DB_HOST",
"MCP_KEY_PORT": "DB_PORT",
"MCP_KEY_NAME": "DB_NAME",
"MCP_KEY_SSLMODE": "DB_SSLMODE",
"MCP_KEY_USER": "DB_USER",
"MCP_KEY_PASS": "DB_PASS"
}
}
}
}
π‘ The upward search behavior handles the common case automatically. Use
env-file=when you need explicit control (CI, monorepos, Docker bind-mounts).
| Tool | Description |
|---|---|
pg_execute_query | SELECT / COUNT / EXISTS with write and multi-statement guards |
pg_manage_query | EXPLAIN plans, slow query analysis, pg_stat_statements |
pg_inspect_schema | Schema info and ENUM types (read-only introspection) |
pg_get_setup_instructions | Setup instructions per platform |
pg_analyze_database | Performance, configuration, and storage analysis |
pg_monitor_database | Real-time monitoring of connections, queries, locks, and replication |
pg_debug_database | Diagnose connections, locks, performance, and replication |
pg_inspect_database_graph | Build a full knowledge graph of the database: schemas, tables, columns, FKs, indexes, inferred relations, and business domains |
pg_describe_table_semantics | Describe a table with semantic context: risk level, column roles, sensitive columns, and related tables |
pg_find_related_tables | Find tables related to a given table via explicit FKs and inferred naming patterns, with path explanation |
pg_classify_query_risk | Classify query risk (safe / warning / review / blocked) without executing it |
tool= argument + POSTGRES_MCP_ALLOW_WRITE=true)| Tool | Description |
|---|---|
pg_manage_schema | Schema info, create/alter tables, manage ENUMs |
pg_manage_indexes | List, create, drop, reindex, analyze index usage |
pg_manage_constraints | List, create, and drop constraints and foreign keys |
pg_manage_functions | List, create, and drop functions and procedures |
pg_manage_triggers | List, create, drop, enable/disable triggers |
pg_manage_rls | Row-Level Security policies |
pg_manage_users | User permissions, create/drop/alter users, grant/revoke |
pg_execute_mutation | INSERT / UPDATE / DELETE / UPSERT with parameterized queries |
pg_execute_sql | Arbitrary SQL execution with optional transaction support |
The four pg_*_graph / pg_*_semantics / pg_*_risk tools build an in-memory knowledge graph of your database at runtime. This gives the LLM a structured map - schemas, tables, columns, foreign keys, inferred relations, risk levels, and business domains - without executing any query against your data.
All inferred fields (column semantic roles, table probable types, inferred relations) are clearly tagged so the LLM knows to treat them as hints, not schema facts.
mcp-config.json)Place a mcp-config.json file beside your .env to tune the semantic layer and security limits. All fields are optional - omitting the file applies safe defaults.
{
"security": {
"defaultLimit": 100,
"maxLimit": 1000,
"blockedSchemas": ["pg_catalog", "information_schema"],
"blockedTables": [],
"requireLimit": true
},
"semanticLayer": {
"enabled": true,
"inferRelationsWithoutForeignKeys": true,
"inferBusinessEntities": true,
"sensitiveKeywords": ["password", "secret", "token", "api_key", "ssn", "hash"]
}
}
For a detailed view of the communication flow between the MCP client, the proxy, and PostgreSQL - including the full sequence diagram - see ARCHITECT.md.
MIT Β© Edelcio Molina
Be the first to review this server!
by Toleno Β· Developer Tools
Toleno Network MCP Server β Manage your Toleno mining account with Claude AI using natural language.
by mcp-marketplace Β· Developer Tools
Create, build, and publish Python MCP servers to PyPI β conversationally.
by Microsoft Β· Content & Media
Convert files (PDF, Word, Excel, images, audio) to Markdown for LLM consumption