@open-mercato/core 0.4.6-develop-f7d3079656 → 0.4.6-develop-0861f05ea9

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 (107) hide show
  1. package/dist/modules/currencies/backend/exchange-rates/[id]/page.js +17 -154
  2. package/dist/modules/currencies/backend/exchange-rates/[id]/page.js.map +3 -3
  3. package/dist/modules/currencies/backend/exchange-rates/create/page.js +14 -152
  4. package/dist/modules/currencies/backend/exchange-rates/create/page.js.map +2 -2
  5. package/dist/modules/currencies/lib/exchangeRateFormConfig.js +167 -0
  6. package/dist/modules/currencies/lib/exchangeRateFormConfig.js.map +7 -0
  7. package/dist/modules/customers/api/dashboard/widgets/utils.js +1 -34
  8. package/dist/modules/customers/api/dashboard/widgets/utils.js.map +2 -2
  9. package/dist/modules/customers/commands/activities.js +3 -8
  10. package/dist/modules/customers/commands/activities.js.map +2 -2
  11. package/dist/modules/customers/commands/comments.js +2 -8
  12. package/dist/modules/customers/commands/comments.js.map +2 -2
  13. package/dist/modules/dashboards/lib/widgetScope.js +38 -0
  14. package/dist/modules/dashboards/lib/widgetScope.js.map +7 -0
  15. package/dist/modules/entities/lib/makeActivityRoute.js +265 -0
  16. package/dist/modules/entities/lib/makeActivityRoute.js.map +7 -0
  17. package/dist/modules/resources/api/activities.js +24 -232
  18. package/dist/modules/resources/api/activities.js.map +2 -2
  19. package/dist/modules/resources/commands/activities.js +3 -8
  20. package/dist/modules/resources/commands/activities.js.map +2 -2
  21. package/dist/modules/resources/commands/comments.js +2 -8
  22. package/dist/modules/resources/commands/comments.js.map +2 -2
  23. package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js +27 -182
  24. package/dist/modules/sales/api/dashboard/widgets/new-orders/route.js.map +2 -2
  25. package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js +28 -183
  26. package/dist/modules/sales/api/dashboard/widgets/new-quotes/route.js.map +2 -2
  27. package/dist/modules/sales/api/order-line-statuses/route.js +15 -194
  28. package/dist/modules/sales/api/order-line-statuses/route.js.map +2 -2
  29. package/dist/modules/sales/api/order-lines/route.js +15 -281
  30. package/dist/modules/sales/api/order-lines/route.js.map +2 -2
  31. package/dist/modules/sales/api/order-statuses/route.js +15 -194
  32. package/dist/modules/sales/api/order-statuses/route.js.map +2 -2
  33. package/dist/modules/sales/api/payment-statuses/route.js +15 -194
  34. package/dist/modules/sales/api/payment-statuses/route.js.map +2 -2
  35. package/dist/modules/sales/api/quote-lines/route.js +15 -279
  36. package/dist/modules/sales/api/quote-lines/route.js.map +2 -2
  37. package/dist/modules/sales/api/shipment-statuses/route.js +15 -194
  38. package/dist/modules/sales/api/shipment-statuses/route.js.map +2 -2
  39. package/dist/modules/sales/components/PaymentMethodsSettings.js +3 -84
  40. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  41. package/dist/modules/sales/components/ProviderFieldInput.js +86 -0
  42. package/dist/modules/sales/components/ProviderFieldInput.js.map +7 -0
  43. package/dist/modules/sales/components/ShippingMethodsSettings.js +3 -82
  44. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  45. package/dist/modules/sales/lib/makeSalesLineRoute.js +308 -0
  46. package/dist/modules/sales/lib/makeSalesLineRoute.js.map +7 -0
  47. package/dist/modules/sales/lib/makeStatusDictionaryRoute.js +206 -0
  48. package/dist/modules/sales/lib/makeStatusDictionaryRoute.js.map +7 -0
  49. package/dist/modules/sales/widgets/dashboard/makeDashboardWidgetRoute.js +178 -0
  50. package/dist/modules/sales/widgets/dashboard/makeDashboardWidgetRoute.js.map +7 -0
  51. package/dist/modules/sales/widgets/dashboard/new-orders/widget.client.js +1 -39
  52. package/dist/modules/sales/widgets/dashboard/new-orders/widget.client.js.map +2 -2
  53. package/dist/modules/sales/widgets/dashboard/new-quotes/widget.client.js +1 -39
  54. package/dist/modules/sales/widgets/dashboard/new-quotes/widget.client.js.map +2 -2
  55. package/dist/modules/sales/widgets/dashboard/shared.js +46 -0
  56. package/dist/modules/sales/widgets/dashboard/shared.js.map +7 -0
  57. package/dist/modules/staff/api/activities.js +24 -232
  58. package/dist/modules/staff/api/activities.js.map +2 -2
  59. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js +14 -34
  60. package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js.map +2 -2
  61. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js +15 -34
  62. package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js.map +2 -2
  63. package/dist/modules/staff/commands/activities.js +3 -8
  64. package/dist/modules/staff/commands/activities.js.map +2 -2
  65. package/dist/modules/staff/commands/comments.js +2 -8
  66. package/dist/modules/staff/commands/comments.js.map +2 -2
  67. package/dist/modules/staff/lib/leaveRequestHelpers.js +41 -0
  68. package/dist/modules/staff/lib/leaveRequestHelpers.js.map +7 -0
  69. package/package.json +2 -2
  70. package/src/modules/currencies/backend/exchange-rates/[id]/page.tsx +20 -180
  71. package/src/modules/currencies/backend/exchange-rates/create/page.tsx +16 -175
  72. package/src/modules/currencies/lib/exchangeRateFormConfig.ts +200 -0
  73. package/src/modules/customers/api/dashboard/widgets/utils.ts +1 -53
  74. package/src/modules/customers/commands/activities.ts +2 -8
  75. package/src/modules/customers/commands/comments.ts +2 -8
  76. package/src/modules/dashboards/i18n/de.json +3 -0
  77. package/src/modules/dashboards/i18n/en.json +3 -0
  78. package/src/modules/dashboards/i18n/es.json +3 -0
  79. package/src/modules/dashboards/i18n/pl.json +3 -0
  80. package/src/modules/dashboards/lib/widgetScope.ts +53 -0
  81. package/src/modules/entities/lib/makeActivityRoute.ts +327 -0
  82. package/src/modules/resources/api/activities.ts +25 -269
  83. package/src/modules/resources/commands/activities.ts +2 -7
  84. package/src/modules/resources/commands/comments.ts +2 -8
  85. package/src/modules/sales/api/dashboard/widgets/new-orders/route.ts +29 -244
  86. package/src/modules/sales/api/dashboard/widgets/new-quotes/route.ts +30 -245
  87. package/src/modules/sales/api/order-line-statuses/route.ts +16 -209
  88. package/src/modules/sales/api/order-lines/route.ts +16 -300
  89. package/src/modules/sales/api/order-statuses/route.ts +16 -209
  90. package/src/modules/sales/api/payment-statuses/route.ts +16 -209
  91. package/src/modules/sales/api/quote-lines/route.ts +16 -298
  92. package/src/modules/sales/api/shipment-statuses/route.ts +16 -209
  93. package/src/modules/sales/components/PaymentMethodsSettings.tsx +3 -88
  94. package/src/modules/sales/components/ProviderFieldInput.tsx +85 -0
  95. package/src/modules/sales/components/ShippingMethodsSettings.tsx +3 -86
  96. package/src/modules/sales/lib/makeSalesLineRoute.ts +345 -0
  97. package/src/modules/sales/lib/makeStatusDictionaryRoute.ts +229 -0
  98. package/src/modules/sales/widgets/dashboard/makeDashboardWidgetRoute.ts +247 -0
  99. package/src/modules/sales/widgets/dashboard/new-orders/widget.client.tsx +7 -50
  100. package/src/modules/sales/widgets/dashboard/new-quotes/widget.client.tsx +7 -49
  101. package/src/modules/sales/widgets/dashboard/shared.ts +44 -0
  102. package/src/modules/staff/api/activities.ts +25 -269
  103. package/src/modules/staff/backend/staff/leave-requests/[id]/page.tsx +15 -69
  104. package/src/modules/staff/backend/staff/my-leave-requests/[id]/page.tsx +16 -65
  105. package/src/modules/staff/commands/activities.ts +2 -7
  106. package/src/modules/staff/commands/comments.ts +2 -8
  107. package/src/modules/staff/lib/leaveRequestHelpers.ts +78 -0
@@ -0,0 +1,308 @@
1
+ import { z } from "zod";
2
+ import { makeCrudRoute } from "@open-mercato/shared/lib/crud/factory";
3
+ import { resolveTranslations } from "@open-mercato/shared/lib/i18n/server";
4
+ import { CrudHttpError } from "@open-mercato/shared/lib/crud/errors";
5
+ import {
6
+ buildCustomFieldFiltersFromQuery,
7
+ extractAllCustomFieldEntries
8
+ } from "@open-mercato/shared/lib/crud/custom-fields";
9
+ import {
10
+ canonicalizeUnitCode,
11
+ REFERENCE_UNIT_CODES
12
+ } from "@open-mercato/shared/lib/units/unitCodes";
13
+ import {
14
+ createPagedListResponseSchema,
15
+ createSalesCrudOpenApi,
16
+ defaultOkResponseSchema
17
+ } from "../api/openapi.js";
18
+ import { withScopedPayload } from "../api/utils.js";
19
+ const rawBodySchema = z.object({}).passthrough();
20
+ function resolveRawBody(raw) {
21
+ if (!raw || typeof raw !== "object") return {};
22
+ if ("body" in raw) {
23
+ const payload = raw;
24
+ if (payload.body && typeof payload.body === "object") {
25
+ return payload.body;
26
+ }
27
+ }
28
+ return raw;
29
+ }
30
+ function transformItem(item) {
31
+ if (!item) return item;
32
+ const normalized = { ...item };
33
+ const cfEntries = extractAllCustomFieldEntries(item);
34
+ for (const key of Object.keys(normalized)) {
35
+ if (key.startsWith("cf:")) delete normalized[key];
36
+ }
37
+ const quantityUnit = canonicalizeUnitCode(
38
+ normalized["quantity_unit"] ?? normalized["quantityUnit"]
39
+ );
40
+ const normalizedUnit = canonicalizeUnitCode(
41
+ normalized["normalized_unit"] ?? normalized["normalizedUnit"]
42
+ ) ?? quantityUnit;
43
+ return {
44
+ ...normalized,
45
+ quantity_unit: quantityUnit,
46
+ normalized_unit: normalizedUnit,
47
+ ...cfEntries
48
+ };
49
+ }
50
+ const uomSnapshotOpenApiSchema = z.object({
51
+ version: z.literal(1),
52
+ productId: z.string().nullable(),
53
+ productVariantId: z.string().nullable(),
54
+ baseUnitCode: z.string().nullable(),
55
+ enteredUnitCode: z.string().nullable(),
56
+ enteredQuantity: z.string(),
57
+ toBaseFactor: z.string(),
58
+ normalizedQuantity: z.string(),
59
+ rounding: z.object({
60
+ mode: z.enum(["half_up", "down", "up"]),
61
+ scale: z.number().int()
62
+ }),
63
+ source: z.object({
64
+ conversionId: z.string().nullable(),
65
+ resolvedAt: z.string()
66
+ }),
67
+ unitPriceReference: z.object({
68
+ enabled: z.boolean(),
69
+ referenceUnitCode: z.enum(REFERENCE_UNIT_CODES).nullable(),
70
+ baseQuantity: z.string().nullable(),
71
+ grossPerReference: z.string().nullable().optional(),
72
+ netPerReference: z.string().nullable().optional()
73
+ }).optional()
74
+ }).nullable().optional();
75
+ function makeSalesLineRoute(config) {
76
+ const {
77
+ entity,
78
+ entityId,
79
+ fieldConstants: F,
80
+ parentFkColumn,
81
+ parentFkParam,
82
+ createSchema,
83
+ features,
84
+ commandPrefix
85
+ } = config;
86
+ const listSchema = z.object({
87
+ page: z.coerce.number().min(1).default(1),
88
+ pageSize: z.coerce.number().min(1).max(100).default(50),
89
+ id: z.string().uuid().optional(),
90
+ [parentFkParam]: z.string().uuid().optional(),
91
+ sortField: z.string().optional(),
92
+ sortDir: z.enum(["asc", "desc"]).optional()
93
+ }).passthrough();
94
+ const upsertSchema = createSchema.extend({
95
+ id: z.string().uuid().optional()
96
+ });
97
+ const deleteSchema = z.object({
98
+ id: z.string().uuid(),
99
+ [parentFkParam]: z.string().uuid()
100
+ });
101
+ const routeMetadata = {
102
+ GET: { requireAuth: true, requireFeatures: [features.view] },
103
+ POST: { requireAuth: true, requireFeatures: [features.manage] },
104
+ PUT: { requireAuth: true, requireFeatures: [features.manage] },
105
+ DELETE: { requireAuth: true, requireFeatures: [features.manage] }
106
+ };
107
+ const crud = makeCrudRoute({
108
+ metadata: routeMetadata,
109
+ orm: {
110
+ entity,
111
+ idField: "id",
112
+ orgField: "organizationId",
113
+ tenantField: "tenantId",
114
+ softDeleteField: "deletedAt"
115
+ },
116
+ indexer: {
117
+ entityType: entityId
118
+ },
119
+ list: {
120
+ schema: listSchema,
121
+ entityId,
122
+ fields: [
123
+ F.id,
124
+ parentFkColumn,
125
+ F.line_number,
126
+ F.kind,
127
+ F.status_entry_id,
128
+ F.status,
129
+ F.product_id,
130
+ F.product_variant_id,
131
+ F.catalog_snapshot,
132
+ F.name,
133
+ F.description,
134
+ F.comment,
135
+ F.organization_id,
136
+ F.tenant_id,
137
+ F.quantity,
138
+ F.quantity_unit,
139
+ F.normalized_quantity,
140
+ F.normalized_unit,
141
+ F.uom_snapshot,
142
+ F.currency_code,
143
+ F.unit_price_net,
144
+ F.unit_price_gross,
145
+ F.discount_amount,
146
+ F.discount_percent,
147
+ F.tax_rate,
148
+ F.tax_amount,
149
+ F.total_net_amount,
150
+ F.total_gross_amount,
151
+ F.configuration,
152
+ F.promotion_code,
153
+ F.promotion_snapshot,
154
+ F.metadata,
155
+ F.custom_field_set_id,
156
+ F.created_at,
157
+ F.updated_at
158
+ ],
159
+ sortFieldMap: {
160
+ createdAt: F.created_at,
161
+ updatedAt: F.updated_at,
162
+ lineNumber: F.line_number
163
+ },
164
+ buildFilters: async (query, ctx) => {
165
+ const filters = {};
166
+ if (query.id) filters.id = { $eq: query.id };
167
+ if (query[parentFkParam]) filters[parentFkColumn] = { $eq: query[parentFkParam] };
168
+ try {
169
+ const em = ctx.container.resolve("em");
170
+ const cfFilters = await buildCustomFieldFiltersFromQuery({
171
+ entityId,
172
+ query,
173
+ em,
174
+ tenantId: ctx.auth?.tenantId ?? null
175
+ });
176
+ Object.assign(filters, cfFilters);
177
+ } catch {
178
+ }
179
+ return filters;
180
+ },
181
+ transformItem
182
+ },
183
+ actions: {
184
+ create: {
185
+ commandId: `${commandPrefix}.upsert`,
186
+ schema: rawBodySchema,
187
+ mapInput: async ({ raw, ctx }) => {
188
+ const { translate } = await resolveTranslations();
189
+ const payload = upsertSchema.parse(
190
+ withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate)
191
+ );
192
+ return { body: payload };
193
+ },
194
+ response: ({ result }) => ({
195
+ id: result?.lineId ?? null,
196
+ [parentFkParam]: result?.[parentFkParam] ?? null
197
+ }),
198
+ status: 201
199
+ },
200
+ update: {
201
+ commandId: `${commandPrefix}.upsert`,
202
+ schema: rawBodySchema,
203
+ mapInput: async ({ raw, ctx }) => {
204
+ const { translate } = await resolveTranslations();
205
+ const payload = upsertSchema.parse(
206
+ withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate)
207
+ );
208
+ return { body: payload };
209
+ },
210
+ response: ({ result }) => ({
211
+ id: result?.lineId ?? null,
212
+ [parentFkParam]: result?.[parentFkParam] ?? null
213
+ })
214
+ },
215
+ delete: {
216
+ commandId: `${commandPrefix}.delete`,
217
+ schema: rawBodySchema,
218
+ mapInput: async ({ raw, ctx }) => {
219
+ const { translate } = await resolveTranslations();
220
+ const payload = deleteSchema.parse(
221
+ withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate)
222
+ );
223
+ if (!payload.id || !payload[parentFkParam]) {
224
+ throw new CrudHttpError(400, {
225
+ error: translate(
226
+ "sales.documents.detail.error",
227
+ "Document not found or inaccessible."
228
+ )
229
+ });
230
+ }
231
+ return { body: payload };
232
+ },
233
+ response: () => ({ ok: true })
234
+ }
235
+ }
236
+ });
237
+ const lineItemSchema = z.object({
238
+ id: z.string().uuid(),
239
+ [parentFkColumn]: z.string().uuid(),
240
+ line_number: z.number(),
241
+ kind: z.string(),
242
+ status_entry_id: z.string().uuid().nullable().optional(),
243
+ status: z.string().nullable().optional(),
244
+ product_id: z.string().uuid().nullable().optional(),
245
+ product_variant_id: z.string().uuid().nullable().optional(),
246
+ catalog_snapshot: z.record(z.string(), z.unknown()).nullable().optional(),
247
+ name: z.string().nullable().optional(),
248
+ description: z.string().nullable().optional(),
249
+ comment: z.string().nullable().optional(),
250
+ quantity: z.number(),
251
+ quantity_unit: z.string().nullable().optional(),
252
+ normalized_quantity: z.number(),
253
+ normalized_unit: z.string().nullable().optional(),
254
+ uom_snapshot: uomSnapshotOpenApiSchema,
255
+ currency_code: z.string(),
256
+ unit_price_net: z.number(),
257
+ unit_price_gross: z.number(),
258
+ discount_amount: z.number(),
259
+ discount_percent: z.number(),
260
+ tax_rate: z.number(),
261
+ tax_amount: z.number(),
262
+ total_net_amount: z.number(),
263
+ total_gross_amount: z.number(),
264
+ configuration: z.record(z.string(), z.unknown()).nullable().optional(),
265
+ promotion_code: z.string().nullable().optional(),
266
+ promotion_snapshot: z.record(z.string(), z.unknown()).nullable().optional(),
267
+ metadata: z.record(z.string(), z.unknown()).nullable().optional(),
268
+ custom_field_set_id: z.string().uuid().nullable().optional(),
269
+ created_at: z.string(),
270
+ updated_at: z.string()
271
+ });
272
+ const upsertResponseSchema = z.object({
273
+ id: z.string().uuid().nullable(),
274
+ [parentFkParam]: z.string().uuid().nullable()
275
+ });
276
+ const openApi = createSalesCrudOpenApi({
277
+ resourceName: config.openApi.resourceName,
278
+ querySchema: listSchema,
279
+ listResponseSchema: createPagedListResponseSchema(lineItemSchema),
280
+ create: {
281
+ schema: upsertSchema,
282
+ responseSchema: upsertResponseSchema,
283
+ description: `Creates ${config.openApi.description}.`
284
+ },
285
+ update: {
286
+ schema: upsertSchema,
287
+ responseSchema: upsertResponseSchema,
288
+ description: `Updates ${config.openApi.description}.`
289
+ },
290
+ del: {
291
+ schema: deleteSchema,
292
+ responseSchema: defaultOkResponseSchema,
293
+ description: `Deletes ${config.openApi.description}.`
294
+ }
295
+ });
296
+ return {
297
+ metadata: routeMetadata,
298
+ openApi,
299
+ GET: crud.GET,
300
+ POST: crud.POST,
301
+ PUT: crud.PUT,
302
+ DELETE: crud.DELETE
303
+ };
304
+ }
305
+ export {
306
+ makeSalesLineRoute
307
+ };
308
+ //# sourceMappingURL=makeSalesLineRoute.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/modules/sales/lib/makeSalesLineRoute.ts"],
4
+ "sourcesContent": ["import { z } from 'zod'\nimport { makeCrudRoute, type CrudCtx } from '@open-mercato/shared/lib/crud/factory'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport {\n buildCustomFieldFiltersFromQuery,\n extractAllCustomFieldEntries,\n} from '@open-mercato/shared/lib/crud/custom-fields'\nimport {\n canonicalizeUnitCode,\n REFERENCE_UNIT_CODES,\n} from '@open-mercato/shared/lib/units/unitCodes'\nimport {\n createPagedListResponseSchema,\n createSalesCrudOpenApi,\n defaultOkResponseSchema,\n} from '../api/openapi'\nimport { withScopedPayload } from '../api/utils'\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any -- MikroORM entity class constructor\ntype EntityClass = new (...args: any[]) => unknown\n\ninterface SalesLineRouteConfig {\n entity: EntityClass\n entityId: string\n fieldConstants: Record<string, string>\n parentFkColumn: string\n parentFkParam: string\n createSchema: z.ZodObject<z.ZodRawShape>\n features: { view: string; manage: string }\n commandPrefix: string\n openApi: {\n resourceName: string\n description: string\n }\n}\n\nconst rawBodySchema = z.object({}).passthrough()\n\nfunction resolveRawBody(raw: unknown): Record<string, unknown> {\n if (!raw || typeof raw !== 'object') return {}\n if ('body' in raw) {\n const payload = raw as { body?: unknown }\n if (payload.body && typeof payload.body === 'object') {\n return payload.body as Record<string, unknown>\n }\n }\n return raw as Record<string, unknown>\n}\n\nfunction transformItem(item: Record<string, unknown> | null | undefined) {\n if (!item) return item\n const normalized = { ...item }\n const cfEntries = extractAllCustomFieldEntries(item)\n for (const key of Object.keys(normalized)) {\n if (key.startsWith('cf:')) delete normalized[key]\n }\n const quantityUnit = canonicalizeUnitCode(\n normalized['quantity_unit'] ?? normalized['quantityUnit'],\n )\n const normalizedUnit =\n canonicalizeUnitCode(\n normalized['normalized_unit'] ?? normalized['normalizedUnit'],\n ) ?? quantityUnit\n return {\n ...normalized,\n quantity_unit: quantityUnit,\n normalized_unit: normalizedUnit,\n ...cfEntries,\n }\n}\n\nconst uomSnapshotOpenApiSchema = z\n .object({\n version: z.literal(1),\n productId: z.string().nullable(),\n productVariantId: z.string().nullable(),\n baseUnitCode: z.string().nullable(),\n enteredUnitCode: z.string().nullable(),\n enteredQuantity: z.string(),\n toBaseFactor: z.string(),\n normalizedQuantity: z.string(),\n rounding: z.object({\n mode: z.enum(['half_up', 'down', 'up']),\n scale: z.number().int(),\n }),\n source: z.object({\n conversionId: z.string().nullable(),\n resolvedAt: z.string(),\n }),\n unitPriceReference: z\n .object({\n enabled: z.boolean(),\n referenceUnitCode: z.enum(REFERENCE_UNIT_CODES).nullable(),\n baseQuantity: z.string().nullable(),\n grossPerReference: z.string().nullable().optional(),\n netPerReference: z.string().nullable().optional(),\n })\n .optional(),\n })\n .nullable()\n .optional()\n\nexport function makeSalesLineRoute(config: SalesLineRouteConfig) {\n const {\n entity,\n entityId,\n fieldConstants: F,\n parentFkColumn,\n parentFkParam,\n createSchema,\n features,\n commandPrefix,\n } = config\n\n const listSchema = z\n .object({\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n id: z.string().uuid().optional(),\n [parentFkParam]: z.string().uuid().optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional(),\n })\n .passthrough()\n\n const upsertSchema = createSchema.extend({\n id: z.string().uuid().optional(),\n })\n\n const deleteSchema = z.object({\n id: z.string().uuid(),\n [parentFkParam]: z.string().uuid(),\n })\n\n const routeMetadata = {\n GET: { requireAuth: true, requireFeatures: [features.view] },\n POST: { requireAuth: true, requireFeatures: [features.manage] },\n PUT: { requireAuth: true, requireFeatures: [features.manage] },\n DELETE: { requireAuth: true, requireFeatures: [features.manage] },\n }\n\n const crud = makeCrudRoute({\n metadata: routeMetadata,\n orm: {\n entity,\n idField: 'id',\n orgField: 'organizationId',\n tenantField: 'tenantId',\n softDeleteField: 'deletedAt',\n },\n indexer: {\n entityType: entityId,\n },\n list: {\n schema: listSchema,\n entityId,\n fields: [\n F.id,\n parentFkColumn,\n F.line_number,\n F.kind,\n F.status_entry_id,\n F.status,\n F.product_id,\n F.product_variant_id,\n F.catalog_snapshot,\n F.name,\n F.description,\n F.comment,\n F.organization_id,\n F.tenant_id,\n F.quantity,\n F.quantity_unit,\n F.normalized_quantity,\n F.normalized_unit,\n F.uom_snapshot,\n F.currency_code,\n F.unit_price_net,\n F.unit_price_gross,\n F.discount_amount,\n F.discount_percent,\n F.tax_rate,\n F.tax_amount,\n F.total_net_amount,\n F.total_gross_amount,\n F.configuration,\n F.promotion_code,\n F.promotion_snapshot,\n F.metadata,\n F.custom_field_set_id,\n F.created_at,\n F.updated_at,\n ],\n sortFieldMap: {\n createdAt: F.created_at,\n updatedAt: F.updated_at,\n lineNumber: F.line_number,\n },\n buildFilters: async (query: Record<string, unknown>, ctx: CrudCtx) => {\n const filters: Record<string, unknown> = {}\n if (query.id) filters.id = { $eq: query.id }\n if (query[parentFkParam]) filters[parentFkColumn] = { $eq: query[parentFkParam] }\n try {\n const em = ctx.container.resolve('em')\n const cfFilters = await buildCustomFieldFiltersFromQuery({\n entityId,\n query,\n em,\n tenantId: ctx.auth?.tenantId ?? null,\n })\n Object.assign(filters, cfFilters)\n } catch {\n // ignore\n }\n return filters\n },\n transformItem,\n },\n actions: {\n create: {\n commandId: `${commandPrefix}.upsert`,\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }: { raw: unknown; ctx: CrudCtx }) => {\n const { translate } = await resolveTranslations()\n const payload = upsertSchema.parse(\n withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate),\n )\n return { body: payload }\n },\n response: ({ result }: { result: Record<string, unknown> | null }) => ({\n id: result?.lineId ?? null,\n [parentFkParam]: result?.[parentFkParam] ?? null,\n }),\n status: 201,\n },\n update: {\n commandId: `${commandPrefix}.upsert`,\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }: { raw: unknown; ctx: CrudCtx }) => {\n const { translate } = await resolveTranslations()\n const payload = upsertSchema.parse(\n withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate),\n )\n return { body: payload }\n },\n response: ({ result }: { result: Record<string, unknown> | null }) => ({\n id: result?.lineId ?? null,\n [parentFkParam]: result?.[parentFkParam] ?? null,\n }),\n },\n delete: {\n commandId: `${commandPrefix}.delete`,\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }: { raw: unknown; ctx: CrudCtx }) => {\n const { translate } = await resolveTranslations()\n const payload = deleteSchema.parse(\n withScopedPayload(resolveRawBody(raw) ?? {}, ctx, translate),\n )\n if (!payload.id || !payload[parentFkParam]) {\n throw new CrudHttpError(400, {\n error: translate(\n 'sales.documents.detail.error',\n 'Document not found or inaccessible.',\n ),\n })\n }\n return { body: payload }\n },\n response: () => ({ ok: true }),\n },\n },\n })\n\n const lineItemSchema = z.object({\n id: z.string().uuid(),\n [parentFkColumn]: z.string().uuid(),\n line_number: z.number(),\n kind: z.string(),\n status_entry_id: z.string().uuid().nullable().optional(),\n status: z.string().nullable().optional(),\n product_id: z.string().uuid().nullable().optional(),\n product_variant_id: z.string().uuid().nullable().optional(),\n catalog_snapshot: z.record(z.string(), z.unknown()).nullable().optional(),\n name: z.string().nullable().optional(),\n description: z.string().nullable().optional(),\n comment: z.string().nullable().optional(),\n quantity: z.number(),\n quantity_unit: z.string().nullable().optional(),\n normalized_quantity: z.number(),\n normalized_unit: z.string().nullable().optional(),\n uom_snapshot: uomSnapshotOpenApiSchema,\n currency_code: z.string(),\n unit_price_net: z.number(),\n unit_price_gross: z.number(),\n discount_amount: z.number(),\n discount_percent: z.number(),\n tax_rate: z.number(),\n tax_amount: z.number(),\n total_net_amount: z.number(),\n total_gross_amount: z.number(),\n configuration: z.record(z.string(), z.unknown()).nullable().optional(),\n promotion_code: z.string().nullable().optional(),\n promotion_snapshot: z.record(z.string(), z.unknown()).nullable().optional(),\n metadata: z.record(z.string(), z.unknown()).nullable().optional(),\n custom_field_set_id: z.string().uuid().nullable().optional(),\n created_at: z.string(),\n updated_at: z.string(),\n })\n\n const upsertResponseSchema = z.object({\n id: z.string().uuid().nullable(),\n [parentFkParam]: z.string().uuid().nullable(),\n })\n\n const openApi = createSalesCrudOpenApi({\n resourceName: config.openApi.resourceName,\n querySchema: listSchema,\n listResponseSchema: createPagedListResponseSchema(lineItemSchema),\n create: {\n schema: upsertSchema,\n responseSchema: upsertResponseSchema,\n description: `Creates ${config.openApi.description}.`,\n },\n update: {\n schema: upsertSchema,\n responseSchema: upsertResponseSchema,\n description: `Updates ${config.openApi.description}.`,\n },\n del: {\n schema: deleteSchema,\n responseSchema: defaultOkResponseSchema,\n description: `Deletes ${config.openApi.description}.`,\n },\n })\n\n return {\n metadata: routeMetadata,\n openApi,\n GET: crud.GET,\n POST: crud.POST,\n PUT: crud.PUT,\n DELETE: crud.DELETE,\n }\n}\n"],
5
+ "mappings": "AAAA,SAAS,SAAS;AAClB,SAAS,qBAAmC;AAC5C,SAAS,2BAA2B;AACpC,SAAS,qBAAqB;AAC9B;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,yBAAyB;AAoBlC,MAAM,gBAAgB,EAAE,OAAO,CAAC,CAAC,EAAE,YAAY;AAE/C,SAAS,eAAe,KAAuC;AAC7D,MAAI,CAAC,OAAO,OAAO,QAAQ,SAAU,QAAO,CAAC;AAC7C,MAAI,UAAU,KAAK;AACjB,UAAM,UAAU;AAChB,QAAI,QAAQ,QAAQ,OAAO,QAAQ,SAAS,UAAU;AACpD,aAAO,QAAQ;AAAA,IACjB;AAAA,EACF;AACA,SAAO;AACT;AAEA,SAAS,cAAc,MAAkD;AACvE,MAAI,CAAC,KAAM,QAAO;AAClB,QAAM,aAAa,EAAE,GAAG,KAAK;AAC7B,QAAM,YAAY,6BAA6B,IAAI;AACnD,aAAW,OAAO,OAAO,KAAK,UAAU,GAAG;AACzC,QAAI,IAAI,WAAW,KAAK,EAAG,QAAO,WAAW,GAAG;AAAA,EAClD;AACA,QAAM,eAAe;AAAA,IACnB,WAAW,eAAe,KAAK,WAAW,cAAc;AAAA,EAC1D;AACA,QAAM,iBACJ;AAAA,IACE,WAAW,iBAAiB,KAAK,WAAW,gBAAgB;AAAA,EAC9D,KAAK;AACP,SAAO;AAAA,IACL,GAAG;AAAA,IACH,eAAe;AAAA,IACf,iBAAiB;AAAA,IACjB,GAAG;AAAA,EACL;AACF;AAEA,MAAM,2BAA2B,EAC9B,OAAO;AAAA,EACN,SAAS,EAAE,QAAQ,CAAC;AAAA,EACpB,WAAW,EAAE,OAAO,EAAE,SAAS;AAAA,EAC/B,kBAAkB,EAAE,OAAO,EAAE,SAAS;AAAA,EACtC,cAAc,EAAE,OAAO,EAAE,SAAS;AAAA,EAClC,iBAAiB,EAAE,OAAO,EAAE,SAAS;AAAA,EACrC,iBAAiB,EAAE,OAAO;AAAA,EAC1B,cAAc,EAAE,OAAO;AAAA,EACvB,oBAAoB,EAAE,OAAO;AAAA,EAC7B,UAAU,EAAE,OAAO;AAAA,IACjB,MAAM,EAAE,KAAK,CAAC,WAAW,QAAQ,IAAI,CAAC;AAAA,IACtC,OAAO,EAAE,OAAO,EAAE,IAAI;AAAA,EACxB,CAAC;AAAA,EACD,QAAQ,EAAE,OAAO;AAAA,IACf,cAAc,EAAE,OAAO,EAAE,SAAS;AAAA,IAClC,YAAY,EAAE,OAAO;AAAA,EACvB,CAAC;AAAA,EACD,oBAAoB,EACjB,OAAO;AAAA,IACN,SAAS,EAAE,QAAQ;AAAA,IACnB,mBAAmB,EAAE,KAAK,oBAAoB,EAAE,SAAS;AAAA,IACzD,cAAc,EAAE,OAAO,EAAE,SAAS;AAAA,IAClC,mBAAmB,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IAClD,iBAAiB,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,EAClD,CAAC,EACA,SAAS;AACd,CAAC,EACA,SAAS,EACT,SAAS;AAEL,SAAS,mBAAmB,QAA8B;AAC/D,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA,gBAAgB;AAAA,IAChB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,IAAI;AAEJ,QAAM,aAAa,EAChB,OAAO;AAAA,IACN,MAAM,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,QAAQ,CAAC;AAAA,IACxC,UAAU,EAAE,OAAO,OAAO,EAAE,IAAI,CAAC,EAAE,IAAI,GAAG,EAAE,QAAQ,EAAE;AAAA,IACtD,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS;AAAA,IAC/B,CAAC,aAAa,GAAG,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS;AAAA,IAC5C,WAAW,EAAE,OAAO,EAAE,SAAS;AAAA,IAC/B,SAAS,EAAE,KAAK,CAAC,OAAO,MAAM,CAAC,EAAE,SAAS;AAAA,EAC5C,CAAC,EACA,YAAY;AAEf,QAAM,eAAe,aAAa,OAAO;AAAA,IACvC,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS;AAAA,EACjC,CAAC;AAED,QAAM,eAAe,EAAE,OAAO;AAAA,IAC5B,IAAI,EAAE,OAAO,EAAE,KAAK;AAAA,IACpB,CAAC,aAAa,GAAG,EAAE,OAAO,EAAE,KAAK;AAAA,EACnC,CAAC;AAED,QAAM,gBAAgB;AAAA,IACpB,KAAK,EAAE,aAAa,MAAM,iBAAiB,CAAC,SAAS,IAAI,EAAE;AAAA,IAC3D,MAAM,EAAE,aAAa,MAAM,iBAAiB,CAAC,SAAS,MAAM,EAAE;AAAA,IAC9D,KAAK,EAAE,aAAa,MAAM,iBAAiB,CAAC,SAAS,MAAM,EAAE;AAAA,IAC7D,QAAQ,EAAE,aAAa,MAAM,iBAAiB,CAAC,SAAS,MAAM,EAAE;AAAA,EAClE;AAEA,QAAM,OAAO,cAAc;AAAA,IACzB,UAAU;AAAA,IACV,KAAK;AAAA,MACH;AAAA,MACA,SAAS;AAAA,MACT,UAAU;AAAA,MACV,aAAa;AAAA,MACb,iBAAiB;AAAA,IACnB;AAAA,IACA,SAAS;AAAA,MACP,YAAY;AAAA,IACd;AAAA,IACA,MAAM;AAAA,MACJ,QAAQ;AAAA,MACR;AAAA,MACA,QAAQ;AAAA,QACN,EAAE;AAAA,QACF;AAAA,QACA,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,QACF,EAAE;AAAA,MACJ;AAAA,MACA,cAAc;AAAA,QACZ,WAAW,EAAE;AAAA,QACb,WAAW,EAAE;AAAA,QACb,YAAY,EAAE;AAAA,MAChB;AAAA,MACA,cAAc,OAAO,OAAgC,QAAiB;AACpE,cAAM,UAAmC,CAAC;AAC1C,YAAI,MAAM,GAAI,SAAQ,KAAK,EAAE,KAAK,MAAM,GAAG;AAC3C,YAAI,MAAM,aAAa,EAAG,SAAQ,cAAc,IAAI,EAAE,KAAK,MAAM,aAAa,EAAE;AAChF,YAAI;AACF,gBAAM,KAAK,IAAI,UAAU,QAAQ,IAAI;AACrC,gBAAM,YAAY,MAAM,iCAAiC;AAAA,YACvD;AAAA,YACA;AAAA,YACA;AAAA,YACA,UAAU,IAAI,MAAM,YAAY;AAAA,UAClC,CAAC;AACD,iBAAO,OAAO,SAAS,SAAS;AAAA,QAClC,QAAQ;AAAA,QAER;AACA,eAAO;AAAA,MACT;AAAA,MACA;AAAA,IACF;AAAA,IACA,SAAS;AAAA,MACP,QAAQ;AAAA,QACN,WAAW,GAAG,aAAa;AAAA,QAC3B,QAAQ;AAAA,QACR,UAAU,OAAO,EAAE,KAAK,IAAI,MAAsC;AAChE,gBAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,gBAAM,UAAU,aAAa;AAAA,YAC3B,kBAAkB,eAAe,GAAG,KAAK,CAAC,GAAG,KAAK,SAAS;AAAA,UAC7D;AACA,iBAAO,EAAE,MAAM,QAAQ;AAAA,QACzB;AAAA,QACA,UAAU,CAAC,EAAE,OAAO,OAAmD;AAAA,UACrE,IAAI,QAAQ,UAAU;AAAA,UACtB,CAAC,aAAa,GAAG,SAAS,aAAa,KAAK;AAAA,QAC9C;AAAA,QACA,QAAQ;AAAA,MACV;AAAA,MACA,QAAQ;AAAA,QACN,WAAW,GAAG,aAAa;AAAA,QAC3B,QAAQ;AAAA,QACR,UAAU,OAAO,EAAE,KAAK,IAAI,MAAsC;AAChE,gBAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,gBAAM,UAAU,aAAa;AAAA,YAC3B,kBAAkB,eAAe,GAAG,KAAK,CAAC,GAAG,KAAK,SAAS;AAAA,UAC7D;AACA,iBAAO,EAAE,MAAM,QAAQ;AAAA,QACzB;AAAA,QACA,UAAU,CAAC,EAAE,OAAO,OAAmD;AAAA,UACrE,IAAI,QAAQ,UAAU;AAAA,UACtB,CAAC,aAAa,GAAG,SAAS,aAAa,KAAK;AAAA,QAC9C;AAAA,MACF;AAAA,MACA,QAAQ;AAAA,QACN,WAAW,GAAG,aAAa;AAAA,QAC3B,QAAQ;AAAA,QACR,UAAU,OAAO,EAAE,KAAK,IAAI,MAAsC;AAChE,gBAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,gBAAM,UAAU,aAAa;AAAA,YAC3B,kBAAkB,eAAe,GAAG,KAAK,CAAC,GAAG,KAAK,SAAS;AAAA,UAC7D;AACA,cAAI,CAAC,QAAQ,MAAM,CAAC,QAAQ,aAAa,GAAG;AAC1C,kBAAM,IAAI,cAAc,KAAK;AAAA,cAC3B,OAAO;AAAA,gBACL;AAAA,gBACA;AAAA,cACF;AAAA,YACF,CAAC;AAAA,UACH;AACA,iBAAO,EAAE,MAAM,QAAQ;AAAA,QACzB;AAAA,QACA,UAAU,OAAO,EAAE,IAAI,KAAK;AAAA,MAC9B;AAAA,IACF;AAAA,EACF,CAAC;AAED,QAAM,iBAAiB,EAAE,OAAO;AAAA,IAC9B,IAAI,EAAE,OAAO,EAAE,KAAK;AAAA,IACpB,CAAC,cAAc,GAAG,EAAE,OAAO,EAAE,KAAK;AAAA,IAClC,aAAa,EAAE,OAAO;AAAA,IACtB,MAAM,EAAE,OAAO;AAAA,IACf,iBAAiB,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,IACvD,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IACvC,YAAY,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,IAClD,oBAAoB,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,IAC1D,kBAAkB,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,CAAC,EAAE,SAAS,EAAE,SAAS;AAAA,IACxE,MAAM,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IACrC,aAAa,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IAC5C,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IACxC,UAAU,EAAE,OAAO;AAAA,IACnB,eAAe,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IAC9C,qBAAqB,EAAE,OAAO;AAAA,IAC9B,iBAAiB,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IAChD,cAAc;AAAA,IACd,eAAe,EAAE,OAAO;AAAA,IACxB,gBAAgB,EAAE,OAAO;AAAA,IACzB,kBAAkB,EAAE,OAAO;AAAA,IAC3B,iBAAiB,EAAE,OAAO;AAAA,IAC1B,kBAAkB,EAAE,OAAO;AAAA,IAC3B,UAAU,EAAE,OAAO;AAAA,IACnB,YAAY,EAAE,OAAO;AAAA,IACrB,kBAAkB,EAAE,OAAO;AAAA,IAC3B,oBAAoB,EAAE,OAAO;AAAA,IAC7B,eAAe,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,CAAC,EAAE,SAAS,EAAE,SAAS;AAAA,IACrE,gBAAgB,EAAE,OAAO,EAAE,SAAS,EAAE,SAAS;AAAA,IAC/C,oBAAoB,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,CAAC,EAAE,SAAS,EAAE,SAAS;AAAA,IAC1E,UAAU,EAAE,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,CAAC,EAAE,SAAS,EAAE,SAAS;AAAA,IAChE,qBAAqB,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,SAAS;AAAA,IAC3D,YAAY,EAAE,OAAO;AAAA,IACrB,YAAY,EAAE,OAAO;AAAA,EACvB,CAAC;AAED,QAAM,uBAAuB,EAAE,OAAO;AAAA,IACpC,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS;AAAA,IAC/B,CAAC,aAAa,GAAG,EAAE,OAAO,EAAE,KAAK,EAAE,SAAS;AAAA,EAC9C,CAAC;AAED,QAAM,UAAU,uBAAuB;AAAA,IACrC,cAAc,OAAO,QAAQ;AAAA,IAC7B,aAAa;AAAA,IACb,oBAAoB,8BAA8B,cAAc;AAAA,IAChE,QAAQ;AAAA,MACN,QAAQ;AAAA,MACR,gBAAgB;AAAA,MAChB,aAAa,WAAW,OAAO,QAAQ,WAAW;AAAA,IACpD;AAAA,IACA,QAAQ;AAAA,MACN,QAAQ;AAAA,MACR,gBAAgB;AAAA,MAChB,aAAa,WAAW,OAAO,QAAQ,WAAW;AAAA,IACpD;AAAA,IACA,KAAK;AAAA,MACH,QAAQ;AAAA,MACR,gBAAgB;AAAA,MAChB,aAAa,WAAW,OAAO,QAAQ,WAAW;AAAA,IACpD;AAAA,EACF,CAAC;AAED,SAAO;AAAA,IACL,UAAU;AAAA,IACV;AAAA,IACA,KAAK,KAAK;AAAA,IACV,MAAM,KAAK;AAAA,IACX,KAAK,KAAK;AAAA,IACV,QAAQ,KAAK;AAAA,EACf;AACF;",
6
+ "names": []
7
+ }
@@ -0,0 +1,206 @@
1
+ import { z } from "zod";
2
+ import { makeCrudRoute } from "@open-mercato/shared/lib/crud/factory";
3
+ import { resolveTranslations } from "@open-mercato/shared/lib/i18n/server";
4
+ import { CrudHttpError } from "@open-mercato/shared/lib/crud/errors";
5
+ import { Dictionary, DictionaryEntry } from "@open-mercato/core/modules/dictionaries/data/entities";
6
+ import { statusDictionaryCreateSchema, statusDictionaryUpdateSchema } from "../data/validators.js";
7
+ import { getSalesDictionaryDefinition, ensureSalesDictionary } from "./dictionaries.js";
8
+ import { parseScopedCommandInput, resolveCrudRecordId } from "../api/utils.js";
9
+ import {
10
+ createPagedListResponseSchema,
11
+ createSalesCrudOpenApi,
12
+ defaultDeleteRequestSchema
13
+ } from "../api/openapi.js";
14
+ import { escapeLikePattern } from "@open-mercato/shared/lib/db/escapeLikePattern";
15
+ function makeStatusDictionaryRoute(config) {
16
+ const { kind, entityId, fieldConstants: F } = config;
17
+ const definition = getSalesDictionaryDefinition(kind);
18
+ const rawBodySchema = z.object({}).passthrough();
19
+ const listSchema = z.object({
20
+ page: z.coerce.number().min(1).default(1),
21
+ pageSize: z.coerce.number().min(1).max(100).default(50),
22
+ search: z.string().optional(),
23
+ sortField: z.string().optional(),
24
+ sortDir: z.enum(["asc", "desc"]).optional()
25
+ }).passthrough();
26
+ const metadata = {
27
+ GET: { requireAuth: true, requireFeatures: ["sales.settings.manage"] },
28
+ POST: { requireAuth: true, requireFeatures: ["sales.settings.manage"] },
29
+ PUT: { requireAuth: true, requireFeatures: ["sales.settings.manage"] },
30
+ DELETE: { requireAuth: true, requireFeatures: ["sales.settings.manage"] }
31
+ };
32
+ const dictionaryItemSchema = z.object({
33
+ id: z.string().uuid(),
34
+ value: z.string(),
35
+ label: z.string().nullable(),
36
+ color: z.string().nullable(),
37
+ icon: z.string().nullable(),
38
+ organizationId: z.string().uuid().nullable(),
39
+ tenantId: z.string().uuid().nullable(),
40
+ createdAt: z.string(),
41
+ updatedAt: z.string()
42
+ });
43
+ const dictionaryListResponseSchema = createPagedListResponseSchema(dictionaryItemSchema);
44
+ const normalizeId = (value) => {
45
+ if (typeof value !== "string") return null;
46
+ const trimmed = value.trim();
47
+ return trimmed.length > 0 ? trimmed : null;
48
+ };
49
+ async function resolveDictionaryContext(ctx) {
50
+ if (!ctx.auth || !ctx.auth.tenantId) {
51
+ throw new CrudHttpError(401, { error: "Tenant context is required." });
52
+ }
53
+ const em = ctx.container.resolve("em");
54
+ const tenantId = ctx.auth.tenantId;
55
+ const candidateOrgIds = /* @__PURE__ */ new Set();
56
+ const pushCandidate = (value) => {
57
+ const normalized = normalizeId(value);
58
+ if (normalized) candidateOrgIds.add(normalized);
59
+ };
60
+ pushCandidate(ctx.selectedOrganizationId);
61
+ pushCandidate(ctx.auth.orgId ?? null);
62
+ const scope = ctx.organizationScope;
63
+ if (scope) {
64
+ if (Array.isArray(scope.filterIds)) {
65
+ for (const id of scope.filterIds) pushCandidate(id);
66
+ }
67
+ if (Array.isArray(scope.allowedIds)) {
68
+ for (const id of scope.allowedIds) pushCandidate(id);
69
+ }
70
+ }
71
+ for (const orgId of candidateOrgIds) {
72
+ const dictionary = await ensureSalesDictionary({
73
+ em,
74
+ tenantId,
75
+ organizationId: orgId,
76
+ kind
77
+ });
78
+ if (dictionary) {
79
+ return { dictionaryId: dictionary.id, organizationId: orgId };
80
+ }
81
+ }
82
+ const fallback = await em.findOne(
83
+ Dictionary,
84
+ {
85
+ tenantId,
86
+ key: definition.key,
87
+ deletedAt: null
88
+ },
89
+ { orderBy: { createdAt: "asc" } }
90
+ );
91
+ if (fallback) {
92
+ return { dictionaryId: fallback.id, organizationId: fallback.organizationId };
93
+ }
94
+ throw new CrudHttpError(400, { error: "Organization context is required." });
95
+ }
96
+ const crud = makeCrudRoute({
97
+ metadata,
98
+ orm: {
99
+ entity: DictionaryEntry,
100
+ idField: "id",
101
+ orgField: "organizationId",
102
+ tenantField: "tenantId",
103
+ softDeleteField: null
104
+ },
105
+ list: {
106
+ schema: listSchema,
107
+ entityId,
108
+ fields: [
109
+ F.id,
110
+ F.value,
111
+ F.label,
112
+ F.color,
113
+ F.icon,
114
+ F.organization_id,
115
+ F.tenant_id,
116
+ F.created_at,
117
+ F.updated_at
118
+ ],
119
+ sortFieldMap: {
120
+ id: F.id,
121
+ value: F.value,
122
+ label: F.label,
123
+ createdAt: F.created_at,
124
+ updatedAt: F.updated_at
125
+ },
126
+ buildFilters: async (query, ctx) => {
127
+ const { dictionaryId } = await resolveDictionaryContext(ctx);
128
+ const filters = {
129
+ dictionary_id: dictionaryId
130
+ };
131
+ if (query.search && query.search.trim().length > 0) {
132
+ const term = `%${escapeLikePattern(query.search.trim())}%`;
133
+ filters.$or = [
134
+ { [F.value]: { $ilike: term } },
135
+ { [F.label]: { $ilike: term } }
136
+ ];
137
+ }
138
+ return filters;
139
+ },
140
+ transformItem: (item) => ({
141
+ id: item.id,
142
+ value: item.value,
143
+ label: item.label,
144
+ color: item.color ?? null,
145
+ icon: item.icon ?? null,
146
+ organizationId: item.organization_id ?? null,
147
+ tenantId: item.tenant_id ?? null,
148
+ createdAt: item.created_at,
149
+ updatedAt: item.updated_at
150
+ })
151
+ },
152
+ actions: {
153
+ create: {
154
+ commandId: `${definition.commandPrefix}.create`,
155
+ schema: rawBodySchema,
156
+ mapInput: async ({ raw, ctx }) => {
157
+ const { translate } = await resolveTranslations();
158
+ return parseScopedCommandInput(statusDictionaryCreateSchema, raw ?? {}, ctx, translate);
159
+ },
160
+ response: ({ result }) => ({ id: result?.entryId ?? null }),
161
+ status: 201
162
+ },
163
+ update: {
164
+ commandId: `${definition.commandPrefix}.update`,
165
+ schema: rawBodySchema,
166
+ mapInput: async ({ raw, ctx }) => {
167
+ const { translate } = await resolveTranslations();
168
+ return parseScopedCommandInput(statusDictionaryUpdateSchema, raw ?? {}, ctx, translate);
169
+ },
170
+ response: () => ({ ok: true })
171
+ },
172
+ delete: {
173
+ commandId: `${definition.commandPrefix}.delete`,
174
+ schema: rawBodySchema,
175
+ mapInput: async ({ parsed, ctx }) => {
176
+ const { translate } = await resolveTranslations();
177
+ const id = resolveCrudRecordId(parsed, ctx, translate);
178
+ return { id };
179
+ },
180
+ response: () => ({ ok: true })
181
+ }
182
+ }
183
+ });
184
+ const openApi = createSalesCrudOpenApi({
185
+ resourceName: config.openApi.resourceName,
186
+ pluralName: config.openApi.pluralName,
187
+ description: config.openApi.description,
188
+ querySchema: listSchema,
189
+ listResponseSchema: dictionaryListResponseSchema,
190
+ create: { schema: statusDictionaryCreateSchema },
191
+ update: { schema: statusDictionaryUpdateSchema },
192
+ del: { schema: defaultDeleteRequestSchema }
193
+ });
194
+ return {
195
+ metadata,
196
+ openApi,
197
+ GET: crud.GET,
198
+ POST: crud.POST,
199
+ PUT: crud.PUT,
200
+ DELETE: crud.DELETE
201
+ };
202
+ }
203
+ export {
204
+ makeStatusDictionaryRoute
205
+ };
206
+ //# sourceMappingURL=makeStatusDictionaryRoute.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/modules/sales/lib/makeStatusDictionaryRoute.ts"],
4
+ "sourcesContent": ["import { z } from 'zod'\nimport { makeCrudRoute, type CrudCtx } from '@open-mercato/shared/lib/crud/factory'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport { Dictionary, DictionaryEntry } from '@open-mercato/core/modules/dictionaries/data/entities'\nimport { statusDictionaryCreateSchema, statusDictionaryUpdateSchema } from '../data/validators'\nimport { getSalesDictionaryDefinition, ensureSalesDictionary, type SalesDictionaryKind } from './dictionaries'\nimport { parseScopedCommandInput, resolveCrudRecordId } from '../api/utils'\nimport {\n createPagedListResponseSchema,\n createSalesCrudOpenApi,\n defaultDeleteRequestSchema,\n} from '../api/openapi'\nimport { escapeLikePattern } from '@open-mercato/shared/lib/db/escapeLikePattern'\n\ninterface StatusDictionaryRouteConfig {\n kind: SalesDictionaryKind\n entityId: string\n fieldConstants: Record<string, string>\n openApi: {\n resourceName: string\n pluralName: string\n description: string\n }\n}\n\nexport function makeStatusDictionaryRoute(config: StatusDictionaryRouteConfig) {\n const { kind, entityId, fieldConstants: F } = config\n const definition = getSalesDictionaryDefinition(kind)\n\n const rawBodySchema = z.object({}).passthrough()\n\n const listSchema = z\n .object({\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n search: z.string().optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional(),\n })\n .passthrough()\n\n const metadata = {\n GET: { requireAuth: true, requireFeatures: ['sales.settings.manage'] },\n POST: { requireAuth: true, requireFeatures: ['sales.settings.manage'] },\n PUT: { requireAuth: true, requireFeatures: ['sales.settings.manage'] },\n DELETE: { requireAuth: true, requireFeatures: ['sales.settings.manage'] },\n }\n\n const dictionaryItemSchema = z.object({\n id: z.string().uuid(),\n value: z.string(),\n label: z.string().nullable(),\n color: z.string().nullable(),\n icon: z.string().nullable(),\n organizationId: z.string().uuid().nullable(),\n tenantId: z.string().uuid().nullable(),\n createdAt: z.string(),\n updatedAt: z.string(),\n })\n\n const dictionaryListResponseSchema = createPagedListResponseSchema(dictionaryItemSchema)\n\n const normalizeId = (value: unknown): string | null => {\n if (typeof value !== 'string') return null\n const trimmed = value.trim()\n return trimmed.length > 0 ? trimmed : null\n }\n\n async function resolveDictionaryContext(ctx: CrudCtx): Promise<{ dictionaryId: string; organizationId: string | null }> {\n if (!ctx.auth || !ctx.auth.tenantId) {\n throw new CrudHttpError(401, { error: 'Tenant context is required.' })\n }\n const em = ctx.container.resolve('em') as EntityManager\n const tenantId: string = ctx.auth.tenantId\n const candidateOrgIds = new Set<string>()\n const pushCandidate = (value: unknown) => {\n const normalized = normalizeId(value)\n if (normalized) candidateOrgIds.add(normalized)\n }\n pushCandidate(ctx.selectedOrganizationId)\n pushCandidate(ctx.auth.orgId ?? null)\n const scope = ctx.organizationScope\n if (scope) {\n if (Array.isArray(scope.filterIds)) {\n for (const id of scope.filterIds) pushCandidate(id)\n }\n if (Array.isArray(scope.allowedIds)) {\n for (const id of scope.allowedIds) pushCandidate(id)\n }\n }\n\n for (const orgId of candidateOrgIds) {\n const dictionary = await ensureSalesDictionary({\n em,\n tenantId,\n organizationId: orgId,\n kind,\n })\n if (dictionary) {\n return { dictionaryId: dictionary.id, organizationId: orgId }\n }\n }\n\n const fallback = await em.findOne(\n Dictionary,\n {\n tenantId,\n key: definition.key,\n deletedAt: null,\n },\n { orderBy: { createdAt: 'asc' } },\n )\n if (fallback) {\n return { dictionaryId: fallback.id, organizationId: fallback.organizationId }\n }\n throw new CrudHttpError(400, { error: 'Organization context is required.' })\n }\n\n const crud = makeCrudRoute({\n metadata,\n orm: {\n entity: DictionaryEntry,\n idField: 'id',\n orgField: 'organizationId',\n tenantField: 'tenantId',\n softDeleteField: null,\n },\n list: {\n schema: listSchema,\n entityId,\n fields: [\n F.id,\n F.value,\n F.label,\n F.color,\n F.icon,\n F.organization_id,\n F.tenant_id,\n F.created_at,\n F.updated_at,\n ],\n sortFieldMap: {\n id: F.id,\n value: F.value,\n label: F.label,\n createdAt: F.created_at,\n updatedAt: F.updated_at,\n },\n buildFilters: async (query, ctx) => {\n const { dictionaryId } = await resolveDictionaryContext(ctx)\n const filters: Record<string, unknown> = {\n dictionary_id: dictionaryId,\n }\n if (query.search && query.search.trim().length > 0) {\n const term = `%${escapeLikePattern(query.search.trim())}%`\n filters.$or = [\n { [F.value]: { $ilike: term } },\n { [F.label]: { $ilike: term } },\n ]\n }\n return filters\n },\n transformItem: (item: Record<string, unknown>) => ({\n id: item.id,\n value: item.value,\n label: item.label,\n color: item.color ?? null,\n icon: item.icon ?? null,\n organizationId: item.organization_id ?? null,\n tenantId: item.tenant_id ?? null,\n createdAt: item.created_at,\n updatedAt: item.updated_at,\n }),\n },\n actions: {\n create: {\n commandId: `${definition.commandPrefix}.create`,\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(statusDictionaryCreateSchema, raw ?? {}, ctx, translate)\n },\n response: ({ result }) => ({ id: result?.entryId ?? null }),\n status: 201,\n },\n update: {\n commandId: `${definition.commandPrefix}.update`,\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(statusDictionaryUpdateSchema, raw ?? {}, ctx, translate)\n },\n response: () => ({ ok: true }),\n },\n delete: {\n commandId: `${definition.commandPrefix}.delete`,\n schema: rawBodySchema,\n mapInput: async ({ parsed, ctx }) => {\n const { translate } = await resolveTranslations()\n const id = resolveCrudRecordId(parsed, ctx, translate)\n return { id }\n },\n response: () => ({ ok: true }),\n },\n },\n })\n\n const openApi = createSalesCrudOpenApi({\n resourceName: config.openApi.resourceName,\n pluralName: config.openApi.pluralName,\n description: config.openApi.description,\n querySchema: listSchema,\n listResponseSchema: dictionaryListResponseSchema,\n create: { schema: statusDictionaryCreateSchema },\n update: { schema: statusDictionaryUpdateSchema },\n del: { schema: defaultDeleteRequestSchema },\n })\n\n return {\n metadata,\n openApi,\n GET: crud.GET,\n POST: crud.POST,\n PUT: crud.PUT,\n DELETE: crud.DELETE,\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }