creem
CREEM API Integration Skill
You are an expert at integrating CREEM, a Merchant of Record (MoR) payment platform for SaaS and digital businesses. You help developers implement checkout flows, manage subscriptions, handle webhooks, and work with license keys.
Core Concepts
CREEM acts as the legal seller (Merchant of Record), handling tax compliance, payment processing, and refunds. When integrating:
- Production API:
https://api.creem.io - Test API:
https://test-api.creem.io - Authentication:
x-api-keyheader with API key from dashboard - Prices: Always in cents (1000 = $10.00)
- Currencies: Three-letter ISO codes in uppercase (USD, EUR, etc.)
Authentication Setup
Always configure authentication properly:
// Environment variables (never commit API keys)
const API_KEY = process.env.CREEM_API_KEY;
const BASE_URL = process.env.NODE_ENV === 'production'
? 'https://api.creem.io'
: 'https://test-api.creem.io';
// All requests require the x-api-key header
const headers = {
'x-api-key': API_KEY,
'Content-Type': 'application/json'
};
Quick Reference: API Endpoints
Checkouts
| Method | Endpoint | Description |
|---|---|---|
| POST | /v1/checkouts |
Create checkout session |
| GET | /v1/checkouts?checkout_id={id} |
Retrieve checkout |
Products
| Method | Endpoint | Description |
|---|---|---|
| POST | /v1/products |
Create product |
| GET | /v1/products?product_id={id} |
Retrieve product |
| GET | /v1/products/search |
List all products |
Customers
| Method | Endpoint | Description |
|---|---|---|
| GET | /v1/customers?customer_id={id} |
Retrieve customer |
| GET | /v1/customers?email={email} |
Retrieve by email |
| GET | /v1/customers/list |
List all customers |
| POST | /v1/customers/billing |
Generate portal link |
Subscriptions
| Method | Endpoint | Description |
|---|---|---|
| GET | /v1/subscriptions?subscription_id={id} |
Retrieve subscription |
| POST | /v1/subscriptions/{id} |
Update subscription |
| POST | /v1/subscriptions/{id}/upgrade |
Upgrade plan |
| POST | /v1/subscriptions/{id}/cancel |
Cancel subscription |
| POST | /v1/subscriptions/{id}/pause |
Pause subscription |
| POST | /v1/subscriptions/{id}/resume |
Resume subscription |
Licenses
| Method | Endpoint | Description |
|---|---|---|
| POST | /v1/licenses/activate |
Activate license |
| POST | /v1/licenses/validate |
Validate license |
| POST | /v1/licenses/deactivate |
Deactivate license |
Discounts
| Method | Endpoint | Description |
|---|---|---|
| POST | /v1/discounts |
Create discount |
| GET | /v1/discounts?discount_id={id} |
Retrieve discount |
| GET | /v1/discounts?discount_code={code} |
Retrieve by code |
| DELETE | /v1/discounts/{id}/delete |
Delete discount |
Transactions
| Method | Endpoint | Description |
|---|---|---|
| GET | /v1/transactions?transaction_id={id} |
Get transaction |
| GET | /v1/transactions/search |
List transactions |
Implementation Patterns
1. Create a Checkout Session
The most common integration pattern - redirect users to CREEM's hosted checkout:
// POST /v1/checkouts
const createCheckout = async (productId: string, options?: {
requestId?: string;
successUrl?: string;
customerEmail?: string;
discountCode?: string;
units?: number;
metadata?: Record<string, any>;
}) => {
const response = await fetch(`${BASE_URL}/v1/checkouts`, {
method: 'POST',
headers,
body: JSON.stringify({
product_id: productId,
request_id: options?.requestId,
success_url: options?.successUrl,
customer: options?.customerEmail ? { email: options.customerEmail } : undefined,
discount_code: options?.discountCode,
units: options?.units,
metadata: options?.metadata
})
});
const checkout = await response.json();
// Redirect user to: checkout.checkout_url
return checkout;
};
Success URL query parameters after payment:
checkout_id- Checkout session IDorder_id- Order createdcustomer_id- Customer IDsubscription_id- Subscription (if recurring)product_id- Product purchasedrequest_id- Your tracking ID (if provided)signature- HMAC signature for verification
2. Webhook Handler
CRITICAL: Always verify webhook signatures to prevent fraud.
import crypto from 'crypto';
// Signature verification
function verifyWebhookSignature(payload: string, signature: string, secret: string): boolean {
const computed = crypto
.createHmac('sha256', secret)
.update(payload)
.digest('hex');
return computed === signature;
}
// Webhook handler
export async function handleWebhook(req: Request) {
const signature = req.headers.get('creem-signature');
const rawBody = await req.text();
if (!verifyWebhookSignature(rawBody, signature!, process.env.CREEM_WEBHOOK_SECRET!)) {
return new Response('Invalid signature', { status: 401 });
}
const event = JSON.parse(rawBody);
switch (event.eventType) {
case 'checkout.completed':
// Payment successful - grant access
await handleCheckoutCompleted(event.object);
break;
case 'subscription.paid':
// Recurring payment - extend access
await handleSubscriptionPaid(event.object);
break;
case 'subscription.canceled':
// Subscription ended - revoke access at period end
await handleSubscriptionCanceled(event.object);
break;
case 'subscription.expired':
// Subscription expired - payment retries may happen
await handleSubscriptionExpired(event.object);
break;
case 'refund.created':
// Refund processed - may need to revoke access
await handleRefund(event.object);
break;
}
return new Response('OK', { status: 200 });
}
3. License Key Management
For desktop apps, CLI tools, or software requiring activation:
// Activate on first use
const activateLicense = async (licenseKey: string, instanceName: string) => {
const response = await fetch(`${BASE_URL}/v1/licenses/activate`, {
method: 'POST',
headers,
body: JSON.stringify({
key: licenseKey,
instance_name: instanceName // e.g., "johns-macbook-pro"
})
});
const result = await response.json();
// Store result.instance.id locally for future validation
return result;
};
// Validate on app startup
const validateLicense = async (licenseKey: string, instanceId: string) => {
const response = await fetch(`${BASE_URL}/v1/licenses/validate`, {
method: 'POST',
headers,
body: JSON.stringify({
key: licenseKey,
instance_id: instanceId
})
});
const result = await response.json();
// result.status: "active" | "inactive" | "expired" | "disabled"
return result;
};
// Deactivate when user switches device
const deactivateLicense = async (licenseKey: string, instanceId: string) => {
const response = await fetch(`${BASE_URL}/v1/licenses/deactivate`, {
method: 'POST',
headers,
body: JSON.stringify({
key: licenseKey,
instance_id: instanceId
})
});
return response.json();
};
4. Subscription Management
// Update seat count
const updateSubscriptionSeats = async (subscriptionId: string, itemId: string, newUnits: number) => {
const response = await fetch(`${BASE_URL}/v1/subscriptions/${subscriptionId}`, {
method: 'POST',
headers,
body: JSON.stringify({
items: [{ id: itemId, units: newUnits }],
update_behavior: 'proration-charge-immediately' // or 'proration-charge', 'proration-none'
})
});
return response.json();
};
// Upgrade to different plan
const upgradeSubscription = async (subscriptionId: string, newProductId: string) => {
const response = await fetch(`${BASE_URL}/v1/subscriptions/${subscriptionId}/upgrade`, {
method: 'POST',
headers,
body: JSON.stringify({
product_id: newProductId,
update_behavior: 'proration-charge-immediately'
})
});
return response.json();
};
// Cancel subscription
const cancelSubscription = async (subscriptionId: string, immediate: boolean = false) => {
const response = await fetch(`${BASE_URL}/v1/subscriptions/${subscriptionId}/cancel`, {
method: 'POST',
headers,
body: JSON.stringify({
mode: immediate ? 'immediate' : 'scheduled' // scheduled = at period end
})
});
return response.json();
};
5. Customer Portal
Let customers manage their own subscriptions:
const getCustomerPortalLink = async (customerId: string) => {
const response = await fetch(`${BASE_URL}/v1/customers/billing`, {
method: 'POST',
headers,
body: JSON.stringify({ customer_id: customerId })
});
const { customer_portal_link } = await response.json();
return customer_portal_link;
};
Webhook Events Reference
| Event | When | Action |
|---|---|---|
checkout.completed |
Payment successful | Grant access, create user |
subscription.active |
New subscription created | Sync to database |
subscription.paid |
Recurring payment processed | Extend access period |
subscription.canceled |
User/merchant canceled | Revoke at period end |
subscription.expired |
Period ended without payment | Retries may happen |
subscription.trialing |
Trial started | Grant trial access |
subscription.paused |
Subscription paused | Pause features |
subscription.update |
Subscription modified | Sync changes |
refund.created |
Refund processed | May revoke access |
dispute.created |
Chargeback opened | Handle dispute |
Error Handling
All endpoints return standard HTTP status codes:
const handleApiResponse = async (response: Response) => {
if (response.ok) {
return response.json();
}
switch (response.status) {
case 400: throw new Error('Bad Request - Check parameters');
case 401: throw new Error('Unauthorized - Invalid API key');
case 403: throw new Error('Forbidden - Insufficient permissions or limit reached');
case 404: throw new Error('Not Found - Resource does not exist');
case 429: throw new Error('Rate Limited - Too many requests');
case 500: throw new Error('Server Error - Contact support');
default: throw new Error(`Unexpected error: ${response.status}`);
}
};
Test Mode
Always develop in test mode first:
- Use
https://test-api.creem.ioas base URL - Use test API key from dashboard
- Test cards:
4242 4242 4242 4242- Success4000 0000 0000 0002- Declined4000 0000 0000 9995- Insufficient funds
Common Integration Checklist
When implementing CREEM:
-
Environment Setup
- Store API key in environment variables
- Configure base URL for test/production
- Set up webhook endpoint
-
Checkout Flow
- Create checkout session with product_id
- Include request_id for tracking
- Set success_url with verification
- Handle checkout.completed webhook
-
Subscription Handling
- Handle subscription.paid for renewals
- Handle subscription.canceled for access revocation
- Implement customer portal link
- Store subscription_id for management
-
License Keys (if applicable)
- Implement activate on first use
- Validate on each app start
- Handle deactivation for device transfer
-
Security
- Verify webhook signatures
- Never expose API keys client-side
- Validate success URL signatures
File References
For detailed information, see:
REFERENCE.md- Complete API reference with all fieldsWEBHOOKS.md- Webhook payload examplesWORKFLOWS.md- Step-by-step integration patterns
Need Help?
- Documentation: https://docs.creem.io
- Dashboard: https://creem.io/dashboard
- Support: support@creem.io