@flexprice/sdk
v2.1.3
Published
Type-safe TypeScript/JavaScript client for the FlexPrice API: billing, metering, and subscription management for SaaS and usage-based products.
Downloads
10,113
Readme
FlexPrice TypeScript / JavaScript SDK
Type-safe TypeScript/JavaScript client for the FlexPrice API: billing, metering, and subscription management for SaaS and usage-based products.
Requirements
- Node.js 18+ (or see RUNTIMES.md for supported runtimes)
Installation
npm i @flexprice/sdkWith pnpm, bun, or yarn:
pnpm add @flexprice/sdk
bun add @flexprice/sdk
yarn add @flexprice/sdkThe package supports both CommonJS and ESM.
Then in your code:
import { FlexPrice } from "@flexprice/sdk";All generated modules (shared models, enums, errors, operation types, SDK types) are re-exported from the package root so you can import everything from one place. Run npm run build before publishing so both ESM and CommonJS entry points (dist/esm/index.js and dist/commonjs/index.js) include these re-exports; then import and require("@flexprice/sdk") both expose types/enums from the root.
import {
FlexPrice,
FeatureType,
Status,
FlexpriceError,
CreateCustomerRequest,
createPageIterator,
} from "@flexprice/sdk";
// No need for: .../dist/sdk/models/shared, .../errors, .../operations, .../typesRunnable samples are in the examples/ directory.
Environment
| Variable | Required | Description |
| -------- | -------- | ----------- |
| FLEXPRICE_API_KEY | Yes | API key |
| FLEXPRICE_API_HOST | Optional | Full base URL including https:// and /v1 (default: https://us.api.flexprice.io/v1). No trailing slash. |
Integration tests in api/tests/ts/test_sdk.ts use a different env shape; see api/tests/README.md.
Quick start
Initialize the client, then create a customer and ingest an event:
import { FlexPrice } from "@flexprice/sdk";
const apiKey = process.env.FLEXPRICE_API_KEY ?? "YOUR_API_KEY";
const serverURL =
process.env.FLEXPRICE_API_HOST ?? "https://us.api.flexprice.io/v1";
const flexPrice = new FlexPrice({
serverURL,
apiKeyAuth: apiKey,
});
async function main() {
const externalId = `customer-${Date.now()}`;
const customer = await flexPrice.customers.createCustomer({
externalId,
email: "[email protected]",
name: "Example Customer",
});
console.log(customer);
const eventResult = await flexPrice.events.ingestEvent({
eventName: "Sample Event",
externalCustomerId: externalId,
properties: { source: "ts_sdk", environment: "test" },
source: "ts_sdk",
});
console.log(eventResult);
}
main();For more examples and all API operations, see the API reference and the examples in this repo.
Property names (snake_case)
For request bodies, the API often expects snake_case field names. The SDK may accept camelCase and serialize to snake_case; if you see validation errors, use the API shape:
- Prefer:
event_name,external_customer_id,page_size - Avoid using only camelCase in raw payloads if the API spec uses snake_case
Check the API reference for the exact request shapes.
TypeScript
The package ships with TypeScript definitions. Use the client with full type safety:
import { FlexPrice } from "@flexprice/sdk";
const serverURL =
process.env.FLEXPRICE_API_HOST ?? "https://us.api.flexprice.io/v1";
const flexPrice = new FlexPrice({
serverURL,
apiKeyAuth: process.env.FLEXPRICE_API_KEY!,
});
const result = await flexPrice.events.ingestEvent({
eventName: "usage",
externalCustomerId: "cust_123",
properties: { units: 10 },
source: "backend",
});Authentication
- Set the API key via
apiKeyAuthwhen constructingFlexPrice. The SDK sends it in thex-api-keyheader. - Set
FLEXPRICE_API_HOSTto a full URL (see Environment) or rely on the defaulthttps://us.api.flexprice.io/v1. - Use environment variables and never expose keys in client-side or public code. Get keys from your FlexPrice dashboard or docs.
Features
- Full API coverage (customers, plans, events, invoices, payments, entitlements, etc.)
- TypeScript types for requests and responses
- Built-in retries and error handling
- ESM and CommonJS support
For a full list of operations, see the API reference and the examples in this repo.
Troubleshooting
- Missing or invalid API key: Ensure
apiKeyAuthis set and the key is active. Use server-side only. - Wrong server URL: Use a full URL such as
https://us.api.flexprice.io/v1(include/v1; no trailing slash). - Validation or 4xx errors: Confirm request body field names (snake_case vs camelCase) and required fields against the API docs.
- Parameter passing: Pass the request object directly to methods (e.g.
ingestEvent({ ... })), not wrapped in an extra key, unless the SDK docs say otherwise.
Handling Webhooks
Flexprice sends webhook events to your server for async updates on payments, invoices, subscriptions, wallets, and more.
Flow:
- Register your endpoint URL in the Flexprice dashboard
- Receive
POSTwith raw JSON body - Read
event_typeto route - Parse payload using SDK helpers
- Handle business logic idempotently
- Return
200quickly
import {
WebhookEventName,
webhookDtoPaymentWebhookPayloadFromJSON,
webhookDtoSubscriptionWebhookPayloadFromJSON,
webhookDtoInvoiceWebhookPayloadFromJSON,
} from "@flexprice/sdk";
function handleWebhook(rawBody: string): void {
const env = JSON.parse(rawBody) as { event_type: string };
switch (env.event_type as WebhookEventName) {
case WebhookEventName.PaymentSuccess:
case WebhookEventName.PaymentFailed:
case WebhookEventName.PaymentUpdated: {
const result = webhookDtoPaymentWebhookPayloadFromJSON(rawBody);
if (!result.ok) { console.error("parse error", result.error); break; }
const { payment } = result.value;
console.log("payment", payment?.id);
// TODO: update payment record
break;
}
case WebhookEventName.SubscriptionActivated:
case WebhookEventName.SubscriptionCancelled:
case WebhookEventName.SubscriptionUpdated: {
const result = webhookDtoSubscriptionWebhookPayloadFromJSON(rawBody);
if (!result.ok) { console.error("parse error", result.error); break; }
console.log("subscription", result.value.subscription?.id);
break;
}
case WebhookEventName.InvoiceUpdateFinalized:
case WebhookEventName.InvoicePaymentOverdue: {
const result = webhookDtoInvoiceWebhookPayloadFromJSON(rawBody);
if (!result.ok) { console.error("parse error", result.error); break; }
console.log("invoice", result.value.invoice?.id);
break;
}
default:
console.log("unhandled event:", env.event_type);
}
}Fields are auto-camelCased by the SDK:
event_type→eventType,invoice_status→invoiceStatus. ThefromJSONhelpers return aSafeParseResult<T>— always check.okbefore accessing.value.
Event types
| Category | Events |
|---|---|
| Payment | payment.created · payment.updated · payment.success · payment.failed · payment.pending |
| Invoice | invoice.create.drafted · invoice.update · invoice.update.finalized · invoice.update.payment · invoice.update.voided · invoice.payment.overdue · invoice.communication.triggered |
| Subscription | subscription.created · subscription.draft.created · subscription.activated · subscription.updated · subscription.paused · subscription.resumed · subscription.cancelled · subscription.renewal.due |
| Subscription Phase | subscription.phase.created · subscription.phase.updated · subscription.phase.deleted |
| Customer | customer.created · customer.updated · customer.deleted |
| Wallet | wallet.created · wallet.updated · wallet.terminated · wallet.transaction.created · wallet.credit_balance.dropped · wallet.credit_balance.recovered · wallet.ongoing_balance.dropped · wallet.ongoing_balance.recovered |
| Feature / Entitlement | feature.created · feature.updated · feature.deleted · feature.wallet_balance.alert · entitlement.created · entitlement.updated · entitlement.deleted |
| Credit Note | credit_note.created · credit_note.updated |
Production rules:
- Keep handlers idempotent — Flexprice retries on non-
2xx - Return
200for unknown event types — prevents unnecessary retries - Do heavy processing async — respond fast, queue the work
Documentation
- FlexPrice API documentation
- TypeScript SDK examples in this repo
- SDK integration tests — different
FLEXPRICE_API_HOSTshape for automated tests
