MCP Server
Node.js MCP server for ChatGPT with OAuth 2.0 authorization code flow + PKCE, refresh token rotation, and Supabase-backed persistence. Includes sample tools like calculator, get_weather, and search_web.
README
MCP Server
Node.js + Express MCP server for ChatGPT with:
- OAuth 2.0 Authorization Code Flow + PKCE
S256 - refresh token rotation
- Supabase Postgres persistence
- MCP Streamable HTTP endpoint at
/mcp - sample tools:
calculator,get_weather,search_web - cleanup job for expired OAuth records
Structure
mcp-server/
|-- .env.example
|-- IMPLEMENTATION_REPORT.md
|-- package.json
|-- README.md
|-- supabase/
| `-- schema.sql
`-- src/
|-- app.js
|-- server.js
|-- config/
|-- mcp/
|-- middleware/
|-- routes/
|-- scripts/
|-- services/
|-- tools/
`-- utils/
1. Supabase setup
- Create a Supabase project.
- Open the SQL editor.
- Run
supabase/schema.sql. - Copy the Postgres connection string into
.envasDATABASE_URL.
For IPv4-only networks, use the Supabase Session Pooler URI instead of the direct IPv6 connection string.
Seed an OAuth client
You need at least one record in oauth_clients.
For local development the server can seed one automatically from env:
AUTO_SEED_CLIENT_IDAUTO_SEED_CLIENT_NAMEAUTO_SEED_CLIENT_SECRETAUTO_SEED_REDIRECT_URIS
If AUTO_SEED_CLIENT_SECRET is empty, the client is treated as public and may use none auth at the token endpoint.
If it is set, the client supports both client_secret_basic and client_secret_post.
2. Local setup
cd mcp-server
npm install
copy .env.example .env
Edit .env:
BASE_URLDATABASE_URLCORS_ORIGINSMCP_ALLOWED_ORIGINSMCP_ALLOWED_HOSTSENABLE_CLEANUP_JOBCLEANUP_INTERVAL_MINUTESAUTO_SEED_*
Check database connectivity:
npm run db:check
npm run db:init
Run locally:
npm run dev
Health check:
GET http://localhost:3000/healthz
Run cleanup manually:
npm run cleanup:run
3. Expose with ngrok
ngrok http 3000
Take the HTTPS URL from ngrok and set it as:
BASE_URL- add the ngrok origin to
MCP_ALLOWED_ORIGINS - add the ngrok host to
MCP_ALLOWED_HOSTS - one of the registered redirect URIs in
oauth_clients
Restart the server after updating .env.
4. OAuth endpoints
Authorization endpoint
GET /authorizePOST /authorize/decision- demo consent screen with optional auto-approve mode
- validates:
client_idredirect_uriresponse_type=codecode_challengecode_challenge_method=S256- scopes
Token endpoint
POST /token- supports:
grant_type=authorization_codegrant_type=refresh_token
Client authentication:
noneclient_secret_basicclient_secret_post
Discovery
GET /.well-known/oauth-authorization-serverGET /.well-known/openid-configuration
Revoke
POST /revoke
5. Refresh token rotation
This server rotates refresh tokens on every refresh request:
- the current refresh token row is locked
- the old refresh token is revoked
- a new access token is issued
- a new refresh token is issued
- rotation lineage is stored in Postgres
If a revoked or replaced refresh token is reused:
- the backing
mcp_sessionis markedcompromised - session tokens are revoked
- the request fails with
invalid_grant
6. MCP endpoint
Main MCP endpoint:
POST /mcp
Behavior:
- uses MCP Streamable HTTP transport
- path is
/mcp - bearer access token is required
- host and origin allowlists are enforced
- the implementation is stateless on the transport layer
- each HTTP request creates a fresh MCP server and transport instance
- no in-memory MCP transport session map is retained across requests or restarts
Compatibility note:
GET /mcpandDELETE /mcpcurrently return405- this server is using the stateless streamable HTTP pattern, not SSE session transport
7. Sample tools
calculatorget_weathersearch_web
These are in src/tools/ and registered via src/mcp/tool-registry.js.
8. Cleanup job
The server starts a background cleanup job on boot when ENABLE_CLEANUP_JOB=true.
It removes:
- expired or stale authorization codes
- expired or revoked access tokens older than 1 day
- expired or revoked refresh tokens older than 7 days
Interval is controlled by CLEANUP_INTERVAL_MINUTES.
9. Register in ChatGPT
Use your deployed server values:
- Authorization URL:
https://your-domain/authorize - Token URL:
https://your-domain/token - Revoke URL:
https://your-domain/revoke - MCP URL:
https://your-domain/mcp
Recommended scopes:
mcp.tools.calloffline_access
PKCE:
- required
- method:
S256
If ChatGPT or your connector uses a confidential client:
- create the client in
oauth_clients - store the secret hash
- allow
client_secret_basicand/orclient_secret_post
10. Security notes
- token values are stored hashed in Postgres
- client secrets are stored hashed
- rate limiting is applied to
/authorize,/token,/revoke - OAuth errors use standard fields:
errorerror_description
- raw secrets and tokens are not logged
- refresh token reuse triggers session compromise handling
11. Current status
Verified against real Supabase:
npm run db:checknpm run db:initnpm run cleanup:run
What is still intentionally basic:
- the consent page is a demo flow, not a real user login system
- access tokens are opaque and require a DB lookup
- there is no automated test suite yet
- rate limits and allowlists should be tightened before production
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.