Batch upsert products by external ID
Create or update multiple products by external system ID. Supports up to 100 products per request with stale update protection.
Related endpoints
GET /products— List productsPOST /products— Create productGET /products/{id}— Get product by IDPATCH /products/{id}— Update productPUT /products/external/{source}/{externalId}— Upsert product by external ID
Common errors
400 invalid_request— malformed payload or failed validation.401 unauthenticated— missing, malformed, or revoked API key.403 permission_denied— key lacks the required scope, or the resource belongs to a different merchant.
Idempotency
Pass an Idempotency-Key header (UUID v4 recommended) to make retries safe. Keys are valid for 24 hours; see the idempotency guide.
x-api-key<token>
Your RevKeen API key (powered by Unkey). Get it from Dashboard > Settings > API Keys. Use rk_sandbox_* for test mode and rk_live_* for production.
In: header
Request Body
application/json
TypeScript Definitions
Use the request body type in TypeScript.
Response Body
application/json
curl -X PUT "https://api.revkeen.com/v2/products/external/batch" \ -H "x-api-key: $REVKEEN_API_KEY" \ -H "Content-Type: application/json" \ -d '{ "source": "practicehub", "products": [ { "external_id": "prod_12345", "name": "Monthly Membership", "description": "string", "kind": "subscription", "amount_cents": 9900, "currency": "USD", "is_active": true, "interval": "month", "interval_count": 1, "external_updated_at": "2026-01-01T00:00:00Z", "external_ref": "string", "metadata": {} } ] }'{
"created": 5,
"updated": 10,
"skipped": 2,
"failed": [
{
"external_id": "string",
"error": "string"
}
]
}Empty
Empty
Empty