starlink-enterprise-mcp

starlink-enterprise-mcp

A hosted MCP server that enables AI agents to interact with the Starlink Enterprise API using service accounts, with transparent OAuth token management and 55 auto-generated tools.

Category
Visit Server

README

Starlink Enterprise MCP Server

License: MIT TypeScript Starlink API MCP Protocol Cloud Run

๐Ÿ›ฐ๏ธ Hosted, multi-account MCP for the Starlink Enterprise API Any AI agent โ€” Claude, ChatGPT, anything that speaks MCP โ€” connects with a real Starlink V2 Service Account, drives the full Enterprise API, and stays connected indefinitely. The Client Secret never touches the model.

โšก Features

  • ๐Ÿ” Hosted OAuth proxy with API-key login โ€” The server is the OAuth 2.1 authorization server. But Starlink has no interactive OAuth and no MFA, so the browser login page doesn't ask for a username and password โ€” it asks for a Service Account Client ID + Client Secret. The server validates them with a client_credentials grant; credentials never enter the model's context.
  • ๐Ÿ” Transparent token re-minting โ€” Starlink bearer tokens are short-lived (~15 min) and have no refresh token. The server stores the service-account credentials alongside the issued MCP token and silently re-mints a fresh bearer before expiry, and again on any 401. AI sessions stay alive across long conversations.
  • ๐Ÿช Stateless login state โ€” OAuth pending state rides in HMAC-signed HttpOnly cookies, so logins survive container restarts and Cloud Run instance switches.
  • ๐Ÿ—„๏ธ Firestore persistence โ€” Issued tokens and DCR client registrations survive deploys and scaling events when MCP_PERSISTENCE=firestore.
  • ๐Ÿค Claude and ChatGPT support โ€” Public-client dynamic registration (token_endpoint_auth_method=none, PKCE only) means ChatGPT connects out of the box alongside confidential clients like Claude.
  • ๐Ÿงฌ 55 auto-generated tools from the spec โ€” The Starlink Enterprise v2 OpenAPI spec, regenerated on every build. Drop in a new spec and rebuild to pick up new endpoints.
  • ๐ŸŽฏ No curated layer needed โ€” At 55 operations the full tool surface fits comfortably in a model's working memory, so every tool is exposed directly with read/write/destructive annotations.
  • ๐Ÿช› Operator-tunable โ€” Disable globs (MCP_DISABLED_TOOLS=delete_*,*reboot*), a semantic destructive toggle (MCP_DISABLE_DESTRUCTIVE=true), branded login page (MCP_LOGIN_HEADER, MCP_ICON_URL). No code change for per-deployment policy.
  • ๐Ÿงช A real test suite โ€” including a draft-2020-12 JSON Schema guard that compiles every tool's input schema on every run.

๐Ÿ”‘ How auth differs from a username/password MCP

Username/password OAuth proxy This server (Starlink)
Login page collects username + password Service Account Client ID + Client Secret
Upstream grant password (+ MFA) client_credentials
MFA yes none (service accounts skip MFA)
Refresh upstream refresh token re-run client_credentials (no refresh token)
Token TTL hours ~15 min, re-minted on expiry / 401

The DCR + browser-redirect OAuth shell is identical โ€” what changed is the login form and the upstream grant.

๐Ÿ—๏ธ Architecture

AI client (Claude/ChatGPT)
  โ”‚  OAuth 2.1 DCR + browser login (PKCE)
  โ–ผ
[ Starlink MCP HTTP server (this repo) ]   โ† OAuth proxy, login page (Client ID + Secret), cookies, Firestore
  โ”‚  per-account Starlink bearer (client_credentials)
  โ–ผ
[ Starlink Enterprise API  https://web-api.starlink.com ]

Each issued MCP bearer maps to a stored upstream Starlink token plus the service-account credentials used to mint it, so the server can re-mint silently.

๐Ÿ’ป Running locally (stdio)

npm install
npm run build
export STARLINK_CLIENT_ID=<your-service-account-client-id>
export STARLINK_CLIENT_SECRET=<your-service-account-secret>
npm start                                      # MCP_TRANSPORT defaults to stdio

Create a V2 service account at Account Settings โ†’ API V2 Service Accounts (requires the Admin or Service Account Management role).

Add this entry to your local MCP client config (Claude Desktop, etc.):

{
  "mcpServers": {
    "starlink": {
      "command": "node",
      "args": ["/path/to/starlink-enterprise-mcp/build/index.js"],
      "env": {
        "STARLINK_CLIENT_ID": "...",
        "STARLINK_CLIENT_SECRET": "..."
      }
    }
  }
}

You can also set STARLINK_ACCESS_TOKEN directly to skip the grant if you already hold a bearer.

๐ŸŒ Running as a hosted server (HTTP)

export MCP_TRANSPORT=http
export MCP_PORT=3000
export MCP_BASE_URL=https://mcp.example.com
export MCP_SESSION_SECRET=<32+ random hex>     # signs login-state cookies
npm start

Connect from Claude / ChatGPT by giving it the URL https://mcp.example.com/mcp. The client DCR-registers, redirects the user to /authorize, the user pastes their Service Account Client ID + Secret, and the bearer flows back to the AI automatically. No upstream operator credentials are needed in HTTP mode โ€” each user brings their own service account.

โ˜๏ธ Cloud Run deployment

Ships with a Cloud Run-friendly Dockerfile and cloudbuild.yaml.

Component Purpose
Cloud Run service Runs the HTTP server with session affinity and min-instances=1
Firestore (native mode) Persistent token store and DCR client registry
Cloud Run SA โ†’ roles/datastore.user Firestore access
gcloud builds submit --config cloudbuild.yaml --project=<your-project>

Required env vars on Cloud Run:

Var Notes
MCP_TRANSPORT=http enable the HTTP transport
MCP_BASE_URL public URL, e.g. https://mcp.example.com
MCP_SESSION_SECRET 32+ chars; signs cookies & must be stable across instances
MCP_PERSISTENCE=firestore enable Firestore-backed tokens and clients
GOOGLE_CLOUD_PROJECT Firestore project ID (auto-set on Cloud Run)

Optional: STARLINK_API_URL, STARLINK_TOKEN_URL (defaults are correct for production), MCP_LOGIN_HEADER, MCP_ICON_URL, MCP_LOGIN_LOGO_URL, MCP_DISABLED_TOOLS, MCP_DISABLED_ACTIONS, MCP_DISABLE_DESTRUCTIVE, MCP_CORS_ORIGIN.

Other targets: fly.toml (Fly.io), render.yaml (Render), railway.toml (Railway), docker-compose.yml, and k8s/ manifests (apply with kubectl apply -k k8s/).

Security note on persistence. In HTTP mode the issued-token records hold each user's Starlink service-account Client ID + Secret so the server can re-mint bearers. Protect the token store accordingly โ€” restrict the Firestore collection / file volume, and rotate MCP_SESSION_SECRET and service-account secrets per Starlink's guidance if exposure is suspected.

๐Ÿ” OAuth flow (detailed)

  1. AI client hits GET /.well-known/oauth-protected-resource/mcp and /.well-known/oauth-authorization-server for discovery.
  2. AI client POSTs /register (RFC 7591 DCR). Public clients pass token_endpoint_auth_method=none and get back a client_id only; confidential clients also get a client_secret. Registrations persist in Firestore.
  3. AI redirects the user's browser to /authorize?... with PKCE parameters. The server stores the pending request in a signed cookie (mcp_pending_auth, 15 min TTL) and renders the login page.
  4. User submits their Service Account Client ID + Client Secret โ†’ server runs POST {STARLINK_TOKEN_URL} with grant_type=client_credentials. On success it stores the Starlink token + credentials and issues an authorization code.
  5. The server redirects back to the AI client; cookies are cleared.
  6. AI exchanges the code at /token for the MCP-issued bearer + refresh token.
  7. On every /mcp request, the server verifies the bearer and transparently re-mints the upstream Starlink token if it's near expiry. On a 401 from the API, the client re-mints and retries once.

๐Ÿงฐ Tools

55 tools generated from spec/starlink-enterprise-v2.json, grouped by tag:

Group Examples
Account get_account, get_products, post_data_usage_query
Service Lines get_service_lines, post_service_lines, put_service_line_nickname, post_service_line_data_top_up, patch_service_line_consume_from_pool
User Terminals get_user_terminals, post_user_terminals, post_user_terminal_reboot, put_user_terminal_l2vpn
Routers get_router, get_routers_configs, post_routers_configs, post_router_reboot, *_routers_configs_tls
Addresses get_addresses, post_addresses, get_address, put_address
Contacts get_contacts, post_contacts, put_contact, delete_contact
Data Pools get_data_pools, get_data_pools_usage, post_data_pools_by_data_pool_id_set_automatic_top_up
Flights post_flights_status (aviation accounts)
Managed post_managed_customers (provider accounts)

Each tool is annotated readOnlyHint / destructiveHint. Reboots and deletes are flagged destructive โ€” hide them all with MCP_DISABLE_DESTRUCTIVE=true, or selectively with e.g. MCP_DISABLED_TOOLS=delete_*,*reboot*.

Tool names map 1:1 to operations ({method}_{path}, with the /public/v2 prefix stripped). Two deep service-line paths are abbreviated to fit the MCP 64-character name limit.

๐Ÿ”„ Regenerating tools

The spec lives at spec/starlink-enterprise-v2.json (sourced from https://web-api.starlink.com/enterprise/swagger/v2/swagger.json). To refresh:

# drop a new spec into spec/starlink-enterprise-v2.json, then:
npm run generate      # rewrites src/generated/
npm run build
npm test

npm run build runs generate automatically via the prebuild hook.

๐Ÿงช Tests

npm test

The Firestore-backed tests are emulator-gated and skip cleanly without one.

๐Ÿ“‹ What this server is

  • Two MCP transports. stdio for local CLI integrations and http (Streamable HTTP) for hosted deployments. Production uses http.
  • Auto-generated tools from the Starlink Enterprise v2 OpenAPI spec, regenerated on every build.
  • Hosted OAuth login where the login page collects Starlink Service Account credentials (Client ID + Secret), not a username/password. MFA does not apply to service accounts.
  • Transparent token re-minting via client_credentials (no refresh token).
  • Firestore persistence for tokens and DCR clients when MCP_PERSISTENCE=firestore.

License

MIT

Recommended Servers

playwright-mcp

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.

Official
Featured
TypeScript
Magic Component Platform (MCP)

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.

Official
Featured
Local
TypeScript
Audiense Insights MCP Server

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.

Official
Featured
Local
TypeScript
VeyraX MCP

VeyraX MCP

Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.

Official
Featured
Local
graphlit-mcp-server

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.

Official
Featured
TypeScript
Kagi MCP Server

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.

Official
Featured
Python
E2B

E2B

Using MCP to run code via e2b.

Official
Featured
Neon Database

Neon Database

MCP server for interacting with Neon Management API and databases

Official
Featured
Exa Search

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.

Official
Featured
Qdrant Server

Qdrant Server

This repository is an example of how to create a MCP server for Qdrant, a vector search engine.

Official
Featured