@happyvertical/accounting 0.74.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENT.md +33 -0
- package/LICENSE +7 -0
- package/dist/chunks/OAuthClient-fC3cd77X.js +17475 -0
- package/dist/chunks/OAuthClient-fC3cd77X.js.map +1 -0
- package/dist/chunks/index-D0bqSiCo.js +893 -0
- package/dist/chunks/index-D0bqSiCo.js.map +1 -0
- package/dist/chunks/index-DO-cM79R.js +772 -0
- package/dist/chunks/index-DO-cM79R.js.map +1 -0
- package/dist/cli/claude-context.d.ts +3 -0
- package/dist/cli/claude-context.d.ts.map +1 -0
- package/dist/cli/claude-context.js +21 -0
- package/dist/cli/claude-context.js.map +1 -0
- package/dist/index.d.ts +53 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +55 -0
- package/dist/index.js.map +1 -0
- package/dist/providers/quickbooks/index.d.ts +43 -0
- package/dist/providers/quickbooks/index.d.ts.map +1 -0
- package/dist/providers/stripe/index.d.ts +26 -0
- package/dist/providers/stripe/index.d.ts.map +1 -0
- package/dist/types.d.ts +670 -0
- package/dist/types.d.ts.map +1 -0
- package/metadata.json +30 -0
- package/package.json +58 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index-D0bqSiCo.js","sources":["../../src/providers/quickbooks/index.ts"],"sourcesContent":["/**\n * QuickBooks Online Provider\n *\n * Implements the AccountingProvider interface for QuickBooks Online.\n * Uses the Intuit OAuth2 library for authentication and the QBO API for operations.\n */\n\n/// <reference path=\"../../intuit-oauth.d.ts\" />\n\nimport { createHmac } from 'node:crypto';\nimport type {\n AccountingProvider,\n Address,\n AuditDiscrepancy,\n AuditMatch,\n AuditOperations,\n AuditReport,\n BillInput,\n BillOperations,\n CustomerInput,\n CustomerOperations,\n DateRange,\n ExternalBill,\n ExternalCustomer,\n ExternalInvoice,\n ExternalPayment,\n ExternalRecord,\n ExternalVendor,\n FieldDiff,\n InvoiceInput,\n InvoiceOperations,\n ListOptions,\n PaymentInput,\n PaymentOperations,\n QuickBooksOptions,\n SyncResult,\n TokenSet,\n VendorInput,\n VendorOperations,\n WebhookEvent,\n WebhookOperations,\n} from '../../types.js';\n\n// =============================================================================\n// Utility Functions\n// =============================================================================\n\n/**\n * Format a Date to YYYY-MM-DD in local timezone (not UTC)\n * This prevents timezone conversion issues where dates shift by a day\n */\nfunction formatLocalDate(date: Date): string {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, '0');\n const day = String(date.getDate()).padStart(2, '0');\n return `${year}-${month}-${day}`;\n}\n\n/**\n * Sleep for a given number of milliseconds\n */\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * Compare two values for equality (handles dates, objects, etc.)\n */\nfunction valuesEqual(a: unknown, b: unknown): boolean {\n if (a === b) return true;\n if (a instanceof Date && b instanceof Date) {\n return a.getTime() === b.getTime();\n }\n if (typeof a === 'number' && typeof b === 'number') {\n // Allow for floating point precision issues\n return Math.abs(a - b) < 0.01;\n }\n return false;\n}\n\n// =============================================================================\n// QuickBooks Provider\n// =============================================================================\n\n/**\n * QuickBooks Online accounting provider\n */\nexport class QuickBooksProvider implements AccountingProvider {\n readonly type = 'quickbooks' as const;\n\n private options: QuickBooksOptions;\n private accessToken: string | null = null;\n private tokenExpiresAt: Date | null = null;\n\n readonly customers: CustomerOperations;\n readonly invoices: InvoiceOperations;\n readonly vendors: VendorOperations;\n readonly bills: BillOperations;\n readonly payments: PaymentOperations;\n readonly audit: AuditOperations;\n readonly webhooks: WebhookOperations;\n\n constructor(options: QuickBooksOptions) {\n this.options = {\n timeout: 30000,\n maxRetries: 3,\n ...options,\n };\n\n // Initialize operation handlers\n this.customers = new QuickBooksCustomerOperations(this);\n this.invoices = new QuickBooksInvoiceOperations(this);\n this.vendors = new QuickBooksVendorOperations(this);\n this.bills = new QuickBooksBillOperations(this);\n this.payments = new QuickBooksPaymentOperations(this);\n this.audit = new QuickBooksAuditOperations(this);\n this.webhooks = new QuickBooksWebhookOperations(this.options);\n }\n\n /**\n * Get the QBO API base URL\n */\n get baseUrl(): string {\n return this.options.environment === 'production'\n ? 'https://quickbooks.api.intuit.com'\n : 'https://sandbox-quickbooks.api.intuit.com';\n }\n\n /**\n * Get the company/realm ID\n */\n get realmId(): string {\n return this.options.realmId;\n }\n\n /**\n * Ensure we have a valid access token, refreshing if needed\n */\n async ensureAccessToken(): Promise<string> {\n // Check if current token is still valid (with 5 min buffer)\n if (this.accessToken && this.tokenExpiresAt) {\n const bufferMs = 5 * 60 * 1000;\n if (new Date().getTime() < this.tokenExpiresAt.getTime() - bufferMs) {\n return this.accessToken;\n }\n }\n\n // Refresh the token\n const tokens = await this.refreshAccessToken();\n this.accessToken = tokens.accessToken;\n this.tokenExpiresAt = tokens.expiresAt;\n\n // Update stored refresh token so subsequent refreshes use the new token\n this.options.refreshToken = tokens.refreshToken;\n\n // Notify callback if provided\n if (this.options.onTokenRefresh) {\n await this.options.onTokenRefresh(tokens);\n }\n\n return this.accessToken;\n }\n\n /**\n * Refresh the OAuth access token\n */\n private async refreshAccessToken(): Promise<TokenSet> {\n // Import intuit-oauth dynamically\n const OAuthClient = (await import('intuit-oauth')).default;\n\n const redirectUri =\n this.options.redirectUri ||\n 'https://developer.intuit.com/v2/OAuth2Playground/RedirectUrl';\n\n const oauthClient = new OAuthClient({\n clientId: this.options.clientId,\n clientSecret: this.options.clientSecret,\n environment:\n this.options.environment === 'production' ? 'production' : 'sandbox',\n redirectUri,\n });\n\n oauthClient.setToken({\n refresh_token: this.options.refreshToken,\n access_token: this.accessToken || '',\n });\n\n const response = await oauthClient.refresh();\n const token = response.getJson();\n\n return {\n accessToken: token.access_token,\n refreshToken: token.refresh_token,\n expiresAt: new Date(Date.now() + token.expires_in * 1000),\n tokenType: token.token_type,\n };\n }\n\n /**\n * Make an authenticated request to the QBO API with timeout and retry support\n */\n async request<T>(\n method: 'GET' | 'POST' | 'DELETE',\n endpoint: string,\n body?: unknown,\n ): Promise<T> {\n const accessToken = await this.ensureAccessToken();\n const url = `${this.baseUrl}/v3/company/${this.realmId}/${endpoint}`;\n const timeout = this.options.timeout || 30000;\n const maxRetries = this.options.maxRetries || 3;\n\n let lastError: Error | null = null;\n\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\n try {\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeout);\n\n try {\n const response = await fetch(url, {\n method,\n headers: {\n Authorization: `Bearer ${accessToken}`,\n Accept: 'application/json',\n 'Content-Type': 'application/json',\n },\n body: body ? JSON.stringify(body) : undefined,\n signal: controller.signal,\n });\n\n clearTimeout(timeoutId);\n\n if (!response.ok) {\n const errorText = await response.text();\n\n // Don't retry client errors (4xx) except rate limiting\n if (\n response.status >= 400 &&\n response.status < 500 &&\n response.status !== 429\n ) {\n throw new Error(\n `QBO API error (${response.status}): ${errorText}`,\n );\n }\n\n // Retry on server errors and rate limiting\n throw new Error(`QBO API error (${response.status}): ${errorText}`);\n }\n\n return (await response.json()) as T;\n } finally {\n clearTimeout(timeoutId);\n }\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error));\n\n // Don't retry client errors (4xx except 429) - re-throw immediately\n const errorMessage = lastError.message;\n const statusMatch = errorMessage.match(/QBO API error \\((\\d+)\\)/);\n if (statusMatch) {\n const status = Number.parseInt(statusMatch[1], 10);\n if (status >= 400 && status < 500 && status !== 429) {\n throw lastError;\n }\n }\n\n // Don't retry on abort (timeout) for the last attempt\n if (attempt < maxRetries) {\n // Exponential backoff: 1s, 2s, 4s\n const backoffMs = 2 ** attempt * 1000;\n await sleep(backoffMs);\n }\n }\n }\n\n throw lastError || new Error('Request failed after retries');\n }\n\n /**\n * Execute a QBO query with pagination support\n */\n async queryAll<T>(\n entity: string,\n whereClause?: string,\n maxResults = 1000,\n ): Promise<T[]> {\n const results: T[] = [];\n let startPosition = 1;\n const pageSize = 100; // QBO max is 1000, but smaller pages are safer\n\n while (results.length < maxResults) {\n let query = `SELECT * FROM ${entity}`;\n if (whereClause) {\n query += ` WHERE ${whereClause}`;\n }\n query += ` MAXRESULTS ${pageSize} STARTPOSITION ${startPosition}`;\n\n const response = await this.request<{\n QueryResponse: Record<string, T[] | undefined>;\n }>('GET', `query?query=${encodeURIComponent(query)}`);\n\n const items = response.QueryResponse[entity] || [];\n results.push(...items);\n\n // If we got fewer than pageSize, we've reached the end\n if (items.length < pageSize) {\n break;\n }\n\n startPosition += pageSize;\n }\n\n return results;\n }\n}\n\n// =============================================================================\n// Customer Operations\n// =============================================================================\n\nclass QuickBooksCustomerOperations implements CustomerOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async push(customer: CustomerInput): Promise<SyncResult> {\n const qboCustomer = mapCustomerToQBO(customer);\n\n const response = await this.provider.request<{ Customer: QBOCustomer }>(\n 'POST',\n 'customer',\n qboCustomer,\n );\n\n return {\n action: 'created',\n externalId: response.Customer.Id,\n syncedAt: new Date(),\n };\n }\n\n async pull(externalId: string): Promise<ExternalCustomer> {\n const response = await this.provider.request<{ Customer: QBOCustomer }>(\n 'GET',\n `customer/${externalId}`,\n );\n\n return mapQBOToCustomer(response.Customer);\n }\n\n async list(options?: ListOptions): Promise<ExternalCustomer[]> {\n const limit = options?.limit || 1000;\n const customers = await this.provider.queryAll<QBOCustomer>(\n 'Customer',\n undefined,\n limit,\n );\n return customers.map(mapQBOToCustomer);\n }\n\n async sync(customer: CustomerInput): Promise<SyncResult> {\n if (customer.externalId) {\n // Update existing\n const existing = await this.provider.request<{ Customer: QBOCustomer }>(\n 'GET',\n `customer/${customer.externalId}`,\n );\n\n const qboCustomer: QBOCustomerUpdate = {\n ...mapCustomerToQBO(customer),\n Id: customer.externalId,\n SyncToken: existing.Customer.SyncToken,\n };\n\n await this.provider.request('POST', 'customer', qboCustomer);\n\n return {\n action: 'updated',\n externalId: customer.externalId,\n syncedAt: new Date(),\n };\n }\n\n return this.push(customer);\n }\n}\n\n// =============================================================================\n// Invoice Operations\n// =============================================================================\n\nclass QuickBooksInvoiceOperations implements InvoiceOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async push(invoice: InvoiceInput): Promise<SyncResult> {\n const qboInvoice = mapInvoiceToQBO(invoice);\n\n const response = await this.provider.request<{ Invoice: QBOInvoice }>(\n 'POST',\n 'invoice',\n qboInvoice,\n );\n\n return {\n action: 'created',\n externalId: response.Invoice.Id,\n syncedAt: new Date(),\n };\n }\n\n async pull(externalId: string): Promise<ExternalInvoice> {\n const response = await this.provider.request<{ Invoice: QBOInvoice }>(\n 'GET',\n `invoice/${externalId}`,\n );\n\n return mapQBOToInvoice(response.Invoice);\n }\n\n async list(options?: ListOptions): Promise<ExternalInvoice[]> {\n const limit = options?.limit || 1000;\n let whereClause: string | undefined;\n\n if (options?.startDate) {\n whereClause = `TxnDate >= '${formatLocalDate(options.startDate)}'`;\n if (options?.endDate) {\n whereClause += ` AND TxnDate <= '${formatLocalDate(options.endDate)}'`;\n }\n }\n\n const invoices = await this.provider.queryAll<QBOInvoice>(\n 'Invoice',\n whereClause,\n limit,\n );\n return invoices.map(mapQBOToInvoice);\n }\n\n async sync(invoice: InvoiceInput): Promise<SyncResult> {\n if (invoice.externalId) {\n const existing = await this.provider.request<{ Invoice: QBOInvoice }>(\n 'GET',\n `invoice/${invoice.externalId}`,\n );\n\n const qboInvoice: QBOInvoiceUpdate = {\n ...mapInvoiceToQBO(invoice),\n Id: invoice.externalId,\n SyncToken: existing.Invoice.SyncToken,\n };\n\n await this.provider.request('POST', 'invoice', qboInvoice);\n\n return {\n action: 'updated',\n externalId: invoice.externalId,\n syncedAt: new Date(),\n };\n }\n\n return this.push(invoice);\n }\n\n async send(externalId: string): Promise<void> {\n await this.provider.request('POST', `invoice/${externalId}/send`);\n }\n\n async void(externalId: string): Promise<void> {\n const existing = await this.provider.request<{ Invoice: QBOInvoice }>(\n 'GET',\n `invoice/${externalId}`,\n );\n\n await this.provider.request('POST', 'invoice', {\n Id: externalId,\n SyncToken: existing.Invoice.SyncToken,\n sparse: true,\n // QBO doesn't have a Void field - you delete the invoice instead\n // or create a credit memo. For now, we'll use sparse update to mark private note\n PrivateNote: 'VOIDED',\n });\n }\n}\n\n// =============================================================================\n// Vendor Operations\n// =============================================================================\n\nclass QuickBooksVendorOperations implements VendorOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async push(vendor: VendorInput): Promise<SyncResult> {\n const qboVendor = mapVendorToQBO(vendor);\n\n const response = await this.provider.request<{ Vendor: QBOVendor }>(\n 'POST',\n 'vendor',\n qboVendor,\n );\n\n return {\n action: 'created',\n externalId: response.Vendor.Id,\n syncedAt: new Date(),\n };\n }\n\n async pull(externalId: string): Promise<ExternalVendor> {\n const response = await this.provider.request<{ Vendor: QBOVendor }>(\n 'GET',\n `vendor/${externalId}`,\n );\n\n return mapQBOToVendor(response.Vendor);\n }\n\n async list(options?: ListOptions): Promise<ExternalVendor[]> {\n const limit = options?.limit || 1000;\n const vendors = await this.provider.queryAll<QBOVendor>(\n 'Vendor',\n undefined,\n limit,\n );\n return vendors.map(mapQBOToVendor);\n }\n\n async sync(vendor: VendorInput): Promise<SyncResult> {\n if (vendor.externalId) {\n const existing = await this.provider.request<{ Vendor: QBOVendor }>(\n 'GET',\n `vendor/${vendor.externalId}`,\n );\n\n const qboVendor: QBOVendorUpdate = {\n ...mapVendorToQBO(vendor),\n Id: vendor.externalId,\n SyncToken: existing.Vendor.SyncToken,\n };\n\n await this.provider.request('POST', 'vendor', qboVendor);\n\n return {\n action: 'updated',\n externalId: vendor.externalId,\n syncedAt: new Date(),\n };\n }\n\n return this.push(vendor);\n }\n}\n\n// =============================================================================\n// Bill Operations\n// =============================================================================\n\nclass QuickBooksBillOperations implements BillOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async push(bill: BillInput): Promise<SyncResult> {\n const qboBill = mapBillToQBO(bill);\n\n const response = await this.provider.request<{ Bill: QBOBill }>(\n 'POST',\n 'bill',\n qboBill,\n );\n\n return {\n action: 'created',\n externalId: response.Bill.Id,\n syncedAt: new Date(),\n };\n }\n\n async pull(externalId: string): Promise<ExternalBill> {\n const response = await this.provider.request<{ Bill: QBOBill }>(\n 'GET',\n `bill/${externalId}`,\n );\n\n return mapQBOToBill(response.Bill);\n }\n\n async list(options?: ListOptions): Promise<ExternalBill[]> {\n const limit = options?.limit || 1000;\n let whereClause: string | undefined;\n\n if (options?.startDate) {\n whereClause = `TxnDate >= '${formatLocalDate(options.startDate)}'`;\n if (options?.endDate) {\n whereClause += ` AND TxnDate <= '${formatLocalDate(options.endDate)}'`;\n }\n }\n\n const bills = await this.provider.queryAll<QBOBill>(\n 'Bill',\n whereClause,\n limit,\n );\n return bills.map(mapQBOToBill);\n }\n\n async sync(bill: BillInput): Promise<SyncResult> {\n if (bill.externalId) {\n const existing = await this.provider.request<{ Bill: QBOBill }>(\n 'GET',\n `bill/${bill.externalId}`,\n );\n\n const qboBill: QBOBillUpdate = {\n ...mapBillToQBO(bill),\n Id: bill.externalId,\n SyncToken: existing.Bill.SyncToken,\n };\n\n await this.provider.request('POST', 'bill', qboBill);\n\n return {\n action: 'updated',\n externalId: bill.externalId,\n syncedAt: new Date(),\n };\n }\n\n return this.push(bill);\n }\n}\n\n// =============================================================================\n// Payment Operations\n// =============================================================================\n\nclass QuickBooksPaymentOperations implements PaymentOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async pull(externalId: string): Promise<ExternalPayment> {\n const response = await this.provider.request<{ Payment: QBOPayment }>(\n 'GET',\n `payment/${externalId}`,\n );\n\n return mapQBOToPayment(response.Payment);\n }\n\n async list(options?: ListOptions): Promise<ExternalPayment[]> {\n const limit = options?.limit || 1000;\n let whereClause: string | undefined;\n\n if (options?.startDate) {\n whereClause = `TxnDate >= '${formatLocalDate(options.startDate)}'`;\n if (options?.endDate) {\n whereClause += ` AND TxnDate <= '${formatLocalDate(options.endDate)}'`;\n }\n }\n\n const payments = await this.provider.queryAll<QBOPayment>(\n 'Payment',\n whereClause,\n limit,\n );\n return payments.map(mapQBOToPayment);\n }\n}\n\n// =============================================================================\n// Audit Operations\n// =============================================================================\n\nclass QuickBooksAuditOperations implements AuditOperations {\n constructor(private provider: QuickBooksProvider) {}\n\n async reconcileCustomers(\n locals: CustomerInput[],\n ): Promise<AuditReport<CustomerInput>> {\n const externals = await this.provider.customers.list({ limit: 10000 });\n return reconcileRecords(\n locals,\n externals,\n (l) => l.externalId,\n (e) => e.externalId,\n compareCustomer,\n );\n }\n\n async reconcileInvoices(\n locals: InvoiceInput[],\n dateRange?: DateRange,\n ): Promise<AuditReport<InvoiceInput>> {\n const externals = await this.provider.invoices.list({\n limit: 10000,\n startDate: dateRange?.start,\n endDate: dateRange?.end,\n });\n return reconcileRecords(\n locals,\n externals,\n (l) => l.externalId,\n (e) => e.externalId,\n compareInvoice,\n );\n }\n\n async reconcileVendors(\n locals: VendorInput[],\n ): Promise<AuditReport<VendorInput>> {\n const externals = await this.provider.vendors.list({ limit: 10000 });\n return reconcileRecords(\n locals,\n externals,\n (l) => l.externalId,\n (e) => e.externalId,\n compareVendor,\n );\n }\n\n async reconcileBills(\n locals: BillInput[],\n dateRange?: DateRange,\n ): Promise<AuditReport<BillInput>> {\n const externals = await this.provider.bills.list({\n limit: 10000,\n startDate: dateRange?.start,\n endDate: dateRange?.end,\n });\n return reconcileRecords(\n locals,\n externals,\n (l) => l.externalId,\n (e) => e.externalId,\n compareBill,\n );\n }\n\n async reconcilePayments(\n locals: PaymentInput[],\n dateRange?: DateRange,\n ): Promise<AuditReport<PaymentInput>> {\n const externals = await this.provider.payments.list({\n limit: 10000,\n startDate: dateRange?.start,\n endDate: dateRange?.end,\n });\n return reconcileRecords(\n locals,\n externals,\n (l) => l.externalId,\n (e) => e.externalId,\n comparePayment,\n );\n }\n}\n\n/**\n * Generic reconciliation function with field comparison\n */\nfunction reconcileRecords<TLocal, TExternal extends ExternalRecord>(\n locals: TLocal[],\n externals: TExternal[],\n getLocalExternalId: (local: TLocal) => string | undefined,\n getExternalId: (external: TExternal) => string,\n compareFields: (local: TLocal, external: TExternal) => FieldDiff[],\n): AuditReport<TLocal> {\n const externalMap = new Map(externals.map((e) => [getExternalId(e), e]));\n const matchedExternalIds = new Set<string>();\n\n const matched: AuditMatch<TLocal>[] = [];\n const localOnly: TLocal[] = [];\n const discrepancies: AuditDiscrepancy<TLocal>[] = [];\n\n for (const local of locals) {\n const externalId = getLocalExternalId(local);\n if (externalId && externalMap.has(externalId)) {\n const external = externalMap.get(externalId)!;\n matchedExternalIds.add(externalId);\n\n // Compare fields\n const differences = compareFields(local, external);\n\n if (differences.length === 0) {\n matched.push({\n local,\n external,\n status: 'identical',\n });\n } else {\n discrepancies.push({\n local,\n external,\n differences,\n });\n }\n } else {\n localOnly.push(local);\n }\n }\n\n const externalOnly = externals.filter(\n (e) => !matchedExternalIds.has(getExternalId(e)),\n );\n\n return {\n provider: 'quickbooks',\n auditedAt: new Date(),\n matched,\n localOnly,\n externalOnly,\n discrepancies,\n summary: {\n total: locals.length + externalOnly.length,\n matched: matched.length,\n localOnly: localOnly.length,\n externalOnly: externalOnly.length,\n discrepancies: discrepancies.length,\n },\n };\n}\n\n// Field comparison functions\nfunction compareCustomer(\n local: CustomerInput,\n external: ExternalCustomer,\n): FieldDiff[] {\n const diffs: FieldDiff[] = [];\n\n if (local.name !== external.name) {\n diffs.push({\n field: 'name',\n localValue: local.name,\n externalValue: external.name,\n });\n }\n if (local.email !== external.email) {\n diffs.push({\n field: 'email',\n localValue: local.email,\n externalValue: external.email,\n });\n }\n\n return diffs;\n}\n\nfunction compareInvoice(\n local: InvoiceInput,\n external: ExternalInvoice,\n): FieldDiff[] {\n const diffs: FieldDiff[] = [];\n\n if (!valuesEqual(local.totalAmount, external.totalAmount)) {\n diffs.push({\n field: 'totalAmount',\n localValue: local.totalAmount,\n externalValue: external.totalAmount,\n });\n }\n if (local.invoiceNumber !== external.invoiceNumber) {\n diffs.push({\n field: 'invoiceNumber',\n localValue: local.invoiceNumber,\n externalValue: external.invoiceNumber,\n });\n }\n\n return diffs;\n}\n\nfunction compareVendor(\n local: VendorInput,\n external: ExternalVendor,\n): FieldDiff[] {\n const diffs: FieldDiff[] = [];\n\n if (local.name !== external.name) {\n diffs.push({\n field: 'name',\n localValue: local.name,\n externalValue: external.name,\n });\n }\n if (local.email !== external.email) {\n diffs.push({\n field: 'email',\n localValue: local.email,\n externalValue: external.email,\n });\n }\n\n return diffs;\n}\n\nfunction compareBill(local: BillInput, external: ExternalBill): FieldDiff[] {\n const diffs: FieldDiff[] = [];\n\n if (!valuesEqual(local.totalAmount, external.totalAmount)) {\n diffs.push({\n field: 'totalAmount',\n localValue: local.totalAmount,\n externalValue: external.totalAmount,\n });\n }\n\n return diffs;\n}\n\nfunction comparePayment(\n local: PaymentInput,\n external: ExternalPayment,\n): FieldDiff[] {\n const diffs: FieldDiff[] = [];\n\n if (!valuesEqual(local.amount, external.amount)) {\n diffs.push({\n field: 'amount',\n localValue: local.amount,\n externalValue: external.amount,\n });\n }\n\n return diffs;\n}\n\n// =============================================================================\n// Webhook Operations\n// =============================================================================\n\nclass QuickBooksWebhookOperations implements WebhookOperations {\n constructor(private options: QuickBooksOptions) {}\n\n /**\n * Verify webhook signature using HMAC-SHA256\n * @see https://developer.intuit.com/app/developer/qbo/docs/develop/webhooks/verify-webhooks\n */\n verify(payload: string, signature: string, secret: string): boolean {\n // Use provided secret or fall back to configured webhook verifier token\n const verifierToken = secret || this.options.webhookVerifierToken;\n\n if (!verifierToken) {\n throw new Error(\n 'Webhook verifier token not provided. Pass it as the secret parameter or configure webhookVerifierToken in options.',\n );\n }\n\n // QBO uses HMAC-SHA256 and base64 encodes the signature\n const hmac = createHmac('sha256', verifierToken);\n hmac.update(payload);\n const expectedSignature = hmac.digest('base64');\n\n // Constant-time comparison to prevent timing attacks\n if (signature.length !== expectedSignature.length) {\n return false;\n }\n\n let result = 0;\n for (let i = 0; i < signature.length; i++) {\n result |= signature.charCodeAt(i) ^ expectedSignature.charCodeAt(i);\n }\n return result === 0;\n }\n\n parse(payload: string): WebhookEvent {\n let data: QBOWebhookPayload;\n try {\n data = JSON.parse(payload) as QBOWebhookPayload;\n } catch (error) {\n const message =\n error instanceof Error ? error.message : 'Unknown JSON parse error';\n throw new Error(`Invalid QuickBooks webhook payload: ${message}`);\n }\n\n // QBO webhook payload structure\n const eventNotifications = data.eventNotifications || [];\n const firstEvent = eventNotifications[0]?.dataChangeEvent?.entities?.[0];\n\n return {\n type: firstEvent?.operation || 'unknown',\n provider: 'quickbooks',\n timestamp: new Date(),\n payload: data,\n resourceType: mapQBOResourceType(firstEvent?.name),\n resourceId: firstEvent?.id,\n };\n }\n}\n\nfunction mapQBOResourceType(\n name?: string,\n): WebhookEvent['resourceType'] | undefined {\n switch (name?.toLowerCase()) {\n case 'customer':\n return 'customer';\n case 'invoice':\n return 'invoice';\n case 'payment':\n return 'payment';\n case 'vendor':\n return 'vendor';\n case 'bill':\n return 'bill';\n default:\n return undefined;\n }\n}\n\n// =============================================================================\n// Mapping Functions\n// =============================================================================\n\nfunction mapCustomerToQBO(customer: CustomerInput): QBOCustomerCreate {\n return {\n DisplayName: customer.name,\n PrimaryEmailAddr: customer.email ? { Address: customer.email } : undefined,\n PrimaryPhone: customer.phone\n ? { FreeFormNumber: customer.phone }\n : undefined,\n BillAddr: mapAddressToQBO(customer.billingAddress),\n ShipAddr: mapAddressToQBO(customer.shippingAddress),\n CurrencyRef: customer.currency ? { value: customer.currency } : undefined,\n };\n}\n\nfunction mapQBOToCustomer(qbo: QBOCustomer): ExternalCustomer {\n return {\n externalId: qbo.Id,\n provider: 'quickbooks',\n syncedAt: new Date(),\n name: qbo.DisplayName,\n email: qbo.PrimaryEmailAddr?.Address,\n phone: qbo.PrimaryPhone?.FreeFormNumber,\n billingAddress: mapQBOToAddress(qbo.BillAddr),\n balance: qbo.Balance,\n currency: qbo.CurrencyRef?.value,\n raw: qbo,\n };\n}\n\nfunction mapInvoiceToQBO(invoice: InvoiceInput): QBOInvoiceCreate {\n return {\n CustomerRef: { value: invoice.customerExternalId || invoice.customerId },\n DocNumber: invoice.invoiceNumber,\n TxnDate: formatLocalDate(invoice.issueDate),\n DueDate: formatLocalDate(invoice.dueDate),\n Line: invoice.lineItems.map((item, idx) => ({\n LineNum: idx + 1,\n Description: item.description,\n Amount: item.amount ?? item.quantity * item.unitPrice,\n DetailType: 'SalesItemLineDetail' as const,\n SalesItemLineDetail: {\n Qty: item.quantity,\n UnitPrice: item.unitPrice,\n },\n })),\n CurrencyRef: invoice.currency ? { value: invoice.currency } : undefined,\n CustomerMemo: invoice.memo ? { value: invoice.memo } : undefined,\n };\n}\n\nfunction mapQBOToInvoice(qbo: QBOInvoice): ExternalInvoice {\n const balance = qbo.Balance ?? 0;\n const totalAmount = qbo.TotalAmt ?? 0;\n\n return {\n externalId: qbo.Id,\n provider: 'quickbooks',\n syncedAt: new Date(),\n invoiceNumber: qbo.DocNumber || '',\n customerExternalId: qbo.CustomerRef?.value || '',\n issueDate: new Date(qbo.TxnDate),\n dueDate: new Date(qbo.DueDate),\n subtotal: totalAmount - (qbo.TxnTaxDetail?.TotalTax ?? 0),\n taxAmount: qbo.TxnTaxDetail?.TotalTax ?? 0,\n totalAmount,\n amountPaid: totalAmount - balance,\n balance,\n status: mapInvoiceStatus(balance, totalAmount, qbo.DueDate),\n currency: qbo.CurrencyRef?.value || 'USD',\n raw: qbo,\n };\n}\n\nfunction mapInvoiceStatus(\n balance: number,\n total: number,\n dueDate: string,\n): ExternalInvoice['status'] {\n if (balance === 0) return 'paid';\n if (balance < total) return 'viewed'; // Partial payment\n if (new Date(dueDate) < new Date()) return 'overdue';\n return 'sent';\n}\n\nfunction mapVendorToQBO(vendor: VendorInput): QBOVendorCreate {\n return {\n DisplayName: vendor.name,\n PrimaryEmailAddr: vendor.email ? { Address: vendor.email } : undefined,\n PrimaryPhone: vendor.phone ? { FreeFormNumber: vendor.phone } : undefined,\n BillAddr: mapAddressToQBO(vendor.address),\n TaxIdentifier: vendor.taxId,\n CurrencyRef: vendor.currency ? { value: vendor.currency } : undefined,\n };\n}\n\nfunction mapQBOToVendor(qbo: QBOVendor): ExternalVendor {\n return {\n externalId: qbo.Id,\n provider: 'quickbooks',\n syncedAt: new Date(),\n name: qbo.DisplayName,\n email: qbo.PrimaryEmailAddr?.Address,\n phone: qbo.PrimaryPhone?.FreeFormNumber,\n address: mapQBOToAddress(qbo.BillAddr),\n balance: qbo.Balance,\n currency: qbo.CurrencyRef?.value,\n raw: qbo,\n };\n}\n\nfunction mapBillToQBO(bill: BillInput): QBOBillCreate {\n return {\n VendorRef: { value: bill.vendorExternalId || bill.vendorId },\n DocNumber: bill.billNumber,\n TxnDate: formatLocalDate(bill.billDate),\n DueDate: formatLocalDate(bill.dueDate),\n Line: bill.lineItems.map((item, idx) => ({\n LineNum: idx + 1,\n Description: item.description,\n Amount: item.amount ?? item.quantity * item.unitPrice,\n DetailType: 'AccountBasedExpenseLineDetail' as const,\n AccountBasedExpenseLineDetail: {\n AccountRef: item.accountCode ? { value: item.accountCode } : undefined,\n },\n })),\n CurrencyRef: bill.currency ? { value: bill.currency } : undefined,\n };\n}\n\nfunction mapQBOToBill(qbo: QBOBill): ExternalBill {\n const balance = qbo.Balance ?? 0;\n const totalAmount = qbo.TotalAmt ?? 0;\n\n return {\n externalId: qbo.Id,\n provider: 'quickbooks',\n syncedAt: new Date(),\n billNumber: qbo.DocNumber,\n vendorExternalId: qbo.VendorRef?.value || '',\n billDate: new Date(qbo.TxnDate),\n dueDate: new Date(qbo.DueDate),\n subtotal: totalAmount - (qbo.TxnTaxDetail?.TotalTax ?? 0),\n taxAmount: qbo.TxnTaxDetail?.TotalTax ?? 0,\n totalAmount,\n amountPaid: totalAmount - balance,\n balance,\n status: mapBillStatus(balance, totalAmount, qbo.DueDate),\n currency: qbo.CurrencyRef?.value || 'USD',\n raw: qbo,\n };\n}\n\nfunction mapBillStatus(\n balance: number,\n total: number,\n dueDate: string,\n): ExternalBill['status'] {\n if (balance === 0) return 'paid';\n if (new Date(dueDate) < new Date()) return 'overdue';\n return 'pending';\n}\n\nfunction mapQBOToPayment(qbo: QBOPayment): ExternalPayment {\n return {\n externalId: qbo.Id,\n provider: 'quickbooks',\n syncedAt: new Date(),\n amount: qbo.TotalAmt ?? 0,\n currency: qbo.CurrencyRef?.value || 'USD',\n paidAt: new Date(qbo.TxnDate),\n method: qbo.PaymentMethodRef?.name,\n transactionId: qbo.PaymentRefNum,\n invoiceExternalIds: qbo.Line?.filter((l) => l.LinkedTxn).flatMap(\n (l) => l.LinkedTxn?.map((t) => t.TxnId) || [],\n ),\n status: 'completed',\n raw: qbo,\n };\n}\n\nfunction mapAddressToQBO(address?: Address): QBOAddress | undefined {\n if (!address) return undefined;\n return {\n Line1: address.street1,\n Line2: address.street2,\n City: address.city,\n CountrySubDivisionCode: address.state,\n PostalCode: address.postalCode,\n Country: address.country,\n };\n}\n\nfunction mapQBOToAddress(qbo?: QBOAddress): Address | undefined {\n if (!qbo) return undefined;\n return {\n street1: qbo.Line1,\n street2: qbo.Line2,\n city: qbo.City,\n state: qbo.CountrySubDivisionCode,\n postalCode: qbo.PostalCode,\n country: qbo.Country,\n };\n}\n\n// =============================================================================\n// QBO API Types (internal, strongly typed)\n// =============================================================================\n\ninterface QBORef {\n value: string;\n name?: string;\n}\n\ninterface QBOAddress {\n Line1?: string;\n Line2?: string;\n City?: string;\n CountrySubDivisionCode?: string;\n PostalCode?: string;\n Country?: string;\n}\n\ninterface QBOEmail {\n Address: string;\n}\n\ninterface QBOPhone {\n FreeFormNumber: string;\n}\n\n// Customer types\ninterface QBOCustomerBase {\n DisplayName: string;\n PrimaryEmailAddr?: QBOEmail;\n PrimaryPhone?: QBOPhone;\n BillAddr?: QBOAddress;\n ShipAddr?: QBOAddress;\n Balance?: number;\n CurrencyRef?: QBORef;\n}\n\ninterface QBOCustomerCreate extends QBOCustomerBase {}\n\ninterface QBOCustomerUpdate extends QBOCustomerBase {\n Id: string;\n SyncToken: string;\n}\n\ninterface QBOCustomer extends QBOCustomerBase {\n Id: string;\n SyncToken: string;\n}\n\n// Invoice types\ninterface QBOInvoiceLine {\n LineNum: number;\n Description?: string;\n Amount: number;\n DetailType: 'SalesItemLineDetail';\n SalesItemLineDetail?: {\n Qty?: number;\n UnitPrice?: number;\n ItemRef?: QBORef;\n };\n}\n\ninterface QBOInvoiceBase {\n CustomerRef: QBORef;\n DocNumber?: string;\n TxnDate: string;\n DueDate: string;\n Line: QBOInvoiceLine[];\n TotalAmt?: number;\n Balance?: number;\n TxnTaxDetail?: {\n TotalTax?: number;\n };\n CurrencyRef?: QBORef;\n CustomerMemo?: { value: string };\n}\n\ninterface QBOInvoiceCreate extends QBOInvoiceBase {}\n\ninterface QBOInvoiceUpdate extends QBOInvoiceBase {\n Id: string;\n SyncToken: string;\n}\n\ninterface QBOInvoice extends QBOInvoiceBase {\n Id: string;\n SyncToken: string;\n}\n\n// Vendor types\ninterface QBOVendorBase {\n DisplayName: string;\n PrimaryEmailAddr?: QBOEmail;\n PrimaryPhone?: QBOPhone;\n BillAddr?: QBOAddress;\n Balance?: number;\n TaxIdentifier?: string;\n CurrencyRef?: QBORef;\n}\n\ninterface QBOVendorCreate extends QBOVendorBase {}\n\ninterface QBOVendorUpdate extends QBOVendorBase {\n Id: string;\n SyncToken: string;\n}\n\ninterface QBOVendor extends QBOVendorBase {\n Id: string;\n SyncToken: string;\n}\n\n// Bill types\ninterface QBOBillLine {\n LineNum: number;\n Description?: string;\n Amount: number;\n DetailType: 'AccountBasedExpenseLineDetail';\n AccountBasedExpenseLineDetail?: {\n AccountRef?: QBORef;\n };\n}\n\ninterface QBOBillBase {\n VendorRef: QBORef;\n DocNumber?: string;\n TxnDate: string;\n DueDate: string;\n Line: QBOBillLine[];\n TotalAmt?: number;\n Balance?: number;\n TxnTaxDetail?: {\n TotalTax?: number;\n };\n CurrencyRef?: QBORef;\n}\n\ninterface QBOBillCreate extends QBOBillBase {}\n\ninterface QBOBillUpdate extends QBOBillBase {\n Id: string;\n SyncToken: string;\n}\n\ninterface QBOBill extends QBOBillBase {\n Id: string;\n SyncToken: string;\n}\n\n// Payment types\ninterface QBOPaymentLine {\n Amount: number;\n LinkedTxn?: {\n TxnId: string;\n TxnType: string;\n }[];\n}\n\ninterface QBOPayment {\n Id: string;\n SyncToken: string;\n TxnDate: string;\n TotalAmt?: number;\n CustomerRef?: QBORef;\n PaymentMethodRef?: QBORef;\n PaymentRefNum?: string;\n Line?: QBOPaymentLine[];\n CurrencyRef?: QBORef;\n}\n\n// Webhook types\ninterface QBOWebhookPayload {\n eventNotifications?: {\n realmId: string;\n dataChangeEvent?: {\n entities?: {\n name: string;\n id: string;\n operation: string;\n lastUpdated: string;\n }[];\n };\n }[];\n}\n"],"names":[],"mappings":"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"}
|