@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,37 @@
1
+ import type { AgingReportQuery, PostgresJsDatabase, ProfitabilityQuery, RevenueReportQuery } from "./service-shared.js";
2
+ export declare const financeReportService: {
3
+ getRevenueReport(db: PostgresJsDatabase, query: RevenueReportQuery): Omit<import("drizzle-orm/pg-core").PgSelectBase<"invoices", {
4
+ month: import("drizzle-orm").SQL<string>;
5
+ totalCents: import("drizzle-orm").SQL<number>;
6
+ count: import("drizzle-orm").SQL<number>;
7
+ }, "partial", Record<"invoices", "not-null">, false, "where" | "groupBy" | "orderBy", {
8
+ month: string;
9
+ totalCents: number;
10
+ count: number;
11
+ }[], {
12
+ month: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
13
+ totalCents: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
14
+ count: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
15
+ }>, "where" | "groupBy" | "orderBy">;
16
+ getAgingReport(db: PostgresJsDatabase, query: AgingReportQuery): Omit<import("drizzle-orm/pg-core").PgSelectBase<"invoices", {
17
+ bucket: import("drizzle-orm").SQL<string>;
18
+ totalCents: import("drizzle-orm").SQL<number>;
19
+ count: import("drizzle-orm").SQL<number>;
20
+ }, "partial", Record<"invoices", "not-null">, false, "where" | "groupBy", {
21
+ bucket: string;
22
+ totalCents: number;
23
+ count: number;
24
+ }[], {
25
+ bucket: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
26
+ totalCents: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
27
+ count: import("drizzle-orm").DrizzleTypeError<"You cannot reference this field without assigning it an alias first - use `.as(<alias>)`">;
28
+ }>, "where" | "groupBy">;
29
+ getProfitabilityReport(db: PostgresJsDatabase, query: ProfitabilityQuery): Promise<{
30
+ bookingId: string;
31
+ bookingNumber: string;
32
+ sellAmountCents: number | null;
33
+ costAmountCents: number | null;
34
+ marginPercent: number | null;
35
+ }[]>;
36
+ };
37
+ //# sourceMappingURL=service-reports.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"service-reports.d.ts","sourceRoot":"","sources":["../src/service-reports.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,gBAAgB,EAChB,kBAAkB,EAClB,kBAAkB,EAClB,kBAAkB,EACnB,MAAM,qBAAqB,CAAA;AAG5B,eAAO,MAAM,oBAAoB;yBACV,kBAAkB,SAAS,kBAAkB;;;;;;;;;;;;;uBAiB/C,kBAAkB,SAAS,gBAAgB;;;;;;;;;;;;;+BA8B7B,kBAAkB,SAAS,kBAAkB;mBAsB/D,MAAM;uBACF,MAAM;yBACJ,MAAM,GAAG,IAAI;yBACb,MAAM,GAAG,IAAI;uBACf,MAAM,GAAG,IAAI;;CAGjC,CAAA"}
@@ -0,0 +1,62 @@
1
+ import { and, asc, bookings, gte, invoices, lte, sql } from "./service-shared.js";
2
+ export const financeReportService = {
3
+ getRevenueReport(db, query) {
4
+ return (db
5
+ .select({
6
+ month: sql `to_char(date_trunc('month', ${invoices.issueDate}::date), 'YYYY-MM')`,
7
+ totalCents: sql `coalesce(sum(${invoices.totalCents}), 0)::int`,
8
+ count: sql `count(*)::int`,
9
+ })
10
+ .from(invoices)
11
+ .where(and(gte(invoices.issueDate, query.from), lte(invoices.issueDate, query.to)))
12
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
13
+ .groupBy(sql `date_trunc('month', ${invoices.issueDate}::date)`)
14
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
15
+ .orderBy(sql `date_trunc('month', ${invoices.issueDate}::date)`));
16
+ },
17
+ getAgingReport(db, query) {
18
+ const asOf = query.asOf ?? new Date().toISOString().slice(0, 10);
19
+ return db
20
+ .select({
21
+ bucket: sql `
22
+ case
23
+ when ${invoices.dueDate}::date >= ${asOf}::date then 'current'
24
+ when ${asOf}::date - ${invoices.dueDate}::date <= 30 then '1-30'
25
+ when ${asOf}::date - ${invoices.dueDate}::date <= 60 then '31-60'
26
+ when ${asOf}::date - ${invoices.dueDate}::date <= 90 then '61-90'
27
+ else '90+'
28
+ end`,
29
+ totalCents: sql `coalesce(sum(${invoices.balanceDueCents}), 0)::int`,
30
+ count: sql `count(*)::int`,
31
+ })
32
+ .from(invoices)
33
+ .where(and(
34
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
35
+ sql `${invoices.balanceDueCents} > 0`,
36
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
37
+ sql `${invoices.status} != 'void'`,
38
+ // agent-quality: raw-sql reviewed -- owner: finance; dynamic SQL interpolation uses Drizzle parameter binding or vetted SQL identifiers.
39
+ sql `${invoices.status} != 'paid'`))
40
+ .groupBy(sql `1`);
41
+ },
42
+ async getProfitabilityReport(db, query) {
43
+ const conditions = [];
44
+ if (query.from) {
45
+ conditions.push(gte(bookings.startDate, query.from));
46
+ }
47
+ if (query.to) {
48
+ conditions.push(lte(bookings.startDate, query.to));
49
+ }
50
+ return (await db
51
+ .select({
52
+ bookingId: bookings.id,
53
+ bookingNumber: bookings.bookingNumber,
54
+ sellAmountCents: bookings.sellAmountCents,
55
+ costAmountCents: bookings.costAmountCents,
56
+ marginPercent: bookings.marginPercent,
57
+ })
58
+ .from(bookings)
59
+ .where(conditions.length > 0 ? and(...conditions) : undefined)
60
+ .orderBy(asc(bookings.startDate), asc(bookings.createdAt)));
61
+ },
62
+ };
@@ -0,0 +1,46 @@
1
+ import type { EventBus } from "@voyant-travel/core";
2
+ import type { PostgresJsDatabase } from "drizzle-orm/postgres-js";
3
+ import type { Invoice as FinanceInvoice, InvoiceExternalRef } from "./schema.js";
4
+ import type { PolledInvoiceSettlementResult, PollInvoiceSettlementInput } from "./validation.js";
5
+ type SettlementInvoice = FinanceInvoice;
6
+ type SettlementExternalRef = InvoiceExternalRef;
7
+ export interface InvoiceSettlementPollerContext {
8
+ db: PostgresJsDatabase;
9
+ invoice: SettlementInvoice;
10
+ externalRef: SettlementExternalRef;
11
+ bindings: Record<string, unknown>;
12
+ }
13
+ export interface InvoiceSettlementPollerResult {
14
+ externalId?: string | null;
15
+ externalNumber?: string | null;
16
+ externalUrl?: string | null;
17
+ status?: string | null;
18
+ paidAmountCents?: number | null;
19
+ unpaidAmountCents?: number | null;
20
+ syncedAt?: string | Date | null;
21
+ settledAt?: string | Date | null;
22
+ referenceNumber?: string | null;
23
+ syncError?: string | null;
24
+ metadata?: Record<string, unknown> | null;
25
+ }
26
+ export type InvoiceSettlementPoller = (context: InvoiceSettlementPollerContext) => Promise<InvoiceSettlementPollerResult>;
27
+ export interface FinanceSettlementRuntimeOptions {
28
+ bindings?: Record<string, unknown>;
29
+ invoiceSettlementPollers?: Record<string, InvoiceSettlementPoller>;
30
+ eventBus?: EventBus;
31
+ }
32
+ export interface InvoiceSettledEvent {
33
+ invoiceId: string;
34
+ paymentId: string;
35
+ provider: string;
36
+ newlyAppliedAmountCents: number;
37
+ paidCents: number;
38
+ balanceDueCents: number;
39
+ }
40
+ export declare const financeSettlementService: {
41
+ pollInvoiceSettlement(db: PostgresJsDatabase, invoiceId: string, input: PollInvoiceSettlementInput, runtime?: FinanceSettlementRuntimeOptions): Promise<PolledInvoiceSettlementResult | {
42
+ status: "not_found";
43
+ }>;
44
+ };
45
+ export {};
46
+ //# sourceMappingURL=service-settlement.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"service-settlement.d.ts","sourceRoot":"","sources":["../src/service-settlement.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAA;AACnD,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAA;AAEjE,OAAO,KAAK,EAAE,OAAO,IAAI,cAAc,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAA;AAEhF,OAAO,KAAK,EAAE,6BAA6B,EAAE,0BAA0B,EAAE,MAAM,iBAAiB,CAAA;AAEhG,KAAK,iBAAiB,GAAG,cAAc,CAAA;AACvC,KAAK,qBAAqB,GAAG,kBAAkB,CAAA;AAE/C,MAAM,WAAW,8BAA8B;IAC7C,EAAE,EAAE,kBAAkB,CAAA;IACtB,OAAO,EAAE,iBAAiB,CAAA;IAC1B,WAAW,EAAE,qBAAqB,CAAA;IAClC,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;CAClC;AAED,MAAM,WAAW,6BAA6B;IAC5C,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC1B,cAAc,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC9B,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC3B,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IACtB,eAAe,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC/B,iBAAiB,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IACjC,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI,CAAA;IAC/B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI,CAAA;IAChC,eAAe,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC/B,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IACzB,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,IAAI,CAAA;CAC1C;AAED,MAAM,MAAM,uBAAuB,GAAG,CACpC,OAAO,EAAE,8BAA8B,KACpC,OAAO,CAAC,6BAA6B,CAAC,CAAA;AAE3C,MAAM,WAAW,+BAA+B;IAC9C,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;IAClC,wBAAwB,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,uBAAuB,CAAC,CAAA;IAClE,QAAQ,CAAC,EAAE,QAAQ,CAAA;CACpB;AAED,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,EAAE,MAAM,CAAA;IACjB,QAAQ,EAAE,MAAM,CAAA;IAChB,uBAAuB,EAAE,MAAM,CAAA;IAC/B,SAAS,EAAE,MAAM,CAAA;IACjB,eAAe,EAAE,MAAM,CAAA;CACxB;AAyED,eAAO,MAAM,wBAAwB;8BAE7B,kBAAkB,aACX,MAAM,SACV,0BAA0B,YACxB,+BAA+B,GACvC,OAAO,CAAC,6BAA6B,GAAG;QAAE,MAAM,EAAE,WAAW,CAAA;KAAE,CAAC;CA0JpE,CAAA"}
@@ -0,0 +1,185 @@
1
+ import { financeService } from "./service.js";
2
+ function coerceRecord(value) {
3
+ return value && typeof value === "object" && !Array.isArray(value)
4
+ ? value
5
+ : null;
6
+ }
7
+ function toIsoString(value) {
8
+ if (!value)
9
+ return null;
10
+ if (value instanceof Date)
11
+ return value.toISOString();
12
+ const parsed = new Date(value);
13
+ return Number.isNaN(parsed.getTime()) ? null : parsed.toISOString();
14
+ }
15
+ function normalizeMoney(value) {
16
+ if (typeof value !== "number" || Number.isNaN(value))
17
+ return null;
18
+ return Math.round(value);
19
+ }
20
+ function resolveReferenceNumber(input, pollResult, externalRef, invoice) {
21
+ return (input.referenceNumber ??
22
+ pollResult.referenceNumber ??
23
+ pollResult.externalNumber ??
24
+ pollResult.externalId ??
25
+ externalRef.externalNumber ??
26
+ externalRef.externalId ??
27
+ invoice.invoiceNumber);
28
+ }
29
+ function resolvePaymentDate(input, pollResult) {
30
+ return (input.paymentDate ??
31
+ toIsoString(pollResult.settledAt) ??
32
+ toIsoString(pollResult.syncedAt) ??
33
+ new Date().toISOString());
34
+ }
35
+ async function synchronizeExternalRef(db, invoiceId, externalRef, pollResult) {
36
+ const syncedAt = toIsoString(pollResult.syncedAt) ?? new Date().toISOString();
37
+ const row = await financeService.registerInvoiceExternalRef(db, invoiceId, {
38
+ provider: externalRef.provider,
39
+ externalId: pollResult.externalId ?? externalRef.externalId ?? null,
40
+ externalNumber: pollResult.externalNumber ?? externalRef.externalNumber ?? null,
41
+ externalUrl: pollResult.externalUrl ?? externalRef.externalUrl ?? null,
42
+ status: pollResult.status ?? externalRef.status ?? null,
43
+ metadata: pollResult.metadata ?? coerceRecord(externalRef.metadata) ?? null,
44
+ syncedAt,
45
+ syncError: pollResult.syncError ?? null,
46
+ });
47
+ return {
48
+ row,
49
+ syncedAt,
50
+ };
51
+ }
52
+ export const financeSettlementService = {
53
+ async pollInvoiceSettlement(db, invoiceId, input, runtime = {}) {
54
+ let invoice = await financeService.getInvoiceById(db, invoiceId);
55
+ if (!invoice) {
56
+ return { status: "not_found" };
57
+ }
58
+ const externalRefs = await financeService.listInvoiceExternalRefs(db, invoiceId);
59
+ const refsToPoll = input.provider
60
+ ? externalRefs.filter((externalRef) => externalRef.provider === input.provider)
61
+ : externalRefs;
62
+ const results = [];
63
+ for (const externalRef of refsToPoll) {
64
+ const poller = runtime.invoiceSettlementPollers?.[externalRef.provider];
65
+ if (!poller) {
66
+ const synced = await synchronizeExternalRef(db, invoice.id, externalRef, {
67
+ syncError: "No settlement poller configured",
68
+ });
69
+ results.push({
70
+ provider: externalRef.provider,
71
+ externalRefId: synced.row?.id ?? externalRef.id,
72
+ externalId: synced.row?.externalId ?? externalRef.externalId ?? null,
73
+ externalNumber: synced.row?.externalNumber ?? externalRef.externalNumber ?? null,
74
+ externalUrl: synced.row?.externalUrl ?? externalRef.externalUrl ?? null,
75
+ status: synced.row?.status ?? externalRef.status ?? null,
76
+ paidAmountCents: null,
77
+ unpaidAmountCents: null,
78
+ syncedAt: toIsoString(synced.row?.syncedAt) ?? synced.syncedAt,
79
+ settledAt: null,
80
+ createdPaymentId: null,
81
+ newlyAppliedAmountCents: 0,
82
+ syncError: synced.row?.syncError ?? "No settlement poller configured",
83
+ });
84
+ continue;
85
+ }
86
+ try {
87
+ const pollResult = await poller({
88
+ db,
89
+ invoice,
90
+ externalRef,
91
+ bindings: runtime.bindings ?? {},
92
+ });
93
+ const synced = await synchronizeExternalRef(db, invoice.id, externalRef, pollResult);
94
+ const paidAmountCents = normalizeMoney(pollResult.paidAmountCents);
95
+ const unpaidAmountCents = normalizeMoney(pollResult.unpaidAmountCents);
96
+ let newlyAppliedAmountCents = 0;
97
+ let createdPaymentId = null;
98
+ if (input.reconcilePayment && paidAmountCents !== null) {
99
+ const cappedPaidAmountCents = Math.min(invoice.totalCents, Math.max(0, paidAmountCents));
100
+ const outstandingAmountCents = Math.max(0, invoice.totalCents - invoice.paidCents);
101
+ newlyAppliedAmountCents = Math.min(outstandingAmountCents, Math.max(0, cappedPaidAmountCents - invoice.paidCents));
102
+ if (newlyAppliedAmountCents > 0) {
103
+ const payment = await financeService.createPayment(db, invoice.id, {
104
+ amountCents: newlyAppliedAmountCents,
105
+ currency: invoice.currency,
106
+ baseCurrency: invoice.baseCurrency ?? null,
107
+ baseAmountCents: invoice.baseCurrency && invoice.baseCurrency === invoice.currency
108
+ ? newlyAppliedAmountCents
109
+ : null,
110
+ fxRateSetId: invoice.fxRateSetId ?? null,
111
+ paymentMethod: input.paymentMethod,
112
+ paymentInstrumentId: null,
113
+ paymentAuthorizationId: null,
114
+ paymentCaptureId: null,
115
+ status: "completed",
116
+ referenceNumber: resolveReferenceNumber(input, pollResult, externalRef, invoice),
117
+ paymentDate: resolvePaymentDate(input, pollResult),
118
+ notes: input.notes ??
119
+ `Settlement reconciled from ${externalRef.provider} external reference`,
120
+ });
121
+ createdPaymentId = payment?.id ?? null;
122
+ invoice = (await financeService.getInvoiceById(db, invoice.id)) ?? invoice;
123
+ if (createdPaymentId) {
124
+ await runtime.eventBus?.emit("invoice.settled", {
125
+ invoiceId: invoice.id,
126
+ paymentId: createdPaymentId,
127
+ provider: externalRef.provider,
128
+ newlyAppliedAmountCents,
129
+ paidCents: invoice.paidCents,
130
+ balanceDueCents: invoice.balanceDueCents,
131
+ }, {
132
+ category: "domain",
133
+ source: "service",
134
+ });
135
+ }
136
+ }
137
+ }
138
+ results.push({
139
+ provider: externalRef.provider,
140
+ externalRefId: synced.row?.id ?? externalRef.id,
141
+ externalId: synced.row?.externalId ?? externalRef.externalId ?? null,
142
+ externalNumber: synced.row?.externalNumber ?? externalRef.externalNumber ?? null,
143
+ externalUrl: synced.row?.externalUrl ?? externalRef.externalUrl ?? null,
144
+ status: synced.row?.status ?? externalRef.status ?? null,
145
+ paidAmountCents,
146
+ unpaidAmountCents,
147
+ syncedAt: toIsoString(synced.row?.syncedAt) ?? synced.syncedAt,
148
+ settledAt: toIsoString(pollResult.settledAt),
149
+ createdPaymentId,
150
+ newlyAppliedAmountCents,
151
+ syncError: synced.row?.syncError ?? pollResult.syncError ?? null,
152
+ });
153
+ }
154
+ catch (error) {
155
+ const message = error instanceof Error ? error.message : "Settlement polling failed";
156
+ const synced = await synchronizeExternalRef(db, invoice.id, externalRef, {
157
+ syncError: message,
158
+ });
159
+ results.push({
160
+ provider: externalRef.provider,
161
+ externalRefId: synced.row?.id ?? externalRef.id,
162
+ externalId: synced.row?.externalId ?? externalRef.externalId ?? null,
163
+ externalNumber: synced.row?.externalNumber ?? externalRef.externalNumber ?? null,
164
+ externalUrl: synced.row?.externalUrl ?? externalRef.externalUrl ?? null,
165
+ status: synced.row?.status ?? externalRef.status ?? null,
166
+ paidAmountCents: null,
167
+ unpaidAmountCents: null,
168
+ syncedAt: toIsoString(synced.row?.syncedAt) ?? synced.syncedAt,
169
+ settledAt: null,
170
+ createdPaymentId: null,
171
+ newlyAppliedAmountCents: 0,
172
+ syncError: synced.row?.syncError ?? message,
173
+ });
174
+ }
175
+ }
176
+ invoice = (await financeService.getInvoiceById(db, invoice.id)) ?? invoice;
177
+ return {
178
+ invoiceId: invoice.id,
179
+ invoiceStatus: invoice.status,
180
+ paidCents: invoice.paidCents,
181
+ balanceDueCents: invoice.balanceDueCents,
182
+ results,
183
+ };
184
+ },
185
+ };