postgres-mcp-server
Provides tools for managing and analyzing PostgreSQL databases, including schema exploration, query execution, performance analysis, and database health monitoring.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@postgres-mcp-servershow me the orders table schema"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
PostgreSQL MCP Server
A Model Context Protocol (MCP) server for PostgreSQL database management and analysis. This server provides comprehensive tools for exploring database schemas, executing queries, analyzing performance, and monitoring database health.
Installation
npm install -g postgres-mcp-serverOr run directly with npx:
npx postgres-mcp-serverConfiguration
Configure each server using PG_* environment variables. The suffix (_1, _2, _DEV, _PROD, …) can be any string — the server is detected by the presence of PG_NAME_*.
PG_NAME_1="prod"
PG_HOST_1="prod.example.com"
PG_PORT_1="5432"
PG_USERNAME_1="user"
PG_PASSWORD_1="pass"
PG_DATABASE_1="mydb"
PG_SSL_1="true"
PG_DEFAULT_1="true"
PG_ACCESS_MODE_1="readonly"
PG_DB_ACCESS_MODES_1="analytics:full,staging:rw"Environment Variable Reference:
Variable | Required | Description |
| Yes | Server name (used to identify the server) |
| Yes | PostgreSQL server hostname |
| No | Port number (default: |
| Yes | Database username |
| No | Database password |
| No | Default database (default: |
| No | Default schema (default: |
| No | SSL mode: |
| No | Set to |
| No | AI context/guidance for this server (see below) |
| No | Server-wide access mode: |
| No | Per-DB access mode overrides: |
AI Context for Servers
PG_CONTEXT_{n} provides guidance to AI agents about how to interact with each server. The context is returned in list_servers and get_current_connection responses so AI agents can adjust their behavior accordingly.
PG_CONTEXT_DEV="Development environment. Safe to run any queries. Contains test data only."
PG_CONTEXT_STAGING="Staging with production-like data. Use LIMIT clauses. Avoid bulk operations."
PG_CONTEXT_PROD="PRODUCTION DATABASE - CRITICAL GUIDELINES:
- Read-only queries strongly preferred
- Always use LIMIT (max 1000 rows)
- Avoid full table scans on large tables (users, orders, events)
- Peak hours: 9am-5pm EST - minimize heavy queries
- Main schemas: 'app' (application data), 'analytics' (reporting)
- Contact DBA before any DDL operations"Access Mode Configuration
Access modes control whether write operations are allowed. Configure at three levels with the following priority:
Priority: Database-level > Server-level > Global > default (full)
# Global default for all servers (optional)
POSTGRES_ACCESS_MODE="readonly" # full | readonly
# Server-level override (recommended for production)
PG_ACCESS_MODE_1="readonly"
# Per-database override (most specific). Format: dbname:mode,dbname:mode
PG_DB_ACCESS_MODES_1="production:readonly,analytics:full,staging:rw"Supported values:
full/rw/readwrite— allows all SQL operationsreadonly/ro/read-only— only SELECT and read operations allowed
Claude Code CLI
claude mcp add-json postgres_dbs --scope user '{
"command": "npx",
"args": ["-y", "@tejasanik/postgres-mcp-server"],
"env": {
"PG_NAME_1": "prod",
"PG_HOST_1": "prod.example.com",
"PG_PORT_1": "5432",
"PG_USERNAME_1": "user",
"PG_PASSWORD_1": "pass",
"PG_DATABASE_1": "mydb",
"PG_SSL_1": "true",
"PG_DEFAULT_1": "true",
"PG_ACCESS_MODE_1": "readonly",
"PG_DB_ACCESS_MODES_1": "analytics:full,staging:rw"
}
}'Claude Desktop
Add to claude_desktop_config.json:
{
"mcpServers": {
"postgres": {
"command": "npx",
"args": ["@tejasanik/postgres-mcp-server"],
"env": {
"PG_NAME_1": "prod",
"PG_HOST_1": "prod.example.com",
"PG_PORT_1": "5432",
"PG_USERNAME_1": "user",
"PG_PASSWORD_1": "pass",
"PG_DATABASE_1": "mydb",
"PG_SSL_1": "true",
"PG_DEFAULT_1": "true",
"PG_ACCESS_MODE_1": "readonly",
"PG_DB_ACCESS_MODES_1": "analytics:full,staging:rw"
}
}
}
}Codex CLI
Add to ~/.codex/config.toml:
[mcp_servers.postgres]
command = "npx"
args = ["-y", "@tejasanik/postgres-mcp-server"]
[mcp_servers.postgres.env]
PG_NAME_1 = "prod"
PG_HOST_1 = "prod.example.com"
PG_PORT_1 = "5432"
PG_USERNAME_1 = "user"
PG_PASSWORD_1 = "pass"
PG_DATABASE_1 = "mydb"
PG_SSL_1 = "true"
PG_DEFAULT_1 = "true"
PG_ACCESS_MODE_1 = "readonly"
PG_DB_ACCESS_MODES_1 = "analytics:full,staging:rw"Available Tools
Server & Database Management
list_servers
Lists all configured PostgreSQL servers. Returns server names, hosts, ports, and connection status. Use this first to discover available servers.
Parameters:
filter(optional): Filter servers by name or host (case-insensitive partial match)
Returns:
servers: Array of server information (name, isConnected, isDefault, defaultDatabase, defaultSchema)currentServer: Currently connected server name (or null)currentDatabase: Currently connected database (or null)currentSchema: Current schema (or null)
Note: Host and port are intentionally hidden from responses for security.
list_databases
Lists databases in a specific PostgreSQL server. Always provide the server name to avoid confusion.
Parameters:
serverName(required): Name of the server to list databases from. Uselist_serversto see available servers.filter(optional): Filter databases by name (case-insensitive partial match)includeSystemDbs(optional): Include system databases (template0, template1). Default: falsemaxResults(optional): Maximum number of databases to return (default: 50, max: 200)
Returns:
serverName: The server name that was querieddatabases: Array of database information (name, owner, encoding, size)currentDatabase: Currently connected database on this server (or null)
switch_server_db
Switch to a different PostgreSQL server and optionally a specific database and schema.
Parameters:
server(required): Name of the server to connect todatabase(optional): Name of the database to connect to (uses server's defaultDatabase or "postgres")schema(optional): Default schema to use (uses server's defaultSchema or "public")
Returns:
success: Whether the switch was successfulmessage: Success messagecurrentServer: Name of the connected servercurrentDatabase: Name of the connected databasecurrentSchema: Name of the current schemacontext: (If configured) AI context/guidance for the connected server
get_current_connection
Returns details about the current database connection including server, database, schema, access mode, user, and AI context.
Parameters: None
Returns:
isConnected: Whether currently connected to a databaseserver: Current server namedatabase: Current database nameschema: Current schema nameaccessMode: "readonly" or "full"user: Database username for the current connectioncontext: (If configured) AI context/guidance for the current server
Schema & Object Exploration
list_schemas
Lists all database schemas in the current PostgreSQL database.
Parameters:
includeSystemSchemas(optional): Include system schemasserver,database,schema(optional): One-time connection override
list_objects
Lists database objects within a specified schema.
Parameters:
schema(required): Schema name to list objects fromobjectType(optional): Type of objects to list (table, view, sequence, extension, all)filter(optional): Filter objects by nameserver,database,targetSchema(optional): One-time connection override
get_object_details
Provides detailed information about a database object including columns, constraints, indexes, size, and row count. Auto-detects whether the target is a table, view, materialized view, or sequence and adapts the response accordingly. Returns exists: false early when the object isn't found.
Parameters:
schema(required): Schema name containing the objectobjectName(required): Name of the objectobjectType(optional): Type of the object (table,view,matview,sequence,extension). Auto-detected frompg_class.relkindwhen omitted.server,database,targetSchema(optional): One-time connection override
Returns (varies by detected kind):
exists: Whether the object was found.detectedKind: The actual kind detected (table/view/matview/sequence).columns,constraints(incl.check_clausefor CHECK constraints),indexes,size,rowCount: For relations.definition: For views and materialized views.sequenceDetails: For sequences (start, min/max, cache, last value).
describe_table
v3 single-call table summary. Replaces the ~5-call dance of get_object_details + COUNT(*) + LIMIT 5 + pg_stats. Returns columns (with null %, distinct ratio from pg_stats), primary key, foreign keys going OUT (this table → others), foreign keys coming IN (others → this table), all indexes with definitions, table size, row-count estimate, and a sample of rows.
Parameters:
table(required): Unqualified table name.schema(optional): Default'public'.sample_size(optional): Number of sample rows to fetch (default 5; 0 to skip).profile_columns(optional): Columns to profile (default: all up to 20).server,database,override_schema(optional): One-time connection override.
find_dependents
Walks pg_depend recursively to find every database object that depends on a target — views, foreign keys, functions, materialized views, indexes, rules. Use BEFORE DROP CASCADE to understand the blast radius. Each dependent is returned flat with its depth from the target (1 = direct, 2 = depends on a depth-1 dependent, …). Sets truncatedAtDepth: true if the recursion limit was hit.
Parameters:
name(required): Object name.kind(optional):table/view/matview/sequence/index/function/procedure/type/extension/schema(defaulttable).schema(optional): Default'public'.max_depth(optional): Recursion limit (1-10, default 5).server,database,override_schema(optional): One-time connection override.
Query Execution
execute_sql
Executes SQL statements on the database. Supports pagination and parameterized queries. Read-only mode prevents write operations.
Parameters:
sql(required): SQL statement(s) to execute. Use$1,$2, etc. for parameterized queries.params(optional): Array of parameters for parameterized queries (e.g.,[123, "value"]). Prevents SQL injection. Not supported withallowMultipleStatements.maxRows(optional): Maximum rows to return (default: 1000, max: 100000). Use withoffsetfor pagination.offset(optional): Number of rows to skip for pagination (default: 0).allowLargeScript(optional): Set to true to bypass the 100KB SQL length limit for deployment scripts.includeSchemaHint(optional): Include schema information (columns, primary keys, foreign keys) for tables referenced in the query.allowMultipleStatements(optional): Allow multiple SQL statements separated by semicolons. Returns results for each statement with line numbers.transactionId(optional): Execute within an active transaction. Get this frombegin_transaction.maxEstimatedRows(optional): Query-budget pre-flight check. If the planner estimates more rows than this, the query is refused without executing. Only applies to SELECT.maxEstimatedCost(optional): Query-budget pre-flight check. If the planner estimates a higher cost than this, the query is refused without executing. Only applies to SELECT.server,database,schema(optional): One-time connection override. Execute on a different server/database/schema without changing the main connection. Cannot be used withtransactionId.
Returns:
rows: Result rows (paginated)rowCount: Total number of rows in the resultfields: Column namesexecutionTimeMs: Query execution time in millisecondsoffset: Current offsethasMore: Whether more rows are availableoutputFile: (Only if output is too large) Path to temp file with full resultsschemaHint: (When includeSchemaHint=true) Schema information for referenced tables:tables: Array of table schemas with columns, primary keys, foreign keys, and row count estimates
Note: Large outputs are automatically written to a temp file, and the file path is returned. This prevents token wastage when dealing with large result sets.
execute_sql_file
Executes a .sql file from the filesystem. Useful for running migration scripts, schema changes, or data imports. Supports SQL files from various tools like Liquibase, Flyway, and SQL Server migrations.
Parameters:
filePath(required): Absolute or relative path to the.sqlfile to executeuseTransaction(optional): Wrap execution in a transaction (default: true). If any statement fails, all changes are rolled back.stopOnError(optional): Stop execution on first error (default: true). If false, continues with remaining statements and collects all errors.stripPatterns(optional): Array of patterns to remove from SQL before execution. Useful for stripping tool-specific delimiters (e.g., Liquibase's/, SQL Server'sGO).stripAsRegex(optional): If true, treatstripPatternsas regular expressions; if false, as literal strings (default: false).validateOnly(optional): If true, parse and validate the file without executing (default: false). Returns a preview of all statements.
Returns:
success: Whether all statements executed successfullyfilePath: Resolved file pathfileSize: File size in bytestotalStatements: Total executable statements in the filestatementsExecuted: Number of successfully executed statementsstatementsFailed: Number of failed statementsexecutionTimeMs: Total execution time in millisecondsrowsAffected: Total rows affected by all statementserrors: (When stopOnError=false) Array of error details:statementIndex: Which statement failed (1-based)sql: The failing SQL (truncated to 200 chars)error: Error message
rollback: Whether a rollback was performedvalidateOnly: (When validateOnly=true) Set to truepreview: (When validateOnly=true) Array of statement previews:index: Statement index (1-based)lineNumber: Line number in the filesql: The SQL statement (truncated to 200 chars)type: Detected statement type (SELECT, INSERT, UPDATE, DELETE, CREATE, etc.)
Limits: Max file size: 50MB. Supports PostgreSQL-specific syntax including dollar-quoted strings and block comments.
Examples:
# Preview a file without executing
execute_sql_file({ filePath: "/path/to/migration.sql", validateOnly: true })
# Strip Liquibase delimiters (literal "/" on its own line)
execute_sql_file({ filePath: "/path/to/liquibase.sql", stripPatterns: ["/"] })
# Strip SQL Server GO statements (regex pattern)
execute_sql_file({
filePath: "/path/to/sqlserver.sql",
stripPatterns: ["^\\s*GO\\s*$"],
stripAsRegex: true
})
# Strip multiple patterns
execute_sql_file({
filePath: "/path/to/migration.sql",
stripPatterns: ["/", "GO", "\\"]
})preview_sql_file
Preview a SQL file without executing it. Similar to mutation_preview but for SQL files. Shows statement counts by type and warnings for potentially dangerous operations. Use this before execute_sql_file to understand what a migration will do.
Parameters:
filePath(required): Absolute or relative path to the.sqlfile to previewstripPatterns(optional): Patterns to strip from SQL before parsing (same as execute_sql_file)stripAsRegex(optional): If true, treat patterns as regex (default: false)maxStatements(optional): Maximum statements to show in preview (default: 20, max: 100)
Returns:
filePath: Resolved file pathfileSize: File size in bytesfileSizeFormatted: Human-readable file size (e.g., "15.2 KB")totalStatements: Total executable statements in the filestatementsByType: Breakdown by statement type (e.g.,{ "CREATE": 5, "INSERT": 10, "ALTER": 2 })statements: Array of statement previews (up to maxStatements):index: Statement number (1-based)lineNumber: Line number in filesql: Statement SQL (truncated to 300 chars)type: Statement type (SELECT, INSERT, CREATE, etc.)
warnings: Array of warnings for dangerous operations:DROP statements
TRUNCATE statements
DELETE/UPDATE without WHERE clause
summary: Human-readable summary (e.g., "File contains 17 statements: 10 INSERT, 5 CREATE, 2 ALTER")
Example:
preview_sql_file({ filePath: "/path/to/migration.sql" })
// Returns:
// {
// "filePath": "/path/to/migration.sql",
// "fileSize": 15234,
// "fileSizeFormatted": "14.9 KB",
// "totalStatements": 17,
// "statementsByType": { "CREATE": 5, "INSERT": 10, "ALTER": 2 },
// "statements": [...],
// "warnings": ["Statement 15 (line 142): DROP statement detected - will permanently remove database object"],
// "summary": "File contains 17 statements: 10 INSERT, 5 CREATE, 2 ALTER"
// }mutation_preview
Preview the effect of INSERT, UPDATE, or DELETE statements without executing them. Shows estimated rows affected and a sample of rows that would be modified. Essential for verifying destructive queries before running them.
Parameters:
sql(required): The INSERT, UPDATE, or DELETE statement to previewsampleSize(optional): Number of sample rows to show (default: 5, max: 20)
Returns:
mutationType: Type of mutation (INSERT, UPDATE, DELETE)estimatedRowsAffected: Estimated number of rows that would be affectedsampleAffectedRows: Sample of rows that would be modified (for UPDATE/DELETE)targetTable: The table being modifiedwhereClause: The WHERE clause from the query (if present)warning: Warning message if no WHERE clause (all rows affected) or for INSERT previews
Example:
mutation_preview({ sql: "DELETE FROM orders WHERE status = 'cancelled'" })
// Returns: { mutationType: "DELETE", estimatedRowsAffected: 150, sampleAffectedRows: [...5 rows...] }mutation_dry_run
Transaction-based dry-run for mutations. Actually executes the INSERT/UPDATE/DELETE within a transaction, captures REAL results, then ROLLBACK so nothing persists. More accurate than mutation_preview because it catches actual constraint violations, trigger effects, and exact row counts.
Non-Rollbackable Operations: Statements containing explicit NEXTVAL() or SETVAL() are skipped to prevent sequence values from being permanently consumed. For skipped statements, an EXPLAIN query plan is provided instead.
Parameters:
sql(required): The INSERT, UPDATE, or DELETE statement to dry-runsampleSize(optional): Number of sample rows to return (default: 10, max: 20)
Returns:
mutationType: Type of mutation (INSERT, UPDATE, DELETE)success: Whether the dry-run executed successfullyskipped: Iftrue, statement was skipped (contains non-rollbackable operation)skipReason: Why the statement was skippedrowsAffected: Actual number of rows that would be affectedbeforeRows: Sample of rows before the change (for UPDATE/DELETE)affectedRows: Sample of rows after the change (for INSERT/UPDATE) or deleted rowstargetTable: The table being modifiedwhereClause: The WHERE clause (if present)executionTimeMs: Execution time in millisecondserror: Detailed PostgreSQL error information if failed:message: Error messagecode: PostgreSQL error code (e.g., '23505' for unique violation)detail: Detailed error descriptionhint: Hint for fixing the errorconstraint: Constraint name that caused the errortable,column,schema: Related database objects
nonRollbackableWarnings: Warnings about side effects:operation: Type of operation (SEQUENCE, VACUUM, etc.)message: Warning messagemustSkip: Iftrue, operation was skipped; iffalse, just a warning
warnings: General warnings (e.g., no WHERE clause)explainPlan: Query plan from EXPLAIN (for skipped DML statements with NEXTVAL/SETVAL)
Example:
mutation_dry_run({ sql: "INSERT INTO users (email) VALUES ('test@test.com')" })
// On success: { success: true, mutationType: "INSERT", rowsAffected: 1, affectedRows: [{id: 5, email: "test@test.com"}] }
// On failure: { success: false, error: { code: "23505", constraint: "users_email_key", detail: "Key already exists" } }
// With explicit NEXTVAL (skipped):
mutation_dry_run({ sql: "INSERT INTO users (id) VALUES (nextval('users_id_seq'))" })
// Returns: { success: true, skipped: true, skipReason: "NEXTVAL increments sequence...", explainPlan: [...] }dry_run_sql_file
Transaction-based dry-run for SQL files. Actually executes ALL statements within a transaction, captures REAL results for each statement (row counts, errors with line numbers, constraint violations), then ROLLBACK so nothing persists. Perfect for testing migrations before deploying.
Non-Rollbackable Operations: The following operations are automatically skipped (not executed):
VACUUM, CLUSTER, REINDEX CONCURRENTLY: Cannot run inside a transaction
CREATE INDEX CONCURRENTLY: Cannot run inside a transaction
CREATE/DROP DATABASE: Cannot run inside a transaction
NEXTVAL(), SETVAL(): Would permanently consume sequence values
For skipped DML statements (INSERT/UPDATE/DELETE/SELECT with NEXTVAL/SETVAL), an EXPLAIN query plan is provided so you can still see what the query would do.
Parameters:
filePath(required): Absolute or relative path to the.sqlfilestripPatterns(optional): Patterns to strip from SQL before execution (e.g.,["/"]for Liquibase)stripAsRegex(optional): If true, treat patterns as regex (default: false)maxStatements(optional): Maximum statements to include in results (default: 50, max: 200)stopOnError(optional): Stop on first error (default: false - continues to show ALL errors)
Returns:
success: Whether all statements executed successfully (skipped statements don't count as failures)filePath: Resolved file pathfileSize: File size in bytesfileSizeFormatted: Human-readable file sizetotalStatements: Total statements in filesuccessCount: Number of successful statementsfailureCount: Number of failed statementsskippedCount: Number of skipped statements (non-rollbackable operations)totalRowsAffected: Total rows affected across all statementsstatementsByType: Breakdown by statement type (e.g.,{"CREATE": 5, "INSERT": 10})executionTimeMs: Total execution timestatementResults: Array of results for each statement:index: Statement number (1-based)lineNumber: Line number in filesql: The SQL statement (truncated)type: Statement type (SELECT, INSERT, CREATE, etc.)success: Whether statement succeededskipped: Iftrue, statement was skipped (non-rollbackable operation)skipReason: Why the statement was skippedrowCount: Rows affected/returnedrows: Sample rows (for SELECT or RETURNING)executionTimeMs: Statement execution timeerror: Detailed PostgreSQL error if failed (same fields asmutation_dry_run)warnings: Warnings for this statementexplainPlan: Query plan from EXPLAIN (for skipped DML statements)
nonRollbackableWarnings: Warnings about operations that can't be fully rolled back:operation: Type (SEQUENCE, VACUUM, CLUSTER, etc.)message: Warning messagemustSkip: Iftrue, operation was skipped; iffalse, just a warningstatementIndex,lineNumber: Location in file
summary: Human-readable summaryrolledBack: Alwaystrue- confirms changes were rolled back
Example:
dry_run_sql_file({ filePath: "/path/to/migration.sql", stripPatterns: ["/"] })
// Returns:
// {
// "success": false,
// "totalStatements": 15,
// "successCount": 12,
// "failureCount": 2,
// "skippedCount": 1,
// "statementResults": [
// { "index": 1, "lineNumber": 1, "type": "CREATE", "success": true },
// { "index": 5, "lineNumber": 23, "type": "INSERT", "success": false,
// "error": { "code": "23505", "constraint": "users_pkey", "detail": "Key already exists" } },
// { "index": 8, "lineNumber": 45, "type": "SELECT", "success": true, "skipped": true,
// "skipReason": "NEXTVAL increments sequence...", "explainPlan": [...] },
// ...
// ],
// "nonRollbackableWarnings": [
// { "operation": "SEQUENCE", "message": "INSERT may consume sequence values...", "mustSkip": false },
// { "operation": "SEQUENCE", "message": "NEXTVAL increments sequence...", "mustSkip": true }
// ],
// "summary": "Dry-run of 15 statements: 12 succeeded, 2 failed, 1 skipped (non-rollbackable). All changes rolled back.",
// "rolledBack": true
// }When to use dry_run_sql_file vs preview_sql_file:
Feature |
|
|
Speed | Fast (just parsing) | Slower (actual execution) |
Detects syntax errors | Basic | Actual PostgreSQL errors |
Detects constraint violations | No | Yes |
Detects trigger effects | No | Yes |
Accurate row counts | No (estimates) | Yes (actual) |
Shows error details | No | Yes (code, constraint, hint) |
Consumes sequences | No | No (NEXTVAL/SETVAL skipped) |
Shows query plan for skipped ops | N/A | Yes (EXPLAIN) |
batch_execute
Execute multiple SQL queries in parallel. Returns all results keyed by query name. Efficient for fetching multiple independent pieces of data in a single call.
Parameters:
queries(required): Array of queries to execute (max 20):name: Unique name for this query (used as key in results)sql: SQL query to executeparams(optional): Query parameters
stopOnError(optional): Stop on first error (default: false, continues with all queries)
Returns:
totalQueries: Total number of queries in the batchsuccessCount: Number of successful queriesfailureCount: Number of failed queriestotalExecutionTimeMs: Total execution time in millisecondsresults: Object with query results keyed by name:success: Whether the query succeededrows: Result rows (if successful)rowCount: Number of rows returnederror: Error message (if failed)executionTimeMs: Individual query execution time
Example:
batch_execute({
queries: [
{ name: "user_count", sql: "SELECT COUNT(*) FROM users" },
{ name: "order_total", sql: "SELECT SUM(total) FROM orders" },
{ name: "recent_signups", sql: "SELECT COUNT(*) FROM users WHERE created_at > NOW() - INTERVAL '7 days'" }
]
})
// Returns all three results in parallel, keyed by nameTransaction Control
begin_transaction
Start a new database transaction. Returns a transactionId to use with execute_sql, commit_transaction, or rollback_transaction.
Parameters: None
Returns:
transactionId: Unique ID for this transactionstatus: "started"message: Instructions for using the transaction
commit_transaction
Commit an active transaction, making all changes permanent.
Parameters:
transactionId(required): The transaction ID returned bybegin_transaction
rollback_transaction
Rollback an active transaction, undoing all changes made within it.
Parameters:
transactionId(required): The transaction ID returned bybegin_transaction
Example - Transaction Usage:
1. Call begin_transaction to get a transactionId
2. Call execute_sql with transactionId for each statement
3. Call commit_transaction to save changes, OR rollback_transaction to undoexplain_query
Gets the execution plan for a SQL query.
Parameters:
sql(required): SQL query to explainanalyze(optional): Execute query to get real timingbuffers(optional): Include buffer usage statisticsformat(optional): Output format (text, json, yaml, xml)hypotheticalIndexes(optional): Simulate indexes (requires hypopg extension)server,database,schema(optional): One-time connection override (see below)
Connection Override (One-Time Execution)
Most query execution tools support one-time connection override parameters that allow executing a query on a different server/database/schema without changing the main connection. This is useful for:
Querying multiple databases in a single workflow
Running read queries against a replica while keeping the main connection to primary
Comparing schemas across different servers
Supported tools: execute_sql, explain_query, list_schemas, list_objects, get_object_details, describe_table, find_dependents, execute_sql_file, mutation_preview, mutation_dry_run, dry_run_sql_file, batch_execute, lock_check, detect_migration_state, column_profile, generate_seed_data, find_blocking_queries, kill_query, export_to_sql_file
Override Parameters:
server(optional): Execute on this server instead of the current onedatabase(optional): Execute on this database instead of the current oneschema(optional): Set search_path to this schema for this execution only
Important Notes:
The main connection remains unchanged after the query completes
Connection override cannot be used with transactions (
transactionId)Override connections use a separate connection pool with LRU eviction
Maximum 10 cached override pools, each limited to 2 connections
Total connections across all pools limited to 50
Examples:
# Query another database without switching
execute_sql({
sql: "SELECT * FROM users LIMIT 10",
database: "analytics_db"
})
# Query a different server entirely
execute_sql({
sql: "SELECT COUNT(*) FROM orders",
server: "reporting",
database: "warehouse"
})
# List schemas on a different server
list_schemas({
server: "production",
database: "myapp"
})
# Compare table structure across environments
get_object_details({
schema: "public",
objectName: "users",
server: "staging"
})Connection Pool Management:
Override connections are managed efficiently:
Pools are cached and reused for repeated queries to the same server/database
LRU eviction removes oldest pools when limit (10) is reached
Connections are properly released after each query
Global connection limit prevents resource exhaustion
Performance Analysis
get_top_queries
Reports the slowest SQL queries based on execution time.
Parameters:
limit(optional): Number of queries to return (default: 10)orderBy(optional): Order by total_time, mean_time, or callsminCalls(optional): Minimum number of calls to include
Requires: pg_stat_statements extension
analyze_workload_indexes
Analyzes database workload and recommends optimal indexes.
Parameters:
topQueriesCount(optional): Number of top queries to analyzeincludeHypothetical(optional): Include hypothetical index analysis
analyze_query_indexes
Analyzes specific SQL queries and recommends indexes.
Parameters:
queries(required): Array of SQL queries to analyze (max 10)
DDL Safety & Migration
lock_check
Static analysis of a SQL statement to determine the PostgreSQL lock level it will require, whether it forces a full-table rewrite, and an estimated duration based on the target table's current size. Knows lock semantics for ALTER TABLE variants, CREATE/DROP INDEX (concurrent vs not), VACUUM, CLUSTER, REFRESH MATERIALIZED VIEW, and more. Returns warnings for ACCESS EXCLUSIVE locks on busy production tables and concrete recommendations (e.g., use CREATE INDEX CONCURRENTLY, NOT VALID + VALIDATE CONSTRAINT, etc.).
Parameters:
sql(required): DDL statement to analyze.estimate_duration(optional): Look up target table size to estimate duration. Defaulttrue.server,database,schema(optional): One-time connection override.
safe_alter_table
Convert a high-level intent into a multi-step zero-downtime DDL recipe. Each step has its own SQL, expected lock level, and notes. Pipe the resulting scriptSql through dry_run_sql_file for verification, then through execute_sql_file({ useTransaction: false }) for the production rollout (CONCURRENTLY operations cannot run inside a transaction).
Supported intents:
add_not_null_column_with_default— backfill before flipping NOT NULL.add_not_null—NOT VALIDthenVALIDATE CONSTRAINT.add_foreign_key—NOT VALIDthenVALIDATE.add_check—NOT VALIDthenVALIDATE.create_index—CREATE INDEX CONCURRENTLY(with allowlisted index method:btree/hash/gist/spgist/gin/brin).drop_index—DROP INDEX CONCURRENTLY.
Parameters:
intent(required):{ kind, ... }— see intent list above.
detect_migration_state
Probe the database for migration tool tracker tables (Liquibase, Flyway, Alembic, Prisma, Knex, Sequelize, Django, Rails, Goose, TypeORM). Returns which tools are detected, the schema and table holding their state, the count of applied migrations, and the latest version. Use this first to learn whether a DB is managed by a migration tool before suggesting changes.
Parameters:
schemas(optional): Schemas to probe (default: all non-system schemas).server,database,schema(optional): One-time connection override.
Cross-Database Operations
export_to_sql_file
Export schema (DDL) and/or data from the connected database to a .sql file. The header banner records timestamp and source server alias (host/port hidden). Use this before transfer_objects or for migration-script generation.
Parameters:
filePath(required): Path to the.sqlfile. Must end with.sql.mode(optional):'append'(default — preserves existing content with a separator banner) or'overwrite'.what(required): One of:{ kind: 'objects', objects: [{ kind, name, schema? }, ...] }— DDL of an explicit list, topologically ordered by dependency.{ kind: 'data', tables, where?, orderBy?, limit? }— INSERT statements for tables.{ kind: 'schema_dump', schema?, include_data? }— full schema, optionally with data.{ kind: 'query_result', sql, target_table }— SELECT result emitted as INSERTs into a target table.
confirm_overwrite(optional): Whenmode='overwrite'and the file was modified <60s ago, settrueto confirm. Foot-gun guard.server,database,schema(optional): One-time connection override.
transfer_objects
Transfer DDL and/or data from one configured server/database to another (same server, different DB, or fully remote). Both endpoints must be configured servers (PG_NAME_*); ad-hoc connection strings are not accepted (security). Refuses if the target's effective access mode is readonly. FK constraints between tables are emitted as ALTER TABLE statements appended after tables to handle inter-table dependency cycles.
Parameters:
from,to(required):{ server, database?, schema? }source and target endpoints.objects(required):'*'(all objects in source schema) or array of{ kind, name, schema? }.include(optional):'ddl'/'data'/'both'(default).if_exists(optional):'skip'/'replace'/'error'(default — fails fast).dry_run(optional): Generate SQL without applying. Use withoutput_file.output_file(optional): Path to write generated SQL whendry_run: true.
schema_diff
Compute the DDL delta between two { server, database, schema } endpoints. Returns objects to CREATE (in source but not target), DROP (in target but not source), and MODIFY (in both, but DDL differs), plus a single migrationSql script that converges the target with the source. Source is the source of truth.
Parameters:
source,target(required):{ server, database?, schema? }.
Data Generation & Profiling
column_profile
Single-pass profile per column: null %, distinct count, top-K values with frequencies, min/max, and type-aware stats (avg/stddev for numeric, length distribution for text, range for temporal). Uses TABLESAMPLE BERNOULLI for tables larger than sample_threshold (default 1M rows) to keep latency bounded.
Parameters:
table(required),schema(optional, default'public').columns(optional): Specific columns to profile (default: all up to 30).sample_percent(optional): Default10.sample_threshold(optional): Default1_000_000rows.top_k(optional): Top-K values per column (default10, max25).server,database,override_schema(optional): One-time connection override.
generate_seed_data
Generate schema-aware fake seed data for a table. Respects NOT NULL, UNIQUE/PK (with retry-with-collision-suffix), enum types (cycles through labels), defaults, text length limits, and FK columns (skipped or filled — caller's choice). Generates type-appropriate values for numeric, text, boolean, uuid, date/timestamp, bytea, JSON, inet, cidr, macaddr.
Parameters:
table(required),schema(optional, default'public').count(required): 1 to 100,000.column_values(optional): Per-column SQL value override, e.g.{ country: "'US'", priority: '1' }. Quoted as PG literals.skip_fks(optional): Defaultfalse.apply(optional): Apply directly to DB (defaulttrue) or return SQL only (false).server,database,override_schema(optional): One-time connection override.
Operational Tools
find_blocking_queries
Show currently-blocking sessions in a friendly tree (blocker → blocked) using pg_stat_activity ⨝ pg_blocking_pids(). Returns each session's pid, user, database, application name, state, current query, time in state, and wait_event. Use to diagnose slowdowns and pick a candidate for kill_query.
Parameters:
include_idle(optional): Defaulttrue.limit(optional): Default50.server,database,schema(optional): One-time connection override.
kill_query
Cancel or terminate a backend session by PID. Returns a snapshot of the target session before signaling.
Parameters:
pid(required): Backend PID to signal.mode(required):'cancel'(soft —pg_cancel_backend, interrupts current statement) or'terminate'(hard —pg_terminate_backend, kills the entire backend).confirm(required): Must betrue. Foot-gun guard.server,database,schema(optional): One-time connection override.
Note: Refused if the target server's effective access mode is readonly.
Health Monitoring
analyze_db_health
Performs comprehensive database health checks including:
Buffer Cache Hit Rate: Checks cache efficiency
Connection Health: Monitors connection usage
Invalid Indexes: Detects broken indexes
Unused Indexes: Identifies indexes that aren't being used
Duplicate Indexes: Finds redundant indexes
Vacuum Health: Monitors dead tuple ratios
Sequence Limits: Warns about sequences approaching limits
Constraint Validation: Checks for unvalidated constraints
Usage Examples
Connect to a Server and List Databases
1. Use list_servers to see available servers
2. Use list_databases with serverName="dev" to see databases in the dev server
3. Use switch_server_db with server="dev", database="myapp" to connectExplore Database Schema
1. Use list_schemas to see all schemas
2. Use list_objects with schema="public" to see tables
3. Use get_object_details with schema="public", objectName="users" to see table structureAnalyze Query Performance
1. Use explain_query with your SQL to see the execution plan
2. Use get_top_queries to find slow queries
3. Use analyze_query_indexes to get index recommendationsHealth Check
1. Use analyze_db_health to run all health checks
2. Review warnings and critical issues
3. Take action on recommendationsExecute SQL Migration File
1. Use execute_sql_file with filePath="/path/to/migration.sql"
2. By default, runs in a transaction - all changes rolled back on error
3. Set stopOnError=false to continue on errors and get a full report
4. Set useTransaction=false for DDL statements that can't run in transactionsFeatures
Auto-Reconnect on Connection Errors
The server automatically handles stale database connections. When a connection error occurs (e.g., server went inactive, connection reset, timeout), the server will:
Detect the connection error
Invalidate the stale connection
Automatically reconnect using the stored server/database/schema
Retry the operation once
This is particularly useful for:
Staging/development servers that go idle
Cloud databases with connection timeouts
Network interruptions
Supported error patterns include: Connection terminated, ECONNRESET, ETIMEDOUT, server closed the connection unexpectedly, and PostgreSQL error codes like 57P01 (admin_shutdown), 08003 (connection_does_not_exist), etc.
Hidden Connection Details
Host URLs, ports, and credentials are never exposed in tool responses. Only server names (aliases) are visible, preventing accidental exposure of infrastructure details.
Connection Context in Responses
All tool responses include a connection object showing which server, database, and schema the operation ran on:
{
"rows": [...],
"connection": {
"server": "production",
"database": "myapp",
"schema": "public"
}
}Multi-Statement Execution
Execute multiple SQL statements in a single call using allowMultipleStatements: true:
execute_sql({
sql: "INSERT INTO logs VALUES (1); INSERT INTO logs VALUES (2); SELECT * FROM logs;",
allowMultipleStatements: true
})Returns results for each statement with line numbers for easy debugging.
Transaction Support
Explicit transaction control for atomic multi-statement operations:
1. begin_transaction() → returns transactionId
2. execute_sql({ sql: "UPDATE ...", transactionId: "..." })
3. execute_sql({ sql: "INSERT ...", transactionId: "..." })
4. commit_transaction({ transactionId: "..." }) OR rollback_transaction({ transactionId: "..." })Line Number Tracking
When execute_sql_file or multi-statement execution encounters errors, line numbers are included to help locate issues:
{
"errors": [
{
"statementIndex": 5,
"lineNumber": 42,
"sql": "INSERT INTO...",
"error": "syntax error at or near..."
}
]
}Security
Access Mode: By default, the server runs in full access mode. Configure access at global (
POSTGRES_ACCESS_MODE), server (PG_ACCESS_MODE_*), or database (PG_DB_ACCESS_MODES_*) levels. Database-level settings override server-level, which override global. Recommended: set production servers/databases toreadonly.SQL Injection Protection: All user inputs are validated and parameterized queries are used where possible.
Query Timeout: Default 30-second timeout prevents runaway queries.
Credentials: Managed via environment variables and never logged or exposed through the MCP interface.
File Permissions: Large output files are created with restricted permissions (0600).
Hidden Infrastructure: Host URLs, ports, and passwords are never included in tool responses.
Requirements
Node.js 18.0.0 or higher
PostgreSQL 11 or higher
Optional:
pg_stat_statementsextension for query performance analysisOptional:
hypopgextension for hypothetical index simulation
Development & Testing
The full test suite includes integration tests that exercise the tools against a real PostgreSQL cluster (no mocks). There are two backends:
Local audit cluster (preferred): set
AUDIT_PG_URLto a real PG 14+ instance you control. Tests provision per-test databases on it (e.g.audit_iter1_a,audit_sp1, …) so suites are isolated. The cluster is shared across runs and torn down externally.testcontainers fallback (CI): set
POSTGRES_INTEGRATION_TESTS=1and the suite spins uppostgres:16-alpineper test file.Skipped (default): if neither variable is set, integration tests are silently skipped — unit tests still run.
# Run unit tests only
npm test
# Run against your audit cluster
export AUDIT_PG_URL="postgres://audit_owner:<urlencoded-pw>@127.0.0.1:5433/audit_db"
npm test
# Run only the audit-iteration regression tests
npm test -- --testPathPatterns=audit/iteration
# Run the perf-health deep audit
npm run perfhealthThe audit_owner role on the cluster needs CREATEDB (so each suite can provision its own DB) and CREATEROLE (for test users). It should NOT be a superuser.
🤖 Agent Experience (AX) - Claude Code Review
Tested by: Claude Code (Sonnet 4.5) Use Case: Database deployment, schema exploration, and SQL migration Rating: ⭐⭐⭐⭐⭐ (9.5/10)
What I Loved
1. Clear, Structured Responses
Every response includes connection context (server, database, schema), making it crystal clear which environment I'm working in. This is essential when managing multiple databases - I never have to guess where a query ran.
2. Excellent Error Handling
When I encountered a syntax error with Liquibase's / delimiter, the error message showed:
Exact line number (151)
The failing statement
Transaction rollback confirmation
This made troubleshooting instant. No digging through logs or guessing what failed.
3. Server Management is Intuitive
list_servers→ Shows all available servers with connection statuslist_databases→ Filters databases by server nameswitch_server_db→ Seamless switching with immediate confirmation
The flow is natural: discover → select → connect → execute.
4. SQL File Deployment Made Easy
The stripPatterns feature solved my exact problem:
execute_sql_file({
filePath: "/path/to/liquibase.sql",
stripPatterns: ["/"], // Removes Liquibase delimiters
});Before this feature, I had to manually remove delimiters or use raw execute_sql. Now it's one clean call.
5. Dry-Run Capabilities are Outstanding
dry_run_sql_file is a game-changer:
Executes ALL statements in a transaction
Shows REAL errors with PostgreSQL error codes and constraint names
Automatically skips non-rollbackable operations (VACUUM, NEXTVAL)
Provides EXPLAIN plans for skipped statements
Then rolls back everything
This is way better than just parsing - I can catch constraint violations, trigger issues, and get exact row counts before deployment.
6. Security by Default
Credentials never appear in responses
Host/port intentionally hidden (only server names visible)
Readonly mode available for production safety
Connection context always visible
Improvements Based on My Feedback
The developer implemented several features after I tested the MCP:
✅ SQL File Delimiter Support - Added stripPatterns for Liquibase /, SQL Server GO, etc.
✅ Validate-Only Mode - execute_sql_file({ validateOnly: true }) previews without execution
✅ Enhanced Connection Info - get_current_connection now returns user and AI context
✅ Comprehensive Dry-Run - dry_run_sql_file provides real execution + rollback
✅ Better Error Details - PostgreSQL error codes, constraint names, hints included
v3 Additions — Deeper Database Workflow Support
After live audit testing across multiple iterations against a real PG 17 cluster with complex schemas (FKs, partial indexes, materialized views, cycles), v3 adds a layer of high-leverage tools that reduce typical multi-step AI workflows to a single call:
✅ describe_table - Replaces ~5 calls (object details + sample rows + count + pg_stats) with one rich response including FKs both directions.
✅ find_dependents - Walk pg_depend recursively before any DROP CASCADE to see the blast radius.
✅ lock_check - Static analysis of DDL to predict lock level, table-rewrite, and duration. Knows the semantics of ALTER TABLE, CREATE INDEX [CONCURRENTLY], VACUUM, CLUSTER, REFRESH MATERIALIZED VIEW, etc.
✅ safe_alter_table - Convert intent ("add NOT NULL with default", "add FK", "create index", …) into a multi-step zero-downtime DDL recipe with NOT VALID + VALIDATE patterns.
✅ detect_migration_state - Probe for Liquibase, Flyway, Alembic, Prisma, Knex, Sequelize, Django, Rails, Goose, TypeORM tracker tables. Quoted-identifier safe (e.g. catches Sequelize's "SequelizeMeta").
✅ export_to_sql_file / transfer_objects / schema_diff - First-class cross-database operations: dump, copy, or compare schema/data between configured endpoints. FK constraints emitted as separate ALTER TABLE to handle inter-table cycles.
✅ column_profile - Type-aware column stats with TABLESAMPLE BERNOULLI for tables >1M rows. Replaces a dozen exploratory queries.
✅ generate_seed_data - Schema-aware fake data with NOT NULL, UNIQUE/PK, enum, default, length, and FK awareness.
✅ find_blocking_queries + kill_query - Diagnose lock waits and pg_cancel_backend / pg_terminate_backend with confirmation guard.
✅ Query budget on execute_sql - maxEstimatedRows / maxEstimatedCost refuse the query before execution if the planner says it's too big.
✅ Hardening from audit iterations - per-statement savepoints in dry_run_sql_file (DO-block + embedded COMMIT no longer compromises the dry-run), preserved PG error codes from mutation_dry_run, allowlisted index_type in safe_alter_table, schema validation runs before pool teardown in switch_server_db, detect_migration_state quoted-identifier lookup, findDependents truncatedAtDepth flag actually trips, analyze_db_health reports invalid-index check failures as warning (not silently healthy).
Real-World Experience
Task: Deploy a PostgreSQL function to two databases (dev + analytics)
Discovery:
list_serversshowed all configured serversPreview: Used
preview_sql_fileto check the file structureIssue: Got syntax error from Liquibase's
/delimiterSolution: Switched to direct
execute_sqlto bypass the delimiterDeployment: Successfully deployed to both databases
Verification: Used
get_current_connectionto confirm each deployment
Total time: ~3 minutes. The structured responses and clear errors made it feel effortless.
Minor Suggestions for Future
Batch Cross Servers Deployment - Deploy same script to multiple servers at once
Recent Connections - Quick-switch to recently used databases
Statement Progress - Show progress for large SQL files (e.g., "Executing statement 15/100...")
Bottom Line
This MCP is production-ready and developer-friendly. The combination of clear responses, robust error handling, and powerful features like dry-run make it an essential tool for database work. The developer clearly understands the needs of both AI agents and human operators.
Recommended for: Database migrations, schema exploration, multi-environment management, and production deployments.
License
MIT
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/Teja-sudo/postgres-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server