@happyvertical/smrt-ledgers 0.30.0
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/AGENTS.md +36 -0
- package/CLAUDE.md +1 -0
- package/LICENSE +7 -0
- package/README.md +124 -0
- package/dist/index.d.ts +647 -0
- package/dist/index.js +1172 -0
- package/dist/index.js.map +1 -0
- package/dist/manifest.json +1561 -0
- package/dist/smrt-knowledge.json +871 -0
- package/dist/types.d.ts +179 -0
- package/dist/types.js +5 -0
- package/dist/types.js.map +1 -0
- package/package.json +59 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../src/__smrt-register__.ts","../src/prompts.ts","../src/models/Account.ts","../src/collections/Accounts.ts","../src/models/JournalEntry.ts","../src/collections/JournalEntries.ts","../src/models/Journal.ts","../src/collections/Journals.ts"],"sourcesContent":["/**\n * Self-registers this package's build-time manifest before any @smrt() decorator\n * in the package fires. Fixes issue #1132: in consumer runtimes (tsx, SvelteKit\n * SSR, plain `vite dev`) the decorator's synchronous manifest lookup previously\n * missed because no step populated the global manifest cache — classes got\n * registered with zero fields and `save()` / `toJSON()` silently dropped every\n * declared property.\n *\n * Import this module as the first statement in `src/index.ts` so its top-level\n * side effect runs ahead of any class module's @smrt() decorator.\n *\n * Silent no-op in dev/test, where the vitest plugin already populates manifests\n * via a different path. Only needs to succeed in the published dist output.\n *\n * @see https://github.com/happyvertical/smrt/issues/1132\n */\nimport { ObjectRegistry } from '@happyvertical/smrt-core';\n\n// `new URL('./manifest.json', import.meta.url)` resolves at runtime to the\n// manifest sitting next to this module's compiled output. Vite warns at build\n// time that it cannot pre-resolve the URL; that is the intended behavior —\n// the URL must resolve to dist/manifest.json at runtime, not be inlined.\nObjectRegistry.registerPackageManifest(\n new URL('./manifest.json', import.meta.url),\n);\n","/**\n * Prompt registrations for the @happyvertical/smrt-ledgers package.\n *\n * Prompts are registered at module-load time via `definePrompt()` so that\n * tenant-aware overrides can be applied at call time via `resolvePrompt()`.\n *\n * Mirrors the pattern used by `@happyvertical/smrt-properties` (see\n * `prompts.ts`) and `@happyvertical/smrt-content` (see `content-prompts.ts`).\n */\n\nimport {\n definePrompt,\n type ResolvedPromptAI,\n} from '@happyvertical/smrt-prompts';\n\n// Journal summarization only uses non-PII journal fields\n// (number, date, description, status, balanced flag) plus aggregate totals\n// and entry count. Internal/foreign-key fields like tenantId, sourceRef,\n// individual entry account IDs, and the extensible `metadata` blob are\n// intentionally NOT passed to the AI provider — they may link to identifying\n// records (customer/vendor IDs via sourceRef) or contain tenant-private\n// configuration. If a downstream tenant needs richer context they can\n// override the template via PromptOverride.\n// Note: smrt-prompts substitutes `{token}` placeholders. The currency prefix\n// is folded into the `journalTotal` variable value (e.g. \"$100.00\") rather\n// than written as a literal `$` in the template — earlier revisions used\n// `\\${journalTotal}` (a template-literal escape that renders as the literal\n// `${journalTotal}` at runtime). That technically worked because the regex\n// still matched the inner `{journalTotal}` and produced `$100.00`, but it\n// confused reviewers; folding the prefix into the value avoids the trap.\nexport const smrtLedgersJournalSummarizePrompt = definePrompt({\n key: 'smrtLedgers.journal.summarize',\n template: `Summarize this accounting journal entry:\nNumber: {journalNumber}\nDate: {journalDate}\nDescription: {journalDescription}\nStatus: {journalStatus}\nTotal: {journalTotal}\nEntries: {entryCount}\nBalanced: {journalBalanced}`,\n editable: {\n template: true,\n profile: true,\n model: true,\n params: true,\n },\n});\n\nexport function promptMessageOptions(ai: ResolvedPromptAI) {\n return {\n ...(ai.params || {}),\n ...(ai.model ? { model: ai.model } : {}),\n ...(typeof ai.temperature === 'number'\n ? { temperature: ai.temperature }\n : {}),\n ...(typeof ai.maxTokens === 'number' ? { maxTokens: ai.maxTokens } : {}),\n };\n}\n","/**\n * Account model - Chart of accounts entry\n *\n * Represents a single account in the chart of accounts.\n * The 5 core types: Asset, Liability, Equity, Revenue, Expense\n */\n\nimport { SmrtHierarchical, smrt } from '@happyvertical/smrt-core';\nimport { TenantScoped, tenantId } from '@happyvertical/smrt-tenancy';\nimport type { AccountOptions, AccountTreeNode, AccountType } from '../types';\n\n@TenantScoped({ mode: 'optional' })\n@smrt({\n api: { include: ['list', 'get', 'create', 'update', 'delete'] },\n mcp: { include: ['list', 'get', 'create'] },\n cli: true,\n})\nexport class Account extends SmrtHierarchical {\n /**\n * Tenant ID for multi-tenancy support (nullable for global accounts)\n */\n @tenantId({ nullable: true })\n tenantId: string | null = null;\n\n /**\n * Account number (e.g., \"1000\", \"5030\")\n */\n number: string = '';\n\n /**\n * Account name (e.g., \"Cash\", \"Coffee Expense\")\n */\n name: string = '';\n\n /**\n * Account description\n */\n description: string = '';\n\n /**\n * Account type - one of the 5 core types\n */\n type: AccountType = 'asset';\n\n // parentId inherited from SmrtHierarchical (null = top-level)\n\n /**\n * Whether the account is active\n */\n active: boolean = true;\n\n /**\n * Extensible metadata\n */\n metadata: Record<string, unknown> = {};\n\n constructor(options: AccountOptions = {}) {\n super(options);\n if (options.number !== undefined) this.number = options.number;\n if (options.name !== undefined) this.name = options.name;\n if (options.description !== undefined)\n this.description = options.description;\n if (options.type !== undefined) this.type = options.type;\n if (options.parentId !== undefined) this.parentId = options.parentId;\n if (options.active !== undefined) this.active = options.active;\n if (options.metadata !== undefined) this.metadata = options.metadata;\n }\n\n /**\n * Check if this is a top-level account (no parent)\n */\n isTopLevel(): boolean {\n return this.parentId === null;\n }\n\n /**\n * Check if this is a debit-normal account (Asset, Expense)\n * Debit-normal accounts increase with debits\n */\n isDebitNormal(): boolean {\n return this.type === 'asset' || this.type === 'expense';\n }\n\n /**\n * Check if this is a credit-normal account (Liability, Equity, Revenue)\n * Credit-normal accounts increase with credits\n */\n isCreditNormal(): boolean {\n return (\n this.type === 'liability' ||\n this.type === 'equity' ||\n this.type === 'revenue'\n );\n }\n\n // Hierarchy traversal (getParent / getChildren / getAncestors /\n // getDescendants / getHierarchy / moveTo) provided by SmrtHierarchical.\n // Account-specific helpers (getFullPath / getBalance / createChild /\n // toTreeNode) remain below.\n\n /**\n * Get full account path (e.g., \"Assets > Current > Cash\")\n */\n async getFullPath(): Promise<string> {\n const ancestors = await this.getAncestors();\n const names = [...ancestors.map((a) => a.name), this.name];\n return names.join(' > ');\n }\n\n /**\n * Build tree node for this account and its children\n */\n async toTreeNode(): Promise<AccountTreeNode> {\n const children = await this.getChildren();\n const childNodes = await Promise.all(children.map((c) => c.toTreeNode()));\n return {\n account: this,\n children: childNodes,\n };\n }\n\n /**\n * Get the current balance of this account\n */\n async getBalance(asOfDate?: Date): Promise<number> {\n if (!this.id) return 0;\n const { JournalEntryCollection } = await import(\n '../collections/JournalEntries'\n );\n const collection = await (JournalEntryCollection as any).create(\n this.options,\n );\n return await collection.getAccountBalance(this.id, asOfDate);\n }\n\n /**\n * Create a sub-account under this account\n */\n async createChild(\n options: Omit<AccountOptions, 'parentId' | 'type'>,\n ): Promise<Account> {\n if (!this.id) {\n throw new Error('Account must be saved before creating children');\n }\n const { AccountCollection } = await import('../collections/Accounts');\n const collection = await (AccountCollection as any).create(this.options);\n const account = await collection.create({\n ...options,\n parentId: this.id,\n type: this.type, // Inherit type from parent\n });\n await account.save();\n return account;\n }\n}\n","/**\n * AccountCollection - Collection manager for Account objects\n *\n * Provides querying and tree operations for chart of accounts.\n */\n\nimport { SmrtCollection } from '@happyvertical/smrt-core';\nimport { Account } from '../models/Account';\nimport type { AccountTree, AccountTreeNode, AccountType } from '../types';\n\nexport class AccountCollection extends SmrtCollection<Account> {\n static readonly _itemClass = Account;\n\n /**\n * Find account by number\n *\n * @param number - Account number\n * @returns Account or null\n */\n async findByNumber(number: string): Promise<Account | null> {\n const accounts = await this.list({\n where: { number },\n limit: 1,\n });\n return accounts[0] || null;\n }\n\n /**\n * Find accounts by type\n *\n * @param type - Account type\n * @returns Array of accounts\n */\n async findByType(type: AccountType): Promise<Account[]> {\n return await this.list({\n where: { type },\n orderBy: 'number ASC',\n });\n }\n\n /**\n * Find all active accounts\n *\n * @returns Array of active accounts\n */\n async findActive(): Promise<Account[]> {\n return await this.list({\n where: { active: true },\n orderBy: 'number ASC',\n });\n }\n\n /**\n * Find top-level accounts (no parent)\n *\n * @returns Array of top-level accounts\n */\n async findTopLevel(): Promise<Account[]> {\n return await this.list({\n where: { parentId: null },\n orderBy: 'number ASC',\n });\n }\n\n /**\n * Find direct children of an account\n *\n * @param parentId - Parent account ID\n * @returns Array of child accounts\n */\n async findChildren(parentId: string): Promise<Account[]> {\n return await this.list({\n where: { parentId },\n orderBy: 'number ASC',\n });\n }\n\n /**\n * Get the complete account tree\n *\n * @returns AccountTree structure\n */\n async getTree(): Promise<AccountTree> {\n const allAccounts = await this.list({\n orderBy: 'number ASC',\n });\n\n const accountMap = new Map<string, Account>();\n const childrenMap = new Map<string, Account[]>();\n\n // Build lookup maps\n for (const account of allAccounts) {\n if (account.id) {\n accountMap.set(account.id, account);\n }\n if (!childrenMap.has(account.parentId || '')) {\n childrenMap.set(account.parentId || '', []);\n }\n childrenMap.get(account.parentId || '')?.push(account);\n }\n\n // Build tree nodes recursively\n const buildNode = (account: Account): AccountTreeNode => {\n const children = (account.id ? childrenMap.get(account.id) : []) || [];\n return {\n account,\n children: children.map(buildNode),\n };\n };\n\n // Get root nodes (no parent)\n const roots = childrenMap.get('') || [];\n\n return {\n roots: roots.map(buildNode),\n };\n }\n\n /**\n * Get or create an account by number\n *\n * @param number - Account number\n * @param defaults - Default values if creating\n * @returns Account\n */\n async getOrCreateByNumber(\n number: string,\n defaults: Partial<{\n name: string;\n description: string;\n type: AccountType;\n parentId: string | null;\n }> = {},\n ): Promise<Account> {\n const existing = await this.findByNumber(number);\n if (existing) {\n return existing;\n }\n\n const account = await this.create({\n number,\n name: defaults.name || number,\n description: defaults.description || '',\n type: defaults.type || 'asset',\n parentId: defaults.parentId || null,\n });\n await account.save();\n return account;\n }\n\n /**\n * Find accounts grouped by type\n *\n * @returns Record of type to accounts array\n */\n async groupByType(): Promise<Record<AccountType, Account[]>> {\n const accounts = await this.findActive();\n const grouped: Record<AccountType, Account[]> = {\n asset: [],\n liability: [],\n equity: [],\n revenue: [],\n expense: [],\n };\n\n for (const account of accounts) {\n grouped[account.type].push(account);\n }\n\n return grouped;\n }\n\n /**\n * Get all descendants of an account recursively\n *\n * @param accountId - Account ID\n * @returns Array of all descendant accounts\n */\n async getDescendants(accountId: string): Promise<Account[]> {\n const descendants: Account[] = [];\n const queue: string[] = [accountId];\n\n while (queue.length > 0) {\n const currentId = queue.shift();\n if (!currentId) continue;\n\n const children = await this.findChildren(currentId);\n\n for (const child of children) {\n descendants.push(child);\n if (child.id) {\n queue.push(child.id);\n }\n }\n }\n\n return descendants;\n }\n\n // ─────────────────────────────────────────────────────────────────────────────\n // Tenant Helper Methods\n // ─────────────────────────────────────────────────────────────────────────────\n\n /**\n * Find all accounts belonging to a specific tenant\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's accounts\n */\n async findByTenant(tenantId: string): Promise<Account[]> {\n return this.list({ where: { tenantId } });\n }\n\n /**\n * Find all global accounts (no tenant association)\n *\n * @returns Array of global accounts\n */\n async findGlobal(): Promise<Account[]> {\n return this.list({ where: { tenantId: null } });\n }\n\n /**\n * Find accounts for a tenant plus all global accounts\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's accounts and global accounts\n */\n async findWithGlobals(tenantId: string): Promise<Account[]> {\n return this.query(\n `SELECT * FROM ${this.tableName} WHERE tenant_id = ? OR tenant_id IS NULL`,\n [tenantId],\n );\n }\n}\n","/**\n * JournalEntry model - Individual ledger line\n *\n * The atomic unit of double-entry accounting - a single debit or credit.\n * Each entry belongs to a Journal and references an Account.\n */\n\nimport { foreignKey, SmrtObject, smrt } from '@happyvertical/smrt-core';\nimport { TenantScoped, tenantId } from '@happyvertical/smrt-tenancy';\nimport type { JournalEntryOptions } from '../types';\n\n@TenantScoped({ mode: 'optional' })\n@smrt({\n api: { include: ['list', 'get'] }, // Created via Journal, not directly\n mcp: { include: ['list', 'get'] },\n cli: true,\n})\nexport class JournalEntry extends SmrtObject {\n /**\n * Tenant ID for multi-tenancy support (nullable for global entries)\n */\n @tenantId({ nullable: true })\n tenantId: string | null = null;\n\n /**\n * Parent journal ID (required)\n */\n @foreignKey('Journal')\n journalId: string = '';\n\n /**\n * Account ID (required)\n */\n @foreignKey('Account')\n accountId: string = '';\n\n /**\n * Debit amount (left side)\n */\n debit: number = 0.0;\n\n /**\n * Credit amount (right side)\n */\n credit: number = 0.0;\n\n /**\n * Currency code (e.g., \"USD\", \"CAD\", \"EUR\")\n */\n currency: string = 'USD';\n\n /**\n * Exchange rate to base currency\n */\n exchangeRate: number = 1.0;\n\n /**\n * Line memo/description\n */\n memo: string = '';\n\n /**\n * Extensible metadata\n */\n metadata: Record<string, unknown> = {};\n\n constructor(options: JournalEntryOptions = {}) {\n super(options);\n if (options.journalId !== undefined) this.journalId = options.journalId;\n if (options.accountId !== undefined) this.accountId = options.accountId;\n if (options.debit !== undefined) this.debit = options.debit;\n if (options.credit !== undefined) this.credit = options.credit;\n if (options.currency !== undefined) this.currency = options.currency;\n if (options.exchangeRate !== undefined)\n this.exchangeRate = options.exchangeRate;\n if (options.memo !== undefined) this.memo = options.memo;\n if (options.metadata !== undefined) this.metadata = options.metadata;\n }\n\n /**\n * Validate entry before save\n */\n protected async validateBeforeSave(): Promise<void> {\n await super.validateBeforeSave();\n // Validate amounts are non-negative\n if (this.debit < 0) {\n throw new Error('Debit amount cannot be negative');\n }\n if (this.credit < 0) {\n throw new Error('Credit amount cannot be negative');\n }\n\n // Validate that entry has debit OR credit, not both\n if (this.debit > 0 && this.credit > 0) {\n throw new Error('Entry cannot have both debit and credit amounts');\n }\n\n // Validate that entry has at least one amount\n if (this.debit === 0 && this.credit === 0) {\n throw new Error('Entry must have either a debit or credit amount');\n }\n\n // Validate exchange rate is positive\n if (this.exchangeRate <= 0) {\n throw new Error('Exchange rate must be positive');\n }\n }\n\n /**\n * Check if this is a debit entry\n */\n isDebit(): boolean {\n return this.debit > 0;\n }\n\n /**\n * Check if this is a credit entry\n */\n isCredit(): boolean {\n return this.credit > 0;\n }\n\n /**\n * Get the entry amount (positive value regardless of debit/credit)\n */\n getAmount(): number {\n return this.debit > 0 ? this.debit : this.credit;\n }\n\n /**\n * Get the amount in base currency\n */\n getBaseAmount(): number {\n return this.getAmount() * this.exchangeRate;\n }\n\n /**\n * Get the parent journal\n */\n async getJournal(): Promise<import('./Journal').Journal | null> {\n if (!this.journalId) return null;\n const { JournalCollection } = await import('../collections/Journals');\n const collection = await (JournalCollection as any).create(this.options);\n return await collection.get({ id: this.journalId });\n }\n\n /**\n * Get the account\n */\n async getAccount(): Promise<import('./Account').Account | null> {\n if (!this.accountId) return null;\n const { AccountCollection } = await import('../collections/Accounts');\n const collection = await (AccountCollection as any).create(this.options);\n return await collection.get({ id: this.accountId });\n }\n\n /**\n * Get a formatted description of this entry\n */\n async getDescription(): Promise<string> {\n const account = await this.getAccount();\n const accountName = account?.name || 'Unknown Account';\n const type = this.isDebit() ? 'DR' : 'CR';\n const amount = this.getAmount().toFixed(2);\n\n return `${type} ${accountName}: $${amount}${this.memo ? ` - ${this.memo}` : ''}`;\n }\n}\n","/**\n * JournalEntryCollection - Collection manager for JournalEntry objects\n *\n * Provides querying and balance calculations for ledger entries.\n */\n\nimport { SmrtCollection } from '@happyvertical/smrt-core';\nimport { JournalEntry } from '../models/JournalEntry';\nimport { BALANCE_EPSILON, type TrialBalanceRow } from '../types';\n\nexport class JournalEntryCollection extends SmrtCollection<JournalEntry> {\n static readonly _itemClass = JournalEntry;\n\n /**\n * Find entries by journal\n *\n * @param journalId - Journal ID\n * @returns Array of entries\n */\n async findByJournal(journalId: string): Promise<JournalEntry[]> {\n return await this.list({\n where: { journalId },\n });\n }\n\n /**\n * Find entries by account\n *\n * @param accountId - Account ID\n * @returns Array of entries\n */\n async findByAccount(accountId: string): Promise<JournalEntry[]> {\n return await this.list({\n where: { accountId },\n });\n }\n\n /**\n * Get account balance\n *\n * For debit-normal accounts (Asset, Expense): balance = debits - credits\n * For credit-normal accounts (Liability, Equity, Revenue): balance = credits - debits\n *\n * @param accountId - Account ID\n * @param asOfDate - Optional date to calculate balance as of\n * @returns Account balance\n */\n async getAccountBalance(accountId: string, asOfDate?: Date): Promise<number> {\n // Get entries for this account from posted journals only\n const { JournalCollection } = await import('./Journals');\n const journalCollection = await (JournalCollection as any).create(\n this.options,\n );\n const { AccountCollection } = await import('./Accounts');\n const accountCollection = await (AccountCollection as any).create(\n this.options,\n );\n\n // Get the account to determine if it's debit or credit normal\n const account = await accountCollection.get({ id: accountId });\n if (!account) {\n throw new Error(`Account not found: ${accountId}`);\n }\n\n // Get all entries for this account\n const allEntries = await this.findByAccount(accountId);\n\n let totalDebits = 0;\n let totalCredits = 0;\n\n for (const entry of allEntries) {\n // Get the journal to check status and date\n const journal = await journalCollection.get({ id: entry.journalId });\n if (!journal) continue;\n\n // Only include posted journals\n if (!journal.isPosted()) continue;\n\n // Check date if specified\n if (asOfDate && journal.date > asOfDate) continue;\n\n totalDebits += entry.debit;\n totalCredits += entry.credit;\n }\n\n // Calculate balance based on account type\n if (account.isDebitNormal()) {\n // Asset, Expense: increases with debits\n return totalDebits - totalCredits;\n } else {\n // Liability, Equity, Revenue: increases with credits\n return totalCredits - totalDebits;\n }\n }\n\n /**\n * Get trial balance\n *\n * @param asOfDate - Optional date for trial balance\n * @returns Array of trial balance rows\n */\n async getTrialBalance(asOfDate?: Date): Promise<TrialBalanceRow[]> {\n const { AccountCollection } = await import('./Accounts');\n const accountCollection = await (AccountCollection as any).create(\n this.options,\n );\n\n const accounts = await accountCollection.findActive();\n const rows: TrialBalanceRow[] = [];\n\n for (const account of accounts) {\n if (!account.id) continue;\n\n const balance = await this.getAccountBalance(account.id, asOfDate);\n\n // Skip zero-balance accounts\n if (Math.abs(balance) < BALANCE_EPSILON) continue;\n\n rows.push({\n accountId: account.id,\n accountNumber: account.number,\n accountName: account.name,\n accountType: account.type,\n debitBalance: account.isDebitNormal() && balance > 0 ? balance : 0,\n creditBalance: account.isCreditNormal() && balance > 0 ? balance : 0,\n });\n }\n\n // Sort by account number\n rows.sort((a, b) => a.accountNumber.localeCompare(b.accountNumber));\n\n return rows;\n }\n\n /**\n * Get total debits and credits for a date range\n *\n * @param start - Start date\n * @param end - End date\n * @returns Object with totalDebits and totalCredits\n */\n async getTotalsForDateRange(\n start: Date,\n end: Date,\n ): Promise<{ totalDebits: number; totalCredits: number }> {\n const { JournalCollection } = await import('./Journals');\n const journalCollection = await (JournalCollection as any).create(\n this.options,\n );\n\n const journals = await journalCollection.findByDateRange(start, end);\n\n let totalDebits = 0;\n let totalCredits = 0;\n\n for (const journal of journals) {\n if (!journal.isPosted()) continue;\n\n const entries = await this.findByJournal(journal.id!);\n for (const entry of entries) {\n totalDebits += entry.debit;\n totalCredits += entry.credit;\n }\n }\n\n return { totalDebits, totalCredits };\n }\n\n /**\n * Get entries for multiple accounts\n *\n * @param accountIds - Array of account IDs\n * @returns Array of entries\n */\n async findByAccounts(accountIds: string[]): Promise<JournalEntry[]> {\n if (accountIds.length === 0) {\n return [];\n }\n\n // Use single query with IN clause to avoid N+1 problem\n return await this.list({\n where: { accountId: accountIds },\n });\n }\n\n /**\n * Get the running balance for an account (list of entries with running total)\n *\n * @param accountId - Account ID\n * @returns Array of entries with running balance\n */\n async getAccountLedger(\n accountId: string,\n ): Promise<Array<{ entry: JournalEntry; runningBalance: number }>> {\n const { JournalCollection } = await import('./Journals');\n const { AccountCollection } = await import('./Accounts');\n const journalCollection = await (JournalCollection as any).create(\n this.options,\n );\n const accountCollection = await (AccountCollection as any).create(\n this.options,\n );\n\n const account = await accountCollection.get({ id: accountId });\n if (!account) {\n throw new Error(`Account not found: ${accountId}`);\n }\n\n const entries = await this.findByAccount(accountId);\n const ledger: Array<{ entry: JournalEntry; runningBalance: number }> = [];\n let runningBalance = 0;\n\n // Sort entries by journal date\n const entriesWithJournals = await Promise.all(\n entries.map(async (entry) => ({\n entry,\n journal: await journalCollection.get({ id: entry.journalId }),\n })),\n );\n\n entriesWithJournals.sort((a, b) => {\n if (!a.journal || !b.journal) return 0;\n return a.journal.date.getTime() - b.journal.date.getTime();\n });\n\n for (const { entry, journal } of entriesWithJournals) {\n if (!journal || !journal.isPosted()) continue;\n\n if (account.isDebitNormal()) {\n runningBalance += entry.debit - entry.credit;\n } else {\n runningBalance += entry.credit - entry.debit;\n }\n\n ledger.push({ entry, runningBalance });\n }\n\n return ledger;\n }\n\n // ─────────────────────────────────────────────────────────────────────────────\n // Tenant Helper Methods\n // ─────────────────────────────────────────────────────────────────────────────\n\n /**\n * Find all journal entries belonging to a specific tenant\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's journal entries\n */\n async findByTenant(tenantId: string): Promise<JournalEntry[]> {\n return this.list({ where: { tenantId } });\n }\n\n /**\n * Find all global journal entries (no tenant association)\n *\n * @returns Array of global journal entries\n */\n async findGlobal(): Promise<JournalEntry[]> {\n return this.list({ where: { tenantId: null } });\n }\n\n /**\n * Find journal entries for a tenant plus all global entries\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's entries and global entries\n */\n async findWithGlobals(tenantId: string): Promise<JournalEntry[]> {\n return this.query(\n `SELECT * FROM ${this.tableName} WHERE tenant_id = ? OR tenant_id IS NULL`,\n [tenantId],\n );\n }\n}\n","/**\n * Journal model - Financial event container\n *\n * Represents a single, discrete financial event.\n * Contains one or more JournalEntry records that must balance (debits = credits).\n */\n\nimport { SmrtObject, smrt } from '@happyvertical/smrt-core';\nimport { resolvePrompt } from '@happyvertical/smrt-prompts';\nimport { TenantScoped, tenantId } from '@happyvertical/smrt-tenancy';\nimport {\n promptMessageOptions,\n smrtLedgersJournalSummarizePrompt,\n} from '../prompts';\nimport {\n BALANCE_EPSILON,\n type JournalEntryData,\n type JournalOptions,\n type JournalStatus,\n} from '../types';\n\n@TenantScoped({ mode: 'optional' })\n@smrt({\n api: { include: ['list', 'get', 'create'] }, // No update/delete - immutable after posting\n mcp: { include: ['list', 'get', 'create'] },\n cli: true,\n})\nexport class Journal extends SmrtObject {\n /**\n * Tenant ID for multi-tenancy support (nullable for global journals)\n */\n @tenantId({ nullable: true })\n tenantId: string | null = null;\n\n /**\n * Journal number (auto-generated sequence, e.g., \"JNL-0001\")\n */\n number: string = '';\n\n /**\n * Transaction date\n */\n date: Date = new Date();\n\n /**\n * Description of the financial event\n */\n description: string = '';\n\n /**\n * Source module that created this journal (e.g., \"smrt-commerce\", \"manual\")\n */\n sourceModule: string = '';\n\n /**\n * External reference (e.g., order ID, invoice number)\n */\n sourceRef: string | null = null;\n\n /**\n * Journal status: draft, posted, voided\n */\n status: JournalStatus = 'draft';\n\n /**\n * When the journal was posted (finalized)\n */\n postedAt: Date | null = null;\n\n /**\n * When the journal was voided\n */\n voidedAt: Date | null = null;\n\n /**\n * Reason for voiding (if voided)\n */\n voidReason: string | null = null;\n\n /**\n * Extensible metadata\n */\n metadata: Record<string, unknown> = {};\n\n constructor(options: JournalOptions = {}) {\n super(options);\n if (options.number !== undefined) this.number = options.number;\n if (options.date !== undefined) this.date = options.date;\n if (options.description !== undefined)\n this.description = options.description;\n if (options.sourceModule !== undefined)\n this.sourceModule = options.sourceModule;\n if (options.sourceRef !== undefined) this.sourceRef = options.sourceRef;\n if (options.status !== undefined) this.status = options.status;\n if (options.postedAt !== undefined) this.postedAt = options.postedAt;\n if (options.voidedAt !== undefined) this.voidedAt = options.voidedAt;\n if (options.voidReason !== undefined) this.voidReason = options.voidReason;\n if (options.metadata !== undefined) this.metadata = options.metadata;\n }\n\n /**\n * Check if journal is in draft status\n */\n isDraft(): boolean {\n return this.status === 'draft';\n }\n\n /**\n * Check if journal has been posted\n */\n isPosted(): boolean {\n return this.status === 'posted';\n }\n\n /**\n * Check if journal has been voided\n */\n isVoided(): boolean {\n return this.status === 'voided';\n }\n\n /**\n * Check if journal can be modified\n */\n isEditable(): boolean {\n return this.status === 'draft';\n }\n\n /**\n * Get all entries for this journal\n */\n async getEntries(): Promise<import('./JournalEntry').JournalEntry[]> {\n if (!this.id) return [];\n const { JournalEntryCollection } = await import(\n '../collections/JournalEntries'\n );\n const collection = await (JournalEntryCollection as any).create(\n this.options,\n );\n return await collection.findByJournal(this.id);\n }\n\n /**\n * Calculate total debits\n */\n async getTotalDebits(): Promise<number> {\n const entries = await this.getEntries();\n return entries.reduce((sum, entry) => sum + entry.debit, 0);\n }\n\n /**\n * Calculate total credits\n */\n async getTotalCredits(): Promise<number> {\n const entries = await this.getEntries();\n return entries.reduce((sum, entry) => sum + entry.credit, 0);\n }\n\n /**\n * Check if journal entries are balanced (debits = credits)\n */\n async isBalanced(): Promise<boolean> {\n const debits = await this.getTotalDebits();\n const credits = await this.getTotalCredits();\n return Math.abs(debits - credits) < BALANCE_EPSILON;\n }\n\n /**\n * Add an entry to this journal (only if draft)\n */\n async addEntry(data: JournalEntryData): Promise<void> {\n if (!this.id) {\n throw new Error('Journal must be saved before adding entries');\n }\n if (!this.isEditable()) {\n throw new Error('Cannot add entries to a posted or voided journal');\n }\n\n const { JournalEntryCollection } = await import(\n '../collections/JournalEntries'\n );\n const collection = await (JournalEntryCollection as any).create(\n this.options,\n );\n const entry = await collection.create({\n journalId: this.id,\n accountId: data.accountId,\n debit: data.debit || 0,\n credit: data.credit || 0,\n currency: data.currency || 'USD',\n exchangeRate: data.exchangeRate || 1.0,\n memo: data.memo || '',\n });\n await entry.save();\n }\n\n /**\n * Post the journal (finalize and make immutable)\n */\n async post(): Promise<void> {\n if (!this.isDraft()) {\n throw new Error('Only draft journals can be posted');\n }\n\n const balanced = await this.isBalanced();\n if (!balanced) {\n const debits = await this.getTotalDebits();\n const credits = await this.getTotalCredits();\n throw new Error(\n `Journal is not balanced. Debits: ${debits}, Credits: ${credits}`,\n );\n }\n\n const entries = await this.getEntries();\n if (entries.length === 0) {\n throw new Error('Journal must have at least one entry');\n }\n\n this.status = 'posted';\n this.postedAt = new Date();\n await this.save();\n }\n\n /**\n * Void the journal (mark as cancelled with reason)\n */\n async void(reason: string): Promise<void> {\n if (this.isVoided()) {\n throw new Error('Journal is already voided');\n }\n\n this.status = 'voided';\n this.voidedAt = new Date();\n this.voidReason = reason;\n await this.save();\n }\n\n /**\n * AI-powered: Generate a summary of this journal.\n *\n * Uses the `smrtLedgers.journal.summarize` prompt registered via\n * `@happyvertical/smrt-prompts`, allowing tenant- or instance-level\n * overrides of the template, model, and parameters at runtime.\n *\n * Only non-PII journal fields (number, date, description, status, balanced\n * flag) plus aggregate totals and entry count are sent to the AI provider.\n * Internal foreign-key fields (tenantId, sourceRef, individual entry\n * account IDs) and the extensible `metadata` blob are intentionally\n * excluded.\n *\n * @returns Generated summary text\n */\n async summarize(): Promise<string> {\n const entries = await this.getEntries();\n const debits = await this.getTotalDebits();\n const balanced = await this.isBalanced();\n\n // Resolve `db` from either the canonical `db` option or its `persistence`\n // alias. SmrtClass maps `persistence → db` lazily during `initialize()`,\n // so on a freshly-constructed Journal that has not yet been initialized,\n // `this.options.db` may be undefined while `this.options.persistence` is\n // set. Falling back here ensures stored app- and tenant-level prompt\n // overrides in `_smrt_prompt_overrides` are honored on the first call —\n // before `getAiClient()` triggers full initialization further below.\n const db = this.options.db ?? this.options.persistence;\n\n const resolvedPrompt = await resolvePrompt(\n smrtLedgersJournalSummarizePrompt.key,\n {\n db,\n tenantId: this.tenantId,\n variables: {\n journalNumber: this.number || '',\n journalDate: this.date.toISOString().split('T')[0],\n journalDescription: this.description || '',\n journalStatus: this.status || '',\n // Currency prefix folded into the value rather than the template\n // — see the `smrtLedgersJournalSummarizePrompt` comment block.\n journalTotal: `$${debits.toFixed(2)}`,\n entryCount: String(entries.length),\n journalBalanced: balanced ? 'Yes' : 'No',\n },\n },\n );\n\n const ai = await this.getAiClient();\n const response = await ai.message(\n resolvedPrompt.text,\n promptMessageOptions(resolvedPrompt.ai),\n );\n\n return response.trim();\n }\n}\n","/**\n * JournalCollection - Collection manager for Journal objects\n *\n * Provides querying and operations for financial journals.\n */\n\nimport { SmrtCollection } from '@happyvertical/smrt-core';\nimport { Journal } from '../models/Journal';\nimport {\n BALANCE_EPSILON,\n type CreateJournalData,\n type JournalStatus,\n} from '../types';\n\nexport class JournalCollection extends SmrtCollection<Journal> {\n static readonly _itemClass = Journal;\n\n /**\n * Generate next journal number\n *\n * Uses timestamp and random component to avoid collisions\n * in concurrent environments without relying on shared state.\n */\n private generateJournalNumber(): string {\n const timestamp = Date.now().toString(36);\n const randomPart = Math.random().toString(36).slice(2, 6);\n return `JNL-${timestamp}-${randomPart}`;\n }\n\n /**\n * Find journal by number\n *\n * @param number - Journal number\n * @returns Journal or null\n */\n async findByNumber(number: string): Promise<Journal | null> {\n const journals = await this.list({\n where: { number },\n limit: 1,\n });\n return journals[0] || null;\n }\n\n /**\n * Find journals by date range\n *\n * @param start - Start date\n * @param end - End date\n * @returns Array of journals\n */\n async findByDateRange(start: Date, end: Date): Promise<Journal[]> {\n return await this.list({\n where: {\n 'date >=': start.toISOString(),\n 'date <=': end.toISOString(),\n },\n orderBy: 'date ASC',\n });\n }\n\n /**\n * Find journals by source module\n *\n * @param sourceModule - Source module name\n * @returns Array of journals\n */\n async findBySource(sourceModule: string): Promise<Journal[]> {\n return await this.list({\n where: { sourceModule },\n orderBy: 'date DESC',\n });\n }\n\n /**\n * Find journals by status\n *\n * @param status - Journal status\n * @returns Array of journals\n */\n async findByStatus(status: JournalStatus): Promise<Journal[]> {\n return await this.list({\n where: { status },\n orderBy: 'date DESC',\n });\n }\n\n /**\n * Find draft journals\n *\n * @returns Array of draft journals\n */\n async findDrafts(): Promise<Journal[]> {\n return await this.findByStatus('draft');\n }\n\n /**\n * Find posted journals\n *\n * @returns Array of posted journals\n */\n async findPosted(): Promise<Journal[]> {\n return await this.findByStatus('posted');\n }\n\n /**\n * Create a complete journal with entries\n *\n * @param data - Journal data with entries\n * @returns Created journal\n */\n async createWithEntries(data: CreateJournalData): Promise<Journal> {\n // Validate entries balance\n let totalDebits = 0;\n let totalCredits = 0;\n\n for (const entry of data.entries) {\n totalDebits += entry.debit || 0;\n totalCredits += entry.credit || 0;\n }\n\n if (Math.abs(totalDebits - totalCredits) >= BALANCE_EPSILON) {\n throw new Error(\n `Entries are not balanced. Debits: ${totalDebits}, Credits: ${totalCredits}`,\n );\n }\n\n if (data.entries.length === 0) {\n throw new Error('Journal must have at least one entry');\n }\n\n // Create journal\n const journalNumber = await this.generateJournalNumber();\n const journal = await this.create({\n number: journalNumber,\n date: data.date || new Date(),\n description: data.description,\n sourceModule: data.sourceModule || 'manual',\n sourceRef: data.sourceRef || null,\n metadata: data.metadata || {},\n });\n await journal.save();\n\n // Create entries\n for (const entryData of data.entries) {\n await journal.addEntry(entryData);\n }\n\n return journal;\n }\n\n /**\n * Post a journal by ID\n *\n * @param journalId - Journal ID\n * @returns Posted journal\n */\n async post(journalId: string): Promise<Journal> {\n const journal = await this.get({ id: journalId });\n if (!journal) {\n throw new Error(`Journal not found: ${journalId}`);\n }\n await journal.post();\n return journal;\n }\n\n /**\n * Void a journal by ID\n *\n * @param journalId - Journal ID\n * @param reason - Reason for voiding\n * @returns Voided journal\n */\n async void(journalId: string, reason: string): Promise<Journal> {\n const journal = await this.get({ id: journalId });\n if (!journal) {\n throw new Error(`Journal not found: ${journalId}`);\n }\n await journal.void(reason);\n return journal;\n }\n\n /**\n * Find journals by source reference\n *\n * @param sourceRef - External reference\n * @returns Array of journals\n */\n async findBySourceRef(sourceRef: string): Promise<Journal[]> {\n return await this.list({\n where: { sourceRef },\n orderBy: 'date DESC',\n });\n }\n\n /**\n * Get journals for a specific month\n *\n * @param year - Year\n * @param month - Month (1-12)\n * @returns Array of journals\n */\n async findByMonth(year: number, month: number): Promise<Journal[]> {\n const start = new Date(year, month - 1, 1);\n const end = new Date(year, month, 0, 23, 59, 59, 999);\n return await this.findByDateRange(start, end);\n }\n\n // ─────────────────────────────────────────────────────────────────────────────\n // Tenant Helper Methods\n // ─────────────────────────────────────────────────────────────────────────────\n\n /**\n * Find all journals belonging to a specific tenant\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's journals\n */\n async findByTenant(tenantId: string): Promise<Journal[]> {\n return this.list({ where: { tenantId } });\n }\n\n /**\n * Find all global journals (no tenant association)\n *\n * @returns Array of global journals\n */\n async findGlobal(): Promise<Journal[]> {\n return this.list({ where: { tenantId: null } });\n }\n\n /**\n * Find journals for a tenant plus all global journals\n *\n * @param tenantId - Tenant ID\n * @returns Array of tenant's journals and global journals\n */\n async findWithGlobals(tenantId: string): Promise<Journal[]> {\n return this.query(\n `SELECT * FROM ${this.tableName} WHERE tenant_id = ? OR tenant_id IS NULL`,\n [tenantId],\n );\n }\n}\n"],"names":["JournalEntryCollection","AccountCollection","__decorateClass","tenantId","JournalCollection"],"mappings":"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"}
|