mcp-audit
The mcp-audit server is a security and observability proxy wrapping an upstream filesystem MCP server. It provides two layers of functionality:
Filesystem Operations (via proxied server)
Read files:
read_text_file(full or first/last N lines),read_media_file(images/audio as base64),read_multiple_files(batch),read_file(deprecated)Write/Edit:
write_fileto create or overwrite;edit_filefor line-based edits with diff output and dry-run supportDirectory operations:
create_directory,list_directory,list_directory_with_sizes(sortable),directory_tree(recursive JSON tree),list_allowed_directoriesFile management:
move_fileto move or rename;search_fileswith glob patterns;get_file_infofor size, timestamps, and permissions
Security & Observability (mcp-audit proxy layer)
Audit trails: All tool calls, resource reads, and JSON-RPC methods are logged as signed JSONL or SQLite entries
Data redaction: Sensitive fields are automatically redacted before storage
Policy enforcement: Synchronous allow/deny policies can block specific tools (e.g., destructive operations), with per-tool rate limiting
Monitoring: A local read-only dashboard shows recent audit entries, top tools, and error rates; Prometheus metrics are exposed for external monitoring
mcp-audit
A drop-in security and observability proxy for MCP servers. mcp-audit sits between an MCP client and any upstream MCP server to produce signed audit trails, redact sensitive payloads, enforce allow/deny policies and per-tool rate limits, and expose a local read-only dashboard.
Why mcp-audit?
The MCP 2026 roadmap calls out enterprise needs around audit trails, gateway patterns, and operational visibility. mcp-audit fills that gap as a deployable sidecar or local wrapper: it sits between any MCP client and server, preserves protocol traffic, and records signed audit entries for tool calls, resource reads, prompt requests, and all other JSON-RPC methods.
+-------------+ JSON-RPC / MCP +-----------+ JSON-RPC / MCP +-------------+
| MCP client | <-------------------> | mcp-audit | <---------------------> | MCP server |
+-------------+ +-----------+ +-------------+
|
v
JSONL or SQLite audit log
|
v
Read-only dashboardRelated MCP server: GitHub MCP Server Plus
What This Is / Is Not
mcp-audit is not a domain-specific MCP server. It is a transparent security and observability proxy that wraps any MCP server and audits the JSON-RPC traffic passing through it.
Directories may show the tools exposed by the upstream server, not tools implemented by mcp-audit itself.
Supported Transports
stdiofor local MCP clients such as Claude Desktophttpfor MCP servers exposed over HTTP
Use Cases
Audit tool calls made by AI agents in regulated environments
Detect unexpected or dangerous MCP tool usage
Keep signed JSONL or SQLite logs for incident review
Redact sensitive fields before storing requests and responses
Block disallowed tools and apply per-tool rate limits without modifying the upstream MCP server
Demo

Quick Start
Install Go, then build from source:
brew install go
go install github.com/P4ST4S/mcp-audit/cmd/mcp-audit@v0.2.0Run in stdio mode:
AUDIT_SECRET="$(openssl rand -hex 32)" \
mcp-audit --transport stdio --upstream "npx @modelcontextprotocol/server-filesystem /tmp"Run in HTTP mode:
mcp-audit --transport http --upstream http://localhost:8080 --port 4422Run with Docker Compose:
docker compose up --buildThe dashboard is available at http://localhost:9090 by default.
Prometheus metrics are available at http://localhost:9091/metrics by default.
Configuration
mcp-audit loads config.yaml from the current directory by default. CLI flags override config values, and AUDIT_SECRET overrides audit.secret.
Key | Default | Description |
|
| Proxy transport: |
| required | Stdio command or HTTP upstream URL. |
|
| HTTP listen port. |
|
| Client identifier written to audit entries. |
|
| Server identifier written to audit entries. |
|
| Storage backend: |
|
| JSONL audit log path. |
|
| SQLite database path. |
|
| Enable HMAC-SHA256 signatures when a secret is set. |
| empty | HMAC secret. Prefer |
|
| Enable asynchronous batched audit writes through a bounded ring buffer. |
|
| Maximum queued audit entries before backpressure blocks writers. |
|
| Maximum entries written per storage batch. |
|
| Maximum time before a partial batch is flushed. |
|
| Enable per-client, per-tool token buckets. |
|
| Allowed requests per minute per |
|
| Enable JSON key-based PII redaction. |
| sensitive keys | Case-insensitive key fragments to redact. |
|
| Enable synchronous allow/deny policy checks for |
|
| Fallback action when no policy rule matches: |
| empty | Ordered first-match allow/deny rules for tool calls. |
|
| Serve the dashboard. |
|
| Dashboard listen port. |
|
| Serve Prometheus metrics on a separate HTTP endpoint. |
|
| Metrics listen port. |
|
| Metrics HTTP path. |
|
| Include Go runtime metrics. |
|
| Include process metrics. |
|
| Include |
CLI flags:
--transport stdio | http
--upstream upstream server command or URL
--port proxy port for http mode
--config path to config.yaml
--storage jsonl | sqlite
--no-dashboard disable the web dashboard
--no-metrics disable Prometheus metrics
--log-level debug | info | warn | errorClaude Desktop
Configure Claude Desktop to spawn mcp-audit instead of the upstream MCP server:
{
"mcpServers": {
"filesystem-audited": {
"command": "mcp-audit",
"args": [
"--transport",
"stdio",
"--upstream",
"npx @modelcontextprotocol/server-filesystem /tmp"
],
"env": {
"AUDIT_SECRET": "replace-with-a-long-random-secret"
}
}
}
}Dashboard
The dashboard shows recent entries, filters, expandable request/result JSON, top tools, calls today, and error rate. It refreshes every five seconds.
Prometheus Metrics
mcp-audit exposes Prometheus metrics on a separate endpoint so platform teams can scrape operational data without exposing the dashboard.
scrape_configs:
- job_name: mcp-audit
static_configs:
- targets: ["localhost:9091"]Application metrics use the mcp_audit_ prefix and avoid unbounded labels. Tool-level labels can be disabled with metrics.tool_labels: false for stricter cardinality control. Policy decisions are exposed as mcp_audit_policy_decisions_total{action="allow|deny"}.
Policy Engine
mcp-audit can enforce synchronous allow/deny rules before a tools/call reaches the upstream MCP server. Denied calls return a JSON-RPC error and are still written to the audit log.
policy:
enabled: true
default_action: allow
rules:
- action: deny
client_id: claude-desktop
server_id: filesystem
tool_name: delete_file
reason: "Destructive filesystem operations are blocked"Rules are evaluated in order. Empty fields and * match any value, so default_action: deny can be used with explicit allow rules for stricter deployments.
Audit Entries
Each stored entry includes a ULID, timestamp, direction, transport, JSON-RPC method, tool name when present, redacted params/result, JSON-RPC error when present, duration, client/server identifiers, and an optional HMAC-SHA256 signature.
Example JSONL entry:
{
"id": "01HY8G6Y8S6W9K6ZD7VJ4Q8X4R",
"timestamp": "2026-05-25T12:34:56Z",
"direction": "client_to_server",
"transport": "stdio",
"method": "tools/call",
"tool_name": "read_file",
"params": {
"name": "read_file",
"arguments": {
"path": "/tmp/example.txt",
"token": "[REDACTED]"
}
},
"duration_ms": 18,
"client_id": "claude-desktop",
"server_id": "filesystem",
"signature": "hmac-sha256:..."
}The signature covers:
id + timestamp + method + tool_name + raw_paramsRoadmap
OpenTelemetry export
SIEM-friendly exports
Contributing
This project is experimental. Keep changes small, run go build ./... and go vet ./..., and prefer standard library behavior over new dependencies.
See CONTRIBUTING.md for setup, PR expectations, and project principles. See CHANGELOG.md for release history.
License
Apache-2.0. See LICENSE.
Maintenance
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/P4ST4S/mcp-audit'
If you have feedback or need assistance with the MCP directory API, please join our Discord server