@voyant-travel/finance 0.119.5

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.
Files changed (294) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +192 -0
  3. package/dist/action-ledger-drift.d.ts +29 -0
  4. package/dist/action-ledger-drift.d.ts.map +1 -0
  5. package/dist/action-ledger-drift.js +166 -0
  6. package/dist/booking-tax.d.ts +124 -0
  7. package/dist/booking-tax.d.ts.map +1 -0
  8. package/dist/booking-tax.js +264 -0
  9. package/dist/checkout-routes.d.ts +1154 -0
  10. package/dist/checkout-routes.d.ts.map +1 -0
  11. package/dist/checkout-routes.js +116 -0
  12. package/dist/checkout-service-plan.d.ts +137 -0
  13. package/dist/checkout-service-plan.d.ts.map +1 -0
  14. package/dist/checkout-service-plan.js +119 -0
  15. package/dist/checkout-service.d.ts +9 -0
  16. package/dist/checkout-service.d.ts.map +1 -0
  17. package/dist/checkout-service.js +324 -0
  18. package/dist/checkout-validation.d.ts +1682 -0
  19. package/dist/checkout-validation.d.ts.map +1 -0
  20. package/dist/checkout-validation.js +228 -0
  21. package/dist/document-download.d.ts +3 -0
  22. package/dist/document-download.d.ts.map +1 -0
  23. package/dist/document-download.js +1 -0
  24. package/dist/fx-money.d.ts +17 -0
  25. package/dist/fx-money.d.ts.map +1 -0
  26. package/dist/fx-money.js +194 -0
  27. package/dist/index.d.ts +65 -0
  28. package/dist/index.d.ts.map +1 -0
  29. package/dist/index.js +108 -0
  30. package/dist/invoice-fx.d.ts +134 -0
  31. package/dist/invoice-fx.d.ts.map +1 -0
  32. package/dist/invoice-fx.js +240 -0
  33. package/dist/invoice-number-errors.d.ts +2 -0
  34. package/dist/invoice-number-errors.d.ts.map +1 -0
  35. package/dist/invoice-number-errors.js +58 -0
  36. package/dist/markets-ref.d.ts +149 -0
  37. package/dist/markets-ref.d.ts.map +1 -0
  38. package/dist/markets-ref.js +17 -0
  39. package/dist/payment-link.d.ts +23 -0
  40. package/dist/payment-link.d.ts.map +1 -0
  41. package/dist/payment-link.js +30 -0
  42. package/dist/payment-policy.d.ts +113 -0
  43. package/dist/payment-policy.d.ts.map +1 -0
  44. package/dist/payment-policy.js +193 -0
  45. package/dist/route-runtime.d.ts +22 -0
  46. package/dist/route-runtime.d.ts.map +1 -0
  47. package/dist/route-runtime.js +18 -0
  48. package/dist/routes-action-ledger.d.ts +181 -0
  49. package/dist/routes-action-ledger.d.ts.map +1 -0
  50. package/dist/routes-action-ledger.js +142 -0
  51. package/dist/routes-booking-billing.d.ts +852 -0
  52. package/dist/routes-booking-billing.d.ts.map +1 -0
  53. package/dist/routes-booking-billing.js +223 -0
  54. package/dist/routes-booking-create.d.ts +3 -0
  55. package/dist/routes-booking-create.d.ts.map +1 -0
  56. package/dist/routes-booking-create.js +194 -0
  57. package/dist/routes-booking-reads.d.ts +46 -0
  58. package/dist/routes-booking-reads.d.ts.map +1 -0
  59. package/dist/routes-booking-reads.js +20 -0
  60. package/dist/routes-documents.d.ts +195 -0
  61. package/dist/routes-documents.d.ts.map +1 -0
  62. package/dist/routes-documents.js +93 -0
  63. package/dist/routes-invoice-core.d.ts +794 -0
  64. package/dist/routes-invoice-core.d.ts.map +1 -0
  65. package/dist/routes-invoice-core.js +238 -0
  66. package/dist/routes-invoice-documents.d.ts +401 -0
  67. package/dist/routes-invoice-documents.d.ts.map +1 -0
  68. package/dist/routes-invoice-documents.js +91 -0
  69. package/dist/routes-invoice-issue.d.ts +384 -0
  70. package/dist/routes-invoice-issue.d.ts.map +1 -0
  71. package/dist/routes-invoice-issue.js +208 -0
  72. package/dist/routes-payment-processing.d.ts +1193 -0
  73. package/dist/routes-payment-processing.d.ts.map +1 -0
  74. package/dist/routes-payment-processing.js +238 -0
  75. package/dist/routes-payments.d.ts +309 -0
  76. package/dist/routes-payments.d.ts.map +1 -0
  77. package/dist/routes-payments.js +94 -0
  78. package/dist/routes-public.d.ts +1948 -0
  79. package/dist/routes-public.d.ts.map +1 -0
  80. package/dist/routes-public.js +275 -0
  81. package/dist/routes-reference-data.d.ts +977 -0
  82. package/dist/routes-reference-data.d.ts.map +1 -0
  83. package/dist/routes-reference-data.js +191 -0
  84. package/dist/routes-reports.d.ts +344 -0
  85. package/dist/routes-reports.d.ts.map +1 -0
  86. package/dist/routes-reports.js +93 -0
  87. package/dist/routes-runtime.d.ts +71 -0
  88. package/dist/routes-runtime.d.ts.map +1 -0
  89. package/dist/routes-runtime.js +59 -0
  90. package/dist/routes-settlement.d.ts +67 -0
  91. package/dist/routes-settlement.d.ts.map +1 -0
  92. package/dist/routes-settlement.js +23 -0
  93. package/dist/routes-shared.d.ts +35 -0
  94. package/dist/routes-shared.d.ts.map +1 -0
  95. package/dist/routes-shared.js +10 -0
  96. package/dist/routes-supplier-invoices.d.ts +778 -0
  97. package/dist/routes-supplier-invoices.d.ts.map +1 -0
  98. package/dist/routes-supplier-invoices.js +159 -0
  99. package/dist/routes-vouchers.d.ts +228 -0
  100. package/dist/routes-vouchers.d.ts.map +1 -0
  101. package/dist/routes-vouchers.js +54 -0
  102. package/dist/routes.d.ts +5577 -0
  103. package/dist/routes.d.ts.map +1 -0
  104. package/dist/routes.js +44 -0
  105. package/dist/schema/booking-billing.d.ts +1006 -0
  106. package/dist/schema/booking-billing.d.ts.map +1 -0
  107. package/dist/schema/booking-billing.js +106 -0
  108. package/dist/schema/enums.d.ts +48 -0
  109. package/dist/schema/enums.d.ts.map +1 -0
  110. package/dist/schema/enums.js +237 -0
  111. package/dist/schema/invoice-documents.d.ts +1245 -0
  112. package/dist/schema/invoice-documents.d.ts.map +1 -0
  113. package/dist/schema/invoice-documents.js +140 -0
  114. package/dist/schema/payment-instruments.d.ts +418 -0
  115. package/dist/schema/payment-instruments.d.ts.map +1 -0
  116. package/dist/schema/payment-instruments.js +45 -0
  117. package/dist/schema/payment-processing.d.ts +563 -0
  118. package/dist/schema/payment-processing.d.ts.map +1 -0
  119. package/dist/schema/payment-processing.js +65 -0
  120. package/dist/schema/payment-sessions.d.ts +728 -0
  121. package/dist/schema/payment-sessions.d.ts.map +1 -0
  122. package/dist/schema/payment-sessions.js +79 -0
  123. package/dist/schema/receivables.d.ts +1474 -0
  124. package/dist/schema/receivables.d.ts.map +1 -0
  125. package/dist/schema/receivables.js +179 -0
  126. package/dist/schema/relations.d.ts +82 -0
  127. package/dist/schema/relations.d.ts.map +1 -0
  128. package/dist/schema/relations.js +144 -0
  129. package/dist/schema/supplier-invoices.d.ts +1619 -0
  130. package/dist/schema/supplier-invoices.d.ts.map +1 -0
  131. package/dist/schema/supplier-invoices.js +228 -0
  132. package/dist/schema/tax.d.ts +712 -0
  133. package/dist/schema/tax.d.ts.map +1 -0
  134. package/dist/schema/tax.js +98 -0
  135. package/dist/schema/vouchers.d.ts +444 -0
  136. package/dist/schema/vouchers.d.ts.map +1 -0
  137. package/dist/schema/vouchers.js +64 -0
  138. package/dist/schema.d.ts +12 -0
  139. package/dist/schema.d.ts.map +1 -0
  140. package/dist/schema.js +11 -0
  141. package/dist/service-accountant-shares.d.ts +106 -0
  142. package/dist/service-accountant-shares.d.ts.map +1 -0
  143. package/dist/service-accountant-shares.js +331 -0
  144. package/dist/service-action-ledger-accounting.d.ts +104 -0
  145. package/dist/service-action-ledger-accounting.d.ts.map +1 -0
  146. package/dist/service-action-ledger-accounting.js +386 -0
  147. package/dist/service-action-ledger-booking-payments.d.ts +48 -0
  148. package/dist/service-action-ledger-booking-payments.d.ts.map +1 -0
  149. package/dist/service-action-ledger-booking-payments.js +178 -0
  150. package/dist/service-action-ledger-bookings.d.ts +44 -0
  151. package/dist/service-action-ledger-bookings.d.ts.map +1 -0
  152. package/dist/service-action-ledger-bookings.js +81 -0
  153. package/dist/service-action-ledger-payment-authorizations.d.ts +48 -0
  154. package/dist/service-action-ledger-payment-authorizations.d.ts.map +1 -0
  155. package/dist/service-action-ledger-payment-authorizations.js +209 -0
  156. package/dist/service-action-ledger-payment-sessions.d.ts +83 -0
  157. package/dist/service-action-ledger-payment-sessions.d.ts.map +1 -0
  158. package/dist/service-action-ledger-payment-sessions.js +294 -0
  159. package/dist/service-action-ledger-supplier-invoices.d.ts +27 -0
  160. package/dist/service-action-ledger-supplier-invoices.d.ts.map +1 -0
  161. package/dist/service-action-ledger-supplier-invoices.js +111 -0
  162. package/dist/service-action-ledger-supplier-payments.d.ts +21 -0
  163. package/dist/service-action-ledger-supplier-payments.d.ts.map +1 -0
  164. package/dist/service-action-ledger-supplier-payments.js +97 -0
  165. package/dist/service-action-ledger.d.ts +7 -0
  166. package/dist/service-action-ledger.d.ts.map +1 -0
  167. package/dist/service-action-ledger.js +6 -0
  168. package/dist/service-aggregates.d.ts +96 -0
  169. package/dist/service-aggregates.d.ts.map +1 -0
  170. package/dist/service-aggregates.js +294 -0
  171. package/dist/service-booking-billing.d.ts +2322 -0
  172. package/dist/service-booking-billing.d.ts.map +1 -0
  173. package/dist/service-booking-billing.js +8 -0
  174. package/dist/service-booking-create.d.ts +410 -0
  175. package/dist/service-booking-create.d.ts.map +1 -0
  176. package/dist/service-booking-create.js +1256 -0
  177. package/dist/service-booking-guarantees.d.ts +725 -0
  178. package/dist/service-booking-guarantees.d.ts.map +1 -0
  179. package/dist/service-booking-guarantees.js +153 -0
  180. package/dist/service-booking-item-billing.d.ts +1062 -0
  181. package/dist/service-booking-item-billing.d.ts.map +1 -0
  182. package/dist/service-booking-item-billing.js +77 -0
  183. package/dist/service-booking-payment-schedules.d.ts +557 -0
  184. package/dist/service-booking-payment-schedules.d.ts.map +1 -0
  185. package/dist/service-booking-payment-schedules.js +372 -0
  186. package/dist/service-bookings-dual-create.d.ts +308 -0
  187. package/dist/service-bookings-dual-create.d.ts.map +1 -0
  188. package/dist/service-bookings-dual-create.js +131 -0
  189. package/dist/service-boundary-sql.d.ts +6 -0
  190. package/dist/service-boundary-sql.d.ts.map +1 -0
  191. package/dist/service-boundary-sql.js +15 -0
  192. package/dist/service-cost-categories.d.ts +26 -0
  193. package/dist/service-cost-categories.d.ts.map +1 -0
  194. package/dist/service-cost-categories.js +76 -0
  195. package/dist/service-documents.d.ts +80 -0
  196. package/dist/service-documents.d.ts.map +1 -0
  197. package/dist/service-documents.js +228 -0
  198. package/dist/service-invoice-artifacts.d.ts +246 -0
  199. package/dist/service-invoice-artifacts.d.ts.map +1 -0
  200. package/dist/service-invoice-artifacts.js +277 -0
  201. package/dist/service-invoice-core.d.ts +405 -0
  202. package/dist/service-invoice-core.d.ts.map +1 -0
  203. package/dist/service-invoice-core.js +290 -0
  204. package/dist/service-invoice-credit-notes.d.ts +973 -0
  205. package/dist/service-invoice-credit-notes.d.ts.map +1 -0
  206. package/dist/service-invoice-credit-notes.js +142 -0
  207. package/dist/service-invoice-from-booking.d.ts +41 -0
  208. package/dist/service-invoice-from-booking.d.ts.map +1 -0
  209. package/dist/service-invoice-from-booking.js +267 -0
  210. package/dist/service-invoice-line-items.d.ts +432 -0
  211. package/dist/service-invoice-line-items.d.ts.map +1 -0
  212. package/dist/service-invoice-line-items.js +102 -0
  213. package/dist/service-invoice-numbering.d.ts +227 -0
  214. package/dist/service-invoice-numbering.d.ts.map +1 -0
  215. package/dist/service-invoice-numbering.js +260 -0
  216. package/dist/service-invoice-payments.d.ts +673 -0
  217. package/dist/service-invoice-payments.d.ts.map +1 -0
  218. package/dist/service-invoice-payments.js +398 -0
  219. package/dist/service-invoices.d.ts +2501 -0
  220. package/dist/service-invoices.d.ts.map +1 -0
  221. package/dist/service-invoices.js +12 -0
  222. package/dist/service-issue.d.ts +207 -0
  223. package/dist/service-issue.d.ts.map +1 -0
  224. package/dist/service-issue.js +431 -0
  225. package/dist/service-payment-authorizations.d.ts +164 -0
  226. package/dist/service-payment-authorizations.d.ts.map +1 -0
  227. package/dist/service-payment-authorizations.js +227 -0
  228. package/dist/service-payment-instruments.d.ts +116 -0
  229. package/dist/service-payment-instruments.d.ts.map +1 -0
  230. package/dist/service-payment-instruments.js +99 -0
  231. package/dist/service-payment-processing.d.ts +676 -0
  232. package/dist/service-payment-processing.d.ts.map +1 -0
  233. package/dist/service-payment-processing.js +10 -0
  234. package/dist/service-payment-session-completion.d.ts +48 -0
  235. package/dist/service-payment-session-completion.d.ts.map +1 -0
  236. package/dist/service-payment-session-completion.js +238 -0
  237. package/dist/service-payment-sessions.d.ts +361 -0
  238. package/dist/service-payment-sessions.d.ts.map +1 -0
  239. package/dist/service-payment-sessions.js +280 -0
  240. package/dist/service-profitability.d.ts +114 -0
  241. package/dist/service-profitability.d.ts.map +1 -0
  242. package/dist/service-profitability.js +794 -0
  243. package/dist/service-public.d.ts +553 -0
  244. package/dist/service-public.d.ts.map +1 -0
  245. package/dist/service-public.js +583 -0
  246. package/dist/service-reference-data.d.ts +272 -0
  247. package/dist/service-reference-data.d.ts.map +1 -0
  248. package/dist/service-reference-data.js +280 -0
  249. package/dist/service-rendition-wait.d.ts +38 -0
  250. package/dist/service-rendition-wait.d.ts.map +1 -0
  251. package/dist/service-rendition-wait.js +67 -0
  252. package/dist/service-reports.d.ts +37 -0
  253. package/dist/service-reports.d.ts.map +1 -0
  254. package/dist/service-reports.js +62 -0
  255. package/dist/service-settlement.d.ts +46 -0
  256. package/dist/service-settlement.d.ts.map +1 -0
  257. package/dist/service-settlement.js +185 -0
  258. package/dist/service-shared.d.ts +541 -0
  259. package/dist/service-shared.d.ts.map +1 -0
  260. package/dist/service-shared.js +764 -0
  261. package/dist/service-supplier-invoices.d.ts +871 -0
  262. package/dist/service-supplier-invoices.d.ts.map +1 -0
  263. package/dist/service-supplier-invoices.js +744 -0
  264. package/dist/service-supplier-payments.d.ts +69 -0
  265. package/dist/service-supplier-payments.d.ts.map +1 -0
  266. package/dist/service-supplier-payments.js +136 -0
  267. package/dist/service-vouchers-migration.d.ts +44 -0
  268. package/dist/service-vouchers-migration.d.ts.map +1 -0
  269. package/dist/service-vouchers-migration.js +148 -0
  270. package/dist/service-vouchers.d.ts +157 -0
  271. package/dist/service-vouchers.d.ts.map +1 -0
  272. package/dist/service-vouchers.js +191 -0
  273. package/dist/service.d.ts +6490 -0
  274. package/dist/service.d.ts.map +1 -0
  275. package/dist/service.js +29 -0
  276. package/dist/validation-billing.d.ts +2 -0
  277. package/dist/validation-billing.d.ts.map +1 -0
  278. package/dist/validation-billing.js +1 -0
  279. package/dist/validation-payments.d.ts +2 -0
  280. package/dist/validation-payments.d.ts.map +1 -0
  281. package/dist/validation-payments.js +1 -0
  282. package/dist/validation-public.d.ts +2 -0
  283. package/dist/validation-public.d.ts.map +1 -0
  284. package/dist/validation-public.js +1 -0
  285. package/dist/validation-shared.d.ts +2 -0
  286. package/dist/validation-shared.d.ts.map +1 -0
  287. package/dist/validation-shared.js +1 -0
  288. package/dist/validation-vouchers.d.ts +2 -0
  289. package/dist/validation-vouchers.d.ts.map +1 -0
  290. package/dist/validation-vouchers.js +1 -0
  291. package/dist/validation.d.ts +2 -0
  292. package/dist/validation.d.ts.map +1 -0
  293. package/dist/validation.js +1 -0
  294. package/package.json +121 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"service-aggregates.d.ts","sourceRoot":"","sources":["../src/service-aggregates.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,294 @@
1
+ import { and, asc, inArray, ne, sql } from "drizzle-orm";
2
+ import { invoices, paymentSessions, payments } from "./schema.js";
3
+ const ALL_INVOICE_STATUSES = [
4
+ "draft",
5
+ "pending_external_allocation",
6
+ "issued",
7
+ "partially_paid",
8
+ "paid",
9
+ "overdue",
10
+ "void",
11
+ ];
12
+ /** Statuses where balance_due_cents > 0 is meaningful money we're owed. */
13
+ const OUTSTANDING_STATUSES = ["issued", "partially_paid", "overdue"];
14
+ export async function getFinanceAggregates(db, options = {}) {
15
+ const outstandingTopLimit = Math.max(0, Math.min(options.outstandingTopLimit ?? 5, 20));
16
+ const { fromDate, toDate } = resolveAggregateRange(options);
17
+ const rangeConditions = [];
18
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
19
+ if (fromDate)
20
+ rangeConditions.push(sql `${invoices.createdAt} >= ${fromDate.toISOString()}`);
21
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
22
+ if (toDate)
23
+ rangeConditions.push(sql `${invoices.createdAt} < ${toDate.toISOString()}`);
24
+ const invoiceConditions = [...rangeConditions];
25
+ if (options.currency?.length)
26
+ invoiceConditions.push(inArray(invoices.currency, options.currency));
27
+ if (options.invoiceType?.length) {
28
+ invoiceConditions.push(inArray(invoices.invoiceType, options.invoiceType));
29
+ }
30
+ if (options.status?.length)
31
+ invoiceConditions.push(inArray(invoices.status, options.status));
32
+ const rangeWhere = invoiceConditions.length ? and(...invoiceConditions) : undefined;
33
+ const paymentConditions = [];
34
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
35
+ if (fromDate)
36
+ paymentConditions.push(sql `${payments.paymentDate} >= ${dateOnly(fromDate)}`);
37
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
38
+ if (toDate)
39
+ paymentConditions.push(sql `${payments.paymentDate} < ${dateOnly(toDate)}`);
40
+ if (options.currency?.length)
41
+ paymentConditions.push(inArray(payments.currency, options.currency));
42
+ const paymentWhere = paymentConditions.length ? and(...paymentConditions) : undefined;
43
+ const sessionConditions = [];
44
+ if (fromDate)
45
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
46
+ sessionConditions.push(sql `${paymentSessions.createdAt} >= ${fromDate.toISOString()}`);
47
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
48
+ if (toDate)
49
+ sessionConditions.push(sql `${paymentSessions.createdAt} < ${toDate.toISOString()}`);
50
+ if (options.currency?.length) {
51
+ sessionConditions.push(inArray(paymentSessions.currency, options.currency));
52
+ }
53
+ const sessionWhere = sessionConditions.length ? and(...sessionConditions) : undefined;
54
+ // Outstanding + overdue always look at the whole book (not the date range),
55
+ // since "what are we owed right now" is a point-in-time question — bounding
56
+ // it by `from..to` would hide old unpaid invoices.
57
+ const todayUtc = new Date();
58
+ todayUtc.setUTCHours(0, 0, 0, 0);
59
+ const todayDateString = todayUtc.toISOString().slice(0, 10);
60
+ const outstandingWhere = and(inArray(invoices.status, [...OUTSTANDING_STATUSES]),
61
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
62
+ sql `${invoices.balanceDueCents} > 0`);
63
+ const [[totalRow], statusRows, monthlyInvoiceCountsRows, monthlyRevenueRows, invoiceSummaryRows, invoiceTotalsRows, paymentTotalsRows, paymentSessionRows, outstandingRows, overdueRows, outstandingTopRows,] = await Promise.all([
64
+ db.select({ count: sql `count(*)::int` }).from(invoices).where(rangeWhere),
65
+ db
66
+ .select({
67
+ status: invoices.status,
68
+ count: sql `count(*)::int`,
69
+ })
70
+ .from(invoices)
71
+ .where(rangeWhere)
72
+ .groupBy(invoices.status),
73
+ db
74
+ .select({
75
+ yearMonth: sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`,
76
+ count: sql `count(*)::int`,
77
+ })
78
+ .from(invoices)
79
+ .where(rangeWhere)
80
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
81
+ .groupBy(sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`)
82
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
83
+ .orderBy(sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`),
84
+ db
85
+ .select({
86
+ yearMonth: sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`,
87
+ currency: invoices.currency,
88
+ totalCents: sql `coalesce(sum(${invoices.totalCents}), 0)::bigint`,
89
+ })
90
+ .from(invoices)
91
+ .where(and(...invoiceConditions, ne(invoices.status, "void")))
92
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
93
+ .groupBy(sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`, invoices.currency)
94
+ .orderBy(
95
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
96
+ sql `to_char(${invoices.createdAt} at time zone 'UTC', 'YYYY-MM')`, invoices.currency),
97
+ db
98
+ .select({
99
+ invoiceType: invoices.invoiceType,
100
+ status: invoices.status,
101
+ count: sql `count(*)::int`,
102
+ })
103
+ .from(invoices)
104
+ .where(rangeWhere)
105
+ .groupBy(invoices.invoiceType, invoices.status),
106
+ db
107
+ .select({
108
+ currency: invoices.currency,
109
+ invoiced: sql `coalesce(sum(case when ${invoices.status} != 'void' then ${invoices.totalCents} else 0 end), 0)::bigint`,
110
+ outstanding: sql `coalesce(sum(case when ${invoices.status} not in ('paid', 'void') then ${invoices.balanceDueCents} else 0 end), 0)::bigint`,
111
+ })
112
+ .from(invoices)
113
+ .where(rangeWhere)
114
+ .groupBy(invoices.currency)
115
+ .orderBy(invoices.currency),
116
+ db
117
+ .select({
118
+ currency: payments.currency,
119
+ collected: sql `coalesce(sum(case when ${payments.status} = 'completed' then ${payments.amountCents} else 0 end), 0)::bigint`,
120
+ refunded: sql `coalesce(sum(case when ${payments.status} = 'refunded' then ${payments.amountCents} else 0 end), 0)::bigint`,
121
+ })
122
+ .from(payments)
123
+ .where(paymentWhere)
124
+ .groupBy(payments.currency)
125
+ .orderBy(payments.currency),
126
+ db
127
+ .select({
128
+ status: paymentSessions.status,
129
+ count: sql `count(*)::int`,
130
+ })
131
+ .from(paymentSessions)
132
+ .where(sessionWhere)
133
+ .groupBy(paymentSessions.status),
134
+ db
135
+ .select({
136
+ currency: invoices.currency,
137
+ balanceDueCents: sql `coalesce(sum(${invoices.balanceDueCents}), 0)::bigint`,
138
+ count: sql `count(*)::int`,
139
+ })
140
+ .from(invoices)
141
+ .where(outstandingWhere)
142
+ .groupBy(invoices.currency)
143
+ .orderBy(invoices.currency),
144
+ db
145
+ .select({
146
+ currency: invoices.currency,
147
+ balanceDueCents: sql `coalesce(sum(${invoices.balanceDueCents}), 0)::bigint`,
148
+ count: sql `count(*)::int`,
149
+ })
150
+ .from(invoices)
151
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
152
+ .where(and(outstandingWhere, sql `${invoices.dueDate} < ${todayDateString}`))
153
+ .groupBy(invoices.currency)
154
+ .orderBy(invoices.currency),
155
+ outstandingTopLimit === 0
156
+ ? Promise.resolve([])
157
+ : db
158
+ .select({
159
+ id: invoices.id,
160
+ invoiceNumber: invoices.invoiceNumber,
161
+ bookingId: invoices.bookingId,
162
+ status: invoices.status,
163
+ currency: invoices.currency,
164
+ totalCents: invoices.totalCents,
165
+ balanceDueCents: invoices.balanceDueCents,
166
+ issueDate: invoices.issueDate,
167
+ dueDate: invoices.dueDate,
168
+ })
169
+ .from(invoices)
170
+ .where(outstandingWhere)
171
+ // Nulls-last on dueDate so undated invoices don't pretend to be
172
+ // the most overdue. After that, oldest issued first.
173
+ .orderBy(
174
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
175
+ sql `${invoices.dueDate} IS NULL`, asc(invoices.dueDate), asc(invoices.issueDate), asc(invoices.id))
176
+ .limit(outstandingTopLimit),
177
+ ]);
178
+ const countsByStatusMap = new Map(statusRows.map((row) => [row.status, row.count]));
179
+ return {
180
+ total: totalRow?.count ?? 0,
181
+ countsByStatus: ALL_INVOICE_STATUSES.map((status) => ({
182
+ status,
183
+ count: countsByStatusMap.get(status) ?? 0,
184
+ })),
185
+ counts: financeCounts(invoiceSummaryRows, paymentSessionRows),
186
+ totals: financeTotals(invoiceTotalsRows, paymentTotalsRows),
187
+ monthlyRevenue: monthlyRevenueRows.map((row) => ({
188
+ yearMonth: row.yearMonth,
189
+ currency: row.currency,
190
+ totalCents: Number(row.totalCents),
191
+ })),
192
+ monthlyInvoiceCounts: monthlyInvoiceCountsRows.map((row) => ({
193
+ yearMonth: row.yearMonth,
194
+ count: row.count,
195
+ })),
196
+ outstanding: outstandingRows.map((row) => ({
197
+ currency: row.currency,
198
+ balanceDueCents: Number(row.balanceDueCents),
199
+ count: row.count,
200
+ })),
201
+ overdue: overdueRows.map((row) => ({
202
+ currency: row.currency,
203
+ balanceDueCents: Number(row.balanceDueCents),
204
+ count: row.count,
205
+ })),
206
+ outstandingTopN: outstandingTopRows.map((row) => ({
207
+ id: row.id,
208
+ invoiceNumber: row.invoiceNumber ?? null,
209
+ bookingId: row.bookingId ?? null,
210
+ status: row.status,
211
+ currency: row.currency,
212
+ totalCents: Number(row.totalCents),
213
+ balanceDueCents: Number(row.balanceDueCents),
214
+ issueDate: row.issueDate ?? null,
215
+ dueDate: row.dueDate ?? null,
216
+ })),
217
+ };
218
+ }
219
+ function resolveAggregateRange(options) {
220
+ if (options.from || options.to || options.range === "custom") {
221
+ return {
222
+ fromDate: options.from ? new Date(options.from) : undefined,
223
+ toDate: options.to ? new Date(options.to) : undefined,
224
+ };
225
+ }
226
+ const now = new Date();
227
+ const year = now.getUTCFullYear();
228
+ const month = now.getUTCMonth();
229
+ if (options.range === "this_month") {
230
+ return {
231
+ fromDate: new Date(Date.UTC(year, month, 1)),
232
+ toDate: new Date(Date.UTC(year, month + 1, 1)),
233
+ };
234
+ }
235
+ if (options.range === "last_month") {
236
+ return {
237
+ fromDate: new Date(Date.UTC(year, month - 1, 1)),
238
+ toDate: new Date(Date.UTC(year, month, 1)),
239
+ };
240
+ }
241
+ if (options.range === "year_to_date") {
242
+ return {
243
+ fromDate: new Date(Date.UTC(year, 0, 1)),
244
+ toDate: undefined,
245
+ };
246
+ }
247
+ return { fromDate: undefined, toDate: undefined };
248
+ }
249
+ function dateOnly(date) {
250
+ return date.toISOString().slice(0, 10);
251
+ }
252
+ function financeCounts(invoiceRows, paymentSessionRows) {
253
+ const invoiceCount = (invoiceType, statuses) => invoiceRows
254
+ .filter((row) => row.invoiceType === invoiceType && statuses.includes(row.status))
255
+ .reduce((sum, row) => sum + row.count, 0);
256
+ const sessionCount = (statuses) => paymentSessionRows
257
+ .filter((row) => statuses.includes(row.status))
258
+ .reduce((sum, row) => sum + row.count, 0);
259
+ return {
260
+ invoices: {
261
+ issued: invoiceCount("invoice", ["issued", "partially_paid", "overdue"]),
262
+ paid: invoiceCount("invoice", ["paid"]),
263
+ void: invoiceCount("invoice", ["void"]),
264
+ overdue: invoiceCount("invoice", ["overdue"]),
265
+ },
266
+ proformas: {
267
+ issued: invoiceCount("proforma", ["issued", "partially_paid", "overdue", "paid"]),
268
+ converted: invoiceCount("proforma", ["paid"]),
269
+ void: invoiceCount("proforma", ["void"]),
270
+ },
271
+ paymentSessions: {
272
+ pending: sessionCount(["pending", "requires_redirect", "processing", "authorized"]),
273
+ paid: sessionCount(["paid"]),
274
+ failed: sessionCount(["failed", "cancelled", "expired"]),
275
+ },
276
+ };
277
+ }
278
+ function financeTotals(invoiceRows, paymentRows) {
279
+ const currencies = new Set([
280
+ ...invoiceRows.map((row) => row.currency),
281
+ ...paymentRows.map((row) => row.currency),
282
+ ]);
283
+ return [...currencies].sort().map((currency) => {
284
+ const invoice = invoiceRows.find((row) => row.currency === currency);
285
+ const payment = paymentRows.find((row) => row.currency === currency);
286
+ return {
287
+ currency,
288
+ invoiced: Number(invoice?.invoiced ?? 0),
289
+ collected: Number(payment?.collected ?? 0),
290
+ outstanding: Number(invoice?.outstanding ?? 0),
291
+ refunded: Number(payment?.refunded ?? 0),
292
+ };
293
+ });
294
+ }