Usage Events
A usage event is a single measurement: at this time, this customer used this agent to do this much of this thing. Logging usage events is how MarginFront learns what to bill for. Every time your agent does work, you log an event. At the end of the billing period, MarginFront rolls them up, applies the customer’s pricing plan, and generates an invoice. This is the most important endpoint in the whole API. Everything else (agents, signals, plans, subscriptions) is setup. This is the ongoing, every-day traffic.You don’t need to set anything up first. When you fire an event with a newcustomerExternalId,agentCode, orsignalName, MarginFront creates the customer, agent, or signal automatically. Your existing user IDs from your own database flow straight through. The dashboard updates the moment the event lands.
The endpoint
X-API-Key header (secret key required — mf_sk_*).
Batch: Even for a single event, the body wraps records in an array. You can send 1-100 records per request.
Fields per record
Each record uses one of two shapes:- Single-service shape: one event, one underlying service. The 90% case. Required fields: top-level
customerExternalId,agentCode,signalName,model,modelProvider(plus volume). - Multi-service shape: one event, multiple underlying services contributing to one business outcome (e.g., one report that called Claude AND queried Google Maps). Required fields:
customerExternalId,agentCode,signalName,services[]. Top-levelmodel/modelProvider/volume are omitted; each service entry carries its own.
services[], never both, never neither. Mixing shapes returns a 400 Bad Request with a clear English message.
Always required (both shapes)
| Field | Type | Description |
|---|---|---|
customerExternalId | string | Your customer’s ID in your system (not the MarginFront internal UUID). If MarginFront has not seen this ID before, the customer record is created automatically. |
agentCode | string | A stable identifier for the agent or product that did the work. If MarginFront has not seen this code before, the agent record is created automatically. |
signalName | string | The billing unit being tracked (e.g. messages, report-pages). Matches the signal’s shortName. If MarginFront has not seen this name before, the signal record is created automatically. |
Required for single-service shape (or use services[] instead)
| Field | Type | Description |
|---|---|---|
model | string | The model identifier from your provider. Pass whatever your provider SDK returned: response.model from OpenAI/Anthropic, the service SKU for non-LLM tools. Case-insensitive, whitespace trimmed. Examples: "gpt-4o", "claude-sonnet-4-6", "twilio-sms", "textract-standard". |
modelProvider | string | The provider name, lowercase. This tells MarginFront which pricing table to look in. Required because different providers can have models with the same name — without this field, MarginFront can’t tell if "gpt-4o" means OpenAI’s or a fine-tune on another platform. Examples: "openai", "anthropic", "google", "twilio", "aws". |
Required for multi-service shape (or use single-service fields instead)
| Field | Type | Description |
|---|---|---|
services | array (≥ 1 entry) | One entry per underlying service that contributed to this business outcome. Each entry has its own model, modelProvider, and volume fields (same shape as the top-level single-service fields). The parent event represents the customer-facing thing; each entry is one cost line under it. |
services[] accepts:
| Field | Type | Required | Description |
|---|---|---|---|
model | string | Yes | Same shape as the top-level model for this service. |
modelProvider | string | Yes | Same shape as the top-level modelProvider (lowercase). |
inputTokens | integer (≥ 0) | LLM only | Input tokens for this LLM service. Must be a whole integer. |
outputTokens | integer (≥ 0) | LLM only | Output tokens for this LLM service. Must be a whole integer. |
quantity | integer (≥ 0) | non-LLM only | Billing units for this non-LLM service. Must be a whole integer. For fractional units (e.g. 14.137 seconds of compute), round before sending. |
Optional fields
| Field | Type | Default | Description |
|---|---|---|---|
inputTokens | integer (≥ 0) | — | Single-service shape only. Number of input (prompt) tokens. Must be a whole integer. Required for LLM cost calculation. Ignored for non-LLM services. |
outputTokens | integer (≥ 0) | — | Single-service shape only. Number of output (completion) tokens. Must be a whole integer. Required for LLM cost calculation. Ignored for non-LLM services. |
quantity | integer (≥ 0) | 1 | Single-service: billing units for non-LLM services. Multi-service: signal-level count (e.g. 1 for one report). Per-service volume goes inside each services[] entry. Must be a whole integer; round fractional units before sending. |
usageDate | ISO 8601 string | now | When the usage actually happened. Use this for back-filling historical events. |
metadata | object | {} | Custom key-value pairs. Stored but not interpreted. |
LLM vs non-LLM events vs multi-service events
All three patterns use the same endpoint. The difference is which fields carry the “what happened” information. Single-service LLM event (OpenAI, Anthropic, Google, etc.) — cost is based on tokens:services[] becomes one cost line under it. The dashboard shows ONE event in the live feed; the “Cost by service” chart on the Cost tab shows where the rolled-up total split. The Gemini entry carries inputTokens, outputTokens, AND quantity: 1 because LLM services[] entries can track all three at once: tokens for cost, quantity for call-count analytics.
You can mix all three shapes in a single batch.
Example curl calls
Single LLM event (OpenAI):Response format (200 OK)
The endpoint always returns 200 OK — even if some records failed. You must check the response body to know what actually happened.
Success entry fields (single-service)
| Field | Description |
|---|---|
customerExternalId | Echoed from your request |
agentCode | Echoed from your request |
signalName | Echoed from your request |
model / modelProvider | Echoed from your request |
inputTokens / outputTokens | Echoed from your request |
quantity | Echoed (or 1 if you didn’t send it) |
totalCostUsd | Calculated cost in USD (string with 10 decimal places) |
eventId | UUID of the signal_events row — use this for lookups |
rawEventId | UUID of the raw_ingest_events audit row |
timestamp | When the event was processed |
Success entry fields (multi-service)
| Field | Description |
|---|---|
customerExternalId | Echoed |
agentCode | Echoed |
signalName | Echoed |
services | Array. One entry per service that contributed. Each entry has model, modelProvider, volume fields, usageCost, eventStatus. |
totalCostUsd | Rolled-up parent cost (sum of resolved services[].usageCost). String with 10 decimal places. |
eventId | UUID of the parent signal_events row |
rawEventId | UUID of the audit row |
timestamp | When the event was processed |
model / modelProvider / inputTokens / outputTokens fields are NOT present on multi-service success entries (they live inside each services[] entry instead).
Failed entry fields
| Field | Description |
|---|---|
record | The original record you sent (echoed back so you can identify it). For multi-service, this includes the services[] array you sent. |
code | Why it failed: NEEDS_COST_BACKFILL, MISSING_VOLUME_DATA, INTERNAL_ERROR, or VALIDATION_ERROR. For multi-service, this is the parent’s worst-case status (NEEDS_COST_BACKFILL > MISSING_VOLUME_DATA > PROCESSED). |
stored | true = event is saved in the system (don’t retry). false = event was NOT saved (safe to retry). |
eventId | UUID of the parent signal_events row (only present when stored: true) |
rawEventId | UUID of the raw audit row (present for most failures) |
error | Human-readable description of what happened. For multi-service with multiple failing services, messages are joined with ” | ”. |
servicesStatus | Multi-service only. Array of {model, modelProvider, eventStatus} per service so you can see which specific services need fixing. |
Error codes explained
| Code | Stored? | What happened | What to do |
|---|---|---|---|
NEEDS_COST_BACKFILL | Yes | The model+provider combination isn’t in the pricing table. The event is saved with cost: null. | Do NOT retry. Go to the MarginFront dashboard → Usage Events → “Needs attention” and map the model to a known one. Once mapped, cost is backfilled automatically and all future events with that model auto-resolve. |
INTERNAL_ERROR | No | Something broke on our side during processing. The event was not saved to signal_events. | Safe to retry. The raw audit row may exist (check rawEventId). |
VALIDATION_ERROR | No | Bad input — a required field is missing or has the wrong type. | Fix the request and resend. Check the error message for which field. |
What happens when the model isn’t recognized
MarginFront never drops events. If you send amodel + modelProvider combination that isn’t in the pricing table:
- The event is stored in
signal_eventswithusageCost: null(not zero — null preserves the ambiguity for backfill). - The response includes the event in
results.failed[]withcode: "NEEDS_COST_BACKFILL"andstored: true. - The “Needs attention” tile on the dashboard
/metrics-eventspage shows a count of these events. - Click through to see the events grouped by model+provider, with context (which agent, customer, signal sent them).
- Pick a known model from the dropdown, click “Map & backfill” — MarginFront creates a permanent mapping and retroactively calculates cost for every affected event.
- Future events with that model+provider auto-resolve — no manual step needed again.
stored: true. They’re already in the system. Retrying would create duplicates.
Mapping unknown models (API endpoints)
These endpoints power the dashboard drill-down page. You can also call them directly.List unknown-cost event groups
| Param | Type | Default | Description |
|---|---|---|---|
startDate | ISO 8601 | 30 days ago | Filter events after this date |
endDate | ISO 8601 | now | Filter events before this date |
Map an unknown model to a known one
Listing events
Query your recorded usage events. Supports filtering and pagination. Use this when you need to see individual events — per-event drill-downs, audit trails, debugging a specific customer’s bill, or feeding a BI tool.x-api-key header. Either a secret key (mf_sk_*) or a publishable key (mf_pk_*) works — this is a read-only endpoint.
Query parameters
All optional. Without any, you get the most recent 20 events for your org.| Param | Type | Default | Description |
|---|---|---|---|
page | integer (≥ 1) | 1 | Page number. |
limit | integer (1-100) | 20 | Results per page. Capped at 100. |
customerId | UUID | — | Filter to events for one customer. Use MarginFront’s internal customer UUID (not your externalId). |
agentId | UUID | — | Filter to events for one agent. |
signalId | UUID | — | Filter to events for one signal. |
startDate | ISO 8601 string | — | Only events on or after this timestamp. |
endDate | ISO 8601 string | — | Only events on or before this timestamp. |
Example
Response (200 OK)
Understanding the event payload
Most fields are self-explanatory. A few are easy to trip over:-
usageCostis a string, not a number (e.g."0.00225"). We use strings to preserve decimal precision — prices can have many significant digits and JSON numbers would round. Convert withparseFloat()orNumber()before doing math. -
usageCostDatais the itemized cost breakdown — this is where per-model and per-dimension details live. Each key is"<model>/<dimension>"(e.g."gpt-4o/input","gpt-4o/output"). Each value has:cost— dollar amount for this line item (number)units— tokens for LLMs, or whatever quantity dimension was billed (number)costPerUnit— the rate applied (number)
usageCostequals the sum of everycostinsideusageCostData. If you need to ask “how many input tokens did this event use?”, readusageCostData["<model>/input"].units. If the event used multiple models or mixed LLM+non-LLM services, there will be multiple keys. -
quantityis a string (same precision reason asusageCost). -
signalis the nested signal object (id, name, shortName). Handy for display without a second lookup. -
subscriptionIdisnullwhen the customer had no active subscription at the time of the event.
Event processing states
TheeventProcessed field tells you where an event is in its lifecycle:
| Value | Meaning |
|---|---|
"PROCESSED" | Cost calculated and stored. Ready to be included in invoices. |
"NEEDS_COST_BACKFILL" | Event stored but cost is null — the model+provider wasn’t found in the pricing table. Use GET /v1/events/needs-cost-backfill to find these and POST /v1/events/map-model to resolve. |
"PENDING" | Still being processed. Should transition within seconds. |
"ERROR" | Cost calculation failed for a reason other than an unknown model. Rare — inspect the event in the dashboard. |
eventProcessed is "NEEDS_COST_BACKFILL", usageCost is null and usageCostData is empty. See Mapping unknown models above for the resolution flow.
Auto-provisioning
If you log a usage event for acustomerExternalId or agentCode that MarginFront has never seen, it will auto-create a minimal customer and/or agent on the spot. Convenient for prototyping — but means you won’t get an error for typos. Double-check in the dashboard if things “work” but show up with a name you don’t recognize.
Common HTTP errors
| Status | Cause |
|---|---|
400 Bad Request | The batch is empty, has more than 100 records, or a record is missing a required field (customerExternalId, agentCode, signalName, model, or modelProvider). The response body tells you which field. |
401 Unauthorized | API key is missing or invalid. |
403 Forbidden | You used a publishable key (mf_pk_*). Usage recording requires a secret key (mf_sk_*). |
Using the Node SDK
The@marginfront/sdk package wraps this endpoint. See the SDK README for full documentation. Quick example:
fireAndForget: true setting, usage.record() never throws — network errors retry automatically via a local buffer. See the SDK docs for details.

