@unifiedcommerce/core 0.1.0 → 0.1.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 (174) hide show
  1. package/package.json +1 -2
  2. package/src/adapters/console-email.ts +0 -43
  3. package/src/auth/access.ts +0 -187
  4. package/src/auth/auth-schema.ts +0 -139
  5. package/src/auth/middleware.ts +0 -161
  6. package/src/auth/org.ts +0 -41
  7. package/src/auth/permissions.ts +0 -28
  8. package/src/auth/setup.ts +0 -169
  9. package/src/auth/system-actor.ts +0 -19
  10. package/src/auth/types.ts +0 -10
  11. package/src/config/defaults.ts +0 -82
  12. package/src/config/define-config.ts +0 -53
  13. package/src/config/types.ts +0 -299
  14. package/src/generated/plugin-capabilities.d.ts +0 -20
  15. package/src/generated/plugin-manifest.ts +0 -23
  16. package/src/generated/plugin-repositories.d.ts +0 -20
  17. package/src/hooks/checkout-completion.ts +0 -262
  18. package/src/hooks/checkout.ts +0 -677
  19. package/src/hooks/order-emails.ts +0 -62
  20. package/src/index.ts +0 -214
  21. package/src/interfaces/mcp/agent-prompt.ts +0 -174
  22. package/src/interfaces/mcp/context-enrichment.ts +0 -177
  23. package/src/interfaces/mcp/server.ts +0 -617
  24. package/src/interfaces/mcp/transport.ts +0 -68
  25. package/src/interfaces/rest/customer-portal.ts +0 -299
  26. package/src/interfaces/rest/index.ts +0 -74
  27. package/src/interfaces/rest/router.ts +0 -334
  28. package/src/interfaces/rest/routes/admin-jobs.ts +0 -58
  29. package/src/interfaces/rest/routes/audit.ts +0 -50
  30. package/src/interfaces/rest/routes/carts.ts +0 -89
  31. package/src/interfaces/rest/routes/catalog.ts +0 -493
  32. package/src/interfaces/rest/routes/checkout.ts +0 -283
  33. package/src/interfaces/rest/routes/inventory.ts +0 -70
  34. package/src/interfaces/rest/routes/media.ts +0 -86
  35. package/src/interfaces/rest/routes/orders.ts +0 -78
  36. package/src/interfaces/rest/routes/payments.ts +0 -60
  37. package/src/interfaces/rest/routes/pricing.ts +0 -57
  38. package/src/interfaces/rest/routes/promotions.ts +0 -92
  39. package/src/interfaces/rest/routes/search.ts +0 -71
  40. package/src/interfaces/rest/routes/webhooks.ts +0 -46
  41. package/src/interfaces/rest/schemas/admin-jobs.ts +0 -40
  42. package/src/interfaces/rest/schemas/audit.ts +0 -46
  43. package/src/interfaces/rest/schemas/carts.ts +0 -125
  44. package/src/interfaces/rest/schemas/catalog.ts +0 -450
  45. package/src/interfaces/rest/schemas/checkout.ts +0 -66
  46. package/src/interfaces/rest/schemas/customer-portal.ts +0 -195
  47. package/src/interfaces/rest/schemas/inventory.ts +0 -138
  48. package/src/interfaces/rest/schemas/media.ts +0 -75
  49. package/src/interfaces/rest/schemas/orders.ts +0 -104
  50. package/src/interfaces/rest/schemas/pricing.ts +0 -80
  51. package/src/interfaces/rest/schemas/promotions.ts +0 -110
  52. package/src/interfaces/rest/schemas/responses.ts +0 -85
  53. package/src/interfaces/rest/schemas/search.ts +0 -58
  54. package/src/interfaces/rest/schemas/shared.ts +0 -62
  55. package/src/interfaces/rest/schemas/webhooks.ts +0 -68
  56. package/src/interfaces/rest/utils.ts +0 -104
  57. package/src/interfaces/rest/webhook-router.ts +0 -50
  58. package/src/kernel/compensation/executor.ts +0 -61
  59. package/src/kernel/compensation/types.ts +0 -26
  60. package/src/kernel/database/adapter.ts +0 -13
  61. package/src/kernel/database/drizzle-db.ts +0 -56
  62. package/src/kernel/database/migrate.ts +0 -76
  63. package/src/kernel/database/plugin-types.ts +0 -34
  64. package/src/kernel/database/schema.ts +0 -49
  65. package/src/kernel/database/scoped-db.ts +0 -68
  66. package/src/kernel/database/tx-context.ts +0 -46
  67. package/src/kernel/error-mapper.ts +0 -15
  68. package/src/kernel/errors.ts +0 -89
  69. package/src/kernel/factory/repository-factory.ts +0 -242
  70. package/src/kernel/hooks/create-context.ts +0 -43
  71. package/src/kernel/hooks/executor.ts +0 -88
  72. package/src/kernel/hooks/registry.ts +0 -74
  73. package/src/kernel/hooks/types.ts +0 -52
  74. package/src/kernel/http-error.ts +0 -44
  75. package/src/kernel/jobs/adapter.ts +0 -36
  76. package/src/kernel/jobs/drizzle-adapter.ts +0 -58
  77. package/src/kernel/jobs/runner.ts +0 -153
  78. package/src/kernel/jobs/schema.ts +0 -46
  79. package/src/kernel/jobs/types.ts +0 -30
  80. package/src/kernel/local-api.ts +0 -185
  81. package/src/kernel/plugin/manifest.ts +0 -253
  82. package/src/kernel/query/executor.ts +0 -184
  83. package/src/kernel/query/registry.ts +0 -46
  84. package/src/kernel/result.ts +0 -33
  85. package/src/kernel/schema/extra-columns.ts +0 -37
  86. package/src/kernel/service-registry.ts +0 -76
  87. package/src/kernel/service-timing.ts +0 -89
  88. package/src/kernel/state-machine/machine.ts +0 -101
  89. package/src/modules/analytics/drizzle-adapter.ts +0 -426
  90. package/src/modules/analytics/hooks.ts +0 -11
  91. package/src/modules/analytics/models.ts +0 -125
  92. package/src/modules/analytics/repository/index.ts +0 -6
  93. package/src/modules/analytics/service.ts +0 -245
  94. package/src/modules/analytics/types.ts +0 -180
  95. package/src/modules/audit/hooks.ts +0 -78
  96. package/src/modules/audit/schema.ts +0 -33
  97. package/src/modules/audit/service.ts +0 -151
  98. package/src/modules/cart/access.ts +0 -27
  99. package/src/modules/cart/matcher.ts +0 -26
  100. package/src/modules/cart/repository/index.ts +0 -234
  101. package/src/modules/cart/schema.ts +0 -42
  102. package/src/modules/cart/schemas.ts +0 -38
  103. package/src/modules/cart/service.ts +0 -541
  104. package/src/modules/catalog/repository/index.ts +0 -772
  105. package/src/modules/catalog/schema.ts +0 -203
  106. package/src/modules/catalog/schemas.ts +0 -104
  107. package/src/modules/catalog/service.ts +0 -1544
  108. package/src/modules/customers/repository/index.ts +0 -327
  109. package/src/modules/customers/schema.ts +0 -64
  110. package/src/modules/customers/service.ts +0 -171
  111. package/src/modules/fulfillment/repository/index.ts +0 -426
  112. package/src/modules/fulfillment/schema.ts +0 -101
  113. package/src/modules/fulfillment/service.ts +0 -555
  114. package/src/modules/fulfillment/types.ts +0 -59
  115. package/src/modules/inventory/repository/index.ts +0 -509
  116. package/src/modules/inventory/schema.ts +0 -94
  117. package/src/modules/inventory/schemas.ts +0 -38
  118. package/src/modules/inventory/service.ts +0 -490
  119. package/src/modules/media/adapter.ts +0 -17
  120. package/src/modules/media/repository/index.ts +0 -274
  121. package/src/modules/media/schema.ts +0 -41
  122. package/src/modules/media/service.ts +0 -151
  123. package/src/modules/orders/repository/index.ts +0 -287
  124. package/src/modules/orders/schema.ts +0 -66
  125. package/src/modules/orders/service.ts +0 -619
  126. package/src/modules/orders/stale-order-cleanup.ts +0 -76
  127. package/src/modules/organization/service.ts +0 -191
  128. package/src/modules/payments/adapter.ts +0 -47
  129. package/src/modules/payments/repository/index.ts +0 -6
  130. package/src/modules/payments/service.ts +0 -107
  131. package/src/modules/pricing/repository/index.ts +0 -291
  132. package/src/modules/pricing/schema.ts +0 -71
  133. package/src/modules/pricing/schemas.ts +0 -38
  134. package/src/modules/pricing/service.ts +0 -494
  135. package/src/modules/promotions/repository/index.ts +0 -325
  136. package/src/modules/promotions/schema.ts +0 -62
  137. package/src/modules/promotions/schemas.ts +0 -38
  138. package/src/modules/promotions/service.ts +0 -598
  139. package/src/modules/search/adapter.ts +0 -57
  140. package/src/modules/search/hooks.ts +0 -12
  141. package/src/modules/search/repository/index.ts +0 -6
  142. package/src/modules/search/service.ts +0 -315
  143. package/src/modules/shipping/calculator.ts +0 -188
  144. package/src/modules/shipping/repository/index.ts +0 -6
  145. package/src/modules/shipping/service.ts +0 -51
  146. package/src/modules/tax/adapter.ts +0 -60
  147. package/src/modules/tax/repository/index.ts +0 -6
  148. package/src/modules/tax/service.ts +0 -53
  149. package/src/modules/webhooks/hook.ts +0 -34
  150. package/src/modules/webhooks/repository/index.ts +0 -278
  151. package/src/modules/webhooks/schema.ts +0 -56
  152. package/src/modules/webhooks/service.ts +0 -117
  153. package/src/modules/webhooks/signing.ts +0 -6
  154. package/src/modules/webhooks/ssrf-guard.ts +0 -71
  155. package/src/modules/webhooks/tasks.ts +0 -52
  156. package/src/modules/webhooks/worker.ts +0 -134
  157. package/src/runtime/commerce.ts +0 -145
  158. package/src/runtime/kernel.ts +0 -419
  159. package/src/runtime/logger.ts +0 -36
  160. package/src/runtime/server.ts +0 -349
  161. package/src/runtime/shutdown.ts +0 -43
  162. package/src/test-utils/create-pglite-adapter.ts +0 -129
  163. package/src/test-utils/create-plugin-test-app.ts +0 -128
  164. package/src/test-utils/create-repository-test-harness.ts +0 -16
  165. package/src/test-utils/create-test-config.ts +0 -190
  166. package/src/test-utils/create-test-kernel.ts +0 -7
  167. package/src/test-utils/create-test-plugin-context.ts +0 -75
  168. package/src/test-utils/rest-api-test-utils.ts +0 -265
  169. package/src/test-utils/test-actors.ts +0 -62
  170. package/src/test-utils/typed-hooks.ts +0 -54
  171. package/src/types/commerce-types.ts +0 -34
  172. package/src/utils/id.ts +0 -3
  173. package/src/utils/logger.ts +0 -18
  174. 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
- ]);