Access EPA environmental data — facility compliance (ECHO), toxic releases (TRI), Superfund sites, drinking water systems, and real-time air quality (AirNow) via MCP. STDIO or Streamable HTTP.
8 tools spanning EPA facility compliance, toxic chemical releases, Superfund cleanup sites, drinking water safety, and real-time air quality:
| Tool | Description |
|---|---|
epa_search_facilities |
Search EPA-regulated facilities by location, industry, or compliance status across all environmental programs (CAA, CWA, RCRA, TRI, SDWA). Returns facility name, EPA Registry ID, coordinates, county FIPS, per-program compliance flags, inspection counts, penalty totals, and TRI release totals. |
epa_get_facility |
Retrieve a full compliance profile for a single EPA-regulated facility: compliance status per program, inspection dates, formal enforcement actions, penalty amounts, and TRI annual release totals. Aggregates multiple ECHO DFR endpoints in parallel. |
epa_search_violations |
Search EPA civil and criminal enforcement cases by state, regulatory program, or date range. Returns case identifier, facility name and Registry ID, programs involved, penalty assessed, settlement date, and case type. |
epa_get_air_quality |
Get AQI observations or forecasts for a location. Returns per-pollutant AQI values (PM2.5, ozone, CO, SO2, NO2), AQI category (Good through Hazardous), reporting area name, and observation timestamp. |
epa_get_tri_releases |
Query Toxic Release Inventory annual chemical release data for a specific facility by medium (air, water, land, underground injection) and reporting year. |
epa_search_tri_releases |
Search Toxic Release Inventory data across facilities in a state or county for a given year. Returns facility name, TRI ID, chemical name, total releases by medium, and facility coordinates. |
epa_search_superfund |
Search Superfund (CERCLA/SEMS) sites by location or NPL listing status. Accepts state/city/ZIP or lat/lng + radius for proximity searches. Returns site name, EPA ID, NPL status, cleanup status, and coordinates. |
epa_search_water_systems |
Search drinking water systems (SDWIS) by state or ZIP code. Returns system name, PWSID, population served, primary water source, and active violation status. |
Search for EPA-regulated facilities with cross-program compliance data.
- Geographic filters: ZIP code, state, city (city requires state)
- Program filter: narrow to CAA, CWA, RCRA, TRI, or SDWA registrants
- Compliance filter:
has_violationflag to surface only non-compliant facilities - Returns
RegistryID(key forepa_get_facility),FacFIPSCode(county FIPS for Census chaining), and coordinates - Results cap enforced — unscoped searches are prohibited at the input validation layer
Retrieve a comprehensive compliance profile by EPA Registry ID.
- Aggregates 3–5 ECHO DFR endpoints in parallel: program flags and TRI totals, compliance summary, inspection/enforcement history, CAA details (if registered), CWA/NPDES permit details (if registered)
- Uses
Promise.allSettled— partial data returned even if one upstream endpoint fails - Includes formal enforcement actions, penalty amounts, and inspection dates across all programs
Search area-level EPA enforcement cases — distinct from per-facility history in epa_get_facility.
- Program filter: CAA, CWA, RCRA, SDWA, CERCLA, FIFRA, or TSCA
- Case type: civil, criminal, or all
- Date range filtering by filing date (ISO 8601)
- Returns case identifier, affected facility name and Registry ID for downstream
epa_get_facilitylookup
Get current AQI observations or daily forecasts from AirNow.
- Accepts ZIP code or latitude/longitude coordinates
mode: currentreturns the latest observed AQI per pollutant;mode: forecastreturns daily AQI forecasts (requiresforecast_date)- AQI categories: Good (1) through Hazardous (6) with numeric and text category
- Data is preliminary — suitable for awareness, not regulatory or enforcement decisions
- AirNow responses cached at ~1 hour TTL to respect rate limits
Query per-chemical release breakdown for a single TRI facility.
- Accepts TRI facility ID from
epa_search_facilitiesresults - Returns release quantities by medium: air, water, land, underground injection
- Optional year filter (defaults to most recent available); optional chemical name filter
- TRI data lags ~18 months — most recent available year is typically 2 years prior to current
Identify top polluters in a region via TRI data.
- State and optional county scope; required year parameter
- Optional chemical name filter to focus on a specific substance
- Returns facility coordinates for downstream map or proximity analysis
- Complement to
epa_get_tri_releases— use this for area discovery, then drill into a specific facility
Search Superfund (CERCLA/SEMS) sites by location or proximity.
- Two input shapes: state/city/ZIP for administrative filters, or lat/lng + radius for proximity
- NPL status filter: listed, not-listed, proposed, or all
- Returns site cleanup status and coordinates for downstream spatial analysis
Identify drinking water systems with active or recent violations.
- State and optional ZIP code scope
has_violationflag surfaces only systems with current violations- PWS type filter: community (
CWS), non-transient non-community (NTNCWS), or transient non-community (TNCWS)
| Type | Name | Description |
|---|---|---|
| Resource | epa://facility/{registry_id} |
Full compliance profile for a facility by EPA Registry ID (same data as epa_get_facility) |
| Resource | epa://superfund/{site_id} |
Superfund site record by SEMS site ID |
All resource data is also reachable via tools. Use epa_get_facility and epa_search_superfund for programmatic access in tool-only MCP clients.
Built on @cyanheads/mcp-ts-core:
- Declarative tool and resource definitions — single file per primitive, framework handles registration and validation
- Unified error handling — handlers throw, framework catches, classifies, and formats
- Pluggable auth:
none,jwt,oauth - Swappable storage backends:
in-memory,filesystem,Supabase,Cloudflare KV/R2/D1 - Structured logging with optional OpenTelemetry tracing
- STDIO and Streamable HTTP transports
EPA-specific:
- Three complementary EPA APIs unified behind a single
epa_tool surface: ECHO (facility compliance), Envirofacts DMAP (TRI, Superfund, SDWIS), and AirNow (real-time air quality) - Parallel ECHO DFR aggregation in
epa_get_facility— 3–5 upstream calls resolved concurrently withPromise.allSettled - AirNow response caching (~1 hour TTL) to stay within per-key rate limits
- DMAP coordinate normalization —
tri.tri_facilityDDMMSS integers converted to decimal degrees
Agent-friendly output:
- Cross-tool join keys surfaced on every response —
RegistryIDandFacFIPSCodefrom facility search feed directly into compliance, TRI, and Census API workflows - Typed enforcement and compliance status fields — agents branch on data values, not string parsing
- Structured partial failure —
epa_get_facilityreturns available program data even when one DFR endpoint is unavailable, with per-section status
Add the following to your MCP client configuration file. An AirNow API key is required for epa_get_air_quality — register free at docs.airnowapi.org. ECHO and DMAP tools work without authentication.
{
"mcpServers": {
"epa-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["@cyanheads/epa-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"AIRNOW_API_KEY": "your-airnow-key"
}
}
}
}Or with npx (no Bun required):
{
"mcpServers": {
"epa-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@cyanheads/epa-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"AIRNOW_API_KEY": "your-airnow-key"
}
}
}
}For Streamable HTTP, set the transport and start the server:
MCP_TRANSPORT_TYPE=http MCP_HTTP_PORT=3010 AIRNOW_API_KEY=... bun run start:http
# Server listens at http://localhost:3010/mcp- Bun v1.3.0 or higher (or Node.js v24+).
- An AirNow API key for
epa_get_air_quality— register free at docs.airnowapi.org/account/request. ECHO and DMAP tools require no API key.
- Clone the repository:
git clone https://github.com/cyanheads/epa-mcp-server.git- Navigate into the directory:
cd epa-mcp-server- Install dependencies:
bun install- Configure environment:
cp .env.example .env
# edit .env and set AIRNOW_API_KEYAll configuration is validated at startup via Zod schemas in src/config/. Key environment variables:
| Variable | Description | Default |
|---|---|---|
AIRNOW_API_KEY |
Required for epa_get_air_quality. Free registration at docs.airnowapi.org. ECHO and DMAP tools work without it. |
— |
EPA_ECHO_BASE_URL |
ECHO API base URL | https://echodata.epa.gov/echo |
EPA_DMAP_BASE_URL |
Envirofacts DMAP API base URL | https://data.epa.gov/dmapservice |
EPA_AIRNOW_BASE_URL |
AirNow API base URL | https://www.airnowapi.org/aq |
MCP_TRANSPORT_TYPE |
Transport: stdio or http |
stdio |
MCP_HTTP_PORT |
HTTP server port | 3010 |
MCP_HTTP_ENDPOINT_PATH |
HTTP endpoint path | /mcp |
MCP_AUTH_MODE |
Auth mode: none, jwt, or oauth |
none |
MCP_LOG_LEVEL |
Log level (RFC 5424) | info |
LOGS_DIR |
Directory for log files (Node.js only) | <project-root>/logs |
STORAGE_PROVIDER_TYPE |
Storage backend: in-memory, filesystem, supabase, cloudflare-kv/r2/d1 |
in-memory |
OTEL_ENABLED |
Enable OpenTelemetry tracing and metrics | false |
See .env.example for the full list of optional overrides.
-
Build and run:
# One-time build bun run rebuild # Run the built server bun run start:stdio # or bun run start:http
-
Run checks and tests:
bun run devcheck # Lint, format, typecheck, security bun run test # Vitest test suite bun run lint:mcp # Validate MCP definitions against spec
docker build -t epa-mcp-server .
docker run --rm -e AIRNOW_API_KEY=your-key -p 3010:3010 epa-mcp-serverThe Dockerfile defaults to HTTP transport, stateless session mode, and logs to /var/log/epa-mcp-server. OpenTelemetry peer dependencies are installed by default — build with --build-arg OTEL_ENABLED=false to omit them.
| Directory | Purpose |
|---|---|
src/index.ts |
createApp() entry point — registers tools/resources and inits services. |
src/config |
Server-specific environment variable parsing and validation with Zod. |
src/mcp-server/tools |
Tool definitions (*.tool.ts). Eight tools across ECHO, DMAP, and AirNow. |
src/mcp-server/resources |
Resource definitions (*.resource.ts). Facility and Superfund URI handlers. |
src/services/echo |
ECHO REST API service layer — facility search, facility detail, enforcement cases. |
src/services/dmap |
Envirofacts DMAP service layer — TRI releases, Superfund sites, drinking water systems. |
src/services/airnow |
AirNow service layer — current and forecast AQI observations. |
tests/ |
Unit and integration tests mirroring src/. |
See CLAUDE.md for development guidelines and architectural rules. The short version:
- Handlers throw, framework catches — no
try/catchin tool logic - Use
ctx.logfor request-scoped logging,ctx.statefor tenant-scoped storage - Register new tools and resources via the barrels in
src/mcp-server/*/definitions/index.ts - Wrap external API calls: validate raw → normalize to domain type → return output schema; never fabricate missing fields
- ECHO searches must enforce at least one geographic parameter — unscoped queries time out against the live API
Issues and pull requests are welcome. Run checks and tests before submitting:
bun run devcheck
bun run testApache-2.0 — see LICENSE for details.