@unifiedcommerce/core 0.1.0 → 0.2.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.
Files changed (245) 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 +2 -2
  73. package/src/adapters/console-email.ts +0 -43
  74. package/src/auth/access.ts +0 -187
  75. package/src/auth/auth-schema.ts +0 -139
  76. package/src/auth/middleware.ts +0 -161
  77. package/src/auth/org.ts +0 -41
  78. package/src/auth/permissions.ts +0 -28
  79. package/src/auth/setup.ts +0 -169
  80. package/src/auth/system-actor.ts +0 -19
  81. package/src/auth/types.ts +0 -10
  82. package/src/config/defaults.ts +0 -82
  83. package/src/config/define-config.ts +0 -53
  84. package/src/config/types.ts +0 -299
  85. package/src/generated/plugin-capabilities.d.ts +0 -20
  86. package/src/generated/plugin-manifest.ts +0 -23
  87. package/src/generated/plugin-repositories.d.ts +0 -20
  88. package/src/hooks/checkout-completion.ts +0 -262
  89. package/src/hooks/checkout.ts +0 -677
  90. package/src/hooks/order-emails.ts +0 -62
  91. package/src/index.ts +0 -214
  92. package/src/interfaces/mcp/agent-prompt.ts +0 -174
  93. package/src/interfaces/mcp/context-enrichment.ts +0 -177
  94. package/src/interfaces/mcp/server.ts +0 -617
  95. package/src/interfaces/mcp/transport.ts +0 -68
  96. package/src/interfaces/rest/customer-portal.ts +0 -299
  97. package/src/interfaces/rest/index.ts +0 -74
  98. package/src/interfaces/rest/router.ts +0 -334
  99. package/src/interfaces/rest/routes/admin-jobs.ts +0 -58
  100. package/src/interfaces/rest/routes/audit.ts +0 -50
  101. package/src/interfaces/rest/routes/carts.ts +0 -89
  102. package/src/interfaces/rest/routes/catalog.ts +0 -493
  103. package/src/interfaces/rest/routes/checkout.ts +0 -283
  104. package/src/interfaces/rest/routes/inventory.ts +0 -70
  105. package/src/interfaces/rest/routes/media.ts +0 -86
  106. package/src/interfaces/rest/routes/orders.ts +0 -78
  107. package/src/interfaces/rest/routes/payments.ts +0 -60
  108. package/src/interfaces/rest/routes/pricing.ts +0 -57
  109. package/src/interfaces/rest/routes/promotions.ts +0 -92
  110. package/src/interfaces/rest/routes/search.ts +0 -71
  111. package/src/interfaces/rest/routes/webhooks.ts +0 -46
  112. package/src/interfaces/rest/schemas/admin-jobs.ts +0 -40
  113. package/src/interfaces/rest/schemas/audit.ts +0 -46
  114. package/src/interfaces/rest/schemas/carts.ts +0 -125
  115. package/src/interfaces/rest/schemas/catalog.ts +0 -450
  116. package/src/interfaces/rest/schemas/checkout.ts +0 -66
  117. package/src/interfaces/rest/schemas/customer-portal.ts +0 -195
  118. package/src/interfaces/rest/schemas/inventory.ts +0 -138
  119. package/src/interfaces/rest/schemas/media.ts +0 -75
  120. package/src/interfaces/rest/schemas/orders.ts +0 -104
  121. package/src/interfaces/rest/schemas/pricing.ts +0 -80
  122. package/src/interfaces/rest/schemas/promotions.ts +0 -110
  123. package/src/interfaces/rest/schemas/responses.ts +0 -85
  124. package/src/interfaces/rest/schemas/search.ts +0 -58
  125. package/src/interfaces/rest/schemas/shared.ts +0 -62
  126. package/src/interfaces/rest/schemas/webhooks.ts +0 -68
  127. package/src/interfaces/rest/utils.ts +0 -104
  128. package/src/interfaces/rest/webhook-router.ts +0 -50
  129. package/src/kernel/compensation/executor.ts +0 -61
  130. package/src/kernel/compensation/types.ts +0 -26
  131. package/src/kernel/database/adapter.ts +0 -13
  132. package/src/kernel/database/drizzle-db.ts +0 -56
  133. package/src/kernel/database/migrate.ts +0 -76
  134. package/src/kernel/database/plugin-types.ts +0 -34
  135. package/src/kernel/database/schema.ts +0 -49
  136. package/src/kernel/database/scoped-db.ts +0 -68
  137. package/src/kernel/database/tx-context.ts +0 -46
  138. package/src/kernel/error-mapper.ts +0 -15
  139. package/src/kernel/errors.ts +0 -89
  140. package/src/kernel/factory/repository-factory.ts +0 -242
  141. package/src/kernel/hooks/create-context.ts +0 -43
  142. package/src/kernel/hooks/executor.ts +0 -88
  143. package/src/kernel/hooks/registry.ts +0 -74
  144. package/src/kernel/hooks/types.ts +0 -52
  145. package/src/kernel/http-error.ts +0 -44
  146. package/src/kernel/jobs/adapter.ts +0 -36
  147. package/src/kernel/jobs/drizzle-adapter.ts +0 -58
  148. package/src/kernel/jobs/runner.ts +0 -153
  149. package/src/kernel/jobs/schema.ts +0 -46
  150. package/src/kernel/jobs/types.ts +0 -30
  151. package/src/kernel/local-api.ts +0 -185
  152. package/src/kernel/plugin/manifest.ts +0 -253
  153. package/src/kernel/query/executor.ts +0 -184
  154. package/src/kernel/query/registry.ts +0 -46
  155. package/src/kernel/result.ts +0 -33
  156. package/src/kernel/schema/extra-columns.ts +0 -37
  157. package/src/kernel/service-registry.ts +0 -76
  158. package/src/kernel/service-timing.ts +0 -89
  159. package/src/kernel/state-machine/machine.ts +0 -101
  160. package/src/modules/analytics/drizzle-adapter.ts +0 -426
  161. package/src/modules/analytics/hooks.ts +0 -11
  162. package/src/modules/analytics/models.ts +0 -125
  163. package/src/modules/analytics/repository/index.ts +0 -6
  164. package/src/modules/analytics/service.ts +0 -245
  165. package/src/modules/analytics/types.ts +0 -180
  166. package/src/modules/audit/hooks.ts +0 -78
  167. package/src/modules/audit/schema.ts +0 -33
  168. package/src/modules/audit/service.ts +0 -151
  169. package/src/modules/cart/access.ts +0 -27
  170. package/src/modules/cart/matcher.ts +0 -26
  171. package/src/modules/cart/repository/index.ts +0 -234
  172. package/src/modules/cart/schema.ts +0 -42
  173. package/src/modules/cart/schemas.ts +0 -38
  174. package/src/modules/cart/service.ts +0 -541
  175. package/src/modules/catalog/repository/index.ts +0 -772
  176. package/src/modules/catalog/schema.ts +0 -203
  177. package/src/modules/catalog/schemas.ts +0 -104
  178. package/src/modules/catalog/service.ts +0 -1544
  179. package/src/modules/customers/repository/index.ts +0 -327
  180. package/src/modules/customers/schema.ts +0 -64
  181. package/src/modules/customers/service.ts +0 -171
  182. package/src/modules/fulfillment/repository/index.ts +0 -426
  183. package/src/modules/fulfillment/schema.ts +0 -101
  184. package/src/modules/fulfillment/service.ts +0 -555
  185. package/src/modules/fulfillment/types.ts +0 -59
  186. package/src/modules/inventory/repository/index.ts +0 -509
  187. package/src/modules/inventory/schema.ts +0 -94
  188. package/src/modules/inventory/schemas.ts +0 -38
  189. package/src/modules/inventory/service.ts +0 -490
  190. package/src/modules/media/adapter.ts +0 -17
  191. package/src/modules/media/repository/index.ts +0 -274
  192. package/src/modules/media/schema.ts +0 -41
  193. package/src/modules/media/service.ts +0 -151
  194. package/src/modules/orders/repository/index.ts +0 -287
  195. package/src/modules/orders/schema.ts +0 -66
  196. package/src/modules/orders/service.ts +0 -619
  197. package/src/modules/orders/stale-order-cleanup.ts +0 -76
  198. package/src/modules/organization/service.ts +0 -191
  199. package/src/modules/payments/adapter.ts +0 -47
  200. package/src/modules/payments/repository/index.ts +0 -6
  201. package/src/modules/payments/service.ts +0 -107
  202. package/src/modules/pricing/repository/index.ts +0 -291
  203. package/src/modules/pricing/schema.ts +0 -71
  204. package/src/modules/pricing/schemas.ts +0 -38
  205. package/src/modules/pricing/service.ts +0 -494
  206. package/src/modules/promotions/repository/index.ts +0 -325
  207. package/src/modules/promotions/schema.ts +0 -62
  208. package/src/modules/promotions/schemas.ts +0 -38
  209. package/src/modules/promotions/service.ts +0 -598
  210. package/src/modules/search/adapter.ts +0 -57
  211. package/src/modules/search/hooks.ts +0 -12
  212. package/src/modules/search/repository/index.ts +0 -6
  213. package/src/modules/search/service.ts +0 -315
  214. package/src/modules/shipping/calculator.ts +0 -188
  215. package/src/modules/shipping/repository/index.ts +0 -6
  216. package/src/modules/shipping/service.ts +0 -51
  217. package/src/modules/tax/adapter.ts +0 -60
  218. package/src/modules/tax/repository/index.ts +0 -6
  219. package/src/modules/tax/service.ts +0 -53
  220. package/src/modules/webhooks/hook.ts +0 -34
  221. package/src/modules/webhooks/repository/index.ts +0 -278
  222. package/src/modules/webhooks/schema.ts +0 -56
  223. package/src/modules/webhooks/service.ts +0 -117
  224. package/src/modules/webhooks/signing.ts +0 -6
  225. package/src/modules/webhooks/ssrf-guard.ts +0 -71
  226. package/src/modules/webhooks/tasks.ts +0 -52
  227. package/src/modules/webhooks/worker.ts +0 -134
  228. package/src/runtime/commerce.ts +0 -145
  229. package/src/runtime/kernel.ts +0 -419
  230. package/src/runtime/logger.ts +0 -36
  231. package/src/runtime/server.ts +0 -349
  232. package/src/runtime/shutdown.ts +0 -43
  233. package/src/test-utils/create-pglite-adapter.ts +0 -129
  234. package/src/test-utils/create-plugin-test-app.ts +0 -128
  235. package/src/test-utils/create-repository-test-harness.ts +0 -16
  236. package/src/test-utils/create-test-config.ts +0 -190
  237. package/src/test-utils/create-test-kernel.ts +0 -7
  238. package/src/test-utils/create-test-plugin-context.ts +0 -75
  239. package/src/test-utils/rest-api-test-utils.ts +0 -265
  240. package/src/test-utils/test-actors.ts +0 -62
  241. package/src/test-utils/typed-hooks.ts +0 -54
  242. package/src/types/commerce-types.ts +0 -34
  243. package/src/utils/id.ts +0 -3
  244. package/src/utils/logger.ts +0 -18
  245. package/src/utils/pagination.ts +0 -22
@@ -1,287 +0,0 @@
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
- }
@@ -1,66 +0,0 @@
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
- ]);