Skip to Content
SourcesTracking SDKAPI Reference

API Reference

The ingestion API receives events from the SDK and writes them to BigQuery. It runs as a Node.js Express service, typically deployed on Cloud Run.

Base URL

The API runs on the host you configure (e.g., https://track.yourdomain.com). All endpoints are relative to this base.

Endpoints

MethodPathDescription
POST/collectReceive and process events
GET/healthHealth check
GET/v1/sdk.jsServe the JavaScript SDK
GET/v1/snippetGenerate a pre-filled HTML snippet

POST /collect

The primary endpoint. Accepts a single event or a batch of events.

Authentication

Include the write key in one of these locations (checked in order):

MethodExample
X-Write-Key headerX-Write-Key: your-write-key
Authorization headerAuthorization: Bearer your-write-key
Body field{ "writeKey": "your-write-key", ... }

The SDK sends the write key in the request body automatically.

Request: Single Event

{ "writeKey": "your-write-key", "type": "track", "messageId": "550e8400-e29b-41d4-a716-446655440000", "timestamp": "2026-02-08T12:00:00.000Z", "anonymousId": "anon-abc-123", "event": "Product Viewed", "properties": { "sku": "SKU-1", "price": 29.99 } }

Request: Batch

{ "writeKey": "your-write-key", "sentAt": "2026-02-08T12:00:05.000Z", "batch": [ { "type": "page", "messageId": "msg-1", "timestamp": "2026-02-08T12:00:00.000Z", "anonymousId": "anon-abc-123", "properties": { "page_url": "https://example.com/pricing" } }, { "type": "track", "messageId": "msg-2", "timestamp": "2026-02-08T12:00:03.000Z", "anonymousId": "anon-abc-123", "event": "Button Clicked", "properties": { "label": "signup" } } ] }

Maximum batch size: 100 events.

Response: Success

{ "ok": true, "received": 2, "inserted": { "events": { "bigquery": { "inserted": 2, "errors": 0 } } }, "invalid": 0, "errors": [] }

Response: Partial Success

If some events are invalid, valid events are still processed:

{ "ok": true, "received": 1, "inserted": { ... }, "invalid": 1, "errors": [ { "index": 1, "errors": [{ "message": "must have required property 'type'" }] } ] }

Event Schema

Required Fields (All Events)

FieldTypeDescription
typestringEvent type: track, identify, page, group, or alias
messageIdstringUnique event ID (UUID recommended)
timestampstringISO 8601 timestamp
anonymousIdstringAnonymous visitor ID

Conditional Fields

Event TypeRequired Fields
trackevent (event name)
identifyuserId
groupgroupId
aliasuserId, previousId
pageNone additional

Optional Fields

FieldTypeDescription
userIdstringKnown user ID (set after identify)
sessionIdstringSession identifier
writeKeystringWrite key (alternative to header auth)
propertiesobjectEvent properties (for track and page)
traitsobjectUser/group traits (for identify and group)
sentAtstringClient-side send timestamp (for clock correction)

Table Routing

Event TypeBigQuery Table
trackevents
pageevents
identifyusers
groupgroups
aliasaliases

Server-Added Fields

The ingestion API enriches each event with:

FieldDescription
received_atServer timestamp when the event was received
ipClient IP (from X-Forwarded-For or request IP)
user_agentClient user agent string

Error Responses

StatusError CodeDescription
400invalid_payloadRequest body is not valid JSON
400invalid_batchBatch array failed schema validation
400no_valid_eventsNo events in the batch passed validation
401missing_write_keyNo write key found in header or body
401invalid_write_keyWrite key does not match
413batch_too_largeBatch exceeds 100 events
500server_errorInternal server error

All error responses follow this format:

{ "ok": false, "error": "error_code", "details": [] }

GET /health

Returns { "ok": true }. Use for load balancer health checks.

GET /v1/sdk.js

Serves the compiled JavaScript SDK as application/javascript with a 1-hour cache header. The SDK file is loaded into memory at API startup.

Note: The SDK is also available via Vendo’s CDN at https://cdn.vendodata.com/sdk/v1/vendo.js. The CDN-hosted version is identical to the one served by the ingestion API. Use the CDN for the fastest setup, or serve from your own domain for maximum first-party control.

Set the SDK_JS_PATH environment variable to override the default SDK file location.

GET /v1/snippet

Returns a ready-to-paste HTML <script> block with the write key and host pre-filled.

Query ParamDefaultDescription
writeKeyYOUR_WRITE_KEYWrite key to embed in the snippet
hostAuto-detected from requestTracking host URL

Example: GET /v1/snippet?writeKey=my-key


Authentication Modes

ModeConfigUse Case
StaticWRITE_KEY env varSingle-tenant, local dev, demos
FirebaseFIREBASE_AUTH_ENABLED=trueMulti-tenant (Shopify integration)

In static mode, the write key in the request must match the WRITE_KEY environment variable.

In Firebase mode, the write key is looked up in the tracking_write_keys Firestore collection. Each document contains the merchant’s destination configuration (Mixpanel, Segment, Customer.io, OneSignal, BigQuery).


Rate Limits

The API does not enforce rate limits by default. For production deployments, configure rate limiting at the load balancer or CDN level. Recommended limits:

ScopeLimit
Per IP100 requests/second
Per write key1000 events/second
Request body2 MB max

Last updated on