MCP server for Google BigQuery — authenticate via service account JSON env var, run with uvx
Project description
BigQuery MCP Server
A Model Context Protocol (MCP) server that enables LLMs to interact with Google BigQuery. Built with FastMCP (Python) and designed for ephemeral cloud environments where only environment variables are available (no file-based credentials).
Features
| Tool | Description |
|---|---|
bq_list_datasets |
List all datasets in the configured project |
bq_list_tables |
List tables in a dataset with row counts and size in MB |
bq_get_schema |
Get the full schema of a table (columns, types, descriptions) |
bq_dry_run |
Validate SQL and estimate query cost (~$6.25/TB) |
bq_run_query |
Execute SQL and return results (auto-appends LIMIT 1000 if missing) |
Project Structure
bigquery-mcp-server/
├── src/
│ ├── server.py # Entry point — starts the MCP server
│ ├── constants.py # Global constants (BigQuery config, limits)
│ ├── tools/
│ │ └── bigquery.py # All 5 BigQuery tools
│ └── services/
│ ├── bigquery_client.py # BigQuery client factory + row serialization
│ └── error_handler.py # Centralized BigQuery error handling
├── main.py # Root-level entry point wrapper
├── Dockerfile # Multi-stage Docker build
├── docker-compose.yml # Server + MCP Inspector
├── docker-compose.dev.yml # Dev mode with hot-reload
├── mcp.json # MCP client configuration
├── pyproject.toml # Project metadata and dependencies
└── requirements.txt # Pip-compatible dependencies
Environment Variables
| Variable | Required | Default | Description |
|---|---|---|---|
BIGQUERY_PROJECT_ID |
Yes | — | GCP project ID (e.g. my-project-123) |
BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT |
Yes | — | Raw JSON string of the service account key |
BIGQUERY_LOCATION |
No | US |
BigQuery dataset region |
TRANSPORT |
No | stdio |
Transport mode: stdio or http |
PORT |
No | 8000 |
HTTP server port |
HOST |
No | 127.0.0.1 |
HTTP server bind address |
Authentication
This server authenticates using a service account JSON provided directly via environment variable — no file paths, no gcloud CLI.
- Create a service account in the GCP Console with at least:
- BigQuery Data Viewer (
roles/bigquery.dataViewer) - BigQuery Job User (
roles/bigquery.jobUser)
- BigQuery Data Viewer (
- Generate a JSON key for the service account
- Set the full JSON string as
BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT
export BIGQUERY_PROJECT_ID="my-project-123"
export BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT='{"type":"service_account","project_id":"...","private_key":"...","client_email":"...","...":"..."}'
Quick Start
Option A — uvx (Recommended)
No manual install needed. uvx downloads the package from PyPI, creates an isolated environment, and runs the server:
uvx bigquery-mcp-server
For MCP clients (Claude Code, Cursor, etc.), configure the server in your client's MCP config:
{
"mcpServers": {
"bigquery-mcp": {
"command": "uvx",
"args": ["bigquery-mcp-server"],
"env": {
"BIGQUERY_PROJECT_ID": "my-project-123",
"BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT": "{ ... }"
}
}
}
}
Note: Env vars must be passed explicitly via the
envblock in your MCP client config.
Option B — Local (virtualenv)
# Create virtual environment and install
python -m venv .venv
source .venv/bin/activate # Windows: .venv\Scripts\activate
pip install -e .
# Set credentials (or use a .env file)
export BIGQUERY_PROJECT_ID="my-project-123"
export BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT='{ ... }'
# Run in stdio mode
python main.py
# Or HTTP mode (for MCP Inspector)
TRANSPORT=http python main.py
Option C — Docker
# Set env vars in a .env file or export them, then:
docker compose up --build
# Open MCP Inspector at http://localhost:6274
# Connect to: http://mcp-server:8000/mcp (Streamable HTTP)
Option D — Dev mode with hot-reload
docker compose -f docker-compose.dev.yml up --build
# In another terminal:
npx @modelcontextprotocol/inspector
# Connect to: http://localhost:8000/mcp (Streamable HTTP)
Testing with MCP Inspector CLI
# List all tools
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/list
# List datasets
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/call --tool-name bq_list_datasets
# List tables in a dataset
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/call --tool-name bq_list_tables \
--tool-arg 'params={"dataset": "my_dataset"}'
# Get table schema
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/call --tool-name bq_get_schema \
--tool-arg 'params={"dataset": "my_dataset", "table": "my_table"}'
# Dry run a query
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/call --tool-name bq_dry_run \
--tool-arg 'params={"sql": "SELECT * FROM my_dataset.my_table"}'
# Run a query
npx @modelcontextprotocol/inspector --cli \
--config mcp.json --server bigquery-mcp \
--method tools/call --tool-name bq_run_query \
--tool-arg 'params={"sql": "SELECT * FROM my_dataset.my_table LIMIT 10"}'
Connect to Claude Code
Using uvx (recommended):
claude mcp add bigquery-mcp uvx -- bigquery-mcp-server
Or add to ~/.claude/mcp.json:
{
"mcpServers": {
"bigquery-mcp": {
"command": "uvx",
"args": ["bigquery-mcp-server"],
"env": {
"BIGQUERY_PROJECT_ID": "my-project-123",
"BIGQUERY_SERVICE_ACCOUNT_JSON_CONTENT": "{ ... }"
}
}
}
}
Transports
| Transport | How to activate | When to use |
|---|---|---|
| stdio | TRANSPORT=stdio python -m src.server |
Claude Code, Cursor, local integration |
| HTTP | TRANSPORT=http python -m src.server |
MCP Inspector, remote servers |
Safety Features
- Auto LIMIT:
bq_run_queryautomatically appendsLIMIT 1000if no LIMIT clause is detected, preventing accidental large data transfers - Dry run:
bq_dry_runvalidates SQL and estimates cost before execution - Input validation: All tool inputs validated with Pydantic v2
- Error handling: BigQuery exceptions (403, 404, 400, etc.) are caught and returned as human-readable strings — the server never crashes
- Type serialization: Dates, datetimes, decimals, and bytes are automatically converted to JSON-safe types
References
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 bigquery_mcp_server-0.1.1.tar.gz.
File metadata
- Download URL: bigquery_mcp_server-0.1.1.tar.gz
- Upload date:
- Size: 10.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.0 {"installer":{"name":"uv","version":"0.10.0","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4bbbf8a46d734d76a69fb1e65b7c7b612156503c07ce30b3be6e6b13d922fa3a
|
|
| MD5 |
e5d7343ea6ec890e72eb18f9867754ff
|
|
| BLAKE2b-256 |
5cef33994118ddc8132afa69d0d1eebbe80cf196e19269220f52db8d071c0cec
|
File details
Details for the file bigquery_mcp_server-0.1.1-py3-none-any.whl.
File metadata
- Download URL: bigquery_mcp_server-0.1.1-py3-none-any.whl
- Upload date:
- Size: 13.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.10.0 {"installer":{"name":"uv","version":"0.10.0","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8ff724db043d31759857adcd373bb608758f24b77168f553784755d14ac65635
|
|
| MD5 |
c3eb1fc51d2a122bbfcbd5b34c1f8952
|
|
| BLAKE2b-256 |
e6698ee9befe725507e76a6e8e9e0bedb5fa48c2c0186f43f85f2d362ff5ce06
|