Awesome-omni-skill apideck-node
Apideck Unified API integration patterns for TypeScript and Node.js. Use when building integrations with accounting software (QuickBooks, Xero, NetSuite), CRMs (Salesforce, HubSpot, Pipedrive), HRIS platforms (Workday, BambooHR), file storage (Google Drive, Dropbox, Box), ATS systems (Greenhouse, Lever), e-commerce, or any of Apideck's 200+ connectors. Covers the @apideck/unify SDK, authentication, CRUD operations, pagination, filtering, webhooks, and Vault connection management.
git clone https://github.com/diegosouzapw/awesome-omni-skill
T=$(mktemp -d) && git clone --depth=1 https://github.com/diegosouzapw/awesome-omni-skill "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/development/apideck-node" ~/.claude/skills/diegosouzapw-awesome-omni-skill-apideck-node-59199e && rm -rf "$T"
skills/development/apideck-node/SKILL.mdApideck TypeScript SDK Skill
Overview
The Apideck Unified API provides a single integration layer to connect with 200+ third-party services across accounting, CRM, HRIS, file storage, ATS, e-commerce, and more. The official TypeScript SDK (
@apideck/unify) provides typed clients for all unified APIs.
Key capabilities:
- Accounting - Invoices, bills, payments, ledger accounts, journal entries, tax rates, balance sheets, P&L
- CRM - Contacts, companies, leads, opportunities, activities, pipelines, notes
- HRIS - Employees, departments, payrolls, time-off requests, schedules
- File Storage - Files, folders, drives, shared links, upload sessions
- ATS - Jobs, applicants, applications
- Vault - Connection management, OAuth flows, custom field mapping
- Vault JS - Embeddable modal UI for users to authorize connectors and manage settings
- Webhook - Event subscriptions and real-time notifications
Installation
npm add @apideck/unify
Requires Node.js 18+. The SDK is fully typed with TypeScript definitions.
IMPORTANT RULES
- ALWAYS use the
SDK. DO NOT make raw@apideck/unify
/fetch
calls to the Apideck API.axios - ALWAYS pass
,apiKey
, andappId
when initializing the client. These are required for all API calls.consumerId - ALWAYS set the
environment variable rather than hardcoding API keys.APIDECK_API_KEY - USE
to specify which downstream connector to use (e.g.,serviceId
,"salesforce"
,"quickbooks"
). If a consumer has multiple connections for an API,"xero"
is required.serviceId - USE cursor-based pagination with
for iterating large result sets. DO NOT implement manual pagination.for await...of - USE the
parameter to narrow results server-side. DO NOT fetch all records and filter client-side.filter - USE the
parameter to request only the columns you need. This reduces response size and improves performance.fields - ALWAYS handle errors with try/catch. The SDK throws typed errors for different HTTP status codes.
- DO NOT store Apideck API keys, App IDs, or Consumer IDs in source code. Use environment variables or a secrets manager.
Quick Start
import { Apideck } from "@apideck/unify"; const apideck = new Apideck({ apiKey: process.env["APIDECK_API_KEY"] ?? "", appId: "your-app-id", consumerId: "your-consumer-id", }); // List CRM contacts const { data } = await apideck.crm.contacts.list({ limit: 20, filter: { email: "john@example.com" }, }); for (const contact of data) { console.log(contact.name, contact.emails); }
SDK Patterns
Client Setup
import { Apideck } from "@apideck/unify"; const apideck = new Apideck({ apiKey: process.env["APIDECK_API_KEY"] ?? "", appId: "your-app-id", consumerId: "your-consumer-id", });
The
consumerId identifies the end-user whose connections are being used. In multi-tenant apps, set this per-request or per-user session.
CRUD Operations
All resources follow the same pattern:
apideck.{api}.{resource}.{operation}().
// LIST - retrieve multiple records const { data } = await apideck.crm.contacts.list({ serviceId: "salesforce", limit: 20, filter: { email: "john@example.com" }, sort: { by: "updated_at", direction: "desc" }, fields: "id,name,email,phone_numbers", }); // CREATE - create a new record const { data: created } = await apideck.crm.contacts.create({ serviceId: "salesforce", contact: { first_name: "John", last_name: "Doe", emails: [{ email: "john@example.com", type: "primary" }], phone_numbers: [{ number: "+1234567890", type: "mobile" }], }, }); console.log(created.id); // "contact_abc123" // GET - retrieve a single record const { data: contact } = await apideck.crm.contacts.get({ id: "contact_abc123", serviceId: "salesforce", }); // UPDATE - modify an existing record const { data: updated } = await apideck.crm.contacts.update({ id: "contact_abc123", serviceId: "salesforce", contact: { first_name: "Jane" }, }); // DELETE - remove a record await apideck.crm.contacts.delete({ id: "contact_abc123", serviceId: "salesforce", });
Pagination
Use async iteration to automatically handle cursor-based pagination:
const result = await apideck.accounting.invoices.list({ serviceId: "quickbooks", limit: 50, }); // Automatically fetches next pages for await (const page of result) { for (const invoice of page.data) { console.log(invoice.number, invoice.total); } }
Or handle pagination manually:
let cursor: string | undefined; do { const { data, meta } = await apideck.accounting.invoices.list({ serviceId: "quickbooks", limit: 50, cursor, }); for (const invoice of data) { console.log(invoice.number); } cursor = meta?.cursors?.next ?? undefined; } while (cursor);
Error Handling
import { Apideck } from "@apideck/unify"; import * as errors from "@apideck/unify/models/errors"; try { const { data } = await apideck.crm.contacts.get({ id: "invalid" }); } catch (e) { if (e instanceof errors.BadRequestResponse) { console.error("Bad request:", e.message); } else if (e instanceof errors.UnauthorizedResponse) { console.error("Invalid API key or missing credentials"); } else if (e instanceof errors.NotFoundResponse) { console.error("Record not found"); } else if (e instanceof errors.PaymentRequiredResponse) { console.error("API limit reached or payment required"); } else if (e instanceof errors.UnprocessableResponse) { console.error("Validation error:", e.message); } else { throw e; } }
Common Parameters
Most list endpoints accept these parameters:
| Parameter | Type | Description |
|---|---|---|
| | Downstream connector ID (e.g., , ) |
| | Max results per page (1-200, default 20) |
| | Pagination cursor from previous response |
| | Resource-specific filter criteria |
| | |
| | Comma-separated field names to return |
| | Pass-through query parameters for the downstream API |
Pass-Through Parameters
When the unified model doesn't cover a connector-specific field, use
passThrough:
const { data } = await apideck.accounting.invoices.list({ serviceId: "quickbooks", passThrough: { search: "overdue", }, });
For creating/updating, use
pass_through in the request body to send connector-specific fields:
const { data } = await apideck.crm.contacts.create({ serviceId: "salesforce", contact: { first_name: "John", last_name: "Doe", pass_through: [ { service_id: "salesforce", operation_id: "contactsAdd", extend_object: { custom_sf_field__c: "value" }, }, ], }, });
API Namespaces
The SDK organizes APIs by namespace. See the reference files for detailed endpoints:
| Namespace | Reference | Resources |
|---|---|---|
| references/accounting-api.md | invoices, bills, payments, customers, suppliers, ledgerAccounts, journalEntries, taxRates, creditNotes, purchaseOrders, balanceSheet, profitAndLoss, and more |
| references/crm-api.md | contacts, companies, leads, opportunities, activities, notes, pipelines, users |
| references/hris-api.md | employees, companies, departments, payrolls, timeOffRequests |
| references/file-storage-api.md | files, folders, drives, driveGroups, sharedLinks, uploadSessions |
| references/ats-api.md | applicants, applications, jobs |
| references/vault-api.md | connections, connectionSettings, consumers, customMappings, logs, sessions |
| references/webhook-api.md | webhooks, eventLogs |
Vault JS (Embeddable UI)
Use
to embed a pre-built modal that lets your users authorize connectors and manage integration settings. Session creation must happen server-side.@apideck/vault-js
// 1. Server-side: create a session const { data } = await apideck.vault.sessions.create({ session: { consumer_metadata: { account_name: "Acme Corp", user_name: "John Doe", email: "john@acme.com" }, redirect_uri: "https://myapp.com/integrations", settings: { unified_apis: ["accounting", "crm"] }, theme: { vault_name: "My App", primary_color: "#4F46E5" }, }, }); // 2. Client-side: open the modal import { ApideckVault } from "@apideck/vault-js"; ApideckVault.open({ token: sessionToken, onConnectionChange: (connection) => console.log("Changed:", connection), onClose: () => console.log("Closed"), });
See references/vault-js.md for full configuration options, theming, React integration, and event callbacks.