@unifiedcommerce/core 0.1.1 → 0.2.1

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 (257) hide show
  1. package/dist/auth/setup.d.ts.map +1 -1
  2. package/dist/auth/setup.js +8 -3
  3. package/dist/config/types.d.ts +3 -1
  4. package/dist/config/types.d.ts.map +1 -1
  5. package/dist/index.d.ts +1 -0
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.js +1 -0
  8. package/dist/interfaces/mcp/server.d.ts +3 -5
  9. package/dist/interfaces/mcp/server.d.ts.map +1 -1
  10. package/dist/interfaces/mcp/server.js +25 -510
  11. package/dist/interfaces/mcp/tool-builder.d.ts +120 -0
  12. package/dist/interfaces/mcp/tool-builder.d.ts.map +1 -0
  13. package/dist/interfaces/mcp/tool-builder.js +224 -0
  14. package/dist/interfaces/mcp/tools/analytics.d.ts +42 -0
  15. package/dist/interfaces/mcp/tools/analytics.d.ts.map +1 -0
  16. package/dist/interfaces/mcp/tools/analytics.js +70 -0
  17. package/dist/interfaces/mcp/tools/cart.d.ts +14 -0
  18. package/dist/interfaces/mcp/tools/cart.d.ts.map +1 -0
  19. package/dist/interfaces/mcp/tools/cart.js +47 -0
  20. package/dist/interfaces/mcp/tools/catalog.d.ts +53 -0
  21. package/dist/interfaces/mcp/tools/catalog.d.ts.map +1 -0
  22. package/dist/interfaces/mcp/tools/catalog.js +284 -0
  23. package/dist/interfaces/mcp/tools/index.d.ts +3 -0
  24. package/dist/interfaces/mcp/tools/index.d.ts.map +1 -0
  25. package/dist/interfaces/mcp/tools/index.js +20 -0
  26. package/dist/interfaces/mcp/tools/inventory.d.ts +27 -0
  27. package/dist/interfaces/mcp/tools/inventory.d.ts.map +1 -0
  28. package/dist/interfaces/mcp/tools/inventory.js +143 -0
  29. package/dist/interfaces/mcp/tools/orders.d.ts +18 -0
  30. package/dist/interfaces/mcp/tools/orders.d.ts.map +1 -0
  31. package/dist/interfaces/mcp/tools/orders.js +82 -0
  32. package/dist/interfaces/mcp/tools/pricing.d.ts +29 -0
  33. package/dist/interfaces/mcp/tools/pricing.d.ts.map +1 -0
  34. package/dist/interfaces/mcp/tools/pricing.js +90 -0
  35. package/dist/interfaces/mcp/tools/promotions.d.ts +44 -0
  36. package/dist/interfaces/mcp/tools/promotions.d.ts.map +1 -0
  37. package/dist/interfaces/mcp/tools/promotions.js +109 -0
  38. package/dist/interfaces/mcp/tools/registry.d.ts +32 -0
  39. package/dist/interfaces/mcp/tools/registry.d.ts.map +1 -0
  40. package/dist/interfaces/mcp/tools/registry.js +55 -0
  41. package/dist/interfaces/mcp/tools/search.d.ts +14 -0
  42. package/dist/interfaces/mcp/tools/search.d.ts.map +1 -0
  43. package/dist/interfaces/mcp/tools/search.js +39 -0
  44. package/dist/interfaces/mcp/tools/webhooks.d.ts +15 -0
  45. package/dist/interfaces/mcp/tools/webhooks.d.ts.map +1 -0
  46. package/dist/interfaces/mcp/tools/webhooks.js +48 -0
  47. package/dist/interfaces/mcp/transport.d.ts +17 -2
  48. package/dist/interfaces/mcp/transport.d.ts.map +1 -1
  49. package/dist/interfaces/mcp/transport.js +91 -44
  50. package/dist/interfaces/rest/router.d.ts.map +1 -1
  51. package/dist/interfaces/rest/routes/checkout.d.ts.map +1 -1
  52. package/dist/interfaces/rest/routes/checkout.js +1 -1
  53. package/dist/interfaces/rest/routes/promotions.d.ts.map +1 -1
  54. package/dist/interfaces/rest/routes/promotions.js +3 -2
  55. package/dist/kernel/database/adapter.d.ts +8 -0
  56. package/dist/kernel/database/adapter.d.ts.map +1 -1
  57. package/dist/kernel/factory/repository-factory.d.ts.map +1 -1
  58. package/dist/kernel/factory/repository-factory.js +3 -1
  59. package/dist/kernel/local-api.d.ts.map +1 -1
  60. package/dist/kernel/local-api.js +2 -0
  61. package/dist/kernel/plugin/manifest.d.ts +3 -3
  62. package/dist/kernel/plugin/manifest.d.ts.map +1 -1
  63. package/dist/kernel/plugin/manifest.js +36 -7
  64. package/dist/runtime/kernel.d.ts +1 -2
  65. package/dist/runtime/kernel.d.ts.map +1 -1
  66. package/dist/runtime/kernel.js +16 -8
  67. package/dist/runtime/server.d.ts.map +1 -1
  68. package/dist/runtime/server.js +8 -3
  69. package/dist/test-utils/create-pglite-adapter.d.ts.map +1 -1
  70. package/dist/test-utils/create-pglite-adapter.js +7 -6
  71. package/dist/tsconfig.tsbuildinfo +1 -0
  72. package/package.json +3 -1
  73. package/src/adapters/console-email.ts +43 -0
  74. package/src/auth/access.ts +187 -0
  75. package/src/auth/auth-schema.ts +139 -0
  76. package/src/auth/middleware.ts +161 -0
  77. package/src/auth/org.ts +41 -0
  78. package/src/auth/permissions.ts +28 -0
  79. package/src/auth/setup.ts +171 -0
  80. package/src/auth/system-actor.ts +19 -0
  81. package/src/auth/types.ts +10 -0
  82. package/src/config/defaults.ts +82 -0
  83. package/src/config/define-config.ts +53 -0
  84. package/src/config/types.ts +301 -0
  85. package/src/generated/plugin-capabilities.d.ts +20 -0
  86. package/src/generated/plugin-manifest.ts +23 -0
  87. package/src/generated/plugin-repositories.d.ts +20 -0
  88. package/src/hooks/checkout-completion.ts +262 -0
  89. package/src/hooks/checkout.ts +677 -0
  90. package/src/hooks/order-emails.ts +62 -0
  91. package/src/index.ts +215 -0
  92. package/src/interfaces/mcp/agent-prompt.ts +174 -0
  93. package/src/interfaces/mcp/context-enrichment.ts +177 -0
  94. package/src/interfaces/mcp/server.ts +47 -0
  95. package/src/interfaces/mcp/tool-builder.ts +261 -0
  96. package/src/interfaces/mcp/tools/analytics.ts +76 -0
  97. package/src/interfaces/mcp/tools/cart.ts +57 -0
  98. package/src/interfaces/mcp/tools/catalog.ts +299 -0
  99. package/src/interfaces/mcp/tools/index.ts +22 -0
  100. package/src/interfaces/mcp/tools/inventory.ts +161 -0
  101. package/src/interfaces/mcp/tools/orders.ts +104 -0
  102. package/src/interfaces/mcp/tools/pricing.ts +94 -0
  103. package/src/interfaces/mcp/tools/promotions.ts +106 -0
  104. package/src/interfaces/mcp/tools/registry.ts +101 -0
  105. package/src/interfaces/mcp/tools/search.ts +42 -0
  106. package/src/interfaces/mcp/tools/webhooks.ts +48 -0
  107. package/src/interfaces/mcp/transport.ts +128 -0
  108. package/src/interfaces/rest/customer-portal.ts +299 -0
  109. package/src/interfaces/rest/index.ts +74 -0
  110. package/src/interfaces/rest/router.ts +333 -0
  111. package/src/interfaces/rest/routes/admin-jobs.ts +58 -0
  112. package/src/interfaces/rest/routes/audit.ts +50 -0
  113. package/src/interfaces/rest/routes/carts.ts +89 -0
  114. package/src/interfaces/rest/routes/catalog.ts +493 -0
  115. package/src/interfaces/rest/routes/checkout.ts +284 -0
  116. package/src/interfaces/rest/routes/inventory.ts +70 -0
  117. package/src/interfaces/rest/routes/media.ts +86 -0
  118. package/src/interfaces/rest/routes/orders.ts +78 -0
  119. package/src/interfaces/rest/routes/payments.ts +60 -0
  120. package/src/interfaces/rest/routes/pricing.ts +57 -0
  121. package/src/interfaces/rest/routes/promotions.ts +93 -0
  122. package/src/interfaces/rest/routes/search.ts +71 -0
  123. package/src/interfaces/rest/routes/webhooks.ts +46 -0
  124. package/src/interfaces/rest/schemas/admin-jobs.ts +40 -0
  125. package/src/interfaces/rest/schemas/audit.ts +46 -0
  126. package/src/interfaces/rest/schemas/carts.ts +125 -0
  127. package/src/interfaces/rest/schemas/catalog.ts +450 -0
  128. package/src/interfaces/rest/schemas/checkout.ts +66 -0
  129. package/src/interfaces/rest/schemas/customer-portal.ts +195 -0
  130. package/src/interfaces/rest/schemas/inventory.ts +138 -0
  131. package/src/interfaces/rest/schemas/media.ts +75 -0
  132. package/src/interfaces/rest/schemas/orders.ts +104 -0
  133. package/src/interfaces/rest/schemas/pricing.ts +80 -0
  134. package/src/interfaces/rest/schemas/promotions.ts +110 -0
  135. package/src/interfaces/rest/schemas/responses.ts +85 -0
  136. package/src/interfaces/rest/schemas/search.ts +58 -0
  137. package/src/interfaces/rest/schemas/shared.ts +62 -0
  138. package/src/interfaces/rest/schemas/webhooks.ts +68 -0
  139. package/src/interfaces/rest/utils.ts +104 -0
  140. package/src/interfaces/rest/webhook-router.ts +50 -0
  141. package/src/kernel/compensation/executor.ts +61 -0
  142. package/src/kernel/compensation/types.ts +26 -0
  143. package/src/kernel/database/adapter.ts +21 -0
  144. package/src/kernel/database/drizzle-db.ts +56 -0
  145. package/src/kernel/database/migrate.ts +76 -0
  146. package/src/kernel/database/plugin-types.ts +34 -0
  147. package/src/kernel/database/schema.ts +49 -0
  148. package/src/kernel/database/scoped-db.ts +68 -0
  149. package/src/kernel/database/tx-context.ts +46 -0
  150. package/src/kernel/error-mapper.ts +15 -0
  151. package/src/kernel/errors.ts +89 -0
  152. package/src/kernel/factory/repository-factory.ts +244 -0
  153. package/src/kernel/hooks/create-context.ts +43 -0
  154. package/src/kernel/hooks/executor.ts +88 -0
  155. package/src/kernel/hooks/registry.ts +74 -0
  156. package/src/kernel/hooks/types.ts +52 -0
  157. package/src/kernel/http-error.ts +44 -0
  158. package/src/kernel/jobs/adapter.ts +36 -0
  159. package/src/kernel/jobs/drizzle-adapter.ts +58 -0
  160. package/src/kernel/jobs/runner.ts +153 -0
  161. package/src/kernel/jobs/schema.ts +46 -0
  162. package/src/kernel/jobs/types.ts +30 -0
  163. package/src/kernel/local-api.ts +187 -0
  164. package/src/kernel/plugin/manifest.ts +271 -0
  165. package/src/kernel/query/executor.ts +184 -0
  166. package/src/kernel/query/registry.ts +46 -0
  167. package/src/kernel/result.ts +33 -0
  168. package/src/kernel/schema/extra-columns.ts +37 -0
  169. package/src/kernel/service-registry.ts +76 -0
  170. package/src/kernel/service-timing.ts +89 -0
  171. package/src/kernel/state-machine/machine.ts +101 -0
  172. package/src/modules/analytics/drizzle-adapter.ts +426 -0
  173. package/src/modules/analytics/hooks.ts +11 -0
  174. package/src/modules/analytics/models.ts +125 -0
  175. package/src/modules/analytics/repository/index.ts +6 -0
  176. package/src/modules/analytics/service.ts +245 -0
  177. package/src/modules/analytics/types.ts +180 -0
  178. package/src/modules/audit/hooks.ts +78 -0
  179. package/src/modules/audit/schema.ts +33 -0
  180. package/src/modules/audit/service.ts +151 -0
  181. package/src/modules/cart/access.ts +27 -0
  182. package/src/modules/cart/matcher.ts +26 -0
  183. package/src/modules/cart/repository/index.ts +234 -0
  184. package/src/modules/cart/schema.ts +42 -0
  185. package/src/modules/cart/schemas.ts +38 -0
  186. package/src/modules/cart/service.ts +541 -0
  187. package/src/modules/catalog/repository/index.ts +772 -0
  188. package/src/modules/catalog/schema.ts +203 -0
  189. package/src/modules/catalog/schemas.ts +104 -0
  190. package/src/modules/catalog/service.ts +1544 -0
  191. package/src/modules/customers/repository/index.ts +327 -0
  192. package/src/modules/customers/schema.ts +64 -0
  193. package/src/modules/customers/service.ts +171 -0
  194. package/src/modules/fulfillment/repository/index.ts +426 -0
  195. package/src/modules/fulfillment/schema.ts +101 -0
  196. package/src/modules/fulfillment/service.ts +555 -0
  197. package/src/modules/fulfillment/types.ts +59 -0
  198. package/src/modules/inventory/repository/index.ts +509 -0
  199. package/src/modules/inventory/schema.ts +94 -0
  200. package/src/modules/inventory/schemas.ts +38 -0
  201. package/src/modules/inventory/service.ts +490 -0
  202. package/src/modules/media/adapter.ts +17 -0
  203. package/src/modules/media/repository/index.ts +274 -0
  204. package/src/modules/media/schema.ts +41 -0
  205. package/src/modules/media/service.ts +151 -0
  206. package/src/modules/orders/repository/index.ts +287 -0
  207. package/src/modules/orders/schema.ts +66 -0
  208. package/src/modules/orders/service.ts +619 -0
  209. package/src/modules/orders/stale-order-cleanup.ts +76 -0
  210. package/src/modules/organization/service.ts +191 -0
  211. package/src/modules/payments/adapter.ts +47 -0
  212. package/src/modules/payments/repository/index.ts +6 -0
  213. package/src/modules/payments/service.ts +107 -0
  214. package/src/modules/pricing/repository/index.ts +291 -0
  215. package/src/modules/pricing/schema.ts +71 -0
  216. package/src/modules/pricing/schemas.ts +38 -0
  217. package/src/modules/pricing/service.ts +494 -0
  218. package/src/modules/promotions/repository/index.ts +325 -0
  219. package/src/modules/promotions/schema.ts +62 -0
  220. package/src/modules/promotions/schemas.ts +38 -0
  221. package/src/modules/promotions/service.ts +598 -0
  222. package/src/modules/search/adapter.ts +57 -0
  223. package/src/modules/search/hooks.ts +12 -0
  224. package/src/modules/search/repository/index.ts +6 -0
  225. package/src/modules/search/service.ts +315 -0
  226. package/src/modules/shipping/calculator.ts +188 -0
  227. package/src/modules/shipping/repository/index.ts +6 -0
  228. package/src/modules/shipping/service.ts +51 -0
  229. package/src/modules/tax/adapter.ts +60 -0
  230. package/src/modules/tax/repository/index.ts +6 -0
  231. package/src/modules/tax/service.ts +53 -0
  232. package/src/modules/webhooks/hook.ts +34 -0
  233. package/src/modules/webhooks/repository/index.ts +278 -0
  234. package/src/modules/webhooks/schema.ts +56 -0
  235. package/src/modules/webhooks/service.ts +117 -0
  236. package/src/modules/webhooks/signing.ts +6 -0
  237. package/src/modules/webhooks/ssrf-guard.ts +71 -0
  238. package/src/modules/webhooks/tasks.ts +52 -0
  239. package/src/modules/webhooks/worker.ts +134 -0
  240. package/src/runtime/commerce.ts +145 -0
  241. package/src/runtime/kernel.ts +426 -0
  242. package/src/runtime/logger.ts +36 -0
  243. package/src/runtime/server.ts +355 -0
  244. package/src/runtime/shutdown.ts +43 -0
  245. package/src/test-utils/create-pglite-adapter.ts +129 -0
  246. package/src/test-utils/create-plugin-test-app.ts +128 -0
  247. package/src/test-utils/create-repository-test-harness.ts +16 -0
  248. package/src/test-utils/create-test-config.ts +190 -0
  249. package/src/test-utils/create-test-kernel.ts +7 -0
  250. package/src/test-utils/create-test-plugin-context.ts +75 -0
  251. package/src/test-utils/rest-api-test-utils.ts +265 -0
  252. package/src/test-utils/test-actors.ts +62 -0
  253. package/src/test-utils/typed-hooks.ts +54 -0
  254. package/src/types/commerce-types.ts +34 -0
  255. package/src/utils/id.ts +3 -0
  256. package/src/utils/logger.ts +18 -0
  257. package/src/utils/pagination.ts +22 -0
@@ -0,0 +1,287 @@
1
+ import { eq, and, desc, sql } from "drizzle-orm";
2
+ import type { TxContext } from "../../../kernel/database/tx-context.js";
3
+ import type {
4
+ DrizzleDatabase,
5
+ DbOrTx,
6
+ } from "../../../kernel/database/drizzle-db.js";
7
+ import { orders, orderLineItems, orderStatusHistory } from "../schema.js";
8
+
9
+ // Infer types from Drizzle schema
10
+ export type Order = typeof orders.$inferSelect;
11
+ export type OrderInsert = typeof orders.$inferInsert;
12
+ export type OrderLineItem = typeof orderLineItems.$inferSelect;
13
+ export type OrderLineItemInsert = typeof orderLineItems.$inferInsert;
14
+ export type OrderStatusHistory = typeof orderStatusHistory.$inferSelect;
15
+ export type OrderStatusHistoryInsert = typeof orderStatusHistory.$inferInsert;
16
+
17
+ /**
18
+ * OrdersRepository provides type-safe database operations for orders.
19
+ *
20
+ * This repository manages orders, order line items, and order status history.
21
+ * All methods support an optional TxContext parameter for transaction participation.
22
+ */
23
+ export class OrdersRepository {
24
+ constructor(private readonly db: DrizzleDatabase) {}
25
+
26
+ private getDb(ctx?: TxContext): DbOrTx {
27
+ return (ctx?.tx as DbOrTx | undefined) ?? this.db;
28
+ }
29
+
30
+ // ─────────────────────────────────────────────────────────────────────────────
31
+ // Orders
32
+ // ─────────────────────────────────────────────────────────────────────────────
33
+
34
+ async findById(orgId: string, id: string, ctx?: TxContext): Promise<Order | undefined> {
35
+ const db = this.getDb(ctx);
36
+ const rows = await db
37
+ .select()
38
+ .from(orders)
39
+ .where(and(eq(orders.organizationId, orgId), eq(orders.id, id)));
40
+ return rows[0];
41
+ }
42
+
43
+ async findByOrderNumber(
44
+ orgId: string,
45
+ orderNumber: string,
46
+ ctx?: TxContext,
47
+ ): Promise<Order | undefined> {
48
+ const db = this.getDb(ctx);
49
+ const rows = await db
50
+ .select()
51
+ .from(orders)
52
+ .where(and(eq(orders.organizationId, orgId), eq(orders.orderNumber, orderNumber)));
53
+ return rows[0];
54
+ }
55
+
56
+ async findByCustomerId(
57
+ orgId: string,
58
+ customerId: string,
59
+ ctx?: TxContext,
60
+ ): Promise<Order[]> {
61
+ const db = this.getDb(ctx);
62
+ return db
63
+ .select()
64
+ .from(orders)
65
+ .where(and(eq(orders.organizationId, orgId), eq(orders.customerId, customerId)))
66
+ .orderBy(desc(orders.placedAt));
67
+ }
68
+
69
+ async findByStatus(orgId: string, status: string, ctx?: TxContext): Promise<Order[]> {
70
+ const db = this.getDb(ctx);
71
+ return db
72
+ .select()
73
+ .from(orders)
74
+ .where(and(eq(orders.organizationId, orgId), eq(orders.status, status)))
75
+ .orderBy(desc(orders.placedAt));
76
+ }
77
+
78
+ async findAll(
79
+ orgId: string,
80
+ options?: { limit?: number; offset?: number },
81
+ ctx?: TxContext,
82
+ ): Promise<Order[]> {
83
+ const db = this.getDb(ctx);
84
+ let query = db
85
+ .select()
86
+ .from(orders)
87
+ .where(eq(orders.organizationId, orgId))
88
+ .orderBy(desc(orders.placedAt))
89
+ .$dynamic();
90
+
91
+ if (options?.limit !== undefined) {
92
+ query = query.limit(options.limit);
93
+ }
94
+ if (options?.offset !== undefined) {
95
+ query = query.offset(options.offset);
96
+ }
97
+ return query;
98
+ }
99
+
100
+ async create(data: OrderInsert, ctx?: TxContext): Promise<Order> {
101
+ const db = this.getDb(ctx);
102
+ const rows = await db.insert(orders).values(data).returning();
103
+ return rows[0]!;
104
+ }
105
+
106
+ async update(
107
+ id: string,
108
+ data: Partial<Omit<OrderInsert, "id">>,
109
+ ctx?: TxContext,
110
+ ): Promise<Order | undefined> {
111
+ const db = this.getDb(ctx);
112
+ const rows = await db
113
+ .update(orders)
114
+ .set(data)
115
+ .where(eq(orders.id, id))
116
+ .returning();
117
+ return rows[0];
118
+ }
119
+
120
+ async updateStatus(
121
+ id: string,
122
+ currentStatus: string,
123
+ newStatus: string,
124
+ ctx?: TxContext,
125
+ ): Promise<Order | undefined> {
126
+ const db = this.getDb(ctx);
127
+ const data: Partial<OrderInsert> = { status: newStatus };
128
+ if (newStatus === "fulfilled") {
129
+ data.fulfilledAt = new Date();
130
+ } else if (newStatus === "cancelled") {
131
+ data.cancelledAt = new Date();
132
+ }
133
+ // Atomic guard: only update if current status matches expected
134
+ const rows = await db
135
+ .update(orders)
136
+ .set(data)
137
+ .where(and(eq(orders.id, id), eq(orders.status, currentStatus)))
138
+ .returning();
139
+ return rows[0];
140
+ }
141
+
142
+ async delete(id: string, ctx?: TxContext): Promise<boolean> {
143
+ const db = this.getDb(ctx);
144
+ const result = await db.delete(orders).where(eq(orders.id, id)).returning();
145
+ return result.length > 0;
146
+ }
147
+
148
+ // ─────────────────────────────────────────────────────────────────────────────
149
+ // Order Line Items
150
+ // ─────────────────────────────────────────────────────────────────────────────
151
+
152
+ async findAllLineItems(ctx?: TxContext): Promise<OrderLineItem[]> {
153
+ const db = this.getDb(ctx);
154
+ return db.select().from(orderLineItems);
155
+ }
156
+
157
+ async findLineItemById(
158
+ id: string,
159
+ ctx?: TxContext,
160
+ ): Promise<OrderLineItem | undefined> {
161
+ const db = this.getDb(ctx);
162
+ const rows = await db
163
+ .select()
164
+ .from(orderLineItems)
165
+ .where(eq(orderLineItems.id, id));
166
+ return rows[0];
167
+ }
168
+
169
+ async findLineItemsByOrderId(
170
+ orderId: string,
171
+ ctx?: TxContext,
172
+ ): Promise<OrderLineItem[]> {
173
+ const db = this.getDb(ctx);
174
+ return db
175
+ .select()
176
+ .from(orderLineItems)
177
+ .where(eq(orderLineItems.orderId, orderId));
178
+ }
179
+
180
+ async createLineItem(
181
+ data: OrderLineItemInsert,
182
+ ctx?: TxContext,
183
+ ): Promise<OrderLineItem> {
184
+ const db = this.getDb(ctx);
185
+ const rows = await db.insert(orderLineItems).values(data).returning();
186
+ return rows[0]!;
187
+ }
188
+
189
+ async createLineItems(
190
+ data: OrderLineItemInsert[],
191
+ ctx?: TxContext,
192
+ ): Promise<OrderLineItem[]> {
193
+ if (data.length === 0) return [];
194
+ const db = this.getDb(ctx);
195
+ return db.insert(orderLineItems).values(data).returning();
196
+ }
197
+
198
+ async updateLineItem(
199
+ id: string,
200
+ data: Partial<Omit<OrderLineItemInsert, "id">>,
201
+ ctx?: TxContext,
202
+ ): Promise<OrderLineItem | undefined> {
203
+ const db = this.getDb(ctx);
204
+ const rows = await db
205
+ .update(orderLineItems)
206
+ .set(data)
207
+ .where(eq(orderLineItems.id, id))
208
+ .returning();
209
+ return rows[0];
210
+ }
211
+
212
+ async deleteLineItem(id: string, ctx?: TxContext): Promise<boolean> {
213
+ const db = this.getDb(ctx);
214
+ const result = await db
215
+ .delete(orderLineItems)
216
+ .where(eq(orderLineItems.id, id))
217
+ .returning();
218
+ return result.length > 0;
219
+ }
220
+
221
+ async deleteLineItemsByOrderId(
222
+ orderId: string,
223
+ ctx?: TxContext,
224
+ ): Promise<void> {
225
+ const db = this.getDb(ctx);
226
+ await db.delete(orderLineItems).where(eq(orderLineItems.orderId, orderId));
227
+ }
228
+
229
+ // ─────────────────────────────────────────────────────────────────────────────
230
+ // Order Status History
231
+ // ─────────────────────────────────────────────────────────────────────────────
232
+
233
+ async findStatusHistoryByOrderId(
234
+ orderId: string,
235
+ ctx?: TxContext,
236
+ ): Promise<OrderStatusHistory[]> {
237
+ const db = this.getDb(ctx);
238
+ return db
239
+ .select()
240
+ .from(orderStatusHistory)
241
+ .where(eq(orderStatusHistory.orderId, orderId))
242
+ .orderBy(desc(orderStatusHistory.changedAt));
243
+ }
244
+
245
+ async createStatusHistory(
246
+ data: OrderStatusHistoryInsert,
247
+ ctx?: TxContext,
248
+ ): Promise<OrderStatusHistory> {
249
+ const db = this.getDb(ctx);
250
+ const rows = await db.insert(orderStatusHistory).values(data).returning();
251
+ return rows[0]!;
252
+ }
253
+
254
+ // ─────────────────────────────────────────────────────────────────────────────
255
+ // Order with Line Items (Aggregate)
256
+ // ─────────────────────────────────────────────────────────────────────────────
257
+
258
+ async findWithLineItems(
259
+ orgId: string,
260
+ id: string,
261
+ ctx?: TxContext,
262
+ ): Promise<{ order: Order; lineItems: OrderLineItem[] } | undefined> {
263
+ const order = await this.findById(orgId, id, ctx);
264
+ if (!order) return undefined;
265
+ const lineItems = await this.findLineItemsByOrderId(id, ctx);
266
+ return { order, lineItems };
267
+ }
268
+
269
+ // ─────────────────────────────────────────────────────────────────────────────
270
+ // Order Number Generation
271
+ // ─────────────────────────────────────────────────────────────────────────────
272
+
273
+ async getNextOrderNumber(ctx?: TxContext): Promise<string> {
274
+ const year = new Date().getFullYear();
275
+ const db = this.getDb(ctx);
276
+
277
+ // Count orders from this year to generate sequential number
278
+ const result = await db
279
+ .select({ count: sql<number>`count(*)::int` })
280
+ .from(orders)
281
+ .where(sql`EXTRACT(YEAR FROM ${orders.placedAt}) = ${year}`);
282
+
283
+ const count = result[0]?.count ?? 0;
284
+ const sequence = count + 1;
285
+ return `ORD-${year}-${String(sequence).padStart(6, "0")}`;
286
+ }
287
+ }
@@ -0,0 +1,66 @@
1
+ import { index, integer, jsonb, pgTable, text, timestamp, uniqueIndex, uuid } from "drizzle-orm/pg-core";
2
+ import { organization } from "../../auth/auth-schema.js";
3
+ import { sellableEntities, variants } from "../catalog/schema.js";
4
+
5
+ export const orders = pgTable("orders", {
6
+ id: uuid("id").defaultRandom().primaryKey(),
7
+ organizationId: text("organization_id").notNull().references(() => organization.id, { onDelete: "cascade" }),
8
+ orderNumber: text("order_number").notNull(),
9
+ customerId: uuid("customer_id"),
10
+ status: text("status").notNull().default("pending"),
11
+ currency: text("currency").notNull(),
12
+ subtotal: integer("subtotal").notNull(),
13
+ taxTotal: integer("tax_total").notNull(),
14
+ shippingTotal: integer("shipping_total").notNull(),
15
+ discountTotal: integer("discount_total").notNull().default(0),
16
+ grandTotal: integer("grand_total").notNull(),
17
+ paymentIntentId: text("payment_intent_id"),
18
+ paymentMethodId: text("payment_method_id"),
19
+ metadata: jsonb("metadata").$type<Record<string, unknown>>().default({}),
20
+ placedAt: timestamp("placed_at", { withTimezone: true }).defaultNow().notNull(),
21
+ fulfilledAt: timestamp("fulfilled_at", { withTimezone: true }),
22
+ cancelledAt: timestamp("cancelled_at", { withTimezone: true }),
23
+ }, (table) => ({
24
+ orgIdx: index("idx_orders_org").on(table.organizationId),
25
+ orgOrderNumberUnique: uniqueIndex("orders_org_order_number_unique").on(table.organizationId, table.orderNumber),
26
+ statusIdx: index("idx_orders_status").on(table.status),
27
+ customerIdIdx: index("idx_orders_customer_id").on(table.customerId),
28
+ placedAtIdx: index("idx_orders_placed_at").on(table.placedAt),
29
+ paymentIntentIdx: index("idx_orders_payment_intent").on(table.paymentIntentId),
30
+ }));
31
+
32
+ export const orderLineItems = pgTable("order_line_items", {
33
+ id: uuid("id").defaultRandom().primaryKey(),
34
+ orderId: uuid("order_id")
35
+ .references(() => orders.id, { onDelete: "cascade" })
36
+ .notNull(),
37
+ entityId: uuid("entity_id").references(() => sellableEntities.id).notNull(),
38
+ entityType: text("entity_type").notNull(),
39
+ variantId: uuid("variant_id").references(() => variants.id),
40
+ sku: text("sku"),
41
+ title: text("title").notNull(),
42
+ quantity: integer("quantity").notNull(),
43
+ unitPrice: integer("unit_price").notNull(),
44
+ totalPrice: integer("total_price").notNull(),
45
+ taxAmount: integer("tax_amount").notNull().default(0),
46
+ discountAmount: integer("discount_amount").notNull().default(0),
47
+ fulfillmentStatus: text("fulfillment_status").notNull().default("unfulfilled"),
48
+ metadata: jsonb("metadata").$type<Record<string, unknown>>().default({}),
49
+ }, (table) => [
50
+ index("idx_order_line_items_order_id").on(table.orderId),
51
+ index("idx_order_line_items_entity_id").on(table.entityId),
52
+ ]);
53
+
54
+ export const orderStatusHistory = pgTable("order_status_history", {
55
+ id: uuid("id").defaultRandom().primaryKey(),
56
+ orderId: uuid("order_id")
57
+ .references(() => orders.id, { onDelete: "cascade" })
58
+ .notNull(),
59
+ fromStatus: text("from_status").notNull(),
60
+ toStatus: text("to_status").notNull(),
61
+ reason: text("reason"),
62
+ changedBy: text("changed_by").notNull(),
63
+ changedAt: timestamp("changed_at", { withTimezone: true }).defaultNow().notNull(),
64
+ }, (table) => [
65
+ index("idx_order_status_history_order_id").on(table.orderId),
66
+ ]);