# EPA TRI Feed: Facility Toxic Releases by Chemical (`thoob/tri-toxic-release-feed`) Actor

Clean, flat EPA Toxics Release Inventory records from the official Envirofacts API: facility, chemical, CAS, release amounts by medium, year, and location, with a what-changed-since-last-run mode. Billed only per delivered record, no start fee.

- **URL**: https://apify.com/thoob/tri-toxic-release-feed.md
- **Developed by:** [Pono Data](https://apify.com/thoob) (community)
- **Categories:** Developer tools
- **Stats:** 2 total users, 1 monthly users, 100.0% runs succeeded, 0 bookmarks
- **User rating**: No ratings yet

## Pricing

$0.30 / 1,000 release records

This Actor is paid per event. You are not charged for the Apify platform usage, but only a fixed price for specific events.

Learn more: https://docs.apify.com/platform/actors/running/actors-in-store#pay-per-event

## What's an Apify Actor?

Actors are a software tools running on the Apify platform, for all kinds of web data extraction and automation use cases.
In Batch mode, an Actor accepts a well-defined JSON input, performs an action which can take anything from a few seconds to a few hours,
and optionally produces a well-defined JSON output, datasets with results, or files in key-value store.
In Standby mode, an Actor provides a web server which can be used as a website, API, or an MCP server.
Actors are written with capital "A".

## How to integrate an Actor?

If asked about integration, you help developers integrate Actors into their projects.
You adapt to their stack and deliver integrations that are safe, well-documented, and production-ready.
The best way to integrate Actors is as follows.

In JavaScript/TypeScript projects, use official [JavaScript/TypeScript client](https://docs.apify.com/api/client/js.md):

```bash
npm install apify-client
```

In Python projects, use official [Python client library](https://docs.apify.com/api/client/python.md):

```bash
pip install apify-client
```

In shell scripts, use [Apify CLI](https://docs.apify.com/cli/docs.md):

````bash
# MacOS / Linux
curl -fsSL https://apify.com/install-cli.sh | bash
# Windows
irm https://apify.com/install-cli.ps1 | iex
```bash

In AI frameworks, you might use the [Apify MCP server](https://docs.apify.com/platform/integrations/mcp.md).

If your project is in a different language, use the [REST API](https://docs.apify.com/api/v2.md).

For usage examples, see the [API](#api) section below.

For more details, see Apify documentation as [Markdown index](https://docs.apify.com/llms.txt) and [Markdown full-text](https://docs.apify.com/llms-full.txt).


# README

## EPA TRI Toxic Release Feed

Clean, flat toxic-release records from the official EPA Toxics Release Inventory, pulled from the Envirofacts REST service. Filter by US state and reporting year, get one useful row per facility-chemical-year report, and run a changes-only mode that returns just what is new or revised since your last run.

You pay only for a delivered record that carries actual release amounts. Filtered-out rows, unchanged rows in changes mode, malformed records, and forms that reported no quantified release for any medium all cost nothing, and there is no per-run start fee. The release amounts are the product, so a record without them is never billed.

### What a record is

One flat row per TRI reporting form (one facility's report of one chemical for one year), assembled from four Envirofacts tables so you do not have to join them yourself:

- facility: `facilityName`, `streetAddress`, `city`, `county`, `state`, `zipCode`, `fipsCode`, `epaRegion`, `latitude`, `longitude`, `parentCompany`, `isFederalFacility`
- chemical: `chemicalName`, `casRegistryNumber`, `unitOfMeasure`, and the EPA hazard flags `isCarcinogen`, `isPfas`, `isPbt`, `isMetal`
- releases: `totalRelease` and `releasesByMedium` (one entry per environmental medium that reported a quantity, such as fugitive air, stack air, surface water, landfill, with its amount)
- identity and year: `docCtrlNum`, `triFacilityId`, `reportingYear`
- provenance: `sourceUrl`, `retrievedAt`, `confidence`, `dataSource`, `jurisdiction`, and in changes mode `changeType`

Every value is copied from an Envirofacts record. The one derivation is decoding EPA's packed degrees-minutes-seconds coordinates into decimal degrees, and that is only done when the packed value is structurally valid. Nothing is guessed.

### Sample output

A real slice, Colorado facilities reporting for 2022 (abbreviated):

| facilityName | state | chemicalName | casRegistryNumber | reportingYear | sourceUrl |
|---|---|---|---|---|---|
| DHARMACON INC | CO | Acetonitrile | 75-05-8 | 2022 | data.epa.gov/efservice/tri_reporting_form/doc_ctrl_num/1322220634909/JSON |

Each `sourceUrl` is the form's own record in the Envirofacts REST service, so any value can be checked against the source.

### Two modes

- **full**: every reporting form matching your filter.
- **changes**: only forms that are new, or whose release total or medium breakdown changed since this account's last run. The state is tracked per account, so it is your timeline. The first changes run has no baseline, so it returns everything matching once, then tracks deltas from there.

### Filters

- **state**: a two-letter US state or territory code, applied to the facility (CO, TX, CA, PR, and the rest). Leave empty for all states.
- **reportingYear**: a TRI reporting year, applied to the form. Leave empty for all years, which returns far more records.
- **maxRecords**: a cap on how many records a run delivers and bills. 0 means no cap; the platform spend cap is honored regardless.

State and year are applied server-side by the Envirofacts service.

### What to expect

- TRI covers more than 20,000 industrial and federal facilities reporting on a list of over 600 toxic chemicals, going back to 1987. A state-and-year filter narrows that to the slice you care about (Colorado 2022, for example, is a few hundred forms).
- The raw Envirofacts API is public and free directly from the EPA. What this actor sells is the assembled record: the four-table join, the release-by-medium aggregation, the decoded coordinates, the CAS lookup, and the per-account change feed, so you do not have to walk the normalized tables or diff them yourself.
- A state-and-year pull is a few dollars; a periodic changes feed is pennies a run.

### Source and scope

- Source: the official EPA Envirofacts REST service (`data.epa.gov/efservice`), public domain, no key, no anti-bot, paged with a declared identifying User-Agent.
- Scope: United States facilities and subjects only. Every row is stamped `jurisdiction=US`.
- These are public reports filed by regulated facilities about their own operations, at the business level. There is no consumer personal data here.
- This is a data tool, not legal, environmental, investment, or regulatory advice. A TRI report is a self-reported filing, not a finding.

### See also

More clean, pay-only-for-results data tools from Pono Data:

- [Clinical Trials Feed](https://apify.com/thoob/clinical-trials-feed) - studies, sponsors, and changes from ClinicalTrials.gov
- [Federal Spending Feed](https://apify.com/thoob/federal-spending-feed) - US federal award records with a changes mode

Full catalog: https://apify.com/thoob

# Actor input Schema

## `state` (type: `string`):

Two-letter US state or territory code to filter facilities by (for example CO, TX, CA, PR). Applied server-side on the facility. Leave empty to match facilities in any state.
## `reportingYear` (type: `string`):

TRI reporting year to filter by (for example 2022). Applied server-side on the reporting form. Leave empty to match every reporting year, which returns far more records.
## `mode` (type: `string`):

full returns every reporting form matching your filter. changes returns only forms that are new, or whose release total or medium breakdown changed since this account's last run, tracked per account. The first changes run has no baseline, so it returns everything matching once, then tracks deltas.
## `maxRecords` (type: `integer`):

Safety cap on how many reporting forms this run delivers and bills. 0 means no cap. In changes mode, forms beyond the cap are not dropped: they are delivered on a later run, never billed twice. The platform spend cap (ACTOR_MAX_TOTAL_CHARGE_USD) is honored regardless.
## `pageSize` (type: `integer`):

How many reporting forms to request per Envirofacts page (1 to 1000). Larger pages are fewer requests; the default is a polite middle.

## Actor input object example

```json
{
  "state": "CO",
  "reportingYear": "2022",
  "mode": "full",
  "maxRecords": 0,
  "pageSize": 200
}
````

# Actor output Schema

## `records` (type: `string`):

One row per facility-chemical-year reporting form matching your filter.

# API

You can run this Actor programmatically using our API. Below are code examples in JavaScript, Python, and CLI, as well as the OpenAPI specification and MCP server setup.

## JavaScript example

```javascript
import { ApifyClient } from 'apify-client';

// Initialize the ApifyClient with your Apify API token
// Replace the '<YOUR_API_TOKEN>' with your token
const client = new ApifyClient({
    token: '<YOUR_API_TOKEN>',
});

// Prepare Actor input
const input = {
    "state": "CO",
    "reportingYear": "2022"
};

// Run the Actor and wait for it to finish
const run = await client.actor("thoob/tri-toxic-release-feed").call(input);

// Fetch and print Actor results from the run's dataset (if any)
console.log('Results from dataset');
console.log(`💾 Check your data here: https://console.apify.com/storage/datasets/${run.defaultDatasetId}`);
const { items } = await client.dataset(run.defaultDatasetId).listItems();
items.forEach((item) => {
    console.dir(item);
});

// 📚 Want to learn more 📖? Go to → https://docs.apify.com/api/client/js/docs

```

## Python example

```python
from apify_client import ApifyClient

# Initialize the ApifyClient with your Apify API token
# Replace '<YOUR_API_TOKEN>' with your token.
client = ApifyClient("<YOUR_API_TOKEN>")

# Prepare the Actor input
run_input = {
    "state": "CO",
    "reportingYear": "2022",
}

# Run the Actor and wait for it to finish
run = client.actor("thoob/tri-toxic-release-feed").call(run_input=run_input)

# Fetch and print Actor results from the run's dataset (if there are any)
print("💾 Check your data here: https://console.apify.com/storage/datasets/" + run["defaultDatasetId"])
for item in client.dataset(run["defaultDatasetId"]).iterate_items():
    print(item)

# 📚 Want to learn more 📖? Go to → https://docs.apify.com/api/client/python/docs/quick-start

```

## CLI example

```bash
echo '{
  "state": "CO",
  "reportingYear": "2022"
}' |
apify call thoob/tri-toxic-release-feed --silent --output-dataset

```

## MCP server setup

```json
{
    "mcpServers": {
        "apify": {
            "command": "npx",
            "args": [
                "mcp-remote",
                "https://mcp.apify.com/?tools=thoob/tri-toxic-release-feed",
                "--header",
                "Authorization: Bearer <YOUR_API_TOKEN>"
            ]
        }
    }
}

```

## OpenAPI specification

```json
{
    "openapi": "3.0.1",
    "info": {
        "title": "EPA TRI Feed: Facility Toxic Releases by Chemical",
        "description": "Clean, flat EPA Toxics Release Inventory records from the official Envirofacts API: facility, chemical, CAS, release amounts by medium, year, and location, with a what-changed-since-last-run mode. Billed only per delivered record, no start fee.",
        "version": "0.0",
        "x-build-id": "aZ0hBLLDUEjHuCgsZ"
    },
    "servers": [
        {
            "url": "https://api.apify.com/v2"
        }
    ],
    "paths": {
        "/acts/thoob~tri-toxic-release-feed/run-sync-get-dataset-items": {
            "post": {
                "operationId": "run-sync-get-dataset-items-thoob-tri-toxic-release-feed",
                "x-openai-isConsequential": false,
                "summary": "Executes an Actor, waits for its completion, and returns Actor's dataset items in response.",
                "tags": [
                    "Run Actor"
                ],
                "requestBody": {
                    "required": true,
                    "content": {
                        "application/json": {
                            "schema": {
                                "$ref": "#/components/schemas/inputSchema"
                            }
                        }
                    }
                },
                "parameters": [
                    {
                        "name": "token",
                        "in": "query",
                        "required": true,
                        "schema": {
                            "type": "string"
                        },
                        "description": "Enter your Apify token here"
                    }
                ],
                "responses": {
                    "200": {
                        "description": "OK"
                    }
                }
            }
        },
        "/acts/thoob~tri-toxic-release-feed/runs": {
            "post": {
                "operationId": "runs-sync-thoob-tri-toxic-release-feed",
                "x-openai-isConsequential": false,
                "summary": "Executes an Actor and returns information about the initiated run in response.",
                "tags": [
                    "Run Actor"
                ],
                "requestBody": {
                    "required": true,
                    "content": {
                        "application/json": {
                            "schema": {
                                "$ref": "#/components/schemas/inputSchema"
                            }
                        }
                    }
                },
                "parameters": [
                    {
                        "name": "token",
                        "in": "query",
                        "required": true,
                        "schema": {
                            "type": "string"
                        },
                        "description": "Enter your Apify token here"
                    }
                ],
                "responses": {
                    "200": {
                        "description": "OK",
                        "content": {
                            "application/json": {
                                "schema": {
                                    "$ref": "#/components/schemas/runsResponseSchema"
                                }
                            }
                        }
                    }
                }
            }
        },
        "/acts/thoob~tri-toxic-release-feed/run-sync": {
            "post": {
                "operationId": "run-sync-thoob-tri-toxic-release-feed",
                "x-openai-isConsequential": false,
                "summary": "Executes an Actor, waits for completion, and returns the OUTPUT from Key-value store in response.",
                "tags": [
                    "Run Actor"
                ],
                "requestBody": {
                    "required": true,
                    "content": {
                        "application/json": {
                            "schema": {
                                "$ref": "#/components/schemas/inputSchema"
                            }
                        }
                    }
                },
                "parameters": [
                    {
                        "name": "token",
                        "in": "query",
                        "required": true,
                        "schema": {
                            "type": "string"
                        },
                        "description": "Enter your Apify token here"
                    }
                ],
                "responses": {
                    "200": {
                        "description": "OK"
                    }
                }
            }
        }
    },
    "components": {
        "schemas": {
            "inputSchema": {
                "type": "object",
                "properties": {
                    "state": {
                        "title": "State",
                        "type": "string",
                        "description": "Two-letter US state or territory code to filter facilities by (for example CO, TX, CA, PR). Applied server-side on the facility. Leave empty to match facilities in any state."
                    },
                    "reportingYear": {
                        "title": "Reporting year",
                        "type": "string",
                        "description": "TRI reporting year to filter by (for example 2022). Applied server-side on the reporting form. Leave empty to match every reporting year, which returns far more records."
                    },
                    "mode": {
                        "title": "Mode",
                        "enum": [
                            "full",
                            "changes"
                        ],
                        "type": "string",
                        "description": "full returns every reporting form matching your filter. changes returns only forms that are new, or whose release total or medium breakdown changed since this account's last run, tracked per account. The first changes run has no baseline, so it returns everything matching once, then tracks deltas.",
                        "default": "full"
                    },
                    "maxRecords": {
                        "title": "Max delivered records",
                        "minimum": 0,
                        "maximum": 200000,
                        "type": "integer",
                        "description": "Safety cap on how many reporting forms this run delivers and bills. 0 means no cap. In changes mode, forms beyond the cap are not dropped: they are delivered on a later run, never billed twice. The platform spend cap (ACTOR_MAX_TOTAL_CHARGE_USD) is honored regardless.",
                        "default": 0
                    },
                    "pageSize": {
                        "title": "API page size",
                        "minimum": 1,
                        "maximum": 1000,
                        "type": "integer",
                        "description": "How many reporting forms to request per Envirofacts page (1 to 1000). Larger pages are fewer requests; the default is a polite middle.",
                        "default": 200
                    }
                }
            },
            "runsResponseSchema": {
                "type": "object",
                "properties": {
                    "data": {
                        "type": "object",
                        "properties": {
                            "id": {
                                "type": "string"
                            },
                            "actId": {
                                "type": "string"
                            },
                            "userId": {
                                "type": "string"
                            },
                            "startedAt": {
                                "type": "string",
                                "format": "date-time",
                                "example": "2025-01-08T00:00:00.000Z"
                            },
                            "finishedAt": {
                                "type": "string",
                                "format": "date-time",
                                "example": "2025-01-08T00:00:00.000Z"
                            },
                            "status": {
                                "type": "string",
                                "example": "READY"
                            },
                            "meta": {
                                "type": "object",
                                "properties": {
                                    "origin": {
                                        "type": "string",
                                        "example": "API"
                                    },
                                    "userAgent": {
                                        "type": "string"
                                    }
                                }
                            },
                            "stats": {
                                "type": "object",
                                "properties": {
                                    "inputBodyLen": {
                                        "type": "integer",
                                        "example": 2000
                                    },
                                    "rebootCount": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "restartCount": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "resurrectCount": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "computeUnits": {
                                        "type": "integer",
                                        "example": 0
                                    }
                                }
                            },
                            "options": {
                                "type": "object",
                                "properties": {
                                    "build": {
                                        "type": "string",
                                        "example": "latest"
                                    },
                                    "timeoutSecs": {
                                        "type": "integer",
                                        "example": 300
                                    },
                                    "memoryMbytes": {
                                        "type": "integer",
                                        "example": 1024
                                    },
                                    "diskMbytes": {
                                        "type": "integer",
                                        "example": 2048
                                    }
                                }
                            },
                            "buildId": {
                                "type": "string"
                            },
                            "defaultKeyValueStoreId": {
                                "type": "string"
                            },
                            "defaultDatasetId": {
                                "type": "string"
                            },
                            "defaultRequestQueueId": {
                                "type": "string"
                            },
                            "buildNumber": {
                                "type": "string",
                                "example": "1.0.0"
                            },
                            "containerUrl": {
                                "type": "string"
                            },
                            "usage": {
                                "type": "object",
                                "properties": {
                                    "ACTOR_COMPUTE_UNITS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATASET_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATASET_WRITES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "KEY_VALUE_STORE_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "KEY_VALUE_STORE_WRITES": {
                                        "type": "integer",
                                        "example": 1
                                    },
                                    "KEY_VALUE_STORE_LISTS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "REQUEST_QUEUE_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "REQUEST_QUEUE_WRITES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATA_TRANSFER_INTERNAL_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATA_TRANSFER_EXTERNAL_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "PROXY_RESIDENTIAL_TRANSFER_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "PROXY_SERPS": {
                                        "type": "integer",
                                        "example": 0
                                    }
                                }
                            },
                            "usageTotalUsd": {
                                "type": "number",
                                "example": 0.00005
                            },
                            "usageUsd": {
                                "type": "object",
                                "properties": {
                                    "ACTOR_COMPUTE_UNITS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATASET_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATASET_WRITES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "KEY_VALUE_STORE_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "KEY_VALUE_STORE_WRITES": {
                                        "type": "number",
                                        "example": 0.00005
                                    },
                                    "KEY_VALUE_STORE_LISTS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "REQUEST_QUEUE_READS": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "REQUEST_QUEUE_WRITES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATA_TRANSFER_INTERNAL_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "DATA_TRANSFER_EXTERNAL_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "PROXY_RESIDENTIAL_TRANSFER_GBYTES": {
                                        "type": "integer",
                                        "example": 0
                                    },
                                    "PROXY_SERPS": {
                                        "type": "integer",
                                        "example": 0
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }
}
```
