@infuro/cms-core 1.0.14 → 1.0.15

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.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/plugins/erp/erp-response-map.ts","../src/plugins/erp/erp-config-enabled.ts","../src/plugins/erp/erp-order-invoice.ts","../src/plugins/email/email-queue.ts","../src/message-templates/sms-defaults.ts","../src/message-templates/interpolate.ts","../src/plugins/payment/stripe.ts","../src/plugins/payment/razorpay.ts","../src/index.ts","../src/plugins/registry.ts","../src/plugins/erp/erp-submission.ts","../src/plugins/erp/erp-queue.ts","../src/plugins/erp/paid-order-erp.ts","../src/plugins/erp/erp-contact-sync.ts","../src/lib/order-number.ts","../src/lib/order-storefront-metadata.ts","../src/plugins/erp/erp-order-status-map.ts","../src/plugins/erp/erp-order-sync.ts","../src/plugins/erp/index.ts","../src/plugins/erp/erp-product-sync.ts","../src/plugins/email/email-service.ts","../src/plugins/email/templates/layout.ts","../src/plugins/email/templates/types.ts","../src/plugins/email/templates/inline-cta.ts","../src/plugins/email/templates/signup.ts","../src/plugins/email/templates/passwordReset.ts","../src/plugins/email/templates/passwordChange.ts","../src/plugins/email/templates/orderPlaced.ts","../src/plugins/email/templates/returnInitiated.ts","../src/plugins/email/templates/shippingUpdate.ts","../src/plugins/email/templates/invite.ts","../src/plugins/email/templates/formSubmission.ts","../src/plugins/email/templates/otp.ts","../src/plugins/email/templates/index.ts","../src/plugins/email/renderer.ts","../src/plugins/email/index.ts","../src/plugins/analytics/analytics-service.ts","../src/plugins/analytics/index.ts","../src/plugins/sms/sms-service.ts","../src/plugins/sms/sms-queue.ts","../src/plugins/sms/index.ts","../src/plugins/payment/index.ts","../src/plugins/storage/s3.ts","../src/plugins/storage/local.ts","../src/plugins/llm/llm-service.ts","../src/plugins/llm/index.ts","../src/plugins/cache/index.ts","../src/plugins/cache/memory-cache.ts","../src/plugins/cache/redis-cache.ts","../src/plugins/queue/bullmq-queue.ts","../src/plugins/queue/memory-queue.ts","../src/plugins/queue/index.ts","../src/plugins/captcha/captcha-service.ts","../src/plugins/captcha/captcha-plugin.ts","../src/plugins/captcha/assert.ts","../src/lib/utils.ts","../src/lib/seo-metadata.ts","../src/lib/link-contact-to-user.ts","../src/lib/email-recipients.ts","../src/lib/otp-challenge.ts","../src/entities/user.entity.ts","../src/entities/user-group.entity.ts","../src/entities/permission.entity.ts","../src/entities/otp-challenge.entity.ts","../src/entities/password-reset-token.entity.ts","../src/entities/blog.entity.ts","../src/entities/category.entity.ts","../src/entities/seo.entity.ts","../src/entities/comment.entity.ts","../src/entities/tag.entity.ts","../src/entities/contact.entity.ts","../src/entities/form-submission.entity.ts","../src/entities/form.entity.ts","../src/entities/form-field.entity.ts","../src/entities/address.entity.ts","../src/entities/order.entity.ts","../src/entities/payment.entity.ts","../src/entities/chat-conversation.entity.ts","../src/entities/chat-message.entity.ts","../src/entities/config.entity.ts","../src/entities/message-template.entity.ts","../src/entities/media.entity.ts","../src/entities/page.entity.ts","../src/entities/product-category.entity.ts","../src/entities/collection.entity.ts","../src/entities/brand.entity.ts","../src/entities/product.entity.ts","../src/entities/attribute.entity.ts","../src/entities/product-attribute.entity.ts","../src/entities/tax.entity.ts","../src/entities/product-tax.entity.ts","../src/entities/order-item.entity.ts","../src/entities/knowledge-base-document.entity.ts","../src/entities/knowledge-base-chunk.entity.ts","../src/entities/cart.entity.ts","../src/entities/cart-item.entity.ts","../src/entities/wishlist.entity.ts","../src/entities/wishlist-item.entity.ts","../src/entities/index.ts","../src/auth/permission-entities.ts","../src/auth/helpers.ts","../src/auth/seed-permissions.ts","../src/auth/middleware.ts","../src/auth/nextauth-options.ts","../src/api/crud.ts","../src/api/auth-handlers.ts","../src/api/cms-handlers.ts","../src/api/message-template-admin-handlers.ts","../src/api/admin-roles-handlers.ts","../src/api/cms-api-handler.ts","../src/api/storefront-handlers.ts","../src/lib/is-valid-signup-email.ts","../src/admin/config.ts"],"sourcesContent":["import type { OrderFulfillmentEvent, OrderFulfillmentMetadata } from '../../lib/order-storefront-metadata';\n\nfunction pickString(o: Record<string, unknown>, keys: string[]): string | undefined {\n for (const k of keys) {\n const v = o[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\nexport function unwrapErpReadData(json: unknown): Record<string, unknown> | null {\n if (!json || typeof json !== 'object') return null;\n const o = json as Record<string, unknown>;\n const d = o.data;\n if (d && typeof d === 'object' && !Array.isArray(d)) return d as Record<string, unknown>;\n return o;\n}\n\nfunction firstObject(\n data: Record<string, unknown>,\n keys: string[]\n): Record<string, unknown> | null {\n for (const k of keys) {\n const v = data[k];\n if (v && typeof v === 'object' && !Array.isArray(v)) return v as Record<string, unknown>;\n }\n return null;\n}\n\nfunction extractEvents(src: Record<string, unknown>): OrderFulfillmentEvent[] | undefined {\n const timeline = src.timeline ?? src.events ?? src.history ?? src.trackingEvents;\n if (!Array.isArray(timeline) || !timeline.length) return undefined;\n const events: OrderFulfillmentEvent[] = [];\n for (const row of timeline) {\n if (!row || typeof row !== 'object') continue;\n const r = row as Record<string, unknown>;\n const at =\n pickString(r, ['at', 'timestamp', 'date', 'occurredAt']) ??\n (r.at instanceof Date ? (r.at as Date).toISOString() : undefined);\n const label = pickString(r, ['label', 'status', 'title', 'message', 'description']);\n const detail = pickString(r, ['detail', 'notes', 'description']);\n if (at || label || detail) events.push({ at, label, detail });\n }\n return events.length ? events : undefined;\n}\n\nexport function mapErpPayloadToFulfillment(data: Record<string, unknown>): OrderFulfillmentMetadata | undefined {\n const nested = firstObject(data, ['fulfillment', 'packaging', 'shipment', 'shipping', 'delivery']);\n const src = nested || data;\n const status = pickString(src, ['status', 'fulfillmentStatus', 'state', 'label', 'packagingStatus']);\n const trackingId = pickString(src, ['trackingId', 'tracking_id', 'trackingNumber', 'awb', 'trackingUrl']);\n const events = extractEvents(src);\n if (!status && !trackingId && !(events && events.length)) return undefined;\n return { status, trackingId, events };\n}\n\nexport function mapErpPayloadToInvoiceNumber(data: Record<string, unknown>): string | undefined {\n const nested = firstObject(data, ['invoice', 'latestInvoice', 'postedInvoice']);\n const src = nested || data;\n return pickString(src, ['invoiceNumber', 'invoice_number', 'number', 'name', 'id']);\n}\n\nexport type ErpChildOrderRef = { ref: string; orderKind: 'return' | 'replacement' };\n\nexport function extractChildOrderRefsFromSalePayload(data: Record<string, unknown>): ErpChildOrderRef[] {\n const lists = [data.returns, data.returnOrders, data.relatedReturns, data.childOrders, data.children];\n const seen = new Set<string>();\n const out: ErpChildOrderRef[] = [];\n for (const list of lists) {\n if (!Array.isArray(list)) continue;\n for (const item of list) {\n if (!item || typeof item !== 'object') continue;\n const o = item as Record<string, unknown>;\n const ref =\n pickString(o, ['platformReturnId', 'platform_return_id', 'refId', 'ref_id']) ??\n (typeof o.id === 'string' ? o.id : undefined);\n if (!ref || seen.has(ref)) continue;\n seen.add(ref);\n const t = (pickString(o, ['kind', 'type', 'orderKind']) || '').toLowerCase();\n const orderKind: 'return' | 'replacement' = /replac/.test(t) ? 'replacement' : 'return';\n out.push({ ref, orderKind });\n }\n }\n return out;\n}\n","import type { ObjectLiteral } from 'typeorm';\n\nexport type CmsAppLike = { getPlugin(name: string): unknown };\n\nexport type ErpConfigDataSource = {\n getRepository(entity: unknown): {\n find(options?: object): Promise<unknown[]>;\n };\n};\n\n/** True when ERP plugin is loaded and config `erp.enabled` is not `'false'`. */\nexport async function isErpIntegrationEnabled(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>\n): Promise<boolean> {\n if (!cms.getPlugin('erp')) return false;\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return false;\n }\n return true;\n}\n","import type { ObjectLiteral, Repository } from 'typeorm';\nimport { unwrapErpReadData } from './erp-response-map';\nimport type { ERPSubmissionService } from './erp-submission';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nfunction pickInvoiceId(data: Record<string, unknown>): string | undefined {\n const nested =\n data.invoice && typeof data.invoice === 'object' && !Array.isArray(data.invoice)\n ? (data.invoice as Record<string, unknown>)\n : null;\n const src = nested || data;\n for (const k of ['invoiceId', 'invoice_id', 'id']) {\n const v = src[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\n/**\n * Streams invoice PDF for a **sale** order. Resolves `invoiceId` from order metadata or `get-invoice`.\n */\nexport async function streamOrderInvoicePdf(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n orderId: number,\n options: { ownerContactId?: number }\n): Promise<Response> {\n const jsonErr = (msg: string, status: number) =>\n new Response(JSON.stringify({ error: msg }), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return jsonErr('Invoice not available', 503);\n\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp?.submission) return jsonErr('Invoice not available', 503);\n\n const orderRepo = dataSource.getRepository(entityMap.orders) as Repository<ObjectLiteral>;\n const order = await orderRepo.findOne({ where: { id: orderId, deleted: false } as ObjectLiteral });\n if (!order) return jsonErr('Not found', 404);\n\n const kind = (order.orderKind as string) || 'sale';\n if (kind !== 'sale') return jsonErr('Invoice only for sale orders', 400);\n\n if (options.ownerContactId != null && order.contactId !== options.ownerContactId) {\n return jsonErr('Not found', 404);\n }\n\n const meta =\n order.metadata && typeof order.metadata === 'object' && !Array.isArray(order.metadata)\n ? (order.metadata as Record<string, unknown>)\n : {};\n const inv = meta.invoice && typeof meta.invoice === 'object' && !Array.isArray(meta.invoice)\n ? (meta.invoice as Record<string, unknown>)\n : {};\n let invoiceId = typeof inv.invoiceId === 'string' ? inv.invoiceId.trim() : '';\n\n if (!invoiceId) {\n const refId = String(order.orderNumber || '');\n const r = await erp.submission.postErpReadAction('get-invoice', { platformOrderId: refId });\n const d = r.ok ? unwrapErpReadData(r.json) : null;\n invoiceId = d ? pickInvoiceId(d) || '' : '';\n }\n\n if (!invoiceId) return jsonErr('Invoice not ready', 404);\n\n const pdf = await erp.submission.fetchInvoicePdf(invoiceId);\n if (!pdf.ok || !pdf.buffer) return jsonErr(pdf.error || 'PDF fetch failed', 502);\n\n const filename = `invoice-${orderId}.pdf`;\n return new Response(pdf.buffer, {\n status: 200,\n headers: {\n 'Content-Type': pdf.contentType || 'application/pdf',\n 'Content-Disposition': `attachment; filename=\"${filename}\"`,\n },\n });\n}\n","import type { CompanyDetails, EmailTemplateName, OrderPlacedLineItem } from './templates/types';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\n/** Context for template rendering: at least companyDetails; template-specific fields (formName, etc.) are allowed. */\nexport interface EmailJobPayload {\n to: string;\n templateName?: EmailTemplateName;\n ctx?: Record<string, unknown> & { companyDetails: CompanyDetails };\n subject?: string;\n html?: string;\n text?: string;\n}\n\nconst EMAIL_QUEUE_NAME = 'email';\n\nexport function registerEmailQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void>; registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void } | undefined;\n const email = cms.getPlugin('email') as { send: (opts: { to: string; subject: string; html: string; text?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (!queue || !email) return;\n queue.registerProcessor(EMAIL_QUEUE_NAME, async (data: object) => {\n const payload = data as EmailJobPayload;\n const { to, templateName, ctx, subject, html, text } = payload;\n if (!to) return;\n if (templateName && ctx) {\n const rendered = email.renderTemplate(templateName, ctx);\n await email.send({ to, subject: rendered.subject, html: rendered.html, text: rendered.text });\n } else if (subject != null && html != null) {\n await email.send({ to, subject, html, text });\n }\n });\n}\n\nexport async function queueEmail(cms: CmsAppLike, payload: EmailJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n if (queue) {\n await queue.add(EMAIL_QUEUE_NAME, payload);\n return;\n }\n const email = cms.getPlugin('email') as { send: (opts: { to: string; subject: string; html: string; text?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (email && payload.templateName && payload.ctx) {\n const rendered = email.renderTemplate(payload.templateName, payload.ctx);\n await email.send({ to: payload.to, subject: rendered.subject, html: rendered.html, text: rendered.text });\n } else if (email && payload.subject != null && payload.html != null) {\n await email.send({ to: payload.to, subject: payload.subject, html: payload.html, text: payload.text });\n }\n}\n\nexport interface OrderPlacedEmailPayload {\n orderNumber: string;\n total?: string | number;\n currency?: string;\n customerName?: string;\n /** Customer receipt; omitted if empty */\n customerEmail?: string | null;\n /** Parsed list from email plugin “Sales team” settings */\n salesTeamEmails: string[];\n companyDetails: CompanyDetails;\n lineItems: OrderPlacedLineItem[];\n}\n\n/** Queues one `orderPlaced` email per recipient (customer + each unique sales address; skips sales if same as customer). */\nexport async function queueOrderPlacedEmails(cms: CmsAppLike, payload: OrderPlacedEmailPayload): Promise<void> {\n const { orderNumber, total, currency, customerName, customerEmail, salesTeamEmails, companyDetails, lineItems } = payload;\n const base = {\n orderNumber,\n total: total != null ? String(total) : undefined,\n currency,\n customerName,\n companyDetails: companyDetails ?? {},\n lineItems: lineItems ?? [],\n };\n const customerLower = customerEmail?.trim().toLowerCase() ?? '';\n const jobs: Promise<void>[] = [];\n if (customerEmail?.trim()) {\n jobs.push(\n queueEmail(cms, {\n to: customerEmail.trim(),\n templateName: 'orderPlaced',\n ctx: { ...base, audience: 'customer' as const },\n })\n );\n }\n const seen = new Set<string>();\n for (const raw of salesTeamEmails) {\n const to = raw.trim();\n if (!to) continue;\n const key = to.toLowerCase();\n if (seen.has(key)) continue;\n seen.add(key);\n if (customerLower && key === customerLower) continue;\n jobs.push(\n queueEmail(cms, {\n to,\n templateName: 'orderPlaced',\n ctx: {\n ...base,\n audience: 'sales' as const,\n internalCustomerEmail: customerEmail?.trim() || undefined,\n },\n })\n );\n }\n await Promise.all(jobs);\n}\n","export type SmsMessageTemplateDefault = {\n templateKey: string;\n name: string;\n body: string;\n externalTemplateRef?: string;\n providerMeta?: { otpVarKey?: string };\n};\n\nexport const SMS_MESSAGE_TEMPLATE_DEFAULTS: SmsMessageTemplateDefault[] = [\n {\n templateKey: 'auth.otp_login',\n name: 'Sign-in OTP (SMS)',\n body: 'Your sign-in code is {{code}}. Valid 10 minutes.',\n providerMeta: { otpVarKey: 'var1' },\n },\n {\n templateKey: 'auth.otp_verify_phone',\n name: 'Verify phone OTP (SMS)',\n body: 'Your verification code is {{code}}. Valid 10 minutes.',\n providerMeta: { otpVarKey: 'var1' },\n },\n];\n\nexport function getSmsTemplateDefault(templateKey: string): SmsMessageTemplateDefault | undefined {\n return SMS_MESSAGE_TEMPLATE_DEFAULTS.find((d) => d.templateKey === templateKey);\n}\n\nexport type ResolvedSmsTemplate = {\n body: string;\n externalTemplateRef: string | undefined;\n otpVarKey: string;\n};\n\nexport async function resolveSmsTemplateForSend(\n templateKey: string,\n getRow: (channel: string, key: string) => Promise<{\n body: string;\n externalTemplateRef: string | null;\n providerMeta: Record<string, unknown> | null;\n enabled: boolean;\n } | null>\n): Promise<ResolvedSmsTemplate | null> {\n const def = getSmsTemplateDefault(templateKey);\n if (!def) return null;\n\n const row = await getRow('sms', templateKey);\n let body = def.body;\n let ref = def.externalTemplateRef;\n let otpVarKey = def.providerMeta?.otpVarKey ?? 'var1';\n\n if (row && row.enabled !== false) {\n if (row.body.trim()) body = row.body;\n if (row.externalTemplateRef?.trim()) ref = row.externalTemplateRef.trim();\n const pm = row.providerMeta;\n const k = pm && typeof pm.otpVarKey === 'string' ? pm.otpVarKey.trim() : '';\n if (k) otpVarKey = k;\n }\n\n return {\n body,\n externalTemplateRef: ref?.trim() || undefined,\n otpVarKey: otpVarKey || 'var1',\n };\n}\n","/** Replace `{{key}}` placeholders (also accepts legacy `{{otp}}` as alias for `code` when code is set). */\nexport function applyTemplateVars(template: string, variables: Record<string, string>): string {\n let out = template;\n for (const [k, v] of Object.entries(variables)) {\n out = out.split(`{{${k}}}`).join(v);\n }\n if (variables.code !== undefined) {\n out = out.split('{{otp}}').join(variables.code);\n }\n return out;\n}\n","import Stripe from 'stripe';\nimport type { PaymentServiceInterface, PaymentIntent } from './index';\n\nexport interface StripeServiceConfig {\n secretKey: string;\n webhookSecret?: string;\n}\n\nexport class StripePaymentService implements PaymentServiceInterface {\n private stripe: Stripe;\n private webhookSecret?: string;\n\n constructor(config: StripeServiceConfig) {\n this.stripe = new Stripe(config.secretKey, { apiVersion: '2024-06-20' });\n this.webhookSecret = config.webhookSecret;\n }\n\n async createPaymentIntent(\n amount: number,\n currency: string,\n metadata?: Record<string, string>\n ): Promise<PaymentIntent> {\n const intent = await this.stripe.paymentIntents.create({\n amount: Math.round(amount * 100),\n currency: currency.toLowerCase(),\n metadata,\n });\n\n return {\n id: intent.id,\n clientSecret: intent.client_secret || '',\n amount,\n currency,\n status: intent.status,\n };\n }\n\n async capturePayment(paymentId: string, amount?: number): Promise<boolean> {\n try {\n await this.stripe.paymentIntents.capture(paymentId, {\n amount_to_capture: amount ? Math.round(amount * 100) : undefined,\n });\n return true;\n } catch {\n return false;\n }\n }\n\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean {\n if (!this.webhookSecret) return false;\n try {\n this.stripe.webhooks.constructEvent(payload, signature, this.webhookSecret);\n return true;\n } catch {\n return false;\n }\n }\n}\n","import Razorpay from 'razorpay';\nimport crypto from 'crypto';\nimport type { PaymentServiceInterface, PaymentIntent } from './index';\n\nexport interface RazorpayServiceConfig {\n keyId: string;\n keySecret: string;\n webhookSecret?: string;\n}\n\nexport class RazorpayPaymentService implements PaymentServiceInterface {\n private razorpay: Razorpay;\n private keySecret: string;\n private webhookSecret?: string;\n\n constructor(config: RazorpayServiceConfig) {\n this.razorpay = new Razorpay({\n key_id: config.keyId,\n key_secret: config.keySecret,\n });\n this.keySecret = config.keySecret;\n this.webhookSecret = config.webhookSecret;\n }\n\n async createPaymentIntent(\n amount: number,\n currency: string,\n metadata?: Record<string, string>\n ): Promise<PaymentIntent> {\n const order = await this.razorpay.orders.create({\n amount: Math.round(amount * 100),\n currency: currency.toUpperCase(),\n notes: metadata,\n });\n\n return {\n id: order.id,\n clientSecret: order.id,\n amount,\n currency,\n status: order.status,\n };\n }\n\n async capturePayment(paymentId: string, amount?: number): Promise<boolean> {\n try {\n await this.razorpay.payments.capture(paymentId, amount ? Math.round(amount * 100) : 0, 'INR');\n return true;\n } catch {\n return false;\n }\n }\n\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean {\n const secret = this.webhookSecret || this.keySecret;\n const expectedSignature = crypto\n .createHmac('sha256', secret)\n .update(typeof payload === 'string' ? payload : payload.toString('utf8'))\n .digest('hex');\n return crypto.timingSafeEqual(Buffer.from(signature), Buffer.from(expectedSignature));\n }\n}\n","export * from './plugins/types';\nexport * from './plugins/registry';\nexport {\n erpPlugin,\n ERPSubmissionService,\n registerErpQueueProcessor,\n queueErp,\n queueErpPaidOrderForOrderId,\n queueErpCreateContactIfEnabled,\n type ErpPaidOrderDataSource,\n type ErpPaidOrderEntityMap,\n type ErpContactSyncInput,\n} from './plugins/erp';\nexport type {\n ERPPluginConfig,\n ERPPluginInstance,\n ErpJobPayload,\n ErpCreateContactPayload,\n PipelineNames,\n} from './plugins/erp';\nexport { emailPlugin, EmailService, emailTemplates, renderEmail, getCompanyDetailsFromSettings, mergeEmailLayoutCompanyDetails, renderLayout, registerEmailQueueProcessor, queueEmail, queueOrderPlacedEmails } from './plugins/email';\nexport type {\n EmailPluginConfig,\n EmailData,\n EmailServiceInterface,\n RenderedEmail,\n CompanyDetails,\n SocialLinkItem,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n RenderEmailOptions,\n EmailJobPayload,\n OrderPlacedEmailPayload,\n OrderPlacedLineItem,\n} from './plugins/email';\nexport { analyticsPlugin } from './plugins/analytics';\nexport type { AnalyticsPluginConfig } from './plugins/analytics';\nexport { smsPlugin, registerSmsQueueProcessor, queueSms } from './plugins/sms';\nexport type {\n SmsPluginConfig,\n SmsServiceInterface,\n SmsJobPayload,\n SmsProviderId,\n SmsProviderChoice,\n SmsServiceConfig,\n} from './plugins/sms';\nexport { paymentPlugin } from './plugins/payment';\nexport type { PaymentPluginConfig, PaymentServiceInterface, PaymentIntent } from './plugins/payment';\nexport { s3StoragePlugin, localStoragePlugin } from './plugins/storage';\nexport type { StorageService, S3StoragePluginConfig, LocalStoragePluginConfig } from './plugins/storage';\nexport { llmPlugin, LlmService } from './plugins/llm';\nexport type { LlmPluginConfig, LlmServiceInterface, LlmMessage, LlmChatOptions } from './plugins/llm';\nexport { cachePlugin } from './plugins/cache';\nexport type { CacheService, CachePluginConfig } from './plugins/cache';\nexport { queuePlugin } from './plugins/queue';\nexport type { QueueService, QueuePluginConfig } from './plugins/queue';\nexport { captchaPlugin, CaptchaService, buildCaptchaPublicConfig, assertCaptchaOk } from './plugins/captcha';\nexport type { CaptchaPublicConfig, CaptchaProviderId, CaptchaVerifyResult } from './plugins/captcha';\nexport * from './lib';\nexport * from './entities';\nexport * from './auth';\nexport * from './api';\nexport { DEFAULT_ADMIN_NAV } from './admin/config';\nexport type { AdminNavItem } from './admin/config';\n","import type { CmsPlugin, PluginContext } from './types';\n\nexport interface CreateCmsAppOptions {\n dataSource: unknown;\n config?: Record<string, string>;\n plugins?: CmsPlugin[];\n logger?: PluginContext['logger'];\n}\n\nconst noopLogger: PluginContext['logger'] = {\n info: () => {},\n warn: () => {},\n error: () => {},\n};\n\nexport interface CmsApp {\n dataSource: unknown;\n getPlugin<T = unknown>(name: string): T | undefined;\n}\n\nexport async function createCmsApp(options: CreateCmsAppOptions): Promise<CmsApp> {\n const { dataSource, config = {}, plugins = [], logger = noopLogger } = options;\n const context: PluginContext = { dataSource, config, logger };\n const registry = new Map<string, unknown>();\n\n for (const plugin of plugins) {\n try {\n const instance = await plugin.init(context);\n registry.set(plugin.name, instance !== undefined ? instance : plugin);\n } catch (err) {\n logger.warn(`Plugin \"${plugin.name}\" failed to init: ${err instanceof Error ? err.message : err}`);\n }\n }\n\n return {\n dataSource,\n getPlugin<T>(name: string): T | undefined {\n return registry.get(name) as T | undefined;\n },\n };\n}\n","export interface ContactFormData {\n firstName: string;\n lastName: string;\n email: string;\n phone?: string;\n industry?: string;\n message?: string;\n}\n\nexport interface ERPSubmissionResult {\n success: boolean;\n error?: string;\n status?: number;\n}\n\nexport type PipelineNames = { pipelineName: string; pipelineStageName: string };\n\n/** Payload for ERP `create-contact` (§7c) — CRM contact upsert, no lead/opportunity row. */\nexport interface ErpCreateContactPayload {\n email: string;\n firstName: string;\n lastName: string;\n phone?: string;\n companyName?: string;\n type?: string;\n notes?: string;\n tags?: string[];\n}\n\nexport class ERPSubmissionService {\n private webhookUrl: string;\n private webhookJwt: string;\n private getPipelineNames?: () => Promise<PipelineNames>;\n\n constructor(config: {\n webhookUrl: string;\n webhookJwt: string;\n getPipelineNames?: () => Promise<PipelineNames>;\n }) {\n this.webhookUrl = config.webhookUrl.replace(/\\/$/, '');\n this.webhookJwt = config.webhookJwt;\n this.getPipelineNames = config.getPipelineNames;\n }\n\n /** Replace trailing path segment of webhook URL with `action` (e.g. `get-order-status`, `invoice-pdf`). */\n resolveErpActionUrl(action: string): string {\n const base = this.webhookUrl.replace(/\\/$/, '');\n const i = base.lastIndexOf('/');\n if (i === -1) return `${base}/${action}`;\n return `${base.slice(0, i + 1)}${action}`;\n }\n\n /**\n * Synchronous ERP read (§7d): POST JSON to `.../generic-webhook/<action>`.\n * Body is sent as-is (or wrap with envelope if ERP expects `event_type` — caller may pass envelope).\n */\n async postErpReadAction(\n action: string,\n body: Record<string, unknown>\n ): Promise<{ ok: boolean; status?: number; json?: unknown; error?: string }> {\n const url = this.resolveErpActionUrl(action);\n try {\n const res = await fetch(url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-External-Token': this.webhookJwt,\n },\n body: JSON.stringify(body),\n });\n const text = await res.text();\n let parsed: unknown;\n try {\n parsed = text ? JSON.parse(text) : null;\n } catch {\n parsed = null;\n }\n if (!res.ok) {\n return { ok: false, status: res.status, error: `${res.status} ${text.slice(0, 500)}` };\n }\n return { ok: true, status: res.status, json: parsed };\n } catch (e: unknown) {\n return { ok: false, error: e instanceof Error ? e.message : 'ERP read request failed' };\n }\n }\n\n /** GET PDF bytes for `invoice-pdf` action (§7d). */\n async fetchInvoicePdf(invoiceId: string): Promise<{\n ok: boolean;\n buffer?: ArrayBuffer;\n contentType?: string;\n error?: string;\n }> {\n const base = this.resolveErpActionUrl('invoice-pdf');\n let u: URL;\n try {\n u = new URL(base);\n } catch {\n return { ok: false, error: 'Invalid ERP webhook URL' };\n }\n u.searchParams.set('invoiceId', invoiceId);\n u.searchParams.set('token', this.webhookJwt);\n try {\n const res = await fetch(u.toString(), { method: 'GET' });\n if (!res.ok) {\n const t = await res.text();\n return { ok: false, error: `${res.status} ${t.slice(0, 200)}` };\n }\n const buffer = await res.arrayBuffer();\n const contentType = res.headers.get('content-type') || 'application/pdf';\n return { ok: true, buffer, contentType };\n } catch (e: unknown) {\n return { ok: false, error: e instanceof Error ? e.message : 'invoice PDF fetch failed' };\n }\n }\n\n /** `product.updated` envelope → `update-product` (§7e). */\n async submitProductUpsert(productData: Record<string, unknown>): Promise<ERPSubmissionResult> {\n const envelope = {\n event_type: 'product.updated',\n timestamp: new Date().toISOString(),\n data: productData,\n };\n return this.postWebhookJson(envelope);\n }\n\n private async postWebhookJson(body: unknown): Promise<ERPSubmissionResult> {\n try {\n const res = await fetch(this.webhookUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-External-Token': this.webhookJwt,\n },\n body: JSON.stringify(body),\n });\n if (res.ok) return { success: true, status: res.status };\n const text = await res.text();\n return { success: false, error: `${res.status} ${text.slice(0, 500)}`, status: res.status };\n } catch (e: unknown) {\n const message = e instanceof Error ? e.message : 'ERP webhook request failed';\n return { success: false, error: message };\n }\n }\n\n private async resolvePipelineNames(): Promise<PipelineNames> {\n let pipelineName = '';\n let pipelineStageName = '';\n if (this.getPipelineNames) {\n try {\n const n = await this.getPipelineNames();\n pipelineName = (n.pipelineName ?? '').trim();\n pipelineStageName = (n.pipelineStageName ?? '').trim();\n } catch {\n /* ignore */\n }\n }\n return { pipelineName, pipelineStageName };\n }\n\n /** Generic webhook POST (e.g. order.created envelope). */\n async postWebhook(body: unknown): Promise<ERPSubmissionResult> {\n return this.postWebhookJson(body);\n }\n\n /** Shared CRM inbound fields (§7a/b in ERP-plugin.md). */\n private async buildCrmInboundData(formData: ContactFormData): Promise<{ error?: string; data?: Record<string, unknown> }> {\n const email = (formData.email ?? '').trim();\n if (!email) {\n return { error: 'email is required' };\n }\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n if (!emailRegex.test(email)) {\n return { error: 'Invalid email format' };\n }\n\n const firstName = (formData.firstName ?? '').trim() || email.split('@')[0] || 'Contact';\n const lastName = (formData.lastName ?? '').trim() || '';\n\n const { pipelineName, pipelineStageName } = await this.resolvePipelineNames();\n\n const data: Record<string, unknown> = {\n email,\n firstName,\n lastName,\n phone: formData.phone?.trim() || undefined,\n subject: formData.industry?.trim() || undefined,\n message: formData.message?.trim() || undefined,\n };\n if (pipelineName) data.pipelineName = pipelineName;\n if (pipelineStageName) data.pipelineStageName = pipelineStageName;\n return { data };\n }\n\n /** `lead.created` → ERP create-lead (CRM leads table). */\n async submitContact(formData: ContactFormData): Promise<ERPSubmissionResult> {\n const built = await this.buildCrmInboundData(formData);\n if (built.error || !built.data) {\n return { success: false, error: built.error ?? 'invalid payload' };\n }\n built.data.leadSource = 'Website';\n\n const envelope = {\n event_type: 'lead.created',\n timestamp: new Date().toISOString(),\n data: built.data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /** `create-contact` → ERP ContactService upsert (§7c). */\n async submitCreateContact(payload: ErpCreateContactPayload): Promise<ERPSubmissionResult> {\n const email = (payload.email ?? '').trim();\n if (!email) {\n return { success: false, error: 'email is required' };\n }\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n if (!emailRegex.test(email)) {\n return { success: false, error: 'Invalid email format' };\n }\n const firstName = (payload.firstName ?? '').trim() || email.split('@')[0] || 'Contact';\n const lastName = (payload.lastName ?? '').trim() || '';\n\n const data: Record<string, unknown> = {\n email,\n firstName,\n lastName,\n phone: payload.phone?.trim() || undefined,\n companyName: payload.companyName?.trim() || undefined,\n type: payload.type?.trim() || undefined,\n message: payload.notes?.trim() || undefined,\n metadata: { source: 'cms' },\n };\n if (payload.tags?.length) {\n data.tags = payload.tags;\n }\n\n const envelope = {\n event_type: 'create-contact',\n timestamp: new Date().toISOString(),\n data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /** `form.submitted` → ERP create-opportunity (pipeline). */\n async submitFormOpportunity(formData: ContactFormData): Promise<ERPSubmissionResult> {\n const built = await this.buildCrmInboundData(formData);\n if (built.error || !built.data) {\n return { success: false, error: built.error ?? 'invalid payload' };\n }\n\n const envelope = {\n event_type: 'form.submitted',\n timestamp: new Date().toISOString(),\n data: built.data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /**\n * Order payload aligned with ERP generic webhook create-order (§6).\n */\n async submitOrder(orderDto: Record<string, unknown>): Promise<ERPSubmissionResult> {\n const envelope = {\n event_type: 'order.created',\n timestamp: new Date().toISOString(),\n data: orderDto,\n };\n return this.postWebhookJson(envelope);\n }\n\n extractContactData(formData: Record<string, unknown>, formFields: { id: string | number; type: string; label: string }[]): ContactFormData | null {\n const contactData: ContactFormData = {\n firstName: '',\n lastName: '',\n email: '',\n phone: '',\n industry: '',\n message: '',\n };\n let hasEmail = false;\n\n for (const field of formFields) {\n const fieldValue = formData[field.id.toString()];\n if (fieldValue == null) continue;\n const label = field.label.toLowerCase();\n const value = String(fieldValue).trim();\n\n if (field.type === 'email') {\n contactData.email = value;\n hasEmail = true;\n } else if (field.type === 'phone') {\n contactData.phone = value;\n } else if (field.type === 'text' || field.type === 'textarea') {\n if (label.includes('first name') || label.includes('firstname')) {\n contactData.firstName = value;\n } else if (label.includes('last name') || label.includes('lastname')) {\n contactData.lastName = value;\n } else if (label.includes('name') && !contactData.firstName) {\n const nameParts = value.split(' ');\n if (nameParts.length >= 2) {\n contactData.firstName = nameParts[0]!;\n contactData.lastName = nameParts.slice(1).join(' ');\n } else {\n contactData.firstName = value;\n }\n } else if (label.includes('industry')) {\n contactData.industry = value;\n } else if (label.includes('message') || label.includes('comment') || label.includes('description') || label.includes('inquiry')) {\n contactData.message = value;\n }\n }\n }\n\n if (!hasEmail && !contactData.firstName) return null;\n if (!contactData.firstName && contactData.email) {\n contactData.firstName = contactData.email.split('@')[0] ?? '';\n contactData.lastName = '';\n }\n return contactData;\n }\n}\n","import type { ERPSubmissionService } from './erp-submission';\nimport type { ContactFormData, ErpCreateContactPayload } from './erp-submission';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\nconst ERP_QUEUE_NAME = 'erp';\n\nexport type ErpJobPayload =\n | { kind: 'lead'; contact: ContactFormData }\n | { kind: 'formOpportunity'; contact: ContactFormData }\n | { kind: 'createContact'; contact: ErpCreateContactPayload }\n | { kind: 'order'; order: Record<string, unknown> }\n | { kind: 'productUpsert'; product: Record<string, unknown> };\n\nexport async function queueErp(cms: CmsAppLike, payload: ErpJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n if (!queue) return;\n await queue.add(ERP_QUEUE_NAME, payload as object);\n}\n\nexport function registerErpQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as\n | { registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void }\n | undefined;\n if (!queue) return;\n queue.registerProcessor(ERP_QUEUE_NAME, async (data: object) => {\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp) return;\n const payload = data as ErpJobPayload;\n if (payload.kind === 'lead') {\n await erp.submission.submitContact(payload.contact);\n } else if (payload.kind === 'formOpportunity') {\n await erp.submission.submitFormOpportunity(payload.contact);\n } else if (payload.kind === 'createContact') {\n await erp.submission.submitCreateContact(payload.contact);\n } else if (payload.kind === 'order') {\n await erp.submission.submitOrder(payload.order);\n } else if (payload.kind === 'productUpsert') {\n await erp.submission.submitProductUpsert(payload.product);\n }\n });\n}\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp, type CmsAppLike } from './erp-queue';\n\n/**\n * Duck-typed DataSource so host apps (e.g. Next) can pass their own TypeORM instance\n * without duplicate-package type errors vs core’s node_modules/typeorm.\n */\nexport type ErpPaidOrderDataSource = {\n getRepository(entity: unknown): {\n find(options?: object): Promise<unknown[]>;\n findOne(options?: object): Promise<unknown | null>;\n };\n};\n\n/**\n * Entity map with `configs` and `orders` entries (e.g. ENTITY_MAP).\n * Typed as `Record<string, unknown>` so apps that cast their map to `Record<string, EntityTarget<…>>` still pass.\n */\nexport type ErpPaidOrderEntityMap = Record<string, unknown>;\n\nfunction amountToMinorUnits(major: number, currency: string): number {\n const c = currency.toUpperCase();\n const zeroDecimal = new Set([\n 'BIF',\n 'CLP',\n 'DJF',\n 'GNF',\n 'JPY',\n 'KMF',\n 'KRW',\n 'MGA',\n 'PYG',\n 'RWF',\n 'UGX',\n 'VND',\n 'VUV',\n 'XAF',\n 'XOF',\n 'XPF',\n ]);\n if (zeroDecimal.has(c)) return Math.round(major);\n return Math.round(major * 100);\n}\n\nfunction addressToWebhookDto(a: ObjectLiteral | null | undefined): Record<string, unknown> {\n if (!a) return {};\n return {\n line1: a.line1 ?? '',\n line2: a.line2 ?? '',\n city: a.city ?? '',\n state: a.state ?? '',\n postalCode: a.postalCode ?? '',\n country: a.country ?? '',\n };\n}\n\nfunction orderStatusLabel(status: string | undefined): string {\n const s = (status || '').toLowerCase();\n if (s === 'confirmed') return 'Confirmed';\n if (s === 'pending') return 'Pending';\n if (!status) return 'Pending';\n return status.charAt(0).toUpperCase() + status.slice(1);\n}\n\nfunction paymentRowToWebhookDto(p: ObjectLiteral): Record<string, unknown> {\n const currency = String(p.currency || 'INR');\n const amountMajor = Number(p.amount);\n const meta = (p.metadata as Record<string, unknown>) || {};\n return {\n id: String(p.externalReference || `payment_${p.id}`),\n amount: amountToMinorUnits(amountMajor, currency),\n currency_code: currency,\n captured_at: p.paidAt\n ? new Date(p.paidAt as Date | string).toISOString()\n : new Date().toISOString(),\n provider_id: String(p.method || 'unknown'),\n data: { status: 'captured', ...meta },\n };\n}\n\n/**\n * Enqueues ERP `order.created` only when the order has at least one **completed** payment.\n * Include `payments` in the payload per §6 (minor units for typical 2-decimal currencies).\n */\nexport async function queueErpPaidOrderForOrderId(\n cms: CmsAppLike,\n dataSource: ErpPaidOrderDataSource,\n entityMap: ErpPaidOrderEntityMap,\n orderId: number\n): Promise<void> {\n try {\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return;\n }\n if (!cms.getPlugin('erp')) return;\n\n const orderRepo = dataSource.getRepository(entityMap.orders);\n const ord = await orderRepo.findOne({\n where: { id: orderId } as ObjectLiteral,\n relations: ['items', 'items.product', 'contact', 'billingAddress', 'shippingAddress', 'payments'],\n });\n if (!ord) return;\n const o = ord as ObjectLiteral;\n const okKind = o.orderKind === undefined || o.orderKind === null || o.orderKind === 'sale';\n if (!okKind) return;\n const rawPayments = (o.payments as ObjectLiteral[]) ?? [];\n const completedPayments = rawPayments.filter((pay) => pay.status === 'completed' && pay.deleted !== true);\n if (!completedPayments.length) return;\n\n const rawItems = (o.items as ObjectLiteral[]) ?? [];\n const lines = rawItems\n .filter((it) => it.product)\n .map((it) => {\n const p = it.product as ObjectLiteral;\n const sku = (p.sku as string) || `SKU-${p.id}`;\n return {\n sku,\n quantity: Number(it.quantity) || 1,\n unitPrice: Number(it.unitPrice),\n title: (p.name as string) || sku,\n discount: 0,\n tax: Number(it.tax) || 0,\n };\n });\n if (!lines.length) return;\n\n const contact = o.contact as ObjectLiteral | undefined;\n const paymentDtos = completedPayments.map((pay) => paymentRowToWebhookDto(pay));\n const baseMeta =\n o.metadata && typeof o.metadata === 'object' && !Array.isArray(o.metadata)\n ? { ...(o.metadata as Record<string, unknown>) }\n : {};\n\n const orderDto: Record<string, unknown> = {\n platformType: 'website',\n platformOrderId: String(o.orderNumber),\n platformOrderNumber: String(o.orderNumber),\n status: orderStatusLabel(o.status as string | undefined),\n customer: {\n name: (contact?.name as string) || '',\n email: (contact?.email as string) || '',\n phone: (contact?.phone as string) || '',\n },\n shippingAddress: addressToWebhookDto(o.shippingAddress as ObjectLiteral),\n billingAddress: addressToWebhookDto(o.billingAddress as ObjectLiteral),\n items: lines,\n payments: paymentDtos,\n metadata: { ...baseMeta, source: 'storefront' },\n };\n\n await queueErp(cms, { kind: 'order', order: orderDto });\n } catch {\n /* non-fatal */\n }\n}\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp, type CmsAppLike } from './erp-queue';\nimport type { ErpPaidOrderDataSource, ErpPaidOrderEntityMap } from './paid-order-erp';\n\nexport interface ErpContactSyncInput {\n name: string;\n email: string;\n phone?: string | null;\n type?: string | null;\n company?: string | null;\n notes?: string | null;\n /** Passed to ERP as `tags` when non-empty (§7c). */\n tags?: string[];\n}\n\nfunction splitName(full: string): { firstName: string; lastName: string } {\n const t = (full || '').trim();\n if (!t) return { firstName: 'Contact', lastName: '' };\n const parts = t.split(/\\s+/);\n if (parts.length === 1) return { firstName: parts[0]!, lastName: '' };\n return { firstName: parts[0]!, lastName: parts.slice(1).join(' ') };\n}\n\n/**\n * When ERP is enabled and plugin loaded, enqueue `create-contact` (non-fatal on failure).\n * Used for admin CRUD and storefront contact paths — not form submissions (those use lead/opportunity).\n */\nexport async function queueErpCreateContactIfEnabled(\n cms: CmsAppLike,\n dataSource: ErpPaidOrderDataSource,\n entityMap: ErpPaidOrderEntityMap,\n input: ErpContactSyncInput\n): Promise<void> {\n try {\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return;\n }\n if (!cms.getPlugin('erp')) return;\n\n const email = (input.email ?? '').trim();\n if (!email) return;\n\n const { firstName, lastName } = splitName(input.name);\n\n await queueErp(cms, {\n kind: 'createContact',\n contact: {\n email,\n firstName,\n lastName,\n phone: input.phone?.trim() || undefined,\n companyName: input.company?.trim() || undefined,\n type: input.type?.trim() || undefined,\n notes: input.notes?.trim() || undefined,\n tags: input.tags?.length ? [...input.tags] : undefined,\n },\n });\n } catch {\n /* non-fatal */\n }\n}\n","export type OrderKind = 'sale' | 'return' | 'replacement';\n\nconst KIND_PREFIX: Record<OrderKind, string> = {\n sale: 'OSL',\n return: 'ORT',\n replacement: 'ORP',\n};\n\n/** YYMM in UTC (two-digit year + month). */\nexport function orderNumberYymmUtc(at: Date): string {\n const yy = String(at.getUTCFullYear()).slice(-2);\n const mm = String(at.getUTCMonth() + 1).padStart(2, '0');\n return yy + mm;\n}\n\n/** Stable 8-char mask from numeric id (not reversible as plain decimal). */\nexport function maskOrderIdSegment(id: number): string {\n let x = (id >>> 0) ^ 0xa5a5a5a5;\n x = Math.imul(x, 2654435761) >>> 0;\n return x.toString(36).toUpperCase().padStart(8, '0').slice(-8);\n}\n\nexport function buildCanonicalOrderNumber(kind: OrderKind, id: number, at: Date): string {\n return KIND_PREFIX[kind] + orderNumberYymmUtc(at) + maskOrderIdSegment(id);\n}\n\nexport function temporaryOrderNumberPlaceholder(): string {\n return `TMP${Date.now().toString(36)}${Math.random().toString(36).slice(2, 10)}`.toUpperCase();\n}\n","export interface OrderFulfillmentEvent {\n at?: string;\n label?: string;\n detail?: string;\n}\n\nexport interface OrderFulfillmentMetadata {\n status?: string;\n trackingId?: string;\n events?: OrderFulfillmentEvent[];\n}\n\nexport interface OrderInvoiceMetadata {\n invoiceNumber?: string;\n /** ERP invoice UUID for PDF fetch (no JWT). */\n invoiceId?: string;\n /** App path for invoice PDF (storefront or site-relative). */\n link?: string;\n}\n\nexport interface OrderStorefrontMetadataShape {\n fulfillment?: OrderFulfillmentMetadata;\n invoice?: OrderInvoiceMetadata;\n cartId?: number;\n platformRef?: string;\n [key: string]: unknown;\n}\n\nexport function mergeOrderMetadataPatch(\n existing: Record<string, unknown> | null | undefined,\n patch: { fulfillment?: OrderFulfillmentMetadata | null; invoice?: OrderInvoiceMetadata | null }\n): Record<string, unknown> {\n const base =\n existing && typeof existing === 'object' && !Array.isArray(existing) ? { ...existing } : {};\n if (patch.fulfillment !== undefined) {\n if (patch.fulfillment === null) delete base.fulfillment;\n else base.fulfillment = patch.fulfillment;\n }\n if (patch.invoice !== undefined) {\n if (patch.invoice === null) delete base.invoice;\n else base.invoice = patch.invoice;\n }\n return base;\n}\n","import type { Order } from '../../entities/order.entity';\n\n/** Map ERP sale-order status labels into CMS `Order.status`. Unknown → undefined (caller keeps existing). */\nexport function mapErpSaleStatusToOrderStatus(erpLabel: string | undefined): Order['status'] | undefined {\n if (!erpLabel || typeof erpLabel !== 'string') return undefined;\n const k = erpLabel.trim().toLowerCase().replace(/\\s+/g, '_');\n const map: Record<string, Order['status']> = {\n draft: 'pending',\n pending: 'pending',\n open: 'pending',\n new: 'pending',\n unconfirmed: 'pending',\n confirmed: 'confirmed',\n processing: 'processing',\n packed: 'processing',\n shipped: 'processing',\n in_transit: 'processing',\n out_for_delivery: 'processing',\n delivered: 'completed',\n completed: 'completed',\n closed: 'completed',\n fulfilled: 'completed',\n cancelled: 'cancelled',\n canceled: 'cancelled',\n void: 'cancelled',\n };\n return map[k];\n}\n","import type { ObjectLiteral, Repository } from 'typeorm';\nimport { buildCanonicalOrderNumber, temporaryOrderNumberPlaceholder, type OrderKind } from '../../lib/order-number';\nimport { mergeOrderMetadataPatch, type OrderFulfillmentMetadata, type OrderInvoiceMetadata } from '../../lib/order-storefront-metadata';\nimport { mapErpSaleStatusToOrderStatus } from './erp-order-status-map';\nimport {\n extractChildOrderRefsFromSalePayload,\n mapErpPayloadToFulfillment,\n mapErpPayloadToInvoiceNumber,\n unwrapErpReadData,\n} from './erp-response-map';\nimport type { ERPSubmissionService } from './erp-submission';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nfunction pickInvoiceId(data: Record<string, unknown>): string | undefined {\n const nested =\n data.invoice && typeof data.invoice === 'object' && !Array.isArray(data.invoice)\n ? (data.invoice as Record<string, unknown>)\n : null;\n const src = nested || data;\n for (const k of ['invoiceId', 'invoice_id', 'id']) {\n const v = src[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\nasync function ensureChildOrdersFromRefs(\n orderRepo: Repository<ObjectLiteral>,\n parent: ObjectLiteral,\n refs: ReturnType<typeof extractChildOrderRefsFromSalePayload>,\n contactId: number,\n currency: string\n): Promise<void> {\n for (const { ref, orderKind } of refs) {\n const existing = await orderRepo\n .createQueryBuilder('o')\n .where('o.parentOrderId = :pid', { pid: parent.id as number })\n .andWhere('o.deleted = :d', { d: false })\n .andWhere(\"o.metadata->>'platformRef' = :ref\", { ref })\n .getOne();\n if (existing) continue;\n const tmp = temporaryOrderNumberPlaceholder();\n const row = await orderRepo.save(\n orderRepo.create({\n orderNumber: tmp,\n orderKind,\n parentOrderId: parent.id as number,\n contactId,\n billingAddressId: null,\n shippingAddressId: null,\n status: 'pending',\n subtotal: 0,\n tax: 0,\n discount: 0,\n total: 0,\n currency,\n metadata: { platformRef: ref },\n deleted: false,\n } as ObjectLiteral)\n );\n const r = row as { id: number; createdAt: Date };\n await orderRepo.update(r.id, {\n orderNumber: buildCanonicalOrderNumber(orderKind as OrderKind, r.id, r.createdAt ?? new Date()),\n } as ObjectLiteral);\n }\n}\n\nfunction deepMergeFulfillment(\n a: OrderFulfillmentMetadata | undefined,\n b: OrderFulfillmentMetadata | undefined\n): OrderFulfillmentMetadata | undefined {\n if (!a) return b;\n if (!b) return a;\n return {\n ...a,\n ...b,\n events: b.events?.length ? b.events : a.events,\n };\n}\n\n/**\n * Calls ERP §7d reads for the given order row, updates `status` + `metadata` (fulfillment / invoice),\n * and upserts child return/replacement rows when the sale payload lists them.\n */\nexport async function refreshOrderFromErp(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n submission: ERPSubmissionService,\n order: ObjectLiteral\n): Promise<void> {\n const orderRepo = dataSource.getRepository(entityMap.orders) as Repository<ObjectLiteral>;\n\n const kind = (order.orderKind as string) || 'sale';\n const meta =\n order.metadata && typeof order.metadata === 'object' && !Array.isArray(order.metadata)\n ? { ...(order.metadata as Record<string, unknown>) }\n : {};\n\n if (kind === 'sale') {\n const refId = String(order.orderNumber || '');\n let fulfillment: OrderFulfillmentMetadata | undefined;\n let invoiceNumber: string | undefined;\n let invoiceId: string | undefined;\n let newStatus: string | undefined;\n\n const r1 = await submission.postErpReadAction('get-order-status', { platformOrderId: refId });\n const d1 = r1.ok ? unwrapErpReadData(r1.json) : null;\n if (d1) {\n const mapped = mapErpSaleStatusToOrderStatus(\n typeof d1.status === 'string'\n ? d1.status\n : typeof d1.orderStatus === 'string'\n ? d1.orderStatus\n : typeof d1.state === 'string'\n ? d1.state\n : undefined\n );\n if (mapped) newStatus = mapped;\n fulfillment = mapErpPayloadToFulfillment(d1);\n const refs = extractChildOrderRefsFromSalePayload(d1);\n if (refs.length) {\n await ensureChildOrdersFromRefs(\n orderRepo,\n order,\n refs,\n order.contactId as number,\n String(order.currency || 'INR')\n );\n }\n }\n\n const r2 = await submission.postErpReadAction('get-fulfillment-status', { platformOrderId: refId });\n const d2 = r2.ok ? unwrapErpReadData(r2.json) : null;\n if (d2) {\n fulfillment = deepMergeFulfillment(fulfillment, mapErpPayloadToFulfillment(d2));\n }\n\n const r3 = await submission.postErpReadAction('get-invoice', { platformOrderId: refId });\n const d3 = r3.ok ? unwrapErpReadData(r3.json) : null;\n if (d3) {\n invoiceNumber = mapErpPayloadToInvoiceNumber(d3);\n invoiceId = pickInvoiceId(d3);\n }\n\n const oid = order.id as number;\n const prevInv =\n meta.invoice && typeof meta.invoice === 'object' && !Array.isArray(meta.invoice)\n ? { ...(meta.invoice as Record<string, unknown>) }\n : {};\n const nextInvoice: OrderInvoiceMetadata = {\n ...prevInv,\n link: `/api/storefront/orders/${oid}/invoice`,\n ...(invoiceNumber ? { invoiceNumber } : {}),\n ...(invoiceId ? { invoiceId } : {}),\n };\n\n const patch: {\n fulfillment?: OrderFulfillmentMetadata | null;\n invoice?: OrderInvoiceMetadata | null;\n } = { invoice: nextInvoice };\n if (fulfillment !== undefined) patch.fulfillment = fulfillment;\n\n const nextMeta = mergeOrderMetadataPatch(meta, patch);\n\n await orderRepo.update(oid, {\n ...(newStatus ? { status: newStatus } : {}),\n metadata: nextMeta,\n updatedAt: new Date(),\n } as ObjectLiteral);\n return;\n }\n\n if (kind === 'return' || kind === 'replacement') {\n const platformReturnId = String(order.orderNumber || '');\n const r = await submission.postErpReadAction('get-return-status', { platformReturnId });\n const d = r.ok ? unwrapErpReadData(r.json) : null;\n if (!d) return;\n const mapped = mapErpSaleStatusToOrderStatus(\n typeof d.status === 'string' ? d.status : typeof d.returnStatus === 'string' ? d.returnStatus : undefined\n );\n const fulfillment = mapErpPayloadToFulfillment(d);\n const patch: { fulfillment?: OrderFulfillmentMetadata | null } = {};\n if (fulfillment !== undefined) patch.fulfillment = fulfillment;\n const nextMeta = Object.keys(patch).length ? mergeOrderMetadataPatch(meta, patch) : meta;\n await orderRepo.update(order.id as number, {\n ...(mapped ? { status: mapped } : {}),\n metadata: nextMeta,\n updatedAt: new Date(),\n } as ObjectLiteral);\n }\n}\n\nexport async function tryRefreshOrderFromErpForStorefront(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n order: ObjectLiteral\n): Promise<void> {\n try {\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return;\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp?.submission) return;\n await refreshOrderFromErp(cms, dataSource, entityMap, erp.submission, order);\n } catch {\n /* non-fatal */\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { ERPSubmissionService } from './erp-submission';\nimport type { PipelineNames } from './erp-submission';\n\nexport interface ERPPluginConfig {\n webhookUrl: string;\n webhookJwt: string;\n getPipelineNames?: () => Promise<PipelineNames>;\n}\n\nexport interface ERPPluginInstance {\n submission: ERPSubmissionService;\n}\n\nexport function erpPlugin(config: ERPPluginConfig): CmsPlugin<ERPPluginInstance> {\n return {\n name: 'erp',\n version: '2.0.0',\n async init() {\n const submission = new ERPSubmissionService({\n webhookUrl: config.webhookUrl,\n webhookJwt: config.webhookJwt,\n getPipelineNames: config.getPipelineNames,\n });\n return { submission };\n },\n };\n}\n\nexport { ERPSubmissionService } from './erp-submission';\nexport type { ContactFormData, ERPSubmissionResult, ErpCreateContactPayload, PipelineNames } from './erp-submission';\nexport { registerErpQueueProcessor, queueErp } from './erp-queue';\nexport type { ErpJobPayload } from './erp-queue';\nexport { queueErpPaidOrderForOrderId, type ErpPaidOrderDataSource, type ErpPaidOrderEntityMap } from './paid-order-erp';\nexport { queueErpCreateContactIfEnabled, type ErpContactSyncInput } from './erp-contact-sync';\nexport { tryRefreshOrderFromErpForStorefront, refreshOrderFromErp } from './erp-order-sync';\nexport { streamOrderInvoicePdf } from './erp-order-invoice';\nexport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\nexport { queueErpProductUpsertIfEnabled } from './erp-product-sync';\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp } from './erp-queue';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nexport async function queueErpProductUpsertIfEnabled(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n product: ObjectLiteral\n): Promise<void> {\n try {\n const sku = typeof product.sku === 'string' ? product.sku.trim() : '';\n if (!sku) return;\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return;\n const payload: Record<string, unknown> = {\n sku,\n title: (product.name as string) || sku,\n name: product.name,\n description: typeof product.metadata === 'object' && product.metadata && 'description' in (product.metadata as object)\n ? String((product.metadata as Record<string, unknown>).description ?? '')\n : undefined,\n hsn_number: product.hsn,\n is_active: product.status === 'available',\n metadata: product.metadata ?? undefined,\n };\n await queueErp(cms, { kind: 'productUpsert', product: payload });\n } catch {\n /* non-fatal */\n }\n}\n","import { SESClient, SendEmailCommand } from '@aws-sdk/client-ses';\nimport nodemailer from 'nodemailer';\nimport type { EmailTemplateName, TemplateContext } from './templates/types';\nimport type { RenderEmailOptions } from './renderer';\nimport { renderEmail } from './renderer';\n\nexport interface EmailPluginConfig {\n type: 'AWS' | 'SMTP' | 'GMAIL' | 'SENDGRID';\n user?: string;\n password?: string;\n from: string;\n to: string;\n /** When type is SMTP, defaults to env SMTP_HOST or undefined (set host for custom relay). */\n host?: string;\n port?: number;\n secure?: boolean;\n region?: string;\n accessKeyId?: string;\n secretAccessKey?: string;\n templateOptions?: RenderEmailOptions;\n}\n\nexport interface EmailData {\n subject: string;\n html: string;\n text?: string;\n to?: string;\n from?: string;\n}\n\nexport interface EmailServiceInterface {\n send(emailData: EmailData): Promise<boolean>;\n}\n\nexport interface RenderedEmail {\n subject: string;\n html: string;\n text?: string;\n}\n\nexport class EmailService implements EmailServiceInterface {\n private config: EmailPluginConfig;\n private templateOptions?: RenderEmailOptions;\n private sesClient?: SESClient;\n private transporter?: nodemailer.Transporter;\n\n constructor(config: EmailPluginConfig) {\n this.config = config;\n this.templateOptions = config.templateOptions;\n if (config.type === 'AWS') {\n if (!config.region || !config.accessKeyId || !config.secretAccessKey) {\n throw new Error('AWS SES configuration incomplete');\n }\n this.sesClient = new SESClient({\n region: config.region,\n credentials: { accessKeyId: config.accessKeyId, secretAccessKey: config.secretAccessKey },\n });\n } else if (config.type === 'SMTP' || config.type === 'GMAIL') {\n if (!config.user || !config.password) throw new Error('SMTP configuration incomplete');\n const host =\n config.type === 'GMAIL' ? 'smtp.gmail.com' : config.host || undefined;\n const port = config.port ?? 587;\n const secure = config.secure ?? false;\n this.transporter = nodemailer.createTransport({\n ...(host ? { host } : {}),\n port,\n secure,\n auth: { user: config.user, pass: config.password },\n });\n } else {\n throw new Error(`Unsupported email type: ${config.type}`);\n }\n }\n\n async send(emailData: EmailData): Promise<boolean> {\n try {\n if (this.config.type === 'AWS' && this.sesClient) {\n await this.sesClient.send(\n new SendEmailCommand({\n Source: emailData.from || this.config.from,\n Destination: { ToAddresses: [emailData.to || this.config.to] },\n Message: {\n Subject: { Data: emailData.subject, Charset: 'UTF-8' },\n Body: {\n Html: { Data: emailData.html, Charset: 'UTF-8' },\n ...(emailData.text && { Text: { Data: emailData.text, Charset: 'UTF-8' } }),\n },\n },\n })\n );\n return true;\n }\n if ((this.config.type === 'SMTP' || this.config.type === 'GMAIL') && this.transporter) {\n await this.transporter.sendMail({\n from: emailData.from || this.config.from,\n to: emailData.to || this.config.to,\n subject: emailData.subject,\n html: emailData.html,\n text: emailData.text,\n });\n return true;\n }\n return false;\n } catch (error) {\n console.error('Email sending failed:', error);\n return false;\n }\n }\n\n renderTemplate(templateName: EmailTemplateName, ctx: TemplateContext<unknown>): RenderedEmail {\n return renderEmail(templateName, ctx, this.templateOptions);\n }\n}\n\nexport const emailTemplates = {\n formSubmission: (data: { formName: string; contactName: string; contactEmail: string; formData: unknown }) => ({\n subject: `New Form Submission: ${data.formName}`,\n html: `<h2>New Form Submission</h2><p><strong>Form:</strong> ${data.formName}</p><p><strong>Contact:</strong> ${data.contactName} (${data.contactEmail})</p><pre>${JSON.stringify(data.formData, null, 2)}</pre>`,\n text: `New Form Submission\\nForm: ${data.formName}\\nContact: ${data.contactName} (${data.contactEmail})\\n${JSON.stringify(data.formData, null, 2)}`,\n }),\n contactSubmission: (data: { name: string; email: string; phone?: string; message?: string }) => ({\n subject: `New Contact Form Submission from ${data.name}`,\n html: `<h2>New Contact Form Submission</h2><p><strong>Name:</strong> ${data.name}</p><p><strong>Email:</strong> ${data.email}</p>${data.phone ? `<p><strong>Phone:</strong> ${data.phone}</p>` : ''}${data.message ? `<p><strong>Message:</strong></p><p>${data.message}</p>` : ''}`,\n text: `New Contact Form Submission\\nName: ${data.name}\\nEmail: ${data.email}\\n${data.phone ? `Phone: ${data.phone}\\n` : ''}${data.message ? `Message: ${data.message}` : ''}`,\n }),\n passwordReset: (data: { resetLink: string }) => ({\n subject: 'Reset your password',\n html: `<h2>Reset your password</h2><p>Click the link below to set a new password. This link expires in 1 hour.</p><p><a href=\"${data.resetLink}\">${data.resetLink}</a></p>`,\n text: `Reset your password: ${data.resetLink}\\n\\nThis link expires in 1 hour.`,\n }),\n};\n","import type { CompanyDetails, SocialLinkItem } from './types';\n\nfunction escapeHtml(s: string): string {\n return s\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;');\n}\n\nfunction socialLinkInnerHtml(s: SocialLinkItem): string {\n if (s.iconUrl?.trim()) {\n return `<img src=\"${escapeHtml(s.iconUrl.trim())}\" alt=\"\" width=\"20\" height=\"20\" style=\"display:inline-block;vertical-align:middle;border:0;\" />`;\n }\n const label = (s.icon && s.icon.trim()) || '🔗';\n return escapeHtml(label);\n}\n\nfunction supportLinesHtml(supportEmail?: string, supportPhone?: string, opts?: { tightTop?: boolean }): string {\n const parts: string[] = [];\n const top = opts?.tightTop ? 'margin-top:0;' : 'margin-top:4px;';\n if (supportEmail) {\n parts.push(`<div style=\"font-size:13px;color:#444;${top}\">&#128231; ${escapeHtml(supportEmail)}</div>`);\n }\n if (supportPhone) {\n parts.push(`<div style=\"font-size:13px;color:#444;margin-top:2px;\">&#128222; ${escapeHtml(supportPhone)}</div>`);\n }\n return parts.join('');\n}\n\nexport function renderLayout(options: {\n bodyHtml: string;\n companyDetails: CompanyDetails;\n}): string {\n const { bodyHtml, companyDetails } = options;\n const {\n logoUrl,\n companyName,\n supportEmail,\n supportPhone,\n socialLinks,\n footerDisclaimer,\n followUsTitle = 'Follow Us',\n } = companyDetails;\n\n const supportFooterHtml = supportLinesHtml(supportEmail, supportPhone, { tightTop: true });\n\n const hasSocial = Boolean(socialLinks?.length);\n const brandFooterInner: string[] = [];\n if (logoUrl) {\n brandFooterInner.push(\n `<img src=\"${escapeHtml(logoUrl)}\" alt=\"\" style=\"max-height:28px;max-width:160px;height:auto;width:auto;display:inline-block;vertical-align:middle;margin-right:10px;border:0;\" />`\n );\n }\n if (companyName) {\n brandFooterInner.push(\n `<span style=\"font-size:14px;font-weight:600;color:#111;vertical-align:middle;\">${escapeHtml(companyName)}</span>`\n );\n }\n const brandFooterHtml = brandFooterInner.length\n ? `<div style=\"line-height:1.4;\">${brandFooterInner.join('')}</div>`\n : '';\n\n const followTitleHtml =\n hasSocial && followUsTitle\n ? `<div class=\"footer-follow-title\" style=\"font-size:13px;font-weight:600;color:#555;margin:0;\">${escapeHtml(followUsTitle)}</div>`\n : '';\n\n const socialRowHtml =\n hasSocial && socialLinks\n ? `<div class=\"footer-social-icons\" style=\"margin-top:6px;text-align:left;font-size:0;line-height:0;\">${socialLinks\n .map(\n (s) =>\n `<a href=\"${escapeHtml(s.url)}\" style=\"display:inline-block;margin-right:12px;text-decoration:none;color:#333;font-size:18px;line-height:1;vertical-align:middle;\">${socialLinkInnerHtml(s)}</a>`\n )\n .join('')}</div>`\n : '';\n\n const rowSupport = supportFooterHtml\n ? `<tr><td class=\"footer-support-cell\" colspan=\"2\" valign=\"top\" style=\"padding:0 0 10px 0;\">${supportFooterHtml}</td></tr>`\n : '';\n\n let footerBlock = '';\n if (brandFooterHtml || followTitleHtml || socialRowHtml || footerDisclaimer || rowSupport) {\n const rowBrand = brandFooterHtml\n ? `<tr><td class=\"footer-brand-cell\" colspan=\"2\" valign=\"top\" style=\"padding:0 0 10px 0;\">${brandFooterHtml}</td></tr>`\n : '';\n const rowFollowTitle = followTitleHtml\n ? `<tr><td class=\"footer-follow-cell\" colspan=\"2\" valign=\"top\" style=\"padding:14px 0 0 0;text-align:left;\">${followTitleHtml}</td></tr>`\n : '';\n const rowSocial = socialRowHtml\n ? `<tr><td class=\"footer-social-cell\" colspan=\"2\" style=\"padding:0;text-align:left;\">${socialRowHtml}</td></tr>`\n : '';\n footerBlock = `<table role=\"presentation\" class=\"footer-main\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"margin-top:28px;padding-top:16px;\">${rowBrand}${rowSupport}${rowFollowTitle}${rowSocial}</table>`;\n }\n\n const disclaimerBlock = footerDisclaimer\n ? `<div style=\"margin-top:20px;padding-top:12px;font-size:11px;line-height:1.5;color:#888;\">${escapeHtml(footerDisclaimer).replace(/\\n/g, '<br/>')}</div>`\n : '';\n\n const responsiveCss = `\n@media only screen and (max-width: 600px) {\n .email-wrap { padding-left: 12px !important; padding-right: 12px !important; }\n .footer-follow-cell { padding-top: 12px !important; }\n}\n`;\n\n return `<!DOCTYPE html>\n<html>\n<head>\n<meta charset=\"utf-8\">\n<meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n<style type=\"text/css\">${responsiveCss}</style>\n</head>\n<body style=\"margin:0;padding:0;font-family:-apple-system,BlinkMacSystemFont,'Segoe UI',Roboto,sans-serif;font-size:14px;line-height:1.5;color:#333;background:#fff;\">\n <table role=\"presentation\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"background:#fff;\">\n <tr><td class=\"email-wrap\" style=\"padding:16px;\">\n <div>\n ${bodyHtml}\n </div>\n ${footerBlock}\n ${disclaimerBlock}\n </td></tr>\n </table>\n</body>\n</html>`;\n}\n","/** Social entry: use `iconUrl` for an image, or `icon` (emoji/text) as fallback when no image. */\nexport interface SocialLinkItem {\n url: string;\n iconUrl?: string;\n icon?: string;\n}\n\nexport interface CompanyDetails {\n logoUrl?: string;\n companyName?: string;\n supportEmail?: string;\n supportPhone?: string;\n socialLinks?: SocialLinkItem[];\n /** Shown below footer rows; from email plugin settings */\n footerDisclaimer?: string;\n /** Heading above social icons (default \"Follow Us\") */\n followUsTitle?: string;\n}\n\n/** One row in order-placed transactional emails */\nexport interface OrderPlacedLineItem {\n productName: string;\n quantity: number;\n unitPrice: number | string;\n lineTotal: number | string;\n sku?: string | null;\n}\n\nexport function normalizeSocialLinkItem(o: {\n icon?: string;\n iconUrl?: string;\n icon_image?: string;\n url?: string;\n}): SocialLinkItem | null {\n const url = String(o.url ?? '').trim();\n if (!url) return null;\n let iconUrl = String(o.iconUrl ?? o.icon_image ?? '').trim();\n let icon = String(o.icon ?? '').trim();\n if (!iconUrl && /^https?:\\/\\//i.test(icon)) {\n iconUrl = icon;\n icon = '';\n }\n const item: SocialLinkItem = { url };\n if (iconUrl) item.iconUrl = iconUrl;\n if (icon) item.icon = icon;\n return item;\n}\n\nexport function parseSocialLinksJson(raw: string | undefined | null): SocialLinkItem[] | undefined {\n if (raw == null || raw.trim() === '') return undefined;\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (!Array.isArray(parsed)) return undefined;\n const out: SocialLinkItem[] = [];\n for (const item of parsed) {\n if (item && typeof item === 'object' && 'url' in item) {\n const n = normalizeSocialLinkItem(item as { icon?: string; iconUrl?: string; url?: string });\n if (n) out.push(n);\n }\n }\n return out.length ? out : undefined;\n } catch {\n return undefined;\n }\n}\n\n/** Merge branding + email plugin settings for layout (email overrides when set). */\nexport function mergeEmailLayoutCompanyDetails(\n branding: Record<string, string>,\n emailSettings: Record<string, string>\n): CompanyDetails {\n const fromBranding = getCompanyDetailsFromSettings(branding);\n const pick = (emailVal: string | undefined, fallback?: string) => {\n const t = emailVal?.trim();\n return t || fallback?.trim() || undefined;\n };\n const logoUrl = pick(emailSettings.logoUrl ?? emailSettings.emailLogoUrl, fromBranding.logoUrl);\n const companyName = pick(emailSettings.companyName ?? emailSettings.emailCompanyName, fromBranding.companyName);\n const supportEmail = pick(emailSettings.supportEmail ?? emailSettings.emailSupportEmail, fromBranding.supportEmail);\n const supportPhone = pick(emailSettings.supportPhone, undefined);\n const footerDisclaimer = pick(emailSettings.footerDisclaimer, undefined);\n const followUsTitle = pick(emailSettings.followUsTitle, 'Follow Us') || 'Follow Us';\n const socialFromEmail = parseSocialLinksJson(emailSettings.socialLinks);\n const socialLinks = socialFromEmail?.length ? socialFromEmail : fromBranding.socialLinks;\n return {\n logoUrl,\n companyName,\n supportEmail,\n supportPhone,\n socialLinks,\n footerDisclaimer,\n followUsTitle,\n };\n}\n\nexport interface EmailTemplateResult {\n subject: string;\n bodyHtml: string;\n text?: string;\n}\n\nexport type TemplateContext<T = unknown> = T & { companyDetails: CompanyDetails };\n\nexport const EMAIL_TEMPLATE_NAMES = [\n 'signup',\n 'passwordReset',\n 'passwordChange',\n 'orderPlaced',\n 'returnInitiated',\n 'shippingUpdate',\n 'invite',\n 'formSubmission',\n 'otp',\n] as const;\n\nexport type EmailTemplateName = (typeof EMAIL_TEMPLATE_NAMES)[number];\n\nexport function getCompanyDetailsFromSettings(\n settingsGroup: Record<string, string>\n): CompanyDetails {\n const logoUrl = settingsGroup.logo ?? settingsGroup.logoUrl ?? '';\n const companyName = settingsGroup.companyName ?? settingsGroup.company_name ?? '';\n const supportEmail = settingsGroup.supportEmail ?? settingsGroup.support_email ?? '';\n let socialLinks: SocialLinkItem[] = [];\n const raw = settingsGroup.socialLinks ?? settingsGroup.social_links;\n if (typeof raw === 'string') {\n try {\n const arr = JSON.parse(raw) as unknown[];\n if (Array.isArray(arr)) {\n for (const item of arr) {\n if (item && typeof item === 'object') {\n const n = normalizeSocialLinkItem(item as { icon?: string; iconUrl?: string; url?: string });\n if (n) socialLinks.push(n);\n }\n }\n }\n } catch {\n // ignore\n }\n }\n return { logoUrl: logoUrl || undefined, companyName: companyName || undefined, supportEmail: supportEmail || undefined, socialLinks: socialLinks.length ? socialLinks : undefined };\n}\n","/** Email-safe primary button (table-based). */\n\nexport function escapeHtml(s: string): string {\n return s\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;');\n}\n\n/** href for use inside double-quoted HTML attribute */\nexport function escapeAttr(s: string): string {\n return escapeHtml(s);\n}\n\nexport function primaryCtaButton(href: string, label: string): string {\n return `<table role=\"presentation\" cellpadding=\"0\" cellspacing=\"0\" style=\"margin:20px 0;\">\n<tr><td style=\"border-radius:6px;background:#1a1a1a;\">\n<a href=\"${escapeAttr(href)}\" style=\"display:inline-block;padding:12px 22px;font-size:14px;font-weight:600;color:#ffffff;text-decoration:none;\">${escapeHtml(label)}</a>\n</td></tr>\n</table>`;\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(\n ctx: TemplateContext<{ name?: string; loginUrl?: string; verifyEmailUrl?: string }>\n) {\n const { name, loginUrl, verifyEmailUrl, companyDetails } = ctx;\n\n if (verifyEmailUrl) {\n const subject = 'Confirm your email';\n const greeting =\n name && name.trim() ? `Hi ${escapeHtml(name.trim())},` : 'Hi,';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${greeting}</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Thanks for signing up. Please confirm your email address to activate your account. Until you confirm, you won’t be able to sign in.</p>\n${primaryCtaButton(verifyEmailUrl, 'Confirm email address')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">This link expires in a few days. If you didn’t create an account, you can ignore this message.<br/><span style=\"word-break:break-all;\">${escapeHtml(verifyEmailUrl)}</span></p>`;\n const text = [\n greeting,\n '',\n 'Confirm your email to activate your account:',\n verifyEmailUrl,\n '',\n 'You cannot sign in until your email is confirmed.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n }\n\n const subject = 'Welcome';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Welcome${name ? `, ${escapeHtml(name)}` : ''}.</p><p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Your account has been created.</p>${\n loginUrl ? `${primaryCtaButton(loginUrl, 'Sign in')}` : ''\n }`;\n const text = `Welcome${name ? `, ${name}` : ''}. Your account has been created.${loginUrl ? ` Sign in: ${loginUrl}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(ctx: TemplateContext<{ resetLink: string }>) {\n const { resetLink, companyDetails } = ctx;\n const subject = 'Reset your password';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Hello,</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">We received a request to reset the password for your account. If you made this request, use the button below to choose a new password.</p>\n<p style=\"margin:0 0 8px 0;font-size:14px;line-height:1.5;color:#555;\">For your security, this link will stop working after a short time. If it expires, request a new reset from the sign-in page.</p>\n${primaryCtaButton(resetLink, 'Reset password')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If you did not request a password reset, you can safely ignore this email—your password will stay the same.</p>\n<p style=\"margin:12px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If the button does not work, copy and paste this URL into your browser:<br/><span style=\"word-break:break-all;\">${escapeHtml(resetLink)}</span></p>`;\n const text = [\n 'Hello,',\n '',\n 'We received a request to reset your password. Open the link below to set a new password:',\n resetLink,\n '',\n 'This link expires after a limited time.',\n '',\n 'If you did not request this, you can ignore this email.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(ctx: TemplateContext<{ name?: string }>) {\n const { name, companyDetails } = ctx;\n const subject = 'Password changed';\n const bodyHtml = `<h2>Password changed</h2><p>Your password has been updated successfully${name ? `, ${escapeHtml(name)}` : ''}.</p>`;\n const text = `Your password has been updated successfully${name ? `, ${name}` : ''}.`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext, OrderPlacedLineItem } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml } from './inline-cta';\n\nfunction formatMoney(amount: number | string, currency?: string): string {\n const v = typeof amount === 'string' ? parseFloat(amount) : amount;\n const num = Number.isFinite(v) ? Number(v).toFixed(2) : String(amount);\n return currency ? `${num} ${currency}` : num;\n}\n\nfunction renderLineItemsHtml(items: OrderPlacedLineItem[], currency?: string): string {\n if (!items.length) {\n return '<p style=\"margin:12px 0 0 0;font-size:14px;color:#666;\">No line items.</p>';\n }\n const rows = items\n .map((it) => {\n const name = escapeHtml(it.productName);\n const sku =\n it.sku && String(it.sku).trim()\n ? `<span style=\"font-size:12px;color:#888;\"> (${escapeHtml(String(it.sku).trim())})</span>`\n : '';\n return `<tr>\n<td style=\"padding:10px 8px 10px 0;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;\">${name}${sku}</td>\n<td align=\"right\" style=\"padding:10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(String(it.quantity))}</td>\n<td align=\"right\" style=\"padding:10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(formatMoney(it.unitPrice, currency))}</td>\n<td align=\"right\" style=\"padding:10px 0 10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(formatMoney(it.lineTotal, currency))}</td>\n</tr>`;\n })\n .join('');\n return `<p style=\"margin:16px 0 8px 0;font-size:14px;font-weight:600;color:#111;\">Order items</p>\n<table role=\"presentation\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"border-collapse:collapse;\">\n<tr>\n<td style=\"padding:0 8px 8px 0;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;letter-spacing:0.02em;\">Item</td>\n<td align=\"right\" style=\"padding:0 8px 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Qty</td>\n<td align=\"right\" style=\"padding:0 8px 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Price</td>\n<td align=\"right\" style=\"padding:0 0 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Total</td>\n</tr>\n${rows}\n</table>`;\n}\n\nfunction renderLineItemsText(items: OrderPlacedLineItem[], currency?: string): string {\n if (!items.length) return '';\n const lines = items.map(\n (it) =>\n `- ${it.productName} × ${it.quantity} @ ${formatMoney(it.unitPrice, currency)} = ${formatMoney(it.lineTotal, currency)}${it.sku ? ` [${it.sku}]` : ''}`\n );\n return ['Items:', ...lines].join('\\n');\n}\n\nexport function render(\n ctx: TemplateContext<{\n orderNumber: string;\n total?: string;\n currency?: string;\n customerName?: string;\n audience?: 'customer' | 'sales';\n internalCustomerEmail?: string;\n lineItems?: OrderPlacedLineItem[];\n }>\n) {\n const {\n orderNumber,\n total,\n currency,\n customerName,\n companyDetails,\n audience = 'customer',\n internalCustomerEmail,\n lineItems = [],\n } = ctx;\n\n const itemsHtml = renderLineItemsHtml(lineItems, currency);\n const itemsText = renderLineItemsText(lineItems, currency);\n\n const totalLine =\n total != null && String(total).trim() !== ''\n ? `<p style=\"margin:12px 0 0 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order total:</strong> ${escapeHtml(String(total))}${currency ? ` ${escapeHtml(currency)}` : ''}</p>`\n : '';\n\n let subject: string;\n let bodyHtml: string;\n let text: string;\n\n if (audience === 'sales') {\n subject = `New order #${orderNumber}`;\n const who =\n customerName || internalCustomerEmail\n ? `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Customer:</strong> ${escapeHtml(customerName || '—')}${\n internalCustomerEmail\n ? ` <span style=\"color:#555;\">(${escapeHtml(internalCustomerEmail)})</span>`\n : ''\n }</p>`\n : '';\n bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">A new order has been placed and payment completed.</p>\n<p style=\"margin:0 0 8px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order number:</strong> ${escapeHtml(orderNumber)}</p>\n${who}\n${itemsHtml}\n${totalLine}`;\n text = [\n `New order #${orderNumber}`,\n customerName ? `Customer: ${customerName}` : '',\n internalCustomerEmail ? `Email: ${internalCustomerEmail}` : '',\n itemsText,\n total != null ? `Order total: ${total}${currency ? ` ${currency}` : ''}` : '',\n ]\n .filter(Boolean)\n .join('\\n\\n');\n } else {\n subject = `Order confirmed #${orderNumber}`;\n const thanksPlain =\n customerName && customerName.trim()\n ? `Thank you, ${customerName.trim()}.`\n : 'Thank you for your order.';\n const thanksHtml = `${escapeHtml(thanksPlain)} We’ve received your order and will process it shortly.`;\n bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${thanksHtml}</p>\n<p style=\"margin:0 0 8px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order number:</strong> ${escapeHtml(orderNumber)}</p>\n${itemsHtml}\n${totalLine}\n<p style=\"margin:16px 0 0 0;font-size:13px;line-height:1.5;color:#666;\">If you have questions, reply to this email or contact us using the details below.</p>`;\n text = [\n `Order confirmed #${orderNumber}`,\n `${thanksPlain} We’ve received your order and will process it shortly.`,\n '',\n itemsText,\n total != null ? `Order total: ${total}${currency ? ` ${currency}` : ''}` : '',\n '',\n 'We will process your order shortly.',\n ]\n .filter((line, i, arr) => !(line === '' && arr[i - 1] === ''))\n .join('\\n');\n }\n\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{ returnId?: string; orderNumber?: string }>\n) {\n const { returnId, orderNumber, companyDetails } = ctx;\n const subject = 'Return initiated';\n const bodyHtml = `<h2>Return initiated</h2><p>Your return request has been received.</p>${orderNumber ? `<p><strong>Order:</strong> ${escapeHtml(orderNumber)}</p>` : ''}${returnId ? `<p><strong>Return ID:</strong> ${escapeHtml(returnId)}</p>` : ''}`;\n const text = `Return initiated.${orderNumber ? ` Order: ${orderNumber}` : ''}${returnId ? ` Return ID: ${returnId}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{\n orderNumber?: string;\n status?: string;\n trackingUrl?: string;\n }>\n) {\n const { orderNumber, status, trackingUrl, companyDetails } = ctx;\n const subject = status ? `Shipping update: ${status}` : 'Shipping update';\n const bodyHtml = `<h2>Shipping update</h2>${status ? `<p><strong>Status:</strong> ${escapeHtml(status)}</p>` : ''}${orderNumber ? `<p><strong>Order:</strong> ${escapeHtml(orderNumber)}</p>` : ''}${trackingUrl ? `<p><a href=\"${escapeHtml(trackingUrl)}\">Track your order</a></p>` : ''}`;\n const text = `Shipping update.${status ? ` Status: ${status}` : ''}${orderNumber ? ` Order: ${orderNumber}` : ''}${trackingUrl ? ` Track: ${trackingUrl}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(\n ctx: TemplateContext<{ inviteLink: string; email: string; inviteeName?: string }>\n) {\n const { inviteLink, email, inviteeName, companyDetails } = ctx;\n const subject = \"You're invited\";\n const greeting =\n inviteeName && inviteeName.trim()\n ? `Hello ${escapeHtml(inviteeName.trim())},`\n : 'Hello,';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${greeting}</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">You have been invited to create your account for <strong>${escapeHtml(email)}</strong>. Use the secure link below to choose your password and activate access.</p>\n<p style=\"margin:0 0 8px 0;font-size:14px;line-height:1.5;color:#555;\">This link is personal to you. If you did not expect this invitation, you can ignore this message.</p>\n${primaryCtaButton(inviteLink, 'Accept invitation & set password')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If the button does not work, copy and paste this URL into your browser:<br/><span style=\"word-break:break-all;\">${escapeHtml(inviteLink)}</span></p>`;\n const text = [\n inviteeName?.trim() ? `Hello ${inviteeName.trim()},` : 'Hello,',\n '',\n `You have been invited to create your account (${email}).`,\n 'Open this link to set your password:',\n inviteLink,\n '',\n 'If you did not expect this invitation, you can ignore this email.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{\n formName: string;\n contactName: string;\n contactEmail: string;\n formData: unknown;\n formFieldRows?: { label: string; value: string }[];\n }>\n) {\n const { formName, contactName, contactEmail, formData, formFieldRows, companyDetails } = ctx;\n const subject = `New Form Submission: ${formName}`;\n\n const fieldsBlock =\n formFieldRows && formFieldRows.length > 0\n ? formFieldRows\n .map(\n (r) =>\n `<p style=\"margin:10px 0 0 0;line-height:1.5;\"><strong>${escapeHtml(r.label)}</strong><br/>${escapeHtml(r.value)}</p>`\n )\n .join('')\n : `<p style=\"margin:10px 0 0 0;font-size:13px;white-space:pre-wrap;\">${escapeHtml(JSON.stringify(formData, null, 2))}</p>`;\n\n const bodyHtml = `<p style=\"margin:0 0 6px 0;font-size:18px;font-weight:600;color:#111;\">New Form Submission</p>\n<p style=\"margin:0 0 4px 0;\"><strong>Form:</strong> ${escapeHtml(formName)}</p>\n<p style=\"margin:0 0 8px 0;\"><strong>Contact:</strong> ${escapeHtml(contactName)}${contactEmail ? ` (${escapeHtml(contactEmail)})` : ''}</p>\n${fieldsBlock}`;\n\n const textLines = [\n 'New Form Submission',\n `Form: ${formName}`,\n `Contact: ${contactName}${contactEmail ? ` (${contactEmail})` : ''}`,\n '',\n ...(formFieldRows?.length\n ? formFieldRows.map((r) => `${r.label}: ${r.value}`)\n : [JSON.stringify(formData, null, 2)]),\n ];\n const text = textLines.join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport type OtpTemplateCtx = TemplateContext<{\n code: string;\n}>;\n\nexport function render(ctx: OtpTemplateCtx) {\n const { code, companyDetails } = ctx;\n const name = companyDetails.companyName?.trim() || 'Our store';\n const bodyHtml = `\n <p>Your verification code is:</p>\n <p style=\"font-size:24px;font-weight:bold;letter-spacing:4px;\">${escapeHtml(code)}</p>\n <p style=\"color:#666;font-size:14px;\">This code expires in 10 minutes. If you did not request it, you can ignore this email.</p>\n `;\n const html = renderLayout({ bodyHtml, companyDetails });\n return {\n subject: `${name}: your sign-in code`,\n html,\n text: `Your code is ${code}. It expires in 10 minutes.`,\n };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","export * from './types';\nexport { renderLayout } from './layout';\nimport type { EmailTemplateName, TemplateContext } from './types';\nimport { render as renderSignup } from './signup';\nimport { render as renderPasswordReset } from './passwordReset';\nimport { render as renderPasswordChange } from './passwordChange';\nimport { render as renderOrderPlaced } from './orderPlaced';\nimport { render as renderReturnInitiated } from './returnInitiated';\nimport { render as renderShippingUpdate } from './shippingUpdate';\nimport { render as renderInvite } from './invite';\nimport { render as renderFormSubmission } from './formSubmission';\nimport { render as renderOtp } from './otp';\n\nexport { renderSignup, renderPasswordReset, renderPasswordChange, renderOrderPlaced, renderReturnInitiated, renderShippingUpdate, renderInvite, renderFormSubmission, renderOtp };\n\nexport type EmailRenderResult = { subject: string; html: string; text?: string };\n\nconst templateRenderMap: Record<\n EmailTemplateName,\n (ctx: TemplateContext<unknown>) => EmailRenderResult\n> = {\n signup: renderSignup as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n passwordReset: renderPasswordReset as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n passwordChange: renderPasswordChange as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n orderPlaced: renderOrderPlaced as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n returnInitiated: renderReturnInitiated as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n shippingUpdate: renderShippingUpdate as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n invite: renderInvite as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n formSubmission: renderFormSubmission as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n otp: renderOtp as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n};\n\nexport function getTemplateRenderer(name: EmailTemplateName) {\n return templateRenderMap[name];\n}\n","import type {\n CompanyDetails,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n} from './templates/types';\nimport { renderLayout } from './templates/layout';\nimport { getTemplateRenderer } from './templates/index';\n\nexport interface RenderEmailOptions {\n renderLayout?: (opts: { bodyHtml: string; companyDetails: CompanyDetails }) => string;\n getBody?: (templateName: string, ctx: TemplateContext<unknown>) => EmailTemplateResult | null;\n}\n\nexport interface RenderedEmail {\n subject: string;\n html: string;\n text?: string;\n}\n\nexport function renderEmail(\n templateName: EmailTemplateName,\n ctx: TemplateContext<unknown>,\n options?: RenderEmailOptions\n): RenderedEmail {\n const layoutFn = options?.renderLayout ?? renderLayout;\n const getBody = options?.getBody;\n\n if (getBody) {\n const custom = getBody(templateName, ctx);\n if (custom != null) {\n const html = layoutFn({ bodyHtml: custom.bodyHtml, companyDetails: ctx.companyDetails });\n return { subject: custom.subject, html, text: custom.text };\n }\n }\n\n const render = getTemplateRenderer(templateName);\n if (!render) {\n throw new Error(`Unknown email template: ${templateName}`);\n }\n return render(ctx);\n}\n","import type { CmsPlugin } from '../types';\nimport { EmailService } from './email-service';\nimport type { RenderEmailOptions } from './renderer';\n\nexport interface EmailPluginConfig {\n type: 'AWS' | 'SMTP' | 'GMAIL' | 'SENDGRID';\n user?: string;\n password?: string;\n from: string;\n to: string;\n host?: string;\n port?: number;\n secure?: boolean;\n region?: string;\n accessKeyId?: string;\n secretAccessKey?: string;\n templateOptions?: RenderEmailOptions;\n}\n\nexport function emailPlugin(config: EmailPluginConfig): CmsPlugin<EmailService> {\n return {\n name: 'email',\n version: '1.0.0',\n async init(context) {\n const from = config.from || context.config.SMTP_FROM || 'no-reply@example.com';\n const to = config.to || context.config.SMTP_TO || 'info@example.com';\n const type = (config.type || context.config.SMTP_TYPE) as EmailPluginConfig['type'] || 'SMTP';\n const portEnv = context.config.SMTP_PORT;\n const portParsed = portEnv ? parseInt(portEnv, 10) : undefined;\n const merged: EmailPluginConfig = {\n ...config,\n from,\n to,\n type,\n user: config.user ?? context.config.SMTP_USER,\n password: config.password ?? context.config.SMTP_PASSWORD,\n host: config.host ?? context.config.SMTP_HOST,\n port: config.port ?? (Number.isFinite(portParsed) ? portParsed : undefined),\n secure: config.secure ?? context.config.SMTP_SECURE === 'true',\n region: config.region ?? context.config.AWS_REGION,\n accessKeyId: config.accessKeyId ?? context.config.AWS_ACCESS_KEY_ID,\n secretAccessKey: config.secretAccessKey ?? context.config.AWS_SECRET_ACCESS_KEY,\n };\n if (type === 'AWS' && (!merged.region || !merged.accessKeyId || !merged.secretAccessKey)) {\n context.logger.warn('Email plugin skipped: AWS SES configuration incomplete');\n return undefined;\n }\n if ((type === 'SMTP' || type === 'GMAIL') && (!merged.user || !merged.password)) {\n context.logger.warn('Email plugin skipped: SMTP credentials not configured');\n return undefined;\n }\n return new EmailService(merged);\n },\n };\n}\n\nexport { EmailService, emailTemplates } from './email-service';\nexport type { EmailData, EmailPluginConfig as EmailConfig, EmailServiceInterface, RenderedEmail } from './email-service';\nexport { renderEmail } from './renderer';\nexport { getCompanyDetailsFromSettings, mergeEmailLayoutCompanyDetails } from './templates/types';\nexport { renderLayout } from './templates/layout';\nexport { registerEmailQueueProcessor, queueEmail, queueOrderPlacedEmails } from './email-queue';\nexport type { EmailJobPayload, OrderPlacedEmailPayload } from './email-queue';\nexport type {\n CompanyDetails,\n SocialLinkItem,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n OrderPlacedLineItem,\n} from './templates/types';\nexport type { RenderEmailOptions } from './renderer';\n","import { google } from 'googleapis';\n\nexport interface AnalyticsData {\n visitors: number;\n pageViews: number;\n bounceRate: number;\n avgSessionDuration: number;\n topPages: Array<{ page: string; views: number }>;\n trafficSources: Array<{ source: string; sessions: number }>;\n geographicData: Array<{ country: string; sessions: number }>;\n dailyUsers: Array<{ date: string; users: number }>;\n}\n\nexport interface AnalyticsPluginConfig {\n privateKey: string;\n clientEmail: string;\n viewId: string;\n}\n\nexport class AnalyticsService {\n private analytics: ReturnType<typeof google.analyticsdata>;\n private viewId: string;\n\n constructor(config: AnalyticsPluginConfig) {\n const privateKey = config.privateKey.replace(/\\\\n/g, '\\n');\n const auth = new google.auth.JWT({\n email: config.clientEmail,\n key: privateKey,\n scopes: ['https://www.googleapis.com/auth/analytics.readonly'],\n });\n this.analytics = google.analyticsdata({ version: 'v1beta', auth });\n this.viewId = config.viewId;\n }\n\n async getAnalyticsData(days = 30): Promise<AnalyticsData> {\n const endDate = new Date();\n const startDate = new Date();\n startDate.setDate(startDate.getDate() - days);\n\n const [visitors, pageViews, bounceRate, sessionDuration, topPages, trafficSources, geographicData, dailyUsers] =\n await Promise.all([\n this.getVisitors(startDate, endDate),\n this.getPageViews(startDate, endDate),\n this.getBounceRate(startDate, endDate),\n this.getAvgSessionDuration(startDate, endDate),\n this.getTopPages(startDate, endDate),\n this.getTrafficSources(startDate, endDate),\n this.getGeographicData(startDate, endDate),\n this.getDailyUsers(startDate, endDate),\n ]);\n\n return {\n visitors,\n pageViews,\n bounceRate,\n avgSessionDuration: sessionDuration,\n topPages,\n trafficSources,\n geographicData,\n dailyUsers,\n };\n }\n\n private async runReport(requestBody: Record<string, unknown>) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const response = await (this.analytics.properties.runReport as any)({\n property: `properties/${this.viewId}`,\n requestBody,\n });\n return response?.data as { rows?: Array<{ dimensionValues: { value: string }[]; metricValues: { value: string }[] }> };\n }\n\n private async getVisitors(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'totalUsers' }],\n });\n return parseInt(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getPageViews(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'screenPageViews' }],\n });\n return parseInt(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getBounceRate(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'bounceRate' }],\n });\n return parseFloat(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getAvgSessionDuration(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'averageSessionDuration' }],\n });\n return parseFloat(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getTopPages(startDate: Date, endDate: Date): Promise<Array<{ page: string; views: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'pagePath' }],\n metrics: [{ name: 'screenPageViews' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ page: row.dimensionValues[0].value, views: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getTrafficSources(\n startDate: Date,\n endDate: Date\n ): Promise<Array<{ source: string; sessions: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'sessionSource' }],\n metrics: [{ name: 'sessions' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ source: row.dimensionValues[0].value, sessions: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getGeographicData(\n startDate: Date,\n endDate: Date\n ): Promise<Array<{ country: string; sessions: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'country' }],\n metrics: [{ name: 'sessions' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ country: row.dimensionValues[0].value, sessions: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getDailyUsers(startDate: Date, endDate: Date): Promise<Array<{ date: string; users: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'date' }],\n metrics: [{ name: 'totalUsers' }],\n orderBys: [{ dimension: { dimensionName: 'date' } }],\n });\n return (\n data.rows?.map((row) => {\n const dateStr = row.dimensionValues[0].value;\n return { date: `${dateStr.substring(4, 6)}/${dateStr.substring(6, 8)}`, users: parseInt(row.metricValues[0].value) };\n }) || []\n );\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { AnalyticsService } from './analytics-service';\n\nexport interface AnalyticsPluginConfig {\n privateKey?: string;\n clientEmail?: string;\n viewId?: string;\n}\n\nexport function analyticsPlugin(config: AnalyticsPluginConfig = {}): CmsPlugin<AnalyticsService> {\n return {\n name: 'analytics',\n version: '1.0.0',\n async init(context) {\n const privateKey = config.privateKey ?? context.config.GOOGLE_ANALYTICS_PRIVATE_KEY;\n const clientEmail = config.clientEmail ?? context.config.GOOGLE_ANALYTICS_CLIENT_EMAIL;\n const viewId = config.viewId ?? context.config.GOOGLE_ANALYTICS_VIEW_ID ?? '';\n if (!privateKey || !clientEmail || !viewId) {\n throw new Error('Google Analytics credentials not configured');\n }\n return new AnalyticsService({ privateKey, clientEmail, viewId });\n },\n };\n}\n\nexport { AnalyticsService } from './analytics-service';\nexport type { AnalyticsData } from './analytics-service';\n","/**\n * India (TRAI / DLT): promotional and service SMS must use templates registered on operator DLT\n * (e.g. Jio, Airtel, BSNL/VIL portals) and linked to your Principal Entity ID and Header (sender).\n * MSG91 maps your DLT-approved template to a Flow / template_id used by their API. Arbitrary free-text\n * SMS APIs are often rejected (e.g. “template id missing”). Prefer Flow with `template_id` from the\n * `message_templates` row (`externalTemplateRef`) or optional env `MSG91_TEMPLATE_ID`, and variables that\n * match your DLT template placeholders (commonly var1…).\n *\n * References: TRAI TCCCPR-2018; MSG91 DLT / Flow docs; operator DLT portals for template approval.\n */\n\nexport interface SmsSendOptions {\n to: string;\n /** Plain body (Twilio/webhook/sendhttp). With `templateKey`, filled after resolving the template. */\n body?: string;\n /** Resolve copy + Flow ref from DB `message_templates` (channel sms) with code defaults. */\n templateKey?: string;\n /** For MSG91 Flow: value injected into the OTP variable (see template `providerMeta.otpVarKey`). */\n otpCode?: string;\n /** Template variables (e.g. `code`) and extra Flow recipient fields. */\n variables?: Record<string, string>;\n}\n\nexport type MessageTemplateRowLoader = (\n channel: string,\n templateKey: string\n) => Promise<{\n body: string;\n externalTemplateRef: string | null;\n providerMeta: Record<string, unknown> | null;\n enabled: boolean;\n} | null>;\n\n/** Which transport to use for outbound SMS. */\nexport type SmsProviderId = 'msg91' | 'twilio' | 'webhook';\n\n/** Stored in admin `smsProvider` or env `SMS_PROVIDER`. `auto` picks first with credentials: msg91 → twilio → webhook. */\nexport type SmsProviderChoice = SmsProviderId | 'auto';\n\n/** MSG91: use DLT Flow API vs legacy sendhttp (non‑India / legacy only). */\nexport type Msg91ApiMode = 'flow' | 'sendhttp' | 'auto';\n\nexport interface SmsServiceConfig {\n provider?: SmsProviderChoice;\n /** Admin settings key `smsProvider` (msg91 | twilio | webhook | auto). */\n smsProvider?: string;\n /** MSG91: auth key */\n msg91AuthKey?: string;\n /** DLT-approved sender ID (required for sendhttp; Flow uses template/header mapped in MSG91 panel). */\n msg91SenderId?: string;\n /** Transactional route; default 4 (sendhttp only). */\n msg91Route?: string;\n /**\n * MSG91 Flow ID / template_id shown in panel after mapping DLT template (India).\n * Required for `flow` mode.\n */\n msg91TemplateId?: string;\n /** `flow` = POST control.msg91.com/api/v5/flow/ (DLT). `sendhttp` = legacy GET API. `auto` = flow if template id set, else sendhttp. */\n msg91ApiMode?: Msg91ApiMode;\n /** Recipient JSON key for OTP value in Flow API (must match MSG91 template variables). Default var1. */\n msg91OtpVarKey?: string;\n /** Flow API short_url flag; default 0. */\n msg91FlowShortUrl?: string;\n accountSid?: string;\n authToken?: string;\n from?: string;\n webhookUrl?: string;\n webhookSecret?: string;\n}\n\nfunction norm(v: string | undefined): string | undefined {\n const t = v?.trim();\n return t || undefined;\n}\n\nfunction pick(\n plugin: Partial<SmsServiceConfig>,\n db: Record<string, string>,\n env: Record<string, string>,\n ...keys: string[]\n): string | undefined {\n for (const key of keys) {\n const pk = key as keyof SmsServiceConfig;\n if (pk in plugin && plugin[pk] !== undefined && String(plugin[pk]).trim() !== '') {\n const v = norm(String(plugin[pk]));\n if (v) return v;\n }\n if (norm(db[key])) return norm(db[key]);\n if (norm(env[key])) return norm(env[key]);\n }\n return undefined;\n}\n\nfunction parseMsg91ApiMode(raw: string | undefined): Msg91ApiMode {\n const x = (raw || 'auto').toLowerCase().trim();\n if (x === 'flow' || x === 'sendhttp' || x === 'auto') return x;\n return 'auto';\n}\n\nexport function mergeSmsConfigLayers(\n env: Record<string, string>,\n db: Record<string, string>,\n plugin: Partial<SmsServiceConfig>\n): SmsServiceConfig {\n const pRaw = pick(plugin, db, env, 'smsProvider', 'SMS_PROVIDER', 'provider')?.toLowerCase();\n const provider: SmsProviderChoice | undefined =\n pRaw === 'msg91' || pRaw === 'twilio' || pRaw === 'webhook' || pRaw === 'auto' ? (pRaw as SmsProviderChoice) : undefined;\n\n return {\n provider: provider ?? 'auto',\n msg91AuthKey: pick(plugin, db, env, 'msg91AuthKey', 'MSG91_AUTH_KEY'),\n msg91SenderId: pick(plugin, db, env, 'msg91SenderId', 'MSG91_SENDER_ID'),\n msg91Route: pick(plugin, db, env, 'msg91Route', 'MSG91_ROUTE') ?? '4',\n msg91TemplateId: pick(plugin, db, env, 'msg91TemplateId', 'MSG91_TEMPLATE_ID'),\n msg91ApiMode: parseMsg91ApiMode(pick(plugin, db, env, 'msg91ApiMode', 'MSG91_API_MODE')),\n msg91OtpVarKey: pick(plugin, db, env, 'msg91OtpVarKey', 'MSG91_OTP_VAR_KEY') ?? 'var1',\n msg91FlowShortUrl: pick(plugin, db, env, 'msg91FlowShortUrl', 'MSG91_FLOW_SHORT_URL') ?? '0',\n accountSid: pick(plugin, db, env, 'accountSid', 'TWILIO_ACCOUNT_SID'),\n authToken: pick(plugin, db, env, 'authToken', 'TWILIO_AUTH_TOKEN'),\n from: pick(plugin, db, env, 'from', 'TWILIO_FROM_NUMBER', 'TWILIO_MESSAGING_SERVICE_SID'),\n webhookUrl: pick(plugin, db, env, 'webhookUrl', 'SMS_WEBHOOK_URL'),\n webhookSecret: pick(plugin, db, env, 'webhookSecret', 'SMS_WEBHOOK_SECRET'),\n };\n}\n\n/** True when Flow can run: auth key and a template id (per-send or global/env). */\nexport function msg91FlowConfigured(cfg: SmsServiceConfig, flowTemplateIdForSend?: string): boolean {\n const tid = (flowTemplateIdForSend || cfg.msg91TemplateId || '').trim();\n return Boolean(cfg.msg91AuthKey && tid);\n}\n\nexport function msg91SendhttpConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.msg91AuthKey && cfg.msg91SenderId);\n}\n\n/** MSG91 is usable with auth key (Flow id may come per message from `message_templates`). */\nexport function msg91Configured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.msg91AuthKey);\n}\n\nexport function resolveMsg91SendMode(cfg: SmsServiceConfig, flowTemplateIdForSend?: string): 'flow' | 'sendhttp' | null {\n const tid = (flowTemplateIdForSend || cfg.msg91TemplateId || '').trim();\n const mode = cfg.msg91ApiMode ?? 'auto';\n if (mode === 'flow') {\n if (!cfg.msg91AuthKey || !tid) return null;\n return 'flow';\n }\n if (mode === 'sendhttp') return msg91SendhttpConfigured(cfg) ? 'sendhttp' : null;\n if (cfg.msg91AuthKey && tid) return 'flow';\n if (msg91SendhttpConfigured(cfg)) return 'sendhttp';\n return null;\n}\n\nexport function twilioConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.accountSid && cfg.authToken && cfg.from);\n}\n\nexport function webhookConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.webhookUrl);\n}\n\nexport function resolveSmsProvider(cfg: SmsServiceConfig): SmsProviderId | null {\n const want = cfg.provider ?? 'auto';\n if (want === 'msg91' && msg91Configured(cfg)) return 'msg91';\n if (want === 'twilio' && twilioConfigured(cfg)) return 'twilio';\n if (want === 'webhook' && webhookConfigured(cfg)) return 'webhook';\n if (want !== 'auto') return null;\n\n if (msg91Configured(cfg)) return 'msg91';\n if (twilioConfigured(cfg)) return 'twilio';\n if (webhookConfigured(cfg)) return 'webhook';\n return null;\n}\n\nexport function anySmsProviderConfigured(cfg: SmsServiceConfig): boolean {\n return resolveSmsProvider(cfg) !== null;\n}\n\nconst MSG91_FLOW_URL = 'https://control.msg91.com/api/v5/flow/';\n\nasync function sendViaMsg91Flow(\n cfg: SmsServiceConfig,\n to: string,\n body: string,\n opts: Pick<SmsSendOptions, 'otpCode' | 'variables'>,\n overrides?: { templateId?: string; otpVarKey?: string }\n): Promise<boolean> {\n const authkey = cfg.msg91AuthKey!;\n const templateId = (overrides?.templateId || cfg.msg91TemplateId || '').trim();\n if (!templateId) return false;\n const mobile = to.replace(/^\\+/, '').replace(/\\s/g, '');\n if (!mobile) return false;\n\n const varKey = (overrides?.otpVarKey || cfg.msg91OtpVarKey || 'var1').trim() || 'var1';\n const recipient: Record<string, string> = { mobiles: mobile };\n const vars: Record<string, string> = { ...(opts.variables ?? {}) };\n if (opts.otpCode) vars[varKey] = opts.otpCode;\n else if (!(varKey in vars)) {\n const m = body.match(/\\d{4,8}/);\n if (m) vars[varKey] = m[0];\n }\n if (Object.keys(vars).length === 0) return false;\n\n for (const [k, v] of Object.entries(vars)) {\n recipient[k] = v;\n }\n\n const payload = {\n template_id: templateId,\n short_url: cfg.msg91FlowShortUrl ?? '0',\n recipients: [recipient],\n };\n\n try {\n const res = await fetch(MSG91_FLOW_URL, {\n method: 'POST',\n headers: {\n accept: 'application/json',\n 'content-type': 'application/json',\n authkey,\n },\n body: JSON.stringify(payload),\n });\n const text = await res.text();\n if (!res.ok) return false;\n try {\n const j = JSON.parse(text) as { type?: string; message?: string };\n if (j.type === 'error' || (typeof j.message === 'string' && /error|fail/i.test(j.message))) return false;\n } catch {\n const lower = text.toLowerCase();\n if (lower.includes('error') || lower.includes('invalid') || lower.includes('fail')) return false;\n }\n return true;\n } catch {\n return false;\n }\n}\n\nasync function sendViaMsg91Sendhttp(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const authkey = cfg.msg91AuthKey;\n const sender = cfg.msg91SenderId;\n if (!authkey || !sender) return false;\n const route = cfg.msg91Route?.trim() || '4';\n const mobile = to.replace(/^\\+/, '').replace(/\\s/g, '');\n if (!mobile) return false;\n const params = new URLSearchParams({\n authkey,\n mobiles: mobile,\n message: body,\n sender,\n route,\n country: '0',\n });\n const url = `https://api.msg91.com/api/sendhttp.php?${params.toString()}`;\n try {\n const res = await fetch(url, { method: 'GET' });\n const text = (await res.text()).trim();\n if (!res.ok) return false;\n const lower = text.toLowerCase();\n if (lower.includes('error') || lower.includes('invalid') || lower.includes('failed')) return false;\n return true;\n } catch {\n return false;\n }\n}\n\nasync function sendViaMsg91(\n cfg: SmsServiceConfig,\n to: string,\n body: string,\n opts: SmsSendOptions,\n flowOverrides?: { templateId?: string; otpVarKey?: string }\n): Promise<boolean> {\n const flowTid = (flowOverrides?.templateId || cfg.msg91TemplateId || '').trim();\n const mode = resolveMsg91SendMode(cfg, flowTid);\n if (mode === 'flow') {\n return sendViaMsg91Flow(cfg, to, body, { otpCode: opts.otpCode, variables: opts.variables }, flowOverrides);\n }\n if (mode === 'sendhttp') {\n return sendViaMsg91Sendhttp(cfg, to, body);\n }\n return false;\n}\n\nasync function sendViaTwilio(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const { accountSid, authToken, from } = cfg;\n if (!accountSid || !authToken || !from) return false;\n const url = `https://api.twilio.com/2010-04-01/Accounts/${accountSid}/Messages.json`;\n const auth = Buffer.from(`${accountSid}:${authToken}`).toString('base64');\n const params = new URLSearchParams({ To: to, From: from, Body: body });\n const res = await fetch(url, {\n method: 'POST',\n headers: { Authorization: `Basic ${auth}`, 'Content-Type': 'application/x-www-form-urlencoded' },\n body: params.toString(),\n });\n return res.ok;\n}\n\nasync function sendViaWebhook(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const { webhookUrl, webhookSecret } = cfg;\n if (!webhookUrl) return false;\n const headers: Record<string, string> = { 'Content-Type': 'application/json' };\n if (webhookSecret) headers.Authorization = `Bearer ${webhookSecret}`;\n const res = await fetch(webhookUrl, {\n method: 'POST',\n headers,\n body: JSON.stringify({ to, message: body }),\n });\n return res.ok;\n}\n\nexport type SmsSettingsLoader = () => Promise<Record<string, string>>;\n\nexport class SmsService {\n private env: Record<string, string>;\n private plugin: Partial<SmsServiceConfig>;\n private getSmsSettings?: SmsSettingsLoader;\n private getMessageTemplateRow?: MessageTemplateRowLoader;\n\n constructor(\n env: Record<string, string>,\n plugin: Partial<SmsServiceConfig> = {},\n getSmsSettings?: SmsSettingsLoader,\n getMessageTemplateRow?: MessageTemplateRowLoader\n ) {\n this.env = env;\n this.plugin = plugin;\n this.getSmsSettings = getSmsSettings;\n this.getMessageTemplateRow = getMessageTemplateRow;\n }\n\n private async merged(): Promise<SmsServiceConfig> {\n const db = (await this.getSmsSettings?.()) ?? {};\n return mergeSmsConfigLayers(this.env, db, this.plugin);\n }\n\n async send(opts: SmsSendOptions): Promise<boolean> {\n try {\n const dbEarly = (await this.getSmsSettings?.()) ?? {};\n if (dbEarly.enabled === 'false') return false;\n let effective: SmsSendOptions = { ...opts };\n let flowOverrides: { templateId?: string; otpVarKey?: string } | undefined;\n\n if (opts.templateKey?.trim()) {\n if (!this.getMessageTemplateRow) return false;\n const { resolveSmsTemplateForSend } = await import('../../message-templates/sms-defaults');\n const { applyTemplateVars } = await import('../../message-templates/interpolate');\n const resolved = await resolveSmsTemplateForSend(opts.templateKey.trim(), this.getMessageTemplateRow);\n if (!resolved) return false;\n const vars = { ...(opts.variables ?? {}) };\n const code = vars.code ?? vars.otp ?? opts.otpCode ?? '';\n if (code) {\n vars.code = code;\n delete vars.otp;\n }\n effective.body = applyTemplateVars(resolved.body, vars);\n effective.otpCode = code || effective.otpCode;\n effective.variables = vars;\n flowOverrides = {\n templateId: resolved.externalTemplateRef,\n otpVarKey: resolved.otpVarKey,\n };\n }\n\n const body = (effective.body ?? '').trim();\n if (!body) return false;\n\n const cfg = await this.merged();\n const p = resolveSmsProvider(cfg);\n if (!p) return false;\n if (p === 'msg91') return sendViaMsg91(cfg, opts.to, body, effective, flowOverrides);\n if (p === 'twilio') return sendViaTwilio(cfg, opts.to, body);\n return sendViaWebhook(cfg, opts.to, body);\n } catch {\n return false;\n }\n }\n}\n","const SMS_QUEUE_NAME = 'sms';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\nexport interface SmsJobPayload {\n to: string;\n /** Legacy / Twilio / sendhttp plain text. */\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n}\n\nexport function registerSmsQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as\n | { add: (name: string, data: object) => Promise<void>; registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void }\n | undefined;\n const sms = cms.getPlugin('sms') as\n | {\n send: (opts: {\n to: string;\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n }) => Promise<boolean>;\n }\n | undefined;\n if (!queue || !sms || typeof sms.send !== 'function') return;\n queue.registerProcessor(SMS_QUEUE_NAME, async (data: object) => {\n const payload = data as SmsJobPayload;\n if (!payload.to) return;\n if (payload.templateKey?.trim()) {\n await sms.send({\n to: payload.to,\n templateKey: payload.templateKey.trim(),\n variables: payload.variables,\n otpCode: payload.otpCode,\n });\n return;\n }\n if (!payload.body?.trim()) return;\n await sms.send({\n to: payload.to,\n body: payload.body,\n otpCode: payload.otpCode,\n variables: payload.variables,\n });\n });\n}\n\nexport async function queueSms(cms: CmsAppLike, payload: SmsJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n const sms = cms.getPlugin('sms') as\n | {\n send: (opts: {\n to: string;\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n }) => Promise<boolean>;\n }\n | undefined;\n if (queue) {\n await queue.add(SMS_QUEUE_NAME, payload);\n return;\n }\n if (sms && typeof sms.send === 'function') {\n if (payload.templateKey?.trim()) {\n await sms.send({\n to: payload.to,\n templateKey: payload.templateKey.trim(),\n variables: payload.variables,\n otpCode: payload.otpCode,\n });\n return;\n }\n if (payload.body?.trim()) {\n await sms.send({\n to: payload.to,\n body: payload.body,\n otpCode: payload.otpCode,\n variables: payload.variables,\n });\n }\n }\n}\n","import type { CmsPlugin, PluginContext } from '../types';\nimport {\n SmsService,\n type SmsServiceConfig,\n type MessageTemplateRowLoader,\n mergeSmsConfigLayers,\n anySmsProviderConfigured,\n} from './sms-service';\n\nexport interface SmsPluginConfig extends Partial<SmsServiceConfig> {\n /** Load admin settings group `sms` (enabled, smsProvider). Secrets typically stay in env. */\n getSmsSettings?: () => Promise<Record<string, string>>;\n /** Resolve `message_templates` rows for SMS (required for `templateKey` sends). */\n getMessageTemplateRow?: MessageTemplateRowLoader;\n}\n\nexport interface SmsServiceInterface {\n send(opts: {\n to: string;\n body?: string;\n templateKey?: string;\n otpCode?: string;\n variables?: Record<string, string>;\n }): Promise<boolean>;\n}\n\nexport function smsPlugin(config: SmsPluginConfig = {}): CmsPlugin<SmsServiceInterface | null> {\n return {\n name: 'sms',\n version: '1.0.0',\n async init(context: PluginContext) {\n const env = context.config as Record<string, string>;\n const { getSmsSettings, getMessageTemplateRow, ...staticRest } = config;\n let db: Record<string, string> = {};\n try {\n db = (await getSmsSettings?.()) ?? {};\n } catch {\n /* ignore */\n }\n const merged = mergeSmsConfigLayers(env, db, staticRest);\n if (!anySmsProviderConfigured(merged)) {\n context.logger.warn(\n 'SMS plugin skipped: set MSG91_AUTH_KEY (Flow template id per Plugins → SMS templates or MSG91_TEMPLATE_ID; or sendhttp: MSG91_SENDER_ID), or TWILIO_*, or SMS_WEBHOOK_URL'\n );\n return null;\n }\n const svc = new SmsService(env, staticRest, getSmsSettings, getMessageTemplateRow);\n return {\n send: (opts) => svc.send(opts),\n };\n },\n };\n}\n\nexport {\n mergeSmsConfigLayers,\n resolveSmsProvider,\n anySmsProviderConfigured,\n msg91Configured,\n msg91FlowConfigured,\n msg91SendhttpConfigured,\n resolveMsg91SendMode,\n twilioConfigured,\n webhookConfigured,\n} from './sms-service';\nexport type {\n SmsProviderId,\n SmsProviderChoice,\n SmsServiceConfig,\n SmsSendOptions,\n SmsSettingsLoader,\n Msg91ApiMode,\n MessageTemplateRowLoader,\n} from './sms-service';\n\nexport { registerSmsQueueProcessor, queueSms } from './sms-queue';\nexport type { SmsJobPayload } from './sms-queue';\n","import type { CmsPlugin, PluginContext } from '../types';\n\nexport interface PaymentIntent {\n id: string;\n clientSecret: string;\n amount: number;\n currency: string;\n status: string;\n}\n\nexport interface PaymentServiceInterface {\n createPaymentIntent(amount: number, currency: string, metadata?: Record<string, string>): Promise<PaymentIntent>;\n capturePayment(paymentId: string, amount?: number): Promise<boolean>;\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean;\n}\n\nexport interface StripeConfig {\n provider: 'stripe';\n secretKey: string;\n webhookSecret?: string;\n}\n\nexport interface RazorpayConfig {\n provider: 'razorpay';\n keyId: string;\n keySecret: string;\n webhookSecret?: string;\n}\n\nexport type PaymentPluginConfig = StripeConfig | RazorpayConfig;\n\nexport function paymentPlugin(config: PaymentPluginConfig): CmsPlugin<PaymentServiceInterface | undefined> {\n return {\n name: 'payment',\n version: '1.0.0',\n async init(context: PluginContext) {\n if (config.provider === 'stripe') {\n const { StripePaymentService } = await import('./stripe');\n const secretKey = config.secretKey || context.config.STRIPE_SECRET_KEY;\n if (!secretKey) {\n context.logger.warn('Payment plugin skipped: Stripe secret key not configured');\n return undefined;\n }\n return new StripePaymentService({\n secretKey,\n webhookSecret: config.webhookSecret || context.config.STRIPE_WEBHOOK_SECRET,\n });\n }\n\n if (config.provider === 'razorpay') {\n const { RazorpayPaymentService } = await import('./razorpay');\n const keyId = config.keyId || context.config.RAZORPAY_KEY_ID;\n const keySecret = config.keySecret || context.config.RAZORPAY_KEY_SECRET;\n if (!keyId || !keySecret) {\n context.logger.warn('Payment plugin skipped: Razorpay credentials not configured');\n return undefined;\n }\n return new RazorpayPaymentService({\n keyId,\n keySecret,\n webhookSecret: config.webhookSecret || context.config.RAZORPAY_WEBHOOK_SECRET,\n });\n }\n\n context.logger.warn(`Payment plugin skipped: unknown provider \"${(config as { provider: string }).provider}\"`);\n return undefined;\n },\n };\n}\n","import type { CmsPlugin } from '../types';\nimport type { StorageService } from './types';\n\nexport interface S3StoragePluginConfig {\n region: string;\n accessKeyId: string;\n secretAccessKey: string;\n bucket: string;\n baseUrl?: string;\n prefix?: string;\n}\n\nfunction getS3Storage(config: S3StoragePluginConfig): StorageService {\n const prefix = (config.prefix ?? 'uploads').replace(/\\/$/, '');\n const keyPrefix = prefix ? `${prefix}/` : '';\n\n return {\n async upload(buffer: Buffer, key: string, contentType: string): Promise<string> {\n const { S3Client, PutObjectCommand } = await import('@aws-sdk/client-s3');\n const client = new S3Client({\n region: config.region,\n credentials: {\n accessKeyId: config.accessKeyId,\n secretAccessKey: config.secretAccessKey,\n },\n });\n const fullKey = keyPrefix + key.replace(/^\\/+/, '');\n await client.send(\n new PutObjectCommand({\n Bucket: config.bucket,\n Key: fullKey,\n Body: buffer,\n ContentType: contentType,\n })\n );\n if (config.baseUrl) {\n return `${config.baseUrl.replace(/\\/$/, '')}/${fullKey}`;\n }\n return `https://s3.${config.region}.amazonaws.com/${config.bucket}/${fullKey}`;\n },\n };\n}\n\nexport function s3StoragePlugin(\n config: Partial<S3StoragePluginConfig> & { bucket: string }\n): CmsPlugin<StorageService> {\n return {\n name: 'storage',\n version: '1.0.0',\n async init(context) {\n const region = config.region ?? context.config.AWS_REGION ?? '';\n const accessKeyId = config.accessKeyId ?? context.config.AWS_ACCESS_KEY_ID ?? '';\n const secretAccessKey = config.secretAccessKey ?? context.config.AWS_SECRET_ACCESS_KEY ?? '';\n const bucket = config.bucket ?? context.config.AWS_BUCKET_NAME ?? '';\n const baseUrl = config.baseUrl ?? context.config.S3_MEDIA_URL ?? context.config.S3_CUSTOM_DOMAIN;\n const resolvedBaseUrl =\n typeof baseUrl === 'string' && baseUrl\n ? baseUrl.startsWith('http')\n ? baseUrl\n : `https://${baseUrl}`\n : undefined;\n if (!region || !accessKeyId || !secretAccessKey || !bucket) {\n throw new Error('S3 storage plugin: missing region, accessKeyId, secretAccessKey or bucket');\n }\n return getS3Storage({\n region,\n accessKeyId,\n secretAccessKey,\n bucket,\n baseUrl: resolvedBaseUrl,\n prefix: config.prefix ?? 'uploads',\n });\n },\n };\n}\n","import type { CmsPlugin } from '../types';\nimport type { StorageService } from './types';\n\nexport interface LocalStoragePluginConfig {\n /** Directory relative to process.cwd() (e.g. \"public/uploads\"). */\n dir?: string;\n /** Public URL prefix for returned URLs (e.g. \"\" for \"/uploads/...\"). */\n publicPath?: string;\n}\n\nfunction getLocalStorage(config: LocalStoragePluginConfig): StorageService {\n const dir = (config.dir ?? 'public/uploads').replace(/\\/$/, '');\n const publicPath =\n config.publicPath ??\n (dir === 'public/uploads' ? '/uploads' : `/${dir.replace(/^\\/+/, '').replace(/\\\\/g, '/')}`);\n\n return {\n async upload(buffer: Buffer, key: string, _contentType: string): Promise<string> {\n const fs = await import('fs/promises');\n const path = await import('path');\n const normalizedKey = key.replace(/^\\/+/, '').replace(/\\.\\./g, '');\n const fileName = normalizedKey.replace(/^uploads\\//, '');\n const fullDir = path.join(process.cwd(), dir);\n await fs.mkdir(fullDir, { recursive: true });\n const filePath = path.join(fullDir, fileName);\n const fileDir = path.dirname(filePath);\n await fs.mkdir(fileDir, { recursive: true });\n await fs.writeFile(filePath, buffer);\n return `${publicPath.replace(/\\/$/, '')}/${fileName}`;\n },\n };\n}\n\nexport function localStoragePlugin(config: LocalStoragePluginConfig = {}): CmsPlugin<StorageService> {\n return {\n name: 'storage',\n version: '1.0.0',\n async init(context) {\n const dir = config.dir ?? context.config.UPLOAD_DIR ?? 'public/uploads';\n return getLocalStorage({ ...config, dir });\n },\n };\n}\n","export interface LlmMessage {\n role: 'user' | 'assistant' | 'system';\n content: string;\n}\n\nexport interface LlmChatOptions {\n model?: string;\n temperature?: number;\n max_tokens?: number;\n}\n\nexport interface LlmServiceInterface {\n chat(messages: LlmMessage[], options?: LlmChatOptions): Promise<{ content: string }>;\n streamChat(messages: LlmMessage[], options?: LlmChatOptions): AsyncIterable<string>;\n /** OpenAI-compatible embeddings; returns [] if gateway does not support /v1/embeddings */\n embed?(text: string, options?: { model?: string }): Promise<number[]>;\n}\n\nexport class LlmService implements LlmServiceInterface {\n private static embedWarned = false;\n\n constructor(\n private readonly baseURL: string,\n private readonly apiKey: string,\n private readonly defaultEmbeddingModel?: string\n ) {}\n\n private get base(): string {\n return this.baseURL.replace(/\\/$/, '');\n }\n\n private get url(): string {\n return `${this.base}/v1/chat/completions`;\n }\n\n /**\n * OpenAI-compatible embeddings. Returns [] if endpoint not available or unexpected response.\n * Server must implement: POST /v1/embeddings\n * Body: { model: string, input: string }\n * Response: { data: [ { embedding: number[] } ] } or { embedding: number[] }\n */\n async embed(text: string, options: { model?: string } = {}): Promise<number[]> {\n const model = options.model ?? this.defaultEmbeddingModel ?? 'text-embedding-3-small';\n const input = text.slice(0, 8000);\n const res = await fetch(`${this.base}/v1/embeddings`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model,\n input,\n }),\n });\n if (!res.ok) {\n const body = await res.text();\n if (!LlmService.embedWarned) {\n LlmService.embedWarned = true;\n console.warn(`[LLM embed] ${res.status} ${res.statusText}: ${body.slice(0, 400)}`);\n }\n return [];\n }\n let data: Record<string, unknown>;\n try {\n data = (await res.json()) as Record<string, unknown>;\n } catch {\n console.warn('[LLM embed] Response is not JSON');\n return [];\n }\n // OpenAI shape: { data: [ { embedding: number[] } ] }\n let embedding = (data?.data as Array<{ embedding?: number[] }>)?.[0]?.embedding;\n if (!Array.isArray(embedding) && Array.isArray(data?.embedding)) {\n embedding = data.embedding as number[];\n }\n if (!Array.isArray(embedding) && Array.isArray((data?.data as unknown[])?.[0])) {\n const first = (data.data as unknown[])[0];\n embedding = Array.isArray(first) ? first : (first as { embedding?: number[] })?.embedding;\n }\n if (!Array.isArray(embedding) && !LlmService.embedWarned) {\n LlmService.embedWarned = true;\n console.warn('[LLM embed] Unexpected response shape. Keys:', Object.keys(data), 'data[0] type:', Array.isArray(data?.data) ? typeof (data.data as unknown[])[0] : 'n/a');\n }\n return Array.isArray(embedding) ? embedding : [];\n }\n\n async chat(messages: LlmMessage[], options: LlmChatOptions = {}): Promise<{ content: string }> {\n const res = await fetch(this.url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model: options.model ?? 'qwen2.5:3b-instruct',\n messages,\n stream: false,\n temperature: options.temperature,\n max_tokens: options.max_tokens,\n }),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`LLM gateway error: ${res.status} ${text}`);\n }\n const data = (await res.json()) as { choices?: Array<{ message?: { content?: string } }> };\n const content = data.choices?.[0]?.message?.content ?? '';\n return { content };\n }\n\n async *streamChat(messages: LlmMessage[], options: LlmChatOptions = {}): AsyncIterable<string> {\n const res = await fetch(this.url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model: options.model ?? 'qwen2.5:3b-instruct',\n messages,\n stream: true,\n temperature: options.temperature,\n max_tokens: options.max_tokens,\n }),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`LLM gateway error: ${res.status} ${text}`);\n }\n const reader = res.body?.getReader();\n if (!reader) return;\n const decoder = new TextDecoder();\n let buffer = '';\n while (true) {\n const { value, done } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() ?? '';\n for (const line of lines) {\n if (line.startsWith('data: ') && line !== 'data: [DONE]') {\n try {\n const json = JSON.parse(line.slice(6)) as { choices?: Array<{ delta?: { content?: string } }> };\n const content = json.choices?.[0]?.delta?.content;\n if (content) yield content;\n } catch {\n // skip malformed chunk\n }\n }\n }\n }\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { LlmService } from './llm-service';\n\nexport interface LlmPluginConfig {\n baseURL?: string;\n apiKey?: string;\n /** Default model for embed(); also from env EMBEDDING_MODEL (e.g. for Qwen use your server's embedding model name). */\n embeddingModel?: string;\n}\n\nexport function llmPlugin(config: LlmPluginConfig = {}): CmsPlugin<LlmService | undefined> {\n return {\n name: 'llm',\n version: '1.0.0',\n async init(context) {\n const baseURL = config.baseURL ?? context.config.LLM_GATEWAY_URL;\n const apiKey = config.apiKey ?? context.config.LLM_API_KEY;\n if (!baseURL || !apiKey) {\n context.logger.warn('LLM plugin skipped: LLM_GATEWAY_URL or LLM_API_KEY not set');\n return undefined;\n }\n const embeddingModel = config.embeddingModel ?? context.config.EMBEDDING_MODEL;\n return new LlmService(baseURL.replace(/\\/$/, ''), apiKey, embeddingModel);\n },\n };\n}\n\nexport { LlmService } from './llm-service';\nexport type { LlmServiceInterface, LlmMessage, LlmChatOptions } from './llm-service';\n","import Redis from 'ioredis';\nimport type { CmsPlugin } from '../types';\nimport type { CacheService } from './types';\nimport { MemoryCache } from './memory-cache';\nimport { RedisCache } from './redis-cache';\n\nexport type { CacheService } from './types';\nexport { MemoryCache } from './memory-cache';\nexport { RedisCache } from './redis-cache';\n\nexport interface CachePluginConfig {\n redisUrl?: string;\n}\n\nexport function cachePlugin(config?: CachePluginConfig): CmsPlugin<CacheService> {\n return {\n name: 'cache',\n version: '1.0.0',\n async init(context) {\n const redisUrl = config?.redisUrl ?? context.config.REDIS_URL ?? '';\n const url = typeof redisUrl === 'string' ? redisUrl.trim() : '';\n if (url) {\n const client = new Redis(url);\n return new RedisCache(client);\n }\n return new MemoryCache();\n },\n };\n}\n","import { LRUCache } from 'lru-cache';\nimport type { CacheService } from './types';\n\nconst DEFAULT_MAX = 500;\nconst DEFAULT_TTL_MS = 60 * 1000;\n\nexport class MemoryCache implements CacheService {\n private readonly cache: LRUCache<string, string>;\n\n constructor(options?: { max?: number; ttlMs?: number }) {\n const max = options?.max ?? DEFAULT_MAX;\n const ttlMs = options?.ttlMs ?? DEFAULT_TTL_MS;\n this.cache = new LRUCache<string, string>({ max, ttl: ttlMs });\n }\n\n async get(key: string): Promise<string | null> {\n const value = this.cache.get(key);\n return value ?? null;\n }\n\n async set(key: string, value: string, ttlSeconds?: number): Promise<void> {\n const ttlMs = ttlSeconds != null && ttlSeconds > 0 ? ttlSeconds * 1000 : undefined;\n this.cache.set(key, value, { ttl: ttlMs });\n }\n\n async del(key: string): Promise<void> {\n this.cache.delete(key);\n }\n}\n","import type Redis from 'ioredis';\nimport type { CacheService } from './types';\n\nexport class RedisCache implements CacheService {\n constructor(private readonly client: Redis) {}\n\n async get(key: string): Promise<string | null> {\n return this.client.get(key);\n }\n\n async set(key: string, value: string, ttlSeconds?: number): Promise<void> {\n if (ttlSeconds != null && ttlSeconds > 0) {\n await this.client.set(key, value, 'EX', ttlSeconds);\n } else {\n await this.client.set(key, value);\n }\n }\n\n async del(key: string): Promise<void> {\n await this.client.del(key);\n }\n}\n","import { Queue, Worker } from 'bullmq';\nimport type { Job } from 'bullmq';\nimport type { QueueService } from './types';\n\nfunction connectionFromUrl(redisUrl: string): { host: string; port: number; username?: string; password?: string; maxRetriesPerRequest: number | null } {\n const u = new URL(redisUrl);\n return {\n host: u.hostname,\n port: u.port ? parseInt(u.port, 10) : 6379,\n ...(u.username && { username: u.username }),\n ...(u.password && { password: u.password }),\n maxRetriesPerRequest: null,\n };\n}\n\nexport class BullMQQueue implements QueueService {\n private readonly connectionOptions: ReturnType<typeof connectionFromUrl>;\n private readonly queues = new Map<string, Queue>();\n private readonly workers = new Map<string, Worker>();\n\n constructor(redisUrl: string) {\n this.connectionOptions = connectionFromUrl(redisUrl);\n }\n\n async add(queueName: string, data: object): Promise<void> {\n const queue = this.getOrCreateQueue(queueName);\n await queue.add(queueName, data);\n }\n\n registerProcessor(queueName: string, processor: (data: object) => Promise<void>): void {\n if (this.workers.has(queueName)) return;\n const worker = new Worker(\n queueName,\n async (job: Job) => {\n await processor(job.data as object);\n },\n { connection: this.connectionOptions }\n );\n this.workers.set(queueName, worker);\n }\n\n private getOrCreateQueue(queueName: string): Queue {\n let queue = this.queues.get(queueName);\n if (!queue) {\n queue = new Queue(queueName, { connection: this.connectionOptions });\n this.queues.set(queueName, queue);\n }\n return queue;\n }\n\n async destroy(): Promise<void> {\n for (const w of this.workers.values()) await w.close();\n this.workers.clear();\n for (const q of this.queues.values()) await q.close();\n this.queues.clear();\n }\n}\n","import fastq from 'fastq';\nimport type { QueueService } from './types';\n\ntype Task = object;\n\nexport class MemoryQueue implements QueueService {\n private readonly queues = new Map<string, fastq.queueAsPromised<Task>>();\n\n async add(queueName: string, data: object): Promise<void> {\n const q = this.queues.get(queueName);\n if (!q) throw new Error(`No processor registered for queue: ${queueName}`);\n // fastq.promise push resolves when the task finishes — do not await or HTTP handlers block on SMTP etc.\n void Promise.resolve(q.push(data as Task)).catch(() => undefined);\n }\n\n registerProcessor(queueName: string, processor: (data: object) => Promise<void>): void {\n if (this.queues.has(queueName)) return;\n const worker = async (task: Task) => processor(task);\n const q = fastq.promise(worker, 1);\n this.queues.set(queueName, q);\n }\n}\n","import type { CmsPlugin } from '../types';\nimport type { QueueService } from './types';\nimport { BullMQQueue } from './bullmq-queue';\nimport { MemoryQueue } from './memory-queue';\n\nexport type { QueueService } from './types';\nexport { BullMQQueue } from './bullmq-queue';\nexport { MemoryQueue } from './memory-queue';\n\nexport interface QueuePluginConfig {\n redisUrl?: string;\n}\n\nexport function queuePlugin(config?: QueuePluginConfig): CmsPlugin<QueueService> {\n let instance: BullMQQueue | MemoryQueue | undefined;\n return {\n name: 'queue',\n version: '1.0.0',\n async init(context) {\n const redisUrl = config?.redisUrl ?? context.config.REDIS_URL ?? '';\n const url = typeof redisUrl === 'string' ? redisUrl.trim() : '';\n if (url) {\n instance = new BullMQQueue(url);\n return instance;\n }\n instance = new MemoryQueue();\n return instance;\n },\n async destroy() {\n if (instance && 'destroy' in instance && typeof instance.destroy === 'function') {\n await instance.destroy();\n }\n },\n };\n}\n","export type CaptchaProviderId = 'turnstile' | 'recaptcha_v3';\n\nexport interface CaptchaPublicConfig {\n enabled: boolean;\n activeProvider: CaptchaProviderId | null;\n siteKey: string | null;\n availableProviders: Array<{ id: CaptchaProviderId; siteKey: string }>;\n multipleProviders: boolean;\n}\n\nexport type CaptchaVerifyResult =\n | { ok: true }\n | { ok: false; status: number; message: string };\n\ninterface ResolvedKeys {\n turnstile: { site: string; secret: string } | null;\n recaptcha: { site: string; secret: string } | null;\n /** When both providers configured, set only if CAPTCHA_PROVIDER env is set */\n envDefaultProvider: CaptchaProviderId | null;\n minScore: number;\n}\n\nfunction resolveKeys(config: Record<string, string>): ResolvedKeys {\n const turnstile =\n config.TURNSTILE_SITE_KEY?.trim() && config.TURNSTILE_SECRET_KEY?.trim()\n ? { site: config.TURNSTILE_SITE_KEY.trim(), secret: config.TURNSTILE_SECRET_KEY.trim() }\n : null;\n const recaptcha =\n config.RECAPTCHA_SITE_KEY?.trim() && config.RECAPTCHA_SECRET_KEY?.trim()\n ? { site: config.RECAPTCHA_SITE_KEY.trim(), secret: config.RECAPTCHA_SECRET_KEY.trim() }\n : null;\n const raw = config.CAPTCHA_PROVIDER?.trim().toLowerCase();\n let envDefaultProvider: CaptchaProviderId | null = null;\n if (raw === 'turnstile') envDefaultProvider = 'turnstile';\n else if (raw === 'recaptcha_v3' || raw === 'recaptcha') envDefaultProvider = 'recaptcha_v3';\n\n const minScore = parseFloat(config.RECAPTCHA_MIN_SCORE ?? '0.5');\n return {\n turnstile,\n recaptcha,\n envDefaultProvider,\n minScore: Number.isFinite(minScore) ? minScore : 0.5,\n };\n}\n\nexport function buildCaptchaPublicConfig(config: Record<string, string>): CaptchaPublicConfig {\n const r = resolveKeys(config);\n const available: Array<{ id: CaptchaProviderId; siteKey: string }> = [];\n if (r.turnstile) available.push({ id: 'turnstile', siteKey: r.turnstile.site });\n if (r.recaptcha) available.push({ id: 'recaptcha_v3', siteKey: r.recaptcha.site });\n const enabled = available.length > 0;\n const multipleProviders = available.length > 1;\n let activeProvider: CaptchaProviderId | null = null;\n if (r.turnstile && !r.recaptcha) activeProvider = 'turnstile';\n else if (r.recaptcha && !r.turnstile) activeProvider = 'recaptcha_v3';\n else if (multipleProviders) {\n activeProvider = r.envDefaultProvider ?? available[0]!.id;\n }\n let siteKey: string | null = null;\n if (activeProvider === 'turnstile' && r.turnstile) siteKey = r.turnstile.site;\n else if (activeProvider === 'recaptcha_v3' && r.recaptcha) siteKey = r.recaptcha.site;\n if (!siteKey && available.length === 1) siteKey = available[0]!.siteKey;\n return {\n enabled,\n activeProvider,\n siteKey,\n availableProviders: available,\n multipleProviders,\n };\n}\n\nfunction clientIp(req: Request): string | undefined {\n const cf = req.headers.get('cf-connecting-ip')?.trim();\n if (cf) return cf;\n const xff = req.headers.get('x-forwarded-for')?.trim();\n if (xff) return xff.split(',')[0]!.trim();\n const real = req.headers.get('x-real-ip')?.trim();\n if (real) return real;\n return undefined;\n}\n\nfunction normalizeBodyProvider(v: unknown): CaptchaProviderId | null {\n if (v === 'turnstile') return 'turnstile';\n if (v === 'recaptcha_v3' || v === 'recaptcha') return 'recaptcha_v3';\n return null;\n}\n\nfunction pickVerifyProvider(\n body: Record<string, unknown>,\n r: ResolvedKeys\n): { provider: CaptchaProviderId; secret: string } | { error: string } {\n const bodyP = normalizeBodyProvider(body.captchaProvider);\n const hasT = !!r.turnstile;\n const hasR = !!r.recaptcha;\n\n if (hasT && !hasR) {\n if (bodyP && bodyP !== 'turnstile') return { error: 'Invalid captchaProvider' };\n return { provider: 'turnstile', secret: r.turnstile!.secret };\n }\n if (hasR && !hasT) {\n if (bodyP && bodyP !== 'recaptcha_v3') return { error: 'Invalid captchaProvider' };\n return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n }\n if (!hasT && !hasR) return { error: 'Captcha not configured' };\n\n if (bodyP === 'turnstile' && hasT) return { provider: 'turnstile', secret: r.turnstile!.secret };\n if (bodyP === 'recaptcha_v3' && hasR) return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n if (r.envDefaultProvider === 'turnstile' && hasT) return { provider: 'turnstile', secret: r.turnstile!.secret };\n if (r.envDefaultProvider === 'recaptcha_v3' && hasR) return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n return { error: 'captchaProvider required (turnstile or recaptcha_v3)' };\n}\n\nasync function verifyTurnstile(secret: string, token: string, remoteip?: string): Promise<boolean> {\n const body = new URLSearchParams({ secret, response: token });\n if (remoteip) body.set('remoteip', remoteip);\n const res = await fetch('https://challenges.cloudflare.com/turnstile/v0/siteverify', {\n method: 'POST',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body: body.toString(),\n });\n if (!res.ok) return false;\n const data = (await res.json()) as { success?: boolean };\n return data.success === true;\n}\n\nasync function verifyRecaptcha(\n secret: string,\n token: string,\n remoteip: string | undefined,\n minScore: number\n): Promise<{ ok: boolean; score?: number }> {\n const body = new URLSearchParams({ secret, response: token });\n if (remoteip) body.set('remoteip', remoteip);\n const res = await fetch('https://www.google.com/recaptcha/api/siteverify', {\n method: 'POST',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body: body.toString(),\n });\n if (!res.ok) return { ok: false };\n const data = (await res.json()) as { success?: boolean; score?: number };\n if (data.success !== true) return { ok: false };\n const score = typeof data.score === 'number' ? data.score : 0;\n return { ok: score >= minScore, score };\n}\n\nexport class CaptchaService {\n constructor(private readonly env: Record<string, string>) {}\n\n isEnforced(): boolean {\n return buildCaptchaPublicConfig(this.env).enabled;\n }\n\n getPublicConfig(): CaptchaPublicConfig {\n return buildCaptchaPublicConfig(this.env);\n }\n\n async verify(body: Record<string, unknown>, req: Request): Promise<CaptchaVerifyResult> {\n if (!this.isEnforced()) return { ok: true };\n const token = typeof body.captchaToken === 'string' ? body.captchaToken.trim() : '';\n if (!token) return { ok: false, status: 400, message: 'captchaToken required' };\n\n const r = resolveKeys(this.env);\n const picked = pickVerifyProvider(body, r);\n if ('error' in picked) return { ok: false, status: 400, message: picked.error };\n\n const ip = clientIp(req);\n if (picked.provider === 'turnstile') {\n const ok = await verifyTurnstile(picked.secret, token, ip);\n if (!ok) return { ok: false, status: 400, message: 'Captcha verification failed' };\n return { ok: true };\n }\n const vr = await verifyRecaptcha(picked.secret, token, ip, r.minScore);\n if (!vr.ok) return { ok: false, status: 400, message: 'Captcha verification failed or score too low' };\n return { ok: true };\n }\n}\n","import type { CmsPlugin, PluginContext } from '../types';\nimport { CaptchaService } from './captcha-service';\n\nexport function captchaPlugin(): CmsPlugin<CaptchaService | void> {\n return {\n name: 'captcha',\n version: '1.0.0',\n async init(ctx: PluginContext) {\n const s = new CaptchaService(ctx.config);\n return s.isEnforced() ? s : undefined;\n },\n };\n}\n","import type { CaptchaService } from './captcha-service';\n\ntype JsonResponse = (body: unknown, init?: { status?: number; headers?: HeadersInit }) => Response;\n\nexport async function assertCaptchaOk(\n getCms: (() => Promise<{ getPlugin: (name: string) => unknown }>) | undefined,\n body: Record<string, unknown>,\n req: Request,\n json: JsonResponse\n): Promise<Response | null> {\n if (!getCms) return null;\n let cms: { getPlugin: (name: string) => unknown };\n try {\n cms = await getCms();\n } catch {\n return null;\n }\n const svc = cms.getPlugin('captcha') as CaptchaService | undefined;\n if (!svc || typeof (svc as CaptchaService).verify !== 'function') return null;\n const result = await svc.verify(body, req);\n if (result.ok) return null;\n return json({ error: result.message }, { status: result.status });\n}\n","import { type ClassValue, clsx } from 'clsx';\nimport { twMerge } from 'tailwind-merge';\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n\nexport function generateSlug(title: string): string {\n return title\n .toLowerCase()\n .replace(/[^a-z0-9]+/g, '-')\n .replace(/(^-|-$)/g, '');\n}\n\nexport function validateSlug(slug: string): boolean {\n const slugRegex = /^[a-z0-9]+(?:-[a-z0-9]+)*$/;\n return slugRegex.test(slug) && slug.length >= 3 && slug.length <= 50;\n}\n\nexport function formatDate(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });\n}\n\nexport function formatDateTime(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', {\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n });\n}\n\nexport function formatDateOnly(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });\n}\n\nexport function truncateText(text: string, maxLength: number): string {\n if (text.length <= maxLength) return text;\n return text.substring(0, maxLength).trim() + '...';\n}\n","import type { DataSource } from 'typeorm';\nimport type { Metadata } from 'next';\nimport type { EntityMap } from '../api/crud';\n\n/** Minimal SEO row shape (DB or API). */\nexport type SeoLike = {\n title?: string | null;\n description?: string | null;\n keywords?: string | null;\n ogTitle?: string | null;\n ogDescription?: string | null;\n ogImage?: string | null;\n};\n\nexport type SeoMetadataOverrides = {\n title?: string;\n description?: string;\n keywords?: string;\n ogTitle?: string;\n ogDescription?: string;\n ogImage?: string;\n};\n\nfunction pickStr(\n override: string | undefined,\n seoPrimary: string | null | undefined,\n seoSecondary: string | null | undefined,\n fallback: string | undefined,\n): string | undefined {\n const o = override?.trim();\n if (o) return o;\n const a = seoPrimary?.trim();\n if (a) return a;\n const b = seoSecondary?.trim();\n if (b) return b;\n return fallback?.trim() || undefined;\n}\n\nfunction pickSeo(row: unknown): SeoLike | null {\n if (!row || typeof row !== 'object') return null;\n const r = row as Record<string, unknown>;\n return {\n title: (r.title as string) ?? null,\n description: (r.description as string) ?? null,\n keywords: (r.keywords as string) ?? null,\n ogTitle: (r.ogTitle as string) ?? null,\n ogDescription: (r.ogDescription as string) ?? null,\n ogImage: (r.ogImage as string) ?? null,\n };\n}\n\n/** Per-field: join row wins over slug row. */\nexport function mergeSeoBySlug(join: SeoLike | null | undefined, bySlug: SeoLike | null): SeoLike | null {\n const a = join ? pickSeo(join) : null;\n const b = bySlug ? pickSeo(bySlug) : null;\n if (!a && !b) return null;\n return {\n title: a?.title ?? b?.title ?? null,\n description: a?.description ?? b?.description ?? null,\n keywords: a?.keywords ?? b?.keywords ?? null,\n ogTitle: a?.ogTitle ?? b?.ogTitle ?? null,\n ogDescription: a?.ogDescription ?? b?.ogDescription ?? null,\n ogImage: a?.ogImage ?? b?.ogImage ?? null,\n };\n}\n\nexport async function fetchSeoBySlug(\n dataSource: DataSource,\n entityMap: EntityMap,\n slug: string,\n): Promise<SeoLike | null> {\n const trimmed = slug?.trim();\n if (!trimmed) return null;\n const SeoEntity = entityMap.seos;\n if (!SeoEntity) return null;\n const repo = dataSource.getRepository(SeoEntity);\n const row = await repo.findOne({\n where: { slug: trimmed, deleted: false } as import('typeorm').ObjectLiteral,\n });\n return row ? pickSeo(row) : null;\n}\n\nfunction toAbsoluteImage(url: string, metadataBase: URL | undefined): string {\n const u = url.trim();\n if (!u) return u;\n if (u.startsWith('http://') || u.startsWith('https://')) return u;\n if (metadataBase) {\n try {\n const path = u.startsWith('/') ? u : `/${u}`;\n return new URL(path, metadataBase).toString();\n } catch {\n return u;\n }\n }\n return u;\n}\n\n/**\n * Build Next.js Metadata from merged SEO + optional layers.\n * `overrides` win over SEO; `fallbacks` apply when SEO (and overrides) omit a field.\n * Returns a partial object; omit fields so the root layout defaults apply.\n */\nexport async function resolvePublicMetadata(args: {\n dataSource: DataSource;\n entityMap: EntityMap;\n slug: string;\n seoFromJoin?: SeoLike | null;\n overrides?: SeoMetadataOverrides;\n fallbacks?: SeoMetadataOverrides;\n canonicalPath?: string;\n metadataBase?: URL;\n}): Promise<Metadata> {\n const { dataSource, entityMap, slug, seoFromJoin, overrides, fallbacks, canonicalPath, metadataBase } = args;\n const bySlug = await fetchSeoBySlug(dataSource, entityMap, slug);\n const seo = mergeSeoBySlug(seoFromJoin, bySlug);\n const o = overrides ?? {};\n const f = fallbacks ?? {};\n\n const title = pickStr(o.title, seo?.title, seo?.ogTitle, f.title);\n\n const description = pickStr(o.description, seo?.description, seo?.ogDescription, f.description);\n\n const keywordsRaw =\n o.keywords?.trim() || seo?.keywords?.trim() || f.keywords?.trim() || undefined;\n const keywords = keywordsRaw\n ? keywordsRaw.split(',').map((k) => k.trim()).filter(Boolean)\n : undefined;\n\n const ogTitle =\n o.ogTitle?.trim() ||\n seo?.ogTitle?.trim() ||\n seo?.title?.trim() ||\n o.title?.trim() ||\n f.ogTitle?.trim() ||\n title;\n\n const ogDescription =\n o.ogDescription?.trim() ||\n seo?.ogDescription?.trim() ||\n seo?.description?.trim() ||\n o.description?.trim() ||\n f.ogDescription?.trim() ||\n description;\n\n const ogImageRaw =\n o.ogImage?.trim() || seo?.ogImage?.trim() || f.ogImage?.trim() || undefined;\n\n const out: Metadata = {};\n\n if (title) out.title = title;\n if (description) out.description = description;\n if (keywords?.length) out.keywords = keywords;\n\n if (ogTitle || ogDescription || ogImageRaw) {\n const images = ogImageRaw\n ? [{ url: toAbsoluteImage(ogImageRaw, metadataBase), alt: ogTitle || title || '' }]\n : undefined;\n out.openGraph = {\n ...(ogTitle && { title: ogTitle }),\n ...(ogDescription && { description: ogDescription }),\n ...(images && { images }),\n };\n out.twitter = {\n card: 'summary_large_image',\n ...(ogTitle && { title: ogTitle }),\n ...(ogDescription && { description: ogDescription }),\n ...(images?.[0] && { images: [images[0].url] }),\n };\n }\n\n if (canonicalPath?.trim() && metadataBase) {\n try {\n const path = canonicalPath.startsWith('/') ? canonicalPath : `/${canonicalPath}`;\n out.alternates = { canonical: new URL(path, metadataBase).toString() };\n } catch {\n /* ignore */\n }\n }\n\n return out;\n}\n","import type { DataSource, EntityTarget, ObjectLiteral } from 'typeorm';\nimport { IsNull } from 'typeorm';\n\n/** Links an unclaimed contact (same email, no userId) to the new user after signup or invite. */\nexport async function linkUnclaimedContactToUser(\n dataSource: DataSource,\n contactsEntity: EntityTarget<ObjectLiteral>,\n userId: number,\n email: string\n): Promise<void> {\n const repo = dataSource.getRepository(contactsEntity);\n const found = await repo.findOne({\n where: { email, userId: IsNull(), deleted: false } as ObjectLiteral,\n });\n if (found) await repo.update((found as { id: number }).id, { userId });\n}\n","/** Parse stored config value into a list of emails (JSON array, or comma/semicolon-separated string). */\nexport function parseEmailRecipientsFromConfig(raw: string | undefined | null): string[] {\n if (raw == null || raw === '') return [];\n const trimmed = raw.trim();\n if (trimmed.startsWith('[')) {\n try {\n const parsed = JSON.parse(trimmed) as unknown;\n if (Array.isArray(parsed)) {\n return parsed.map((e) => String(e).trim()).filter(Boolean);\n }\n } catch {\n // fall through\n }\n }\n return trimmed\n .split(/[,;]+/)\n .map((s) => s.trim())\n .filter(Boolean);\n}\n\n/** Serialize email list for config storage (JSON array in DB). */\nexport function serializeEmailRecipients(emails: string[]): string {\n return JSON.stringify(emails);\n}\n\n/** Join for SMTP `to` header (multiple recipients). */\nexport function joinRecipientsForSend(emails: string[]): string | null {\n if (!emails.length) return null;\n return emails.join(', ');\n}\n","import { createHmac, randomInt, timingSafeEqual } from 'crypto';\nimport type { DataSource } from 'typeorm';\nimport type { EntityTarget, ObjectLiteral } from 'typeorm';\nimport { IsNull, MoreThan } from 'typeorm';\n\nexport type OtpPurpose = 'login' | 'verify_email' | 'verify_phone';\nexport type OtpChannel = 'email' | 'sms';\n\nconst OTP_TTL_MS = 10 * 60 * 1000;\nconst MAX_SENDS_PER_HOUR = 5;\nconst MAX_VERIFY_ATTEMPTS = 8;\n\nfunction getPepper(explicit?: string): string {\n return (explicit || process.env.OTP_PEPPER || process.env.NEXTAUTH_SECRET || 'dev-otp-pepper').trim();\n}\n\nexport function hashOtpCode(code: string, purpose: string, identifier: string, pepper?: string): string {\n return createHmac('sha256', getPepper(pepper)).update(`${purpose}|${identifier}|${code}`).digest('hex');\n}\n\nexport function verifyOtpCodeHash(code: string, storedHash: string, purpose: string, identifier: string, pepper?: string): boolean {\n const h = hashOtpCode(code, purpose, identifier, pepper);\n try {\n return timingSafeEqual(Buffer.from(h, 'utf8'), Buffer.from(storedHash, 'utf8'));\n } catch {\n return false;\n }\n}\n\nexport function generateNumericOtp(length = 6): string {\n const max = 10 ** length;\n return randomInt(0, max).toString().padStart(length, '0');\n}\n\n/** Normalize to E.164-like +digits */\nexport function normalizePhoneE164(raw: string, defaultCountryCode?: string): string | null {\n const t = raw.trim();\n const digitsOnly = t.replace(/\\D/g, '');\n if (digitsOnly.length < 10) return null;\n if (t.startsWith('+')) return `+${digitsOnly}`;\n const cc = (defaultCountryCode || process.env.DEFAULT_PHONE_COUNTRY_CODE || '91').replace(/\\D/g, '');\n if (digitsOnly.length > 10) return `+${digitsOnly}`;\n return `+${cc}${digitsOnly}`;\n}\n\ntype EntityMap = Record<string, EntityTarget<ObjectLiteral>>;\n\nexport async function countRecentOtpSends(\n dataSource: DataSource,\n entityMap: EntityMap,\n purpose: OtpPurpose,\n identifier: string,\n since: Date\n): Promise<number> {\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n return repo.count({\n where: { purpose, identifier, createdAt: MoreThan(since) } as ObjectLiteral,\n });\n}\n\nexport async function createOtpChallenge(\n dataSource: DataSource,\n entityMap: EntityMap,\n input: { purpose: OtpPurpose; channel: OtpChannel; identifier: string; code: string; pepper?: string }\n): Promise<{ ok: true } | { ok: false; error: string; status: number }> {\n const { purpose, channel, identifier, code, pepper } = input;\n const since = new Date(Date.now() - 60 * 60 * 1000);\n const recent = await countRecentOtpSends(dataSource, entityMap, purpose, identifier, since);\n if (recent >= MAX_SENDS_PER_HOUR) {\n return { ok: false, error: 'Too many codes sent. Try again later.', status: 429 };\n }\n\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n await repo.delete({\n purpose,\n identifier,\n consumedAt: IsNull(),\n } as ObjectLiteral);\n\n const expiresAt = new Date(Date.now() + OTP_TTL_MS);\n const codeHash = hashOtpCode(code, purpose, identifier, pepper);\n await repo.save(\n repo.create({\n purpose,\n channel,\n identifier,\n codeHash,\n expiresAt,\n attempts: 0,\n consumedAt: null,\n } as ObjectLiteral)\n );\n return { ok: true };\n}\n\nexport async function verifyAndConsumeOtpChallenge(\n dataSource: DataSource,\n entityMap: EntityMap,\n input: { purpose: OtpPurpose; identifier: string; code: string; pepper?: string }\n): Promise<{ ok: true } | { ok: false; error: string; status: number }> {\n const { purpose, identifier, code, pepper } = input;\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n const row = await repo.findOne({\n where: { purpose, identifier, consumedAt: IsNull() } as ObjectLiteral,\n order: { id: 'DESC' },\n });\n if (!row) {\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n const r = row as ObjectLiteral;\n if (new Date(r.expiresAt as Date) < new Date()) {\n await repo.delete((row as { id: number }).id);\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n const attempts = (r.attempts as number) || 0;\n if (attempts >= MAX_VERIFY_ATTEMPTS) {\n await repo.delete((row as { id: number }).id);\n return { ok: false, error: 'Too many attempts', status: 400 };\n }\n\n const valid = verifyOtpCodeHash(code, r.codeHash as string, purpose, identifier, pepper);\n if (!valid) {\n await repo.update((row as { id: number }).id, { attempts: attempts + 1 } as ObjectLiteral);\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n\n await repo.update((row as { id: number }).id, { consumedAt: new Date(), attempts: attempts + 1 } as ObjectLiteral);\n return { ok: true };\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { UserGroup } from './user-group.entity';\n\n@Entity('users')\nexport class User {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n email: string;\n\n /** E.164 when set; unique among non-null values (partial index in DB). */\n @Column('varchar', { nullable: true })\n phone: string | null;\n\n @Column({ type: 'timestamp', nullable: true })\n phoneVerifiedAt: Date | null;\n\n @Column({ type: 'timestamp', nullable: true })\n emailVerifiedAt: Date | null;\n\n @Column('varchar', { nullable: true })\n password: string | null;\n\n @Column('boolean', { default: false })\n blocked: boolean;\n\n @Column('boolean', { default: false })\n adminAccess: boolean;\n\n @Column('int', { nullable: true })\n groupId: number | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => UserGroup, (g) => g.users, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'groupId' })\n group: UserGroup | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { Permission } from './permission.entity';\nimport { User } from './user.entity';\n\n@Entity('user_groups')\nexport class UserGroup {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => Permission, (p) => p.group)\n permissions: Permission[];\n\n @OneToMany(() => User, (u) => u.group)\n users: User[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { UserGroup } from './user-group.entity';\n\n@Entity('permissions')\nexport class Permission {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n groupId: number;\n\n @Column('varchar')\n entity: string;\n\n @Column('boolean', { default: false })\n canCreate: boolean;\n\n @Column('boolean', { default: false })\n canRead: boolean;\n\n @Column('boolean', { default: false })\n canUpdate: boolean;\n\n @Column('boolean', { default: false })\n canDelete: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => UserGroup, (g) => g.permissions, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'groupId' })\n group: UserGroup;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';\n\n@Entity('otp_challenges')\n@Index(['purpose', 'identifier'])\nexport class OtpChallenge {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n purpose: string;\n\n @Column('varchar')\n channel: string;\n\n @Column('varchar')\n identifier: string;\n\n @Column('varchar')\n codeHash: string;\n\n @Column({ type: 'timestamp' })\n expiresAt: Date;\n\n @Column('int', { default: 0 })\n attempts: number;\n\n @Column({ type: 'timestamp', nullable: true })\n consumedAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('password_reset_tokens')\nexport class PasswordResetToken {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n email: string;\n\n @Column('varchar', { unique: true })\n token: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n expiresAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n}\n","import {\n Entity,\n PrimaryGeneratedColumn,\n Column,\n ManyToOne,\n OneToMany,\n ManyToMany,\n JoinTable,\n JoinColumn,\n} from 'typeorm';\nimport { User } from './user.entity';\nimport { Category } from './category.entity';\nimport { Seo } from './seo.entity';\nimport { Comment } from './comment.entity';\nimport { Tag } from './tag.entity';\n\n@Entity('blogs')\nexport class Blog {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n title: string;\n\n @Column('text')\n content: string;\n\n @Column('varchar', { nullable: true })\n coverImage: string | null;\n\n @Column('int')\n authorId: number;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @ManyToOne(() => User, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'authorId' })\n author: User;\n\n @ManyToOne(() => Category, (c) => c.blogs, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: Category | null;\n\n @ManyToOne(() => Seo, (s) => s.blogs, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @OneToMany(() => Comment, (c) => c.blog)\n comments: Comment[];\n\n @ManyToMany(() => Tag, (t) => t.blogs)\n @JoinTable({\n name: 'blog_tags',\n joinColumn: { name: 'blogId', referencedColumnName: 'id' },\n inverseJoinColumn: { name: 'tagId', referencedColumnName: 'id' },\n })\n tags: Tag[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport type { Blog } from './blog.entity';\n\n@Entity('categories')\nexport class Category {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany('Blog', 'category')\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { Blog } from './blog.entity';\n\n@Entity('seos')\nexport class Seo {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n title: string | null;\n\n @Column('varchar', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n keywords: string | null;\n\n @Column('varchar', { nullable: true })\n ogTitle: string | null;\n\n @Column('varchar', { nullable: true })\n ogDescription: string | null;\n\n @Column('varchar', { nullable: true })\n ogImage: string | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => Blog, (blog) => blog.seo)\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { User } from './user.entity';\nimport { Blog } from './blog.entity';\n\n@Entity('comments')\nexport class Comment {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('text')\n content: string;\n\n @Column('int')\n blogId: number;\n\n @Column('int')\n authorId: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => User, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'authorId' })\n author: User;\n\n @ManyToOne(() => Blog, (b) => b.comments, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'blogId' })\n blog: Blog;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToMany } from 'typeorm';\nimport { Blog } from './blog.entity';\n\n@Entity('tags')\nexport class Tag {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToMany(() => Blog, (blog) => blog.tags)\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne, JoinColumn } from 'typeorm';\nimport { User } from './user.entity';\nimport { FormSubmission } from './form-submission.entity';\nimport { Address } from './address.entity';\nimport { Order } from './order.entity';\nimport { Payment } from './payment.entity';\nimport { ChatConversation } from './chat-conversation.entity';\n\n@Entity('contacts')\nexport class Contact {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n email: string;\n\n @Column('varchar', { nullable: true })\n phone: string | null;\n\n @Column('varchar', { nullable: true })\n type: string | null;\n\n @Column('varchar', { nullable: true })\n company: string | null;\n\n @Column('varchar', { nullable: true })\n taxId: string | null;\n\n @Column('text', { nullable: true })\n notes: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n userId: number | null;\n\n @ManyToOne(() => User, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'userId' })\n user: User | null;\n\n @OneToMany(() => FormSubmission, (fs) => fs.contact)\n form_submissions: FormSubmission[];\n\n @OneToMany(() => Address, (a) => a.contact)\n addresses: Address[];\n\n @OneToMany(() => Order, (o) => o.contact)\n orders: Order[];\n\n @OneToMany(() => Payment, (p) => p.contact)\n payments: Payment[];\n\n @OneToMany(() => ChatConversation, (c) => c.contact)\n chatConversations: ChatConversation[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Form } from './form.entity';\nimport { Contact } from './contact.entity';\n\n@Entity('form_submissions')\nexport class FormSubmission {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n formId: number;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('jsonb')\n data: Record<string, unknown>;\n\n @Column('varchar', { nullable: true })\n ipAddress: string | null;\n\n @Column('varchar', { nullable: true })\n userAgent: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Form, (f) => f.submissions, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'formId' })\n form: Form;\n\n @ManyToOne(() => Contact, (c) => c.form_submissions, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { FormField } from './form-field.entity';\nimport { FormSubmission } from './form-submission.entity';\n\n@Entity('forms')\nexport class Form {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n campaign: string | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => FormField, (f) => f.form)\n fields: FormField[];\n\n @OneToMany(() => FormSubmission, (s) => s.form)\n submissions: FormSubmission[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Form } from './form.entity';\n\n@Entity('form_fields')\nexport class FormField {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n formId: number;\n\n @Column('varchar')\n label: string;\n\n @Column('varchar')\n type: string;\n\n @Column('varchar', { nullable: true })\n placeholder: string | null;\n\n @Column('varchar', { nullable: true })\n options: string | null;\n\n @Column('boolean', { default: false })\n required: boolean;\n\n @Column('varchar', { nullable: true })\n validation: string | null;\n\n @Column('int')\n order: number;\n\n @Column('int', { default: 1 })\n groupId: number;\n\n @Column('int', { default: 12 })\n columnWidth: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Form, (f) => f.fields, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'formId' })\n form: Form;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\n\n@Entity('addresses')\nexport class Address {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n contactId: number;\n\n @Column('varchar', { nullable: true })\n tag: string | null;\n\n @Column('varchar', { nullable: true })\n line1: string | null;\n\n @Column('varchar', { nullable: true })\n line2: string | null;\n\n @Column('varchar', { nullable: true })\n city: string | null;\n\n @Column('varchar', { nullable: true })\n state: string | null;\n\n @Column('varchar', { nullable: true })\n postalCode: string | null;\n\n @Column('varchar', { nullable: true })\n country: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, (c) => c.addresses, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport { Address } from './address.entity';\nimport type { OrderItem } from './order-item.entity';\nimport type { Payment } from './payment.entity';\n\nexport type OrderKind = 'sale' | 'return' | 'replacement';\n\n@Entity('orders')\nexport class Order {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n orderNumber: string;\n\n @Column('varchar', { default: 'sale' })\n orderKind: OrderKind;\n\n @Column('int', { nullable: true })\n parentOrderId: number | null;\n\n @Column('int')\n contactId: number;\n\n @Column('int', { nullable: true })\n billingAddressId: number | null;\n\n @Column('int', { nullable: true })\n shippingAddressId: number | null;\n\n @Column('varchar', { default: 'pending' })\n status: 'pending' | 'confirmed' | 'processing' | 'completed' | 'cancelled';\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n subtotal: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n tax: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n discount: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n total: number;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Order, (o) => o.children, { nullable: true, onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentOrderId' })\n parentOrder: Order | null;\n\n @OneToMany(() => Order, (o) => o.parentOrder)\n children: Order[];\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n\n @ManyToOne(() => Address, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'billingAddressId' })\n billingAddress: Address | null;\n\n @ManyToOne(() => Address, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'shippingAddressId' })\n shippingAddress: Address | null;\n\n @OneToMany('OrderItem', 'order')\n items: OrderItem[];\n\n @OneToMany('Payment', 'order')\n payments: Payment[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Order } from './order.entity';\nimport { Contact } from './contact.entity';\n\n@Entity('payments')\nexport class Payment {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n orderId: number;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('decimal', { precision: 12, scale: 2 })\n amount: number;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column('varchar', { default: 'pending' })\n status: 'pending' | 'processing' | 'completed' | 'failed' | 'refunded';\n\n @Column('varchar', { nullable: true })\n method: string | null;\n\n @Column('varchar', { nullable: true })\n externalReference: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', nullable: true })\n paidAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Order, (o) => o.payments, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'orderId' })\n order: Order;\n\n @ManyToOne(() => Contact, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport { ChatMessage } from './chat-message.entity';\n\n@Entity('chat_conversations')\nexport class ChatConversation {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n contactId: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, (c) => c.chatConversations, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n\n @OneToMany(() => ChatMessage, (m) => m.conversation)\n messages: ChatMessage[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { ChatConversation } from './chat-conversation.entity';\n\n@Entity('chat_messages')\nexport class ChatMessage {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n conversationId: number;\n\n @Column('varchar')\n role: 'user' | 'assistant' | 'system';\n\n @Column('text')\n content: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @ManyToOne(() => ChatConversation, (c) => c.messages, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'conversationId' })\n conversation: ChatConversation;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, Unique } from 'typeorm';\n\n@Entity('configs')\n@Unique(['settings', 'key'])\nexport class Config {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n settings: string;\n\n @Column('varchar')\n key: string;\n\n @Column('varchar')\n value: string;\n\n @Column('varchar', { default: 'private' })\n type: 'public' | 'private';\n\n @Column('boolean', { default: false })\n encrypted: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('message_templates')\nexport class MessageTemplate {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n channel: string;\n\n @Column('varchar', { name: 'template_key' })\n templateKey: string;\n\n @Column('varchar', { nullable: true })\n name: string | null;\n\n @Column('varchar', { nullable: true })\n subject: string | null;\n\n @Column('text', { default: '' })\n body: string;\n\n @Column('varchar', { name: 'external_template_ref', nullable: true })\n externalTemplateRef: string | null;\n\n @Column({ type: 'jsonb', nullable: true })\n providerMeta: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n enabled: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('media')\nexport class Media {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n filename: string;\n\n @Column('varchar')\n url: string;\n\n @Column('varchar')\n mimeType: string;\n\n @Column('int', { default: 0 })\n size: number;\n\n @Column('varchar', { nullable: true })\n alt: string | null;\n\n @Column('boolean', { default: false })\n isPublic: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Seo } from './seo.entity';\n\n@Entity('pages')\nexport class Page {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n title: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column({ type: 'jsonb', default: {} })\n content: object;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column('varchar', { default: 'default' })\n theme: string;\n\n @Column('int', { nullable: true })\n parentId: number | null;\n\n @ManyToOne(() => Page, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentId' })\n parent: Page | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport type { Product } from './product.entity';\nimport type { Collection } from './collection.entity';\n\n@Entity('product_categories')\nexport class ProductCategory {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('int', { nullable: true })\n parentId: number | null;\n\n @Column('varchar', { nullable: true })\n image: string | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.children, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentId' })\n parent: ProductCategory | null;\n\n @OneToMany(() => ProductCategory, (c) => c.parent)\n children: ProductCategory[];\n\n @OneToMany('Product', 'category')\n products: Product[];\n\n @OneToMany('Collection', 'category')\n collections: Collection[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { ProductCategory } from './product-category.entity';\nimport { Brand } from './brand.entity';\nimport { Seo } from './seo.entity';\nimport type { Product } from './product.entity';\n\n@Entity('collections')\nexport class Collection {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('int', { nullable: true })\n brandId: number | null;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { nullable: true })\n hsn: string | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n image: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.collections, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: ProductCategory | null;\n\n @ManyToOne(() => Brand, (b) => b.collections, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'brandId' })\n brand: Brand | null;\n\n @OneToMany('Product', 'collection')\n products: Product[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne, JoinColumn } from 'typeorm';\nimport { Seo } from './seo.entity';\nimport type { Product } from './product.entity';\nimport type { Collection } from './collection.entity';\n\n@Entity('brands')\nexport class Brand {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { nullable: true })\n logo: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @OneToMany('Product', 'brand')\n products: Product[];\n\n @OneToMany('Collection', 'brand')\n collections: Collection[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Collection } from './collection.entity';\nimport { Brand } from './brand.entity';\nimport { ProductCategory } from './product-category.entity';\nimport { Seo } from './seo.entity';\nimport type { ProductAttribute } from './product-attribute.entity';\nimport type { ProductTax } from './product-tax.entity';\n\n@Entity('products')\nexport class Product {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int', { nullable: true })\n collectionId: number | null;\n\n @Column('int', { nullable: true })\n brandId: number | null;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('varchar', { nullable: true })\n sku: string | null;\n\n @Column('varchar', { nullable: true })\n hsn: string | null;\n\n @Column('varchar', { unique: true, nullable: true })\n slug: string | null;\n\n @Column('varchar', { nullable: true })\n name: string | null;\n\n @Column('decimal', { precision: 12, scale: 2 })\n price: number;\n\n @Column('decimal', { precision: 12, scale: 2, nullable: true })\n compareAtPrice: number | null;\n\n @Column('int', { default: 0 })\n quantity: number;\n\n @Column('varchar', { default: 'draft' })\n status: 'draft' | 'available' | 'reserved' | 'sold';\n\n @Column('boolean', { default: false })\n featured: boolean;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @ManyToOne(() => Collection, (c) => c.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'collectionId' })\n collection: Collection | null;\n\n @ManyToOne(() => Brand, (b) => b.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'brandId' })\n brand: Brand | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: ProductCategory | null;\n\n @OneToMany('ProductAttribute', 'product')\n attributes: ProductAttribute[];\n\n @OneToMany('ProductTax', 'product')\n taxes: ProductTax[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('attributes')\nexport class Attribute {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { default: 'text' })\n type: 'text' | 'number' | 'select' | 'boolean';\n\n @Column('jsonb', { nullable: true })\n options: string[] | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Product } from './product.entity';\nimport { Attribute } from './attribute.entity';\n\n@Entity('product_attributes')\nexport class ProductAttribute {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n productId: number;\n\n @Column('int')\n attributeId: number;\n\n @Column('varchar')\n value: string;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Product, (p) => p.attributes, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n\n @ManyToOne(() => Attribute, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'attributeId' })\n attribute: Attribute;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('taxes')\nexport class Tax {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('decimal', { precision: 5, scale: 2 })\n rate: number;\n\n @Column('boolean', { default: false })\n isDefault: boolean;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Product } from './product.entity';\nimport { Tax } from './tax.entity';\n\n@Entity('product_taxes')\nexport class ProductTax {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n productId: number;\n\n @Column('int')\n taxId: number;\n\n @Column('decimal', { precision: 5, scale: 2, nullable: true })\n rate: number | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Product, (p) => p.taxes, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n\n @ManyToOne(() => Tax, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'taxId' })\n tax: Tax;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Order } from './order.entity';\nimport { Product } from './product.entity';\n\n@Entity('order_items')\nexport class OrderItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n orderId: number;\n\n @Column('int')\n productId: number;\n\n @Column('int', { default: 1 })\n quantity: number;\n\n @Column('decimal', { precision: 12, scale: 2 })\n unitPrice: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n tax: number;\n\n @Column('decimal', { precision: 12, scale: 2 })\n total: number;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Order, (o) => o.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'orderId' })\n order: Order;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { KnowledgeBaseChunk } from './knowledge-base-chunk.entity';\n\n@Entity('knowledge_base_documents')\nexport class KnowledgeBaseDocument {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { nullable: true })\n sourceUrl: string | null;\n\n @Column('text')\n content: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @OneToMany(() => KnowledgeBaseChunk, (c) => c.document)\n chunks: KnowledgeBaseChunk[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { KnowledgeBaseDocument } from './knowledge-base-document.entity';\n\n@Entity('knowledge_base_chunks')\nexport class KnowledgeBaseChunk {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n documentId: number;\n\n @Column('text')\n content: string;\n\n @Column('int', { default: 0 })\n chunkIndex: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @ManyToOne(() => KnowledgeBaseDocument, (d) => d.chunks, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'documentId' })\n document: KnowledgeBaseDocument;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport type { CartItem } from './cart-item.entity';\n\n@Entity('carts')\nexport class Cart {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n guestToken: string | null;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column({ type: 'timestamp', nullable: true })\n expiresAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n\n @OneToMany('CartItem', 'cart')\n items: CartItem[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Cart } from './cart.entity';\nimport { Product } from './product.entity';\n\n@Entity('cart_items')\nexport class CartItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n cartId: number;\n\n @Column('int')\n productId: number;\n\n @Column('int', { default: 1 })\n quantity: number;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Cart, (c) => c.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'cartId' })\n cart: Cart;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport type { WishlistItem } from './wishlist-item.entity';\n\n@Entity('wishlists')\nexport class Wishlist {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n guestId: string | null;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('varchar', { default: 'default' })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n\n @OneToMany('WishlistItem', 'wishlist')\n items: WishlistItem[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Wishlist } from './wishlist.entity';\nimport { Product } from './product.entity';\n\n@Entity('wishlist_items')\nexport class WishlistItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n wishlistId: number;\n\n @Column('int')\n productId: number;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Wishlist, (w) => w.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'wishlistId' })\n wishlist: Wishlist;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import type { EntityTarget } from 'typeorm';\nimport { User } from './user.entity';\nimport { OtpChallenge } from './otp-challenge.entity';\nimport { PasswordResetToken } from './password-reset-token.entity';\nimport { UserGroup } from './user-group.entity';\nimport { Permission } from './permission.entity';\nimport { Blog } from './blog.entity';\nimport { Tag } from './tag.entity';\nimport { Category } from './category.entity';\nimport { Comment } from './comment.entity';\nimport { Contact } from './contact.entity';\nimport { Address } from './address.entity';\nimport { Form } from './form.entity';\nimport { FormField } from './form-field.entity';\nimport { FormSubmission } from './form-submission.entity';\nimport { Seo } from './seo.entity';\nimport { Config } from './config.entity';\nimport { MessageTemplate } from './message-template.entity';\nimport { Media } from './media.entity';\nimport { Page } from './page.entity';\nimport { ProductCategory } from './product-category.entity';\nimport { Collection } from './collection.entity';\nimport { Product } from './product.entity';\nimport { Attribute } from './attribute.entity';\nimport { ProductAttribute } from './product-attribute.entity';\nimport { Tax } from './tax.entity';\nimport { ProductTax } from './product-tax.entity';\nimport { Order } from './order.entity';\nimport { OrderItem } from './order-item.entity';\nimport { Payment } from './payment.entity';\nimport { Brand } from './brand.entity';\nimport { KnowledgeBaseDocument } from './knowledge-base-document.entity';\nimport { KnowledgeBaseChunk } from './knowledge-base-chunk.entity';\nimport { ChatConversation } from './chat-conversation.entity';\nimport { ChatMessage } from './chat-message.entity';\nimport { Cart } from './cart.entity';\nimport { CartItem } from './cart-item.entity';\nimport { Wishlist } from './wishlist.entity';\nimport { WishlistItem } from './wishlist-item.entity';\n\nexport {\n User,\n OtpChallenge,\n PasswordResetToken,\n UserGroup,\n Permission,\n Blog,\n Tag,\n Category,\n Comment,\n Contact,\n Form,\n FormField,\n FormSubmission,\n Seo,\n Config,\n MessageTemplate,\n Media,\n Page,\n ProductCategory,\n Collection,\n Product,\n Attribute,\n ProductAttribute,\n Tax,\n ProductTax,\n Order,\n OrderItem,\n Payment,\n Brand,\n KnowledgeBaseDocument,\n KnowledgeBaseChunk,\n ChatConversation,\n ChatMessage,\n Cart,\n CartItem,\n Wishlist,\n WishlistItem,\n};\n\n/** Map API resource segment (e.g. \"blogs\", \"form_submissions\") to entity. Used by CRUD handler. */\nexport const CMS_ENTITY_MAP: Record<string, EntityTarget<import('typeorm').ObjectLiteral>> = {\n users: User,\n otp_challenges: OtpChallenge,\n password_reset_tokens: PasswordResetToken,\n user_groups: UserGroup,\n permissions: Permission,\n blogs: Blog,\n tags: Tag,\n categories: Category,\n comments: Comment,\n contacts: Contact,\n addresses: Address,\n forms: Form,\n form_fields: FormField,\n form_submissions: FormSubmission,\n seos: Seo,\n configs: Config,\n message_templates: MessageTemplate,\n media: Media,\n pages: Page,\n product_categories: ProductCategory,\n collections: Collection,\n products: Product,\n attributes: Attribute,\n product_attributes: ProductAttribute,\n taxes: Tax,\n product_taxes: ProductTax,\n orders: Order,\n order_items: OrderItem,\n payments: Payment,\n brands: Brand,\n knowledge_base_documents: KnowledgeBaseDocument,\n knowledge_base_chunks: KnowledgeBaseChunk,\n chat_conversations: ChatConversation,\n chat_messages: ChatMessage,\n carts: Cart,\n cart_items: CartItem,\n wishlists: Wishlist,\n wishlist_items: WishlistItem,\n};\n","/** API resource keys excluded from the permission matrix (internal / not admin-CRUD). */\nexport const PERMISSION_ENTITY_INTERNAL_EXCLUDE = new Set([\n 'users',\n 'password_reset_tokens',\n 'user_groups',\n 'permissions',\n 'comments',\n 'form_fields',\n 'configs',\n 'knowledge_base_chunks',\n 'carts',\n 'cart_items',\n 'wishlists',\n 'wishlist_items',\n 'message_templates',\n]);\n\n/** Non-CRUD admin surfaces mapped to entity keys for RBAC. */\nexport const PERMISSION_LOGICAL_ENTITIES = [\n 'users',\n 'forms',\n 'form_submissions',\n 'dashboard',\n 'upload',\n 'settings',\n 'analytics',\n 'chat',\n] as const;\n\n/** Canonical name for new installs / migrations */\nexport const ADMIN_GROUP_NAME = 'Administrator';\n\n/** System administrator group (roles UI + users / user_groups / permissions bypass). */\nexport function isSuperAdminGroupName(name: string | null | undefined): boolean {\n return name === ADMIN_GROUP_NAME;\n}\n\nexport type EntityCrudAction = 'create' | 'read' | 'update' | 'delete';\n\nexport type EntityPermissionFlags = { c: boolean; r: boolean; u: boolean; d: boolean };\n\nexport function getPermissionableEntityKeys(entityMap: Record<string, unknown>): string[] {\n const fromMap = Object.keys(entityMap).filter((k) => !PERMISSION_ENTITY_INTERNAL_EXCLUDE.has(k));\n const logical = PERMISSION_LOGICAL_ENTITIES.filter((k) => !fromMap.includes(k));\n return [...fromMap.sort(), ...logical].filter((k, i, a) => a.indexOf(k) === i);\n}\n\nexport function permissionRowsToRecord(\n rows: Array<{ entity: string; canCreate: boolean; canRead: boolean; canUpdate: boolean; canDelete: boolean }> | undefined\n): Record<string, EntityPermissionFlags> {\n const out: Record<string, EntityPermissionFlags> = {};\n if (!rows?.length) return out;\n for (const p of rows) {\n out[p.entity] = {\n c: !!p.canCreate,\n r: !!p.canRead,\n u: !!p.canUpdate,\n d: !!p.canDelete,\n };\n }\n return out;\n}\n\nexport function hasEntityPermission(\n record: Record<string, EntityPermissionFlags> | undefined,\n entity: string,\n action: EntityCrudAction\n): boolean {\n const p = record?.[entity];\n if (!p) return false;\n if (action === 'create') return p.c;\n if (action === 'read') return p.r;\n if (action === 'update') return p.u;\n return p.d;\n}\n","import type { EntityCrudAction, EntityPermissionFlags } from './permission-entities';\nimport { hasEntityPermission } from './permission-entities';\n\n/** isRBACAdmin bypasses entity checks only for these (users / roles plumbing). */\nexport const RBAC_ADMIN_ONLY_ENTITIES = new Set(['users', 'user_groups', 'permissions']);\n\nexport interface SessionUser {\n id?: string;\n email?: string | null;\n name?: string | null;\n groupId?: number;\n /** @deprecated use entityPerms / isRBACAdmin */\n permissions?: string[];\n /** Administrator group: full access only for users, user_groups, permissions */\n isRBACAdmin?: boolean;\n entityPerms?: Record<string, EntityPermissionFlags>;\n /** When false and not isRBACAdmin, admin API/UI is denied. */\n adminAccess?: boolean;\n}\n\nexport function sessionHasEntityAccess(\n user: SessionUser | null | undefined,\n entity: string,\n action: EntityCrudAction\n): boolean {\n if (!user?.email) return false;\n if (user.isRBACAdmin && RBAC_ADMIN_ONLY_ENTITIES.has(entity)) return true;\n return hasEntityPermission(user.entityPerms, entity, action);\n}\n\nexport function canManageRoles(user: SessionUser | null | undefined): boolean {\n return !!(user?.email && user.isRBACAdmin);\n}\n\nexport type GetSession = () => Promise<{ user?: SessionUser } | null>;\n\nexport const OPEN_ENDPOINTS: Array<Record<string, string[]>> = [\n { '/api/contacts': ['POST'] },\n { '/api/form-submissions': ['POST'] },\n { '/api/blogs': ['GET'] },\n];\n\nexport const PERMISSION_REQUIRED_ENDPOINTS: Record<string, string[]> = {};\n\nexport function isOpenEndpoint(pathname: string): boolean {\n return OPEN_ENDPOINTS.some((endpoint) => pathname.startsWith(Object.keys(endpoint)[0]));\n}\n\nexport function getRequiredPermission(pathname: string): string[] | null {\n return null;\n}\n\nexport function isPublicMethod(pathname: string, method: string): boolean {\n for (const endpoint of OPEN_ENDPOINTS) {\n const key = Object.keys(endpoint)[0];\n if (pathname.startsWith(key) && endpoint[key].includes(method)) return true;\n }\n return false;\n}\n\nexport interface AuthHelpers {\n requireAuth(req: Request): Promise<Response | null>;\n requirePermission(req: Request, permission: string): Promise<Response | null>;\n requireEntityPermission(req: Request, entity: string, action: EntityCrudAction): Promise<Response | null>;\n requireAdminAccess(req: Request): Promise<Response | null>;\n getAuthenticatedUser(): Promise<SessionUser | null>;\n}\n\nexport function createAuthHelpers(getSession: GetSession, NextResponse: { json: (body: unknown, init?: { status?: number }) => Response }): AuthHelpers {\n return {\n async requireAuth() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n return null;\n },\n async requirePermission() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n return null;\n },\n async requireEntityPermission(_req: Request, entity: string, action: EntityCrudAction) {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n const u = session.user as SessionUser;\n if (sessionHasEntityAccess(u, entity, action)) return null;\n return NextResponse.json({ error: 'Forbidden', entity, action }, { status: 403 });\n },\n async requireAdminAccess() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n const u = session.user as SessionUser;\n if (u.isRBACAdmin) return null;\n if (u.adminAccess === false) return NextResponse.json({ error: 'Forbidden', reason: 'admin_access' }, { status: 403 });\n return null;\n },\n async getAuthenticatedUser() {\n const session = await getSession();\n return (session?.user as SessionUser) ?? null;\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityTarget } from 'typeorm';\nimport type { ObjectLiteral } from 'typeorm';\nimport { getPermissionableEntityKeys, ADMIN_GROUP_NAME } from './permission-entities';\n\n/**\n * Ensures the Administrator group has full CRUD on every permissionable entity\n * for the given entity map. Idempotent: upserts per entity.\n */\nexport async function seedAdministratorPermissions(\n dataSource: DataSource,\n entityMap: Record<string, unknown>\n): Promise<void> {\n const entities = getPermissionableEntityKeys(entityMap);\n const groupRepo = dataSource.getRepository(entityMap.user_groups as EntityTarget<ObjectLiteral>);\n const permRepo = dataSource.getRepository(entityMap.permissions as EntityTarget<ObjectLiteral>);\n\n const adminGroup = await groupRepo.findOne({ where: { name: ADMIN_GROUP_NAME, deleted: false } });\n if (!adminGroup) return;\n\n const fullCrud = { canCreate: true, canRead: true, canUpdate: true, canDelete: true };\n\n for (const entity of entities) {\n const existing = await permRepo.findOne({\n where: { groupId: adminGroup.id, entity },\n });\n if (existing) {\n existing.canCreate = true;\n existing.canRead = true;\n existing.canUpdate = true;\n existing.canDelete = true;\n await permRepo.save(existing);\n } else {\n await permRepo.save(\n permRepo.create({\n groupId: adminGroup.id,\n entity,\n ...fullCrud,\n })\n );\n }\n }\n}\n","export interface CmsMiddlewareConfig {\n publicAdminPaths?: string[];\n publicApiPaths?: string[];\n /** path -> allowed methods */\n publicApiMethods?: Record<string, string[]>;\n signInPath?: string;\n getSessionToken?: (request: { cookies: { get: (name: string) => { value?: string } | undefined } }) => string | undefined;\n}\n\n/** Default public API paths (no auth). Sites should extend this with their own routes. */\nexport const defaultPublicApiMethods: Record<string, string[]> = {\n '/api/contacts': ['POST'],\n '/api/form-submissions': ['POST'],\n '/api/blogs': ['GET'],\n '/api/forms': ['GET'],\n '/api/auth': ['GET', 'POST'],\n '/api/health': ['GET'],\n '/api/users/forgot-password': ['POST'],\n '/api/users/set-password': ['POST'],\n '/api/users/invite': ['POST'],\n};\n\nfunction defaultGetSessionToken(request: { cookies: { get: (name: string) => { value?: string } | undefined } }): string | undefined {\n return (\n request.cookies.get('__Secure-next-auth.session-token')?.value ??\n request.cookies.get('next-auth.session-token')?.value\n );\n}\n\nfunction isPublicMethod(pathname: string, method: string, publicApiMethods: Record<string, string[]>): boolean {\n for (const [endpoint, methods] of Object.entries(publicApiMethods)) {\n if (pathname.startsWith(endpoint) && methods.includes(method)) return true;\n }\n return false;\n}\n\n/**\n * Returns middleware logic. Use from Next.js middleware:\n * import { createCmsMiddleware } from '@infuro/cms-core';\n * export const middleware = createCmsMiddleware({ ... });\n * export const config = { matcher: ['/admin/:path*', '/api/:path*'] };\n */\nexport function createCmsMiddleware(config: CmsMiddlewareConfig = {}) {\n const {\n publicAdminPaths = ['/admin/signin', '/admin/forgot-password', '/admin/reset-password', '/admin/invite'],\n publicApiMethods = defaultPublicApiMethods,\n signInPath = '/admin/signin',\n getSessionToken = defaultGetSessionToken,\n } = config;\n\n return function cmsMiddleware(request: {\n nextUrl: { pathname: string };\n url: string;\n method: string;\n cookies: { get: (name: string) => { value?: string } | undefined };\n }): { type: 'next' } | { type: 'redirect'; url: string } | { type: 'json'; status: number; body: unknown } {\n const pathname = request.nextUrl.pathname;\n const method = request.method;\n\n if (publicAdminPaths.some((p) => pathname === p || pathname.startsWith(p + '/'))) {\n return { type: 'next' };\n }\n\n if (pathname.startsWith('/admin')) {\n const token = getSessionToken(request);\n if (!token) {\n return { type: 'redirect', url: new URL(signInPath, request.url).toString() };\n }\n }\n\n if (pathname.startsWith('/api')) {\n if (isPublicMethod(pathname, method, publicApiMethods)) {\n return { type: 'next' };\n }\n const token = getSessionToken(request);\n if (!token) {\n return { type: 'json', status: 401, body: { error: 'Unauthorized' } };\n }\n }\n\n return { type: 'next' };\n };\n}\n","/**\n * Build NextAuth options for credentials auth. App can extend/override via extend().\n */\nimport type { NextAuthOptions } from 'next-auth';\nimport _CredentialsProvider from 'next-auth/providers/credentials';\nconst CredentialsProvider = (_CredentialsProvider as unknown as { default: typeof _CredentialsProvider }).default ?? _CredentialsProvider;\nimport { isSuperAdminGroupName, permissionRowsToRecord } from './permission-entities';\n\nexport interface NextAuthUser {\n id: number;\n email: string;\n name: string | null;\n password: string | null;\n blocked?: boolean;\n deleted?: boolean;\n groupId?: number | null;\n adminAccess?: boolean;\n group?: {\n name?: string;\n permissions?: Array<{\n entity: string;\n canCreate: boolean;\n canRead: boolean;\n canUpdate: boolean;\n canDelete: boolean;\n }>;\n };\n}\n\nexport interface AuthorizeOtpInput {\n identifier: string;\n channel: 'email' | 'sms';\n code: string;\n}\n\nexport interface NextAuthOptionsConfig {\n /** Resolve user by email (e.g. from TypeORM). Return null if not found. */\n getUserByEmail: (email: string) => Promise<NextAuthUser | null>;\n comparePassword: (plain: string, hash: string) => Promise<boolean>;\n signInPage?: string;\n secret?: string;\n extend?: (options: NextAuthOptions) => NextAuthOptions;\n /** When false, password CredentialsProvider is omitted. Default true. */\n enablePasswordLogin?: boolean;\n /** When true, registers CredentialsProvider id `otp`. Requires authorizeOtp. Default false. */\n enableOtpLogin?: boolean;\n /** Validate OTP and return user (same shape as password flow) or null. */\n authorizeOtp?: (input: AuthorizeOtpInput) => Promise<NextAuthUser | null>;\n}\n\nfunction sessionUserFromNextAuthUser(user: NextAuthUser) {\n const g = user.group;\n const isRBACAdmin = isSuperAdminGroupName(g?.name);\n const entityPerms = permissionRowsToRecord(g?.permissions);\n const adminAccess = (user as NextAuthUser & { adminAccess?: boolean }).adminAccess === true;\n return {\n id: user.id.toString(),\n email: user.email,\n name: user.name,\n groupId: user.groupId ?? undefined,\n isRBACAdmin,\n entityPerms,\n adminAccess,\n };\n}\n\nexport function getNextAuthOptions(config: NextAuthOptionsConfig): NextAuthOptions {\n const {\n getUserByEmail,\n comparePassword,\n signInPage = '/admin/signin',\n secret,\n extend,\n enablePasswordLogin = true,\n enableOtpLogin = false,\n authorizeOtp,\n } = config;\n\n const providers: NextAuthOptions['providers'] = [];\n\n if (enablePasswordLogin) {\n providers.push(\n CredentialsProvider({\n name: 'credentials',\n credentials: {\n email: { label: 'Email', type: 'email' },\n password: { label: 'Password', type: 'password' },\n },\n async authorize(credentials) {\n if (!credentials?.email || !credentials?.password) return null;\n try {\n const user = await getUserByEmail(credentials.email);\n if (!user || user.blocked || (user as { deleted?: boolean }).deleted || !user.password) return null;\n const valid = await comparePassword(credentials.password, user.password);\n if (!valid) return null;\n return sessionUserFromNextAuthUser(user);\n } catch {\n return null;\n }\n },\n })\n );\n }\n\n if (enableOtpLogin && authorizeOtp) {\n providers.push(\n CredentialsProvider({\n id: 'otp',\n name: 'otp',\n credentials: {\n identifier: { label: 'Email or phone', type: 'text' },\n code: { label: 'Code', type: 'text' },\n channel: { label: 'Channel', type: 'text' },\n },\n async authorize(credentials) {\n const identifier = typeof credentials?.identifier === 'string' ? credentials.identifier.trim() : '';\n const code = typeof credentials?.code === 'string' ? credentials.code.trim() : '';\n const ch = credentials?.channel === 'sms' ? 'sms' : 'email';\n if (!identifier || !code) return null;\n try {\n const user = await authorizeOtp({ identifier, channel: ch, code });\n if (!user || user.blocked || (user as { deleted?: boolean }).deleted) return null;\n return sessionUserFromNextAuthUser(user);\n } catch {\n return null;\n }\n },\n })\n );\n }\n\n const options: NextAuthOptions = {\n secret: secret ?? process.env.NEXTAUTH_SECRET,\n providers,\n session: { strategy: 'jwt' },\n pages: { signIn: signInPage },\n cookies: {\n sessionToken: {\n name: process.env.NEXTAUTH_URL?.startsWith('https')\n ? '__Secure-next-auth.session-token'\n : 'next-auth.session-token',\n options: {\n httpOnly: true,\n sameSite: 'lax',\n path: '/',\n secure: process.env.NEXTAUTH_URL?.startsWith('https') ?? false,\n },\n },\n },\n callbacks: {\n async jwt({ token, user }) {\n if (user) {\n const u = user as unknown as Record<string, unknown>;\n (token as Record<string, unknown>).id = u.id;\n (token as Record<string, unknown>).groupId = u.groupId;\n (token as Record<string, unknown>).isRBACAdmin = u.isRBACAdmin;\n (token as Record<string, unknown>).entityPerms = u.entityPerms;\n (token as Record<string, unknown>).adminAccess = u.adminAccess;\n }\n return token;\n },\n async session({ session, token }) {\n if (session.user) {\n const t = token as Record<string, unknown>;\n (session.user as Record<string, unknown>).id = t.id;\n (session.user as Record<string, unknown>).groupId = t.groupId;\n (session.user as Record<string, unknown>).isRBACAdmin = t.isRBACAdmin;\n (session.user as Record<string, unknown>).entityPerms = t.entityPerms;\n (session.user as Record<string, unknown>).adminAccess = t.adminAccess;\n }\n return session;\n },\n },\n };\n\n return extend ? extend(options) : options;\n}\n","import type { DataSource } from 'typeorm';\nimport { ILike, Like, MoreThan } from 'typeorm';\nimport type { Repository } from 'typeorm';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport { queueErpCreateContactIfEnabled } from '../plugins/erp/erp-contact-sync';\nimport { queueErpProductUpsertIfEnabled } from '../plugins/erp/erp-product-sync';\n\nexport type EntityMap = Record<string, import('typeorm').EntityTarget<import('typeorm').ObjectLiteral>>;\n\nexport interface CrudHandlerOptions {\n requireAuth: (req: Request) => Promise<Response | null>;\n json: (body: unknown, init?: { status?: number }) => Response;\n requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n /** When set, contact create/update enqueues ERP `create-contact` (non-fatal). */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n}\n\nconst DATE_COLUMN_TYPES = new Set([\n 'date', 'datetime', 'datetime2', 'timestamp', 'timestamptz', 'timetz', 'smalldatetime', 'timestamp with time zone', 'timestamp without time zone',\n]);\n\nconst TIMESTAMP_PROP_NAMES = new Set(['createdAt', 'updatedAt', 'deletedAt']);\n\nfunction isInvalidDateValue(v: unknown): boolean {\n if (v === '' || v == null) return true;\n if (typeof v === 'string') return isNaN(Date.parse(v)) || /NaN|Invalid/i.test(v);\n if (v instanceof Date) return isNaN(v.getTime());\n return false;\n}\n\n/** Strip empty/invalid values for boolean, number, and date columns so DB defaults apply. */\nfunction sanitizeBodyForEntity(repo: Repository<import('typeorm').ObjectLiteral>, body: Record<string, unknown>): void {\n const meta = repo.metadata;\n for (const col of meta.columns) {\n if (!(col.propertyName in body)) continue;\n const v = body[col.propertyName];\n const t = typeof col.type === 'string' ? col.type : (col.type as Function)?.name ?? '';\n const isBoolean = t === 'boolean' || t === 'bool' || col.type === Boolean;\n const isNumber = ['int', 'integer', 'int2', 'int4', 'int8', 'smallint', 'bigint', 'number', 'Number'].includes(t) || col.type === Number;\n const isDate = DATE_COLUMN_TYPES.has(t) || col.type === Date || TIMESTAMP_PROP_NAMES.has(col.propertyName);\n if (v === '' && (isBoolean || isNumber)) {\n delete body[col.propertyName];\n } else if (isDate && isInvalidDateValue(v)) {\n delete body[col.propertyName];\n }\n }\n}\n\n/** Keep only scalar columns (excludes relations like tags) so repo.update() does not throw. */\nfunction pickColumnUpdates(\n repo: Repository<import('typeorm').ObjectLiteral>,\n body: Record<string, unknown>\n): Record<string, unknown> {\n const cols = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const out: Record<string, unknown> = {};\n for (const k of Object.keys(body)) {\n if (cols.has(k)) out[k] = body[k];\n }\n return out;\n}\n\n/** OR search on columns that exist on the entity (avoids Tag/Category etc. without `title`). */\nfunction buildSearchWhereClause(\n repo: Repository<import('typeorm').ObjectLiteral>,\n search: string\n): Record<string, unknown>[] | Record<string, unknown> {\n const cols = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const term = ILike(`%${search}%`);\n const ors: Record<string, unknown>[] = [];\n for (const field of ['name', 'title', 'slug', 'email', 'filename'] as const) {\n if (cols.has(field)) ors.push({ [field]: term });\n }\n if (ors.length === 0) return {};\n return ors.length === 1 ? ors[0]! : ors;\n}\n\nfunction makeContactErpSync(\n dataSource: DataSource,\n entityMap: EntityMap,\n getCms: CrudHandlerOptions['getCms']\n): (row: import('typeorm').ObjectLiteral) => Promise<void> {\n return async function syncContactRowToErp(row: import('typeorm').ObjectLiteral): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n const c = row as {\n name: string;\n email: string;\n phone?: string | null;\n type?: string | null;\n company?: string | null;\n notes?: string | null;\n };\n await queueErpCreateContactIfEnabled(cms, dataSource, entityMap, {\n name: c.name,\n email: c.email,\n phone: c.phone,\n type: c.type,\n company: c.company,\n notes: c.notes,\n });\n } catch {\n /* ignore */\n }\n };\n}\n\nexport function createCrudHandler(\n dataSource: DataSource,\n entityMap: EntityMap,\n options: CrudHandlerOptions\n) {\n const { requireAuth, json, requireEntityPermission: reqPerm, getCms } = options;\n const syncContactRowToErp = makeContactErpSync(dataSource, entityMap, getCms);\n\n async function authz(req: Request, resource: string, action: EntityCrudAction): Promise<Response | null> {\n const authError = await requireAuth(req);\n if (authError) return authError;\n if (reqPerm) {\n const pe = await reqPerm(req, resource, action);\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const { searchParams } = new URL(req.url);\n const page = Number(searchParams.get('page')) || 1;\n const limit = Math.min(Number(searchParams.get('limit')) || 10, 100);\n const skip = (page - 1) * limit;\n const sortFieldRaw = searchParams.get('sortField') || 'createdAt';\n const sortOrder = searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const search = searchParams.get('search');\n\n // Orders list: include contact + itemsSummary; search (order# + customer), status, date, paymentRef\n if (resource === 'orders') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'orderNumber', 'contactId', 'status', 'total', 'currency', 'createdAt', 'updatedAt'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderOrders = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const statusFilter = searchParams.get('status')?.trim();\n const dateFrom = searchParams.get('dateFrom')?.trim();\n const dateTo = searchParams.get('dateTo')?.trim();\n const paymentRef = searchParams.get('paymentRef')?.trim();\n\n let orderIdsFromPayment: number[] | null = null;\n if (paymentRef && entityMap['payments']) {\n const paymentRepo = dataSource.getRepository(entityMap['payments']);\n const payments = await paymentRepo\n .createQueryBuilder('p')\n .select('p.orderId')\n .where('p.externalReference = :ref', { ref: paymentRef })\n .orWhere(\"p.metadata->>'razorpayPaymentId' = :ref\", { ref: paymentRef })\n .getRawMany<{ orderId: number }>();\n orderIdsFromPayment = payments.map((r) => r.orderId);\n if (orderIdsFromPayment.length === 0) {\n return json({ total: 0, page, limit, totalPages: 0, data: [] });\n }\n }\n\n const qb = repo\n .createQueryBuilder('order')\n .leftJoinAndSelect('order.contact', 'contact')\n .leftJoinAndSelect('order.items', 'items')\n .leftJoinAndSelect('items.product', 'product')\n .leftJoinAndSelect('product.collection', 'collection')\n .orderBy(`order.${sortField}`, sortOrderOrders)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere(\n '(order.orderNumber ILIKE :term OR contact.name ILIKE :term OR contact.email ILIKE :term)',\n { term }\n );\n }\n if (statusFilter) qb.andWhere('order.status = :status', { status: statusFilter });\n if (dateFrom) qb.andWhere('order.createdAt >= :dateFrom', { dateFrom: new Date(dateFrom + 'T00:00:00.000Z') });\n if (dateTo) qb.andWhere('order.createdAt <= :dateTo', { dateTo: new Date(dateTo + 'T23:59:59.999Z') });\n if (orderIdsFromPayment && orderIdsFromPayment.length) qb.andWhere('order.id IN (:...orderIds)', { orderIds: orderIdsFromPayment });\n\n const [rows, total] = await qb.getManyAndCount();\n const data = (rows as Record<string, unknown>[]).map((order: Record<string, unknown>) => {\n const contact = order.contact as Record<string, unknown> | undefined;\n const items = (order.items as Array<{ product?: { name?: string; collection?: { name?: string } }; quantity: number }>) ?? [];\n const itemsSummary = items\n .map((i) => {\n const label = i.product?.collection?.name ?? i.product?.name ?? 'Product';\n return `${label} × ${i.quantity}`;\n })\n .join(', ') || '—';\n return {\n ...order,\n contact: contact ? { id: contact.id, name: contact.name, email: contact.email, phone: contact.phone } : null,\n itemsSummary,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Payments list: include order + contact; search (customer), status, date, method, orderNumber\n if (resource === 'payments') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'orderId', 'amount', 'currency', 'status', 'method', 'paidAt', 'createdAt', 'updatedAt'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderPayments = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const statusFilter = searchParams.get('status')?.trim();\n const dateFrom = searchParams.get('dateFrom')?.trim();\n const dateTo = searchParams.get('dateTo')?.trim();\n const methodFilter = searchParams.get('method')?.trim();\n const orderNumberParam = searchParams.get('orderNumber')?.trim();\n\n const qb = repo\n .createQueryBuilder('payment')\n .leftJoinAndSelect('payment.order', 'ord')\n .leftJoinAndSelect('ord.contact', 'orderContact')\n .leftJoinAndSelect('payment.contact', 'contact')\n .orderBy(`payment.${sortField}`, sortOrderPayments)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere(\n '(orderContact.name ILIKE :term OR orderContact.email ILIKE :term OR contact.name ILIKE :term OR contact.email ILIKE :term)',\n { term }\n );\n }\n if (statusFilter) qb.andWhere('payment.status = :status', { status: statusFilter });\n if (dateFrom) qb.andWhere('payment.createdAt >= :dateFrom', { dateFrom: new Date(dateFrom + 'T00:00:00.000Z') });\n if (dateTo) qb.andWhere('payment.createdAt <= :dateTo', { dateTo: new Date(dateTo + 'T23:59:59.999Z') });\n if (methodFilter) qb.andWhere('payment.method = :method', { method: methodFilter });\n if (orderNumberParam) qb.andWhere('ord.orderNumber ILIKE :orderNumber', { orderNumber: `%${orderNumberParam}%` });\n\n const [rows, total] = await qb.getManyAndCount();\n const data = (rows as Record<string, unknown>[]).map((payment: Record<string, unknown>) => {\n const order = payment.order as Record<string, unknown> | undefined;\n const orderContact = order?.contact as Record<string, unknown> | undefined;\n const contact = payment.contact as Record<string, unknown> | undefined;\n const customer = orderContact ?? contact;\n return {\n ...payment,\n order: order ? { id: order.id, orderNumber: order.orderNumber, contact: orderContact ? { name: orderContact.name, email: orderContact.email } : null } : null,\n contact: customer ? { id: customer.id, name: customer.name, email: customer.email } : null,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Products list: status and inventory filters\n if (resource === 'products') {\n const repo = dataSource.getRepository(entity);\n const statusFilter = searchParams.get('status')?.trim();\n const inventory = searchParams.get('inventory')?.trim();\n const productWhere: Record<string, unknown> = {};\n if (statusFilter) productWhere.status = statusFilter;\n if (inventory === 'in_stock') productWhere.quantity = MoreThan(0);\n if (inventory === 'out_of_stock') productWhere.quantity = 0;\n if (search && typeof search === 'string' && search.trim()) {\n productWhere.name = ILike(`%${search.trim()}%`);\n }\n const [data, total] = await repo.findAndCount({\n where: Object.keys(productWhere).length ? productWhere : undefined,\n skip,\n take: limit,\n order: { [sortFieldRaw]: sortOrder },\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Contacts list: filters (type, orderId), optional includeSummary (orderCount, totalPaid)\n if (resource === 'contacts') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'name', 'email', 'createdAt', 'type'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderContacts = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const typeFilter = searchParams.get('type')?.trim();\n const orderIdParam = searchParams.get('orderId')?.trim();\n const includeSummary = searchParams.get('includeSummary') === '1';\n\n const qb = repo\n .createQueryBuilder('contact')\n .orderBy(`contact.${sortField}`, sortOrderContacts)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere('(contact.name ILIKE :term OR contact.email ILIKE :term OR contact.phone ILIKE :term)', { term });\n }\n if (typeFilter) qb.andWhere('contact.type = :type', { type: typeFilter });\n if (orderIdParam) {\n const orderId = Number(orderIdParam);\n if (!Number.isNaN(orderId)) {\n qb.andWhere('contact.id IN (SELECT \"contactId\" FROM orders WHERE id = :orderId)', { orderId });\n }\n }\n\n if (includeSummary && entityMap['orders'] && entityMap['payments']) {\n qb.loadRelationCountAndMap('contact._orderCount', 'contact.orders');\n const [rows, total] = await qb.getManyAndCount();\n const contactIds = (rows as { id: number }[]).map((c) => c.id);\n const paymentRepo = dataSource.getRepository(entityMap['payments']);\n const paidByContact = await paymentRepo\n .createQueryBuilder('p')\n .select('p.contactId', 'contactId')\n .addSelect('COALESCE(SUM(CAST(p.amount AS DECIMAL)), 0)', 'total')\n .where('p.contactId IN (:...ids)', { ids: contactIds.length ? contactIds : [0] })\n .andWhere('p.status = :status', { status: 'completed' })\n .groupBy('p.contactId')\n .getRawMany<{ contactId: number; total: string }>();\n const totalPaidMap = new Map<number, number>(paidByContact.map((r) => [r.contactId, Number(r.total)]));\n const data = (rows as Record<string, unknown>[]).map((c) => {\n const { _orderCount, ...rest } = c as { _orderCount?: number; id: number };\n return {\n ...rest,\n orderCount: _orderCount ?? 0,\n totalPaid: totalPaidMap.get((rest as { id: number }).id) ?? 0,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n const [data, total] = await qb.getManyAndCount();\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n const repo = dataSource.getRepository(entity);\n const typeFilter = searchParams.get('type');\n const columnNames = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const sortField = columnNames.has(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n let where: Record<string, unknown>[] | Record<string, unknown> = {};\n if (resource === 'media') {\n const mediaWhere: Record<string, unknown> = {};\n if (search) mediaWhere.filename = ILike(`%${search}%`);\n if (typeFilter) mediaWhere.mimeType = Like(`${typeFilter}/%`);\n where = Object.keys(mediaWhere).length > 0 ? mediaWhere : {};\n } else if (search) {\n where = buildSearchWhereClause(repo, search);\n }\n const [data, total] = await repo.findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n where,\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n },\n\n async POST(req: Request, resource: string) {\n const authError = await authz(req, resource, 'create');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const body = await req.json();\n if (!body || typeof body !== 'object' || Object.keys(body).length === 0) {\n return json({ error: 'Invalid request payload' }, { status: 400 });\n }\n const repo = dataSource.getRepository(entity);\n sanitizeBodyForEntity(repo, body as Record<string, unknown>);\n const created = await repo.save(repo.create(body as object));\n if (resource === 'contacts') {\n await syncContactRowToErp(created as import('typeorm').ObjectLiteral);\n }\n if (resource === 'products' && getCms) {\n const cms = await getCms();\n await queueErpProductUpsertIfEnabled(cms, dataSource, entityMap, created as import('typeorm').ObjectLiteral);\n }\n return json(created, { status: 201 });\n },\n\n async GET_METADATA(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const repo = dataSource.getRepository(entity);\n const meta = repo.metadata;\n\n // Collect unique column names from indices\n const uniqueFromIndices = new Set<string>();\n for (const idx of meta.indices) {\n if (idx.isUnique && idx.columns.length === 1) {\n uniqueFromIndices.add(idx.columns[0].propertyName);\n }\n }\n for (const uniq of meta.uniques) {\n if (uniq.columns.length === 1) {\n uniqueFromIndices.add(uniq.columns[0].propertyName);\n }\n }\n\n const columns = meta.columns.map((col) => ({\n name: col.propertyName,\n type: typeof col.type === 'string' ? col.type : (col.type as { name?: string })?.name ?? 'unknown',\n nullable: col.isNullable,\n isUnique: uniqueFromIndices.has(col.propertyName),\n isPrimary: col.isPrimary,\n default: col.default,\n }));\n\n const uniqueColumns = [...uniqueFromIndices];\n\n return json({ columns, uniqueColumns });\n },\n\n async BULK_POST(req: Request, resource: string) {\n const authError = await authz(req, resource, 'update');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const body = await req.json();\n const { records, upsertKey = 'id' } = body as { records: object[]; upsertKey?: string };\n\n if (!Array.isArray(records) || records.length === 0) {\n return json({ error: 'Records array is required' }, { status: 400 });\n }\n\n const repo = dataSource.getRepository(entity);\n\n // Sanitize each record\n for (const record of records) {\n sanitizeBodyForEntity(repo, record as Record<string, unknown>);\n }\n\n try {\n const result = await repo.upsert(records, {\n conflictPaths: [upsertKey],\n skipUpdateIfNoValuesChanged: true,\n });\n return json({\n success: true,\n imported: result.identifiers.length,\n identifiers: result.identifiers,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Bulk import failed';\n return json({ error: message }, { status: 400 });\n }\n },\n\n async GET_EXPORT(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n\n const { searchParams } = new URL(req.url);\n const format = searchParams.get('format') || 'csv';\n\n const repo = dataSource.getRepository(entity);\n const meta = repo.metadata;\n\n // Check if entity has 'deleted' column\n const hasDeleted = meta.columns.some((c) => c.propertyName === 'deleted');\n const where = hasDeleted ? { deleted: false } : {};\n\n const data = await repo.find({ where });\n\n // Get exportable columns (exclude soft-delete related)\n const excludeCols = new Set(['deletedAt', 'deletedBy', 'deleted']);\n const columns = meta.columns\n .filter((c) => !excludeCols.has(c.propertyName))\n .map((c) => c.propertyName);\n\n if (format === 'json') {\n return json(data);\n }\n\n // CSV format\n const escapeCSV = (val: unknown): string => {\n if (val === null || val === undefined) return '';\n const str = typeof val === 'object' ? JSON.stringify(val) : String(val);\n if (str.includes(',') || str.includes('\"') || str.includes('\\n')) {\n return `\"${str.replace(/\"/g, '\"\"')}\"`;\n }\n return str;\n };\n\n const header = columns.join(',');\n const rows = data.map((row) =>\n columns.map((col) => escapeCSV((row as Record<string, unknown>)[col])).join(',')\n );\n const csv = [header, ...rows].join('\\n');\n\n return new Response(csv, {\n headers: {\n 'Content-Type': 'text/csv; charset=utf-8',\n 'Content-Disposition': `attachment; filename=\"${resource}.csv\"`,\n },\n });\n },\n };\n}\n\nexport function createCrudByIdHandler(\n dataSource: DataSource,\n entityMap: EntityMap,\n options: CrudHandlerOptions\n) {\n const { requireAuth, json, requireEntityPermission: reqPerm, getCms } = options;\n const syncContactRowToErp = makeContactErpSync(dataSource, entityMap, getCms);\n\n async function authz(req: Request, resource: string, action: EntityCrudAction): Promise<Response | null> {\n const authError = await requireAuth(req);\n if (authError) return authError;\n if (reqPerm) {\n const pe = await reqPerm(req, resource, action);\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const repo = dataSource.getRepository(entity);\n\n if (resource === 'orders') {\n const order = await repo.findOne({\n where: { id: Number(id) },\n relations: ['contact', 'billingAddress', 'shippingAddress', 'items', 'items.product', 'items.product.collection', 'payments'],\n });\n if (!order) return json({ message: 'Not found' }, { status: 404 });\n const relatedOrders = await repo.find({\n where: { parentOrderId: Number(id), deleted: false } as import('typeorm').ObjectLiteral,\n order: { id: 'ASC' },\n });\n return json({ ...order, relatedOrders });\n }\n\n if (resource === 'contacts') {\n const contact = await repo.findOne({\n where: { id: Number(id) },\n relations: ['form_submissions', 'form_submissions.form', 'orders', 'payments', 'addresses'],\n });\n if (!contact) return json({ message: 'Not found' }, { status: 404 });\n const orders = (contact as { orders?: { total?: unknown; createdAt?: string }[] }).orders ?? [];\n const payments = (contact as { payments?: { status: string; amount?: number }[] }).payments ?? [];\n const totalPaid = payments\n .filter((p) => p.status === 'completed')\n .reduce((sum, p) => sum + Number(p.amount ?? 0), 0);\n const lastOrderAt =\n orders.length > 0\n ? orders.reduce((latest, o) => {\n const t = o.createdAt ? new Date(o.createdAt).getTime() : 0;\n return t > latest ? t : latest;\n }, 0)\n : null;\n return json({\n ...contact,\n summary: {\n totalOrders: orders.length,\n totalPaid,\n lastOrderAt: lastOrderAt ? new Date(lastOrderAt).toISOString() : null,\n },\n });\n }\n\n if (resource === 'payments') {\n const payment = await repo.findOne({\n where: { id: Number(id) },\n relations: ['order', 'order.contact', 'contact'],\n });\n if (!payment) return json({ message: 'Not found' }, { status: 404 });\n return json(payment);\n }\n\n if (resource === 'blogs') {\n const blog = await repo.findOne({\n where: { id: Number(id) },\n relations: ['category', 'seo', 'tags'],\n });\n return blog ? json(blog) : json({ message: 'Not found' }, { status: 404 });\n }\n\n const item = await repo.findOne({ where: { id: Number(id) } });\n return item ? json(item) : json({ message: 'Not found' }, { status: 404 });\n },\n\n async PUT(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'update');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const rawBody = (await req.json()) as Record<string, unknown> | null;\n const repo = dataSource.getRepository(entity);\n const numericId = Number(id);\n\n if (\n resource === 'blogs' &&\n rawBody &&\n typeof rawBody === 'object' &&\n entityMap.categories &&\n entityMap.seos &&\n entityMap.tags\n ) {\n const existing = await repo.findOne({ where: { id: numericId } });\n if (!existing) return json({ message: 'Not found' }, { status: 404 });\n\n const updatePayload = pickColumnUpdates(repo, rawBody);\n\n if ('category' in rawBody) {\n const c = rawBody.category;\n if (typeof c === 'string' && c.trim()) {\n const cat = await dataSource\n .getRepository(entityMap.categories)\n .findOne({ where: { name: c.trim() } });\n updatePayload.categoryId = cat?.id ?? null;\n } else {\n updatePayload.categoryId = null;\n }\n }\n\n const blogSlug =\n (typeof updatePayload.slug === 'string' && updatePayload.slug) ||\n (existing as { slug: string }).slug;\n const seoRepo = dataSource.getRepository(entityMap.seos);\n const seoField = (k: string): string | null | undefined => {\n if (!(k in rawBody)) return undefined;\n const v = rawBody[k];\n if (v == null || v === '') return null;\n return String(v);\n };\n if (\n 'metaTitle' in rawBody ||\n 'metaDescription' in rawBody ||\n 'metaKeywords' in rawBody ||\n 'ogImage' in rawBody\n ) {\n const title = seoField('metaTitle');\n const description = seoField('metaDescription');\n const keywords = seoField('metaKeywords');\n const ogImage = seoField('ogImage');\n const exSeoId = (existing as { seoId: number | null }).seoId;\n if (exSeoId) {\n const seo = await seoRepo.findOne({ where: { id: exSeoId } });\n if (seo) {\n const s = seo as Record<string, unknown>;\n if (title !== undefined) s.title = title;\n if (description !== undefined) s.description = description;\n if (keywords !== undefined) s.keywords = keywords;\n if (ogImage !== undefined) s.ogImage = ogImage;\n s.slug = blogSlug;\n await seoRepo.save(seo);\n }\n } else {\n let seoSlug = blogSlug;\n const taken = await seoRepo.findOne({ where: { slug: seoSlug } });\n if (taken) seoSlug = `blog-${numericId}-${blogSlug}`;\n const seo = await seoRepo.save(\n seoRepo.create({\n slug: seoSlug,\n title: title ?? null,\n description: description ?? null,\n keywords: keywords ?? null,\n ogImage: ogImage ?? null,\n })\n );\n updatePayload.seoId = (seo as { id: number }).id;\n }\n }\n\n sanitizeBodyForEntity(repo, updatePayload);\n await repo.update(numericId, updatePayload as object);\n\n if (Array.isArray(rawBody.tags)) {\n const tagNames = (rawBody.tags as unknown[]).map((t) => String(t).trim()).filter(Boolean);\n const tagRepo = dataSource.getRepository(entityMap.tags);\n const tagEntities: import('typeorm').ObjectLiteral[] = [];\n for (const name of tagNames) {\n let tag = await tagRepo.findOne({ where: { name } });\n if (!tag) tag = await tagRepo.save(tagRepo.create({ name }));\n tagEntities.push(tag);\n }\n const blog = await repo.findOne({ where: { id: numericId }, relations: ['tags'] });\n if (blog) {\n (blog as Record<string, unknown>).tags = tagEntities;\n await repo.save(blog);\n }\n }\n\n const updated = await repo.findOne({\n where: { id: numericId },\n relations: ['tags', 'category', 'seo'],\n });\n return updated ? json(updated) : json({ message: 'Not found' }, { status: 404 });\n }\n\n const updatePayload = rawBody && typeof rawBody === 'object' ? pickColumnUpdates(repo, rawBody) : {};\n if (Object.keys(updatePayload).length > 0) {\n sanitizeBodyForEntity(repo, updatePayload);\n await repo.update(numericId, updatePayload as object);\n }\n const updated = await repo.findOne({ where: { id: numericId } });\n if (resource === 'contacts' && updated) {\n await syncContactRowToErp(updated as import('typeorm').ObjectLiteral);\n }\n if (resource === 'products' && updated && getCms) {\n const cms = await getCms();\n await queueErpProductUpsertIfEnabled(cms, dataSource, entityMap, updated as import('typeorm').ObjectLiteral);\n }\n return updated ? json(updated) : json({ message: 'Not found' }, { status: 404 });\n },\n\n async DELETE(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'delete');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const repo = dataSource.getRepository(entity);\n const result = await repo.delete(Number(id));\n if (result.affected === 0) return json({ message: 'Not found' }, { status: 404 });\n return json({ message: 'Deleted successfully' }, { status: 200 });\n },\n };\n}\n","/**\n * Auth API handler factories. Inject dataSource + entityMap, sendEmail, hash/compare; optional hooks to customize.\n */\nimport type { DataSource } from 'typeorm';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport type { EntityMap } from './crud';\n\nexport interface AuthHandlersConfig {\n json: (body: unknown, init?: { status?: number }) => Response;\n baseUrl: string;\n hashPassword: (plain: string) => Promise<string>;\n comparePassword: (plain: string, hash: string) => Promise<boolean>;\n}\n\nexport interface ForgotPasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n sendEmail?: (opts: {\n to: string;\n subject: string;\n html: string;\n text?: string;\n /** Plain reset URL for templated emails (preferred over parsing html). */\n resetLink?: string;\n }) => Promise<void>;\n resetExpiryHours?: number;\n afterCreateToken?: (email: string, resetLink: string) => Promise<void>;\n}\n\nexport function createForgotPasswordHandler(config: ForgotPasswordConfig) {\n const { dataSource, entityMap, json, baseUrl, sendEmail, resetExpiryHours = 1, afterCreateToken } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { email?: string };\n const email = typeof body?.email === 'string' ? body.email.trim().toLowerCase() : '';\n if (!email) return json({ error: 'Email is required' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email }, select: ['email'] });\n const msg = 'If an account exists with this email, you will receive a reset link shortly.';\n if (!user) return json({ message: msg }, { status: 200 });\n\n const crypto = await import('crypto');\n const token = crypto.randomBytes(32).toString('hex');\n const expiresAt = new Date(Date.now() + (resetExpiryHours * 60 * 60 * 1000));\n const tokenRepo = dataSource.getRepository(entityMap.password_reset_tokens);\n await tokenRepo.save(tokenRepo.create({ email: user.email, token, expiresAt }));\n const resetLink = `${baseUrl}/admin/reset-password?token=${token}`;\n\n if (sendEmail)\n await sendEmail({\n to: user.email,\n subject: 'Password reset',\n html: `<a href=\"${resetLink}\">Reset password</a>`,\n text: resetLink,\n resetLink,\n });\n if (afterCreateToken) await afterCreateToken(user.email, resetLink);\n return json({ message: msg }, { status: 200 });\n } catch (err) {\n return json({ error: 'Something went wrong. Please try again.' }, { status: 500 });\n }\n };\n}\n\nexport interface SetPasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n minPasswordLength?: number;\n beforeUpdate?: (email: string, userId: number) => Promise<void>;\n}\n\nexport function createSetPasswordHandler(config: SetPasswordConfig) {\n const { dataSource, entityMap, json, hashPassword, minPasswordLength = 6, beforeUpdate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { token?: string; newPassword?: string };\n const { token, newPassword } = body;\n if (!token || !newPassword) return json({ error: 'Token and new password are required' }, { status: 400 });\n if (newPassword.length < minPasswordLength) return json({ error: 'Password must be at least 6 characters' }, { status: 400 });\n\n const tokenRepo = dataSource.getRepository(entityMap.password_reset_tokens);\n const record = await tokenRepo.findOne({ where: { token } });\n if (!record || record.expiresAt < new Date()) return json({ error: 'Invalid or expired reset link. Please request a new one.' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email: record.email }, select: ['id'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n\n if (beforeUpdate) await beforeUpdate(record.email, user.id);\n const hashedPassword = await hashPassword(newPassword);\n await userRepo.update(user.id, { password: hashedPassword, updatedAt: new Date() });\n await tokenRepo.delete({ email: record.email });\n return json({ message: 'Password updated successfully. You can now sign in.' });\n } catch {\n return json({ error: 'Something went wrong. Please try again.' }, { status: 500 });\n }\n };\n}\n\nexport interface InviteAcceptConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n beforeActivate?: (email: string, userId: number) => Promise<void>;\n}\n\n/** Decode invite token (base64 email) and set password + unblock user */\nexport function createInviteAcceptHandler(config: InviteAcceptConfig) {\n const { dataSource, entityMap, json, hashPassword, beforeActivate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { token?: string; password?: string };\n const { token, password } = body;\n if (!token || !password) return json({ error: 'Missing required fields: token, password' }, { status: 400 });\n\n let email: string;\n try {\n email = Buffer.from(token, 'base64').toString('utf8');\n } catch {\n return json({ error: 'Invalid or expired invite token' }, { status: 400 });\n }\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email }, select: ['id', 'blocked'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n if (!user.blocked) return json({ error: 'User is already active' }, { status: 400 });\n\n if (entityMap.contacts) {\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, user.id, email);\n }\n if (beforeActivate) await beforeActivate(email, user.id);\n const hashedPassword = await hashPassword(password);\n await userRepo.update(user.id, { password: hashedPassword, blocked: false });\n return json({ message: 'User account activated successfully' }, { status: 200 });\n } catch (err) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface ChangePasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n getSession: () => Promise<{ user?: { email?: string | null } } | null>;\n minPasswordLength?: number;\n beforeUpdate?: (email: string) => Promise<void>;\n}\n\nexport function createChangePasswordHandler(config: ChangePasswordConfig) {\n const { dataSource, entityMap, json, comparePassword, hashPassword, getSession, minPasswordLength = 6, beforeUpdate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n\n const body = await request.json().catch(() => ({})) as { currentPassword?: string; newPassword?: string };\n const { currentPassword, newPassword } = body;\n if (!currentPassword || !newPassword) return json({ error: 'Current password and new password are required' }, { status: 400 });\n if (newPassword.length < minPasswordLength) return json({ error: 'New password must be at least 6 characters long' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email: session.user.email }, select: ['password'] });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n if (!user.password) return json({ error: 'Current password is incorrect' }, { status: 400 });\n const valid = await comparePassword(currentPassword, user.password);\n if (!valid) return json({ error: 'Current password is incorrect' }, { status: 400 });\n\n if (beforeUpdate) await beforeUpdate(session.user.email);\n const hashedPassword = await hashPassword(newPassword);\n await userRepo.update({ email: session.user.email }, { password: hashedPassword, updatedAt: new Date() });\n return json({ message: 'Password updated successfully' });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface UserAuthApiConfig\n extends ForgotPasswordConfig,\n Omit<SetPasswordConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword'>,\n Omit<InviteAcceptConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword'>,\n Omit<ChangePasswordConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword' | 'beforeUpdate' | 'getSession'> {\n getSession?: () => Promise<{ user?: { email?: string | null } } | null>;\n beforeChangePasswordUpdate?: (email: string) => Promise<void>;\n}\n\nconst USER_AUTH_PATHS = ['forgot-password', 'invite', 'set-password', 'reset-password'] as const;\n\n/**\n * Single router for all user-auth APIs. Mount in the app once.\n * Use when you have no users/[id] route (else Next.js gives [id] precedence and \"forgot-password\" would match as id).\n * Path is the segment after the mount (e.g. \"forgot-password\"). Returns 404 for unknown paths.\n */\nexport function createUserAuthApiRouter(config: UserAuthApiConfig) {\n const forgot = createForgotPasswordHandler(config);\n const setPass = createSetPasswordHandler(config);\n const invite = createInviteAcceptHandler(config);\n const changePass = config.getSession\n ? createChangePasswordHandler({\n ...config,\n getSession: config.getSession,\n beforeUpdate: config.beforeChangePasswordUpdate,\n })\n : null;\n\n return {\n async POST(req: Request, pathname: string): Promise<Response> {\n const path = pathname.replace(/\\/$/, '');\n if (!USER_AUTH_PATHS.includes(path as (typeof USER_AUTH_PATHS)[number])) {\n return config.json({ error: 'Not found' }, { status: 404 });\n }\n if (path === 'forgot-password') return forgot(req);\n if (path === 'set-password') return setPass(req);\n if (path === 'invite') return invite(req);\n if (path === 'reset-password' && changePass) return changePass(req);\n return config.json({ error: 'Not found' }, { status: 404 });\n },\n };\n}\n","/**\n * CMS API handlers: dashboard, analytics, upload, blog/form by slug, users (list/create/get/update/delete/regenerate-invite/avatar/profile).\n * All accept injectable deps; upload supports S3 or local.\n */\nimport type { DataSource } from 'typeorm';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport { MoreThanOrEqual, ILike } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport { queueEmail } from '../plugins/email/email-queue';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { queueErp } from '../plugins/erp/erp-queue';\nimport type { ERPSubmissionService } from '../plugins/erp/erp-submission';\nimport { assertCaptchaOk } from '../plugins/captcha/assert';\n\nexport type RequireEntityPermissionFn = (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n\nexport interface CmsHandlersBase {\n json: (body: unknown, init?: { status?: number }) => Response;\n requireAuth: (req: Request) => Promise<Response | null>;\n requireEntityPermission?: RequireEntityPermissionFn;\n}\n\nexport interface DashboardStatsConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n requirePermission?: (req: Request, permission: string) => Promise<Response | null>;\n}\n\nexport function createDashboardStatsHandler(config: DashboardStatsConfig) {\n const { dataSource, entityMap, json, requireAuth, requirePermission, requireEntityPermission } = config;\n return async function GET(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'dashboard', 'read');\n if (pe) return pe;\n }\n if (requirePermission) {\n const permErr = await requirePermission(req, 'view_dashboard');\n if (permErr) return permErr;\n }\n try {\n const sevenDaysAgo = new Date(Date.now() - 7 * 24 * 60 * 60 * 1000);\n const repo = (name: string) => entityMap[name] ? dataSource.getRepository(entityMap[name]) : undefined;\n const [contactsCount, formsCount, formSubmissionsCount, usersCount, blogsCount, recentContacts, recentSubmissions] = await Promise.all([\n repo('contacts')?.count() ?? 0,\n repo('forms')?.count({ where: { deleted: false } }) ?? 0,\n repo('form_submissions')?.count() ?? 0,\n repo('users')?.count({ where: { deleted: false } }) ?? 0,\n repo('blogs')?.count({ where: { deleted: false } }) ?? 0,\n repo('contacts')?.count({ where: { createdAt: MoreThanOrEqual(sevenDaysAgo) } }) ?? 0,\n repo('form_submissions')?.count({ where: { createdAt: MoreThanOrEqual(sevenDaysAgo) } }) ?? 0,\n ]);\n return json({\n contacts: { total: contactsCount, recent: recentContacts },\n forms: { total: formsCount, submissions: formSubmissionsCount, recentSubmissions },\n users: usersCount,\n blogs: blogsCount,\n });\n } catch (err) {\n return json({ error: 'Failed to fetch dashboard stats' }, { status: 500 });\n }\n };\n}\n\nexport interface AnalyticsHandlerConfig extends CmsHandlersBase {\n getAnalyticsData?: (days: number) => Promise<unknown>;\n getPropertyId?: () => ({ currentViewId?: string; [k: string]: unknown });\n getPermissions?: () => ({ serviceAccountEmail?: string; currentViewId?: string; [k: string]: unknown });\n}\n\nexport function createAnalyticsHandlers(config: AnalyticsHandlerConfig) {\n const { json, getAnalyticsData, getPropertyId, getPermissions } = config;\n return {\n async GET(req: Request): Promise<Response> {\n if (!getAnalyticsData) return json({ error: 'Analytics not configured' }, { status: 404 });\n try {\n const url = new URL(req.url);\n const days = parseInt(url.searchParams.get('days') || '30', 10);\n const data = await getAnalyticsData(days);\n return json(data);\n } catch (err: unknown) {\n const msg = err instanceof Error ? err.message : '';\n if (msg.includes('authentication credential')) return json({ error: 'Google Analytics authentication failed.' }, { status: 401 });\n if (msg.includes('sufficient permissions')) return json({ error: 'Service account does not have access.' }, { status: 403 });\n return json({ error: 'Failed to fetch analytics data' }, { status: 500 });\n }\n },\n propertyId: async (): Promise<Response> => {\n const payload = getPropertyId ? getPropertyId() : { currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID };\n return json({ message: 'Property ID Information', ...payload });\n },\n permissions: async (): Promise<Response> => {\n const payload = getPermissions ? getPermissions() : {\n serviceAccountEmail: process.env.GOOGLE_ANALYTICS_CLIENT_EMAIL,\n currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID,\n };\n return json({ message: 'Permission Troubleshooting Guide', ...payload });\n },\n };\n}\n\nimport type { StorageService } from '../plugins/storage';\n\nexport type { StorageService };\n\nexport interface UploadHandlerConfig extends CmsHandlersBase {\n /** Storage plugin instance or getter (e.g. () => getCms().then(c => c.getPlugin('storage'))). If not set, uses local fallback. */\n storage?: StorageService | (() => StorageService | undefined) | (() => Promise<StorageService | undefined>);\n /** Fallback when storage not set: dir relative to cwd (e.g. \"public/uploads\") */\n localUploadDir?: string;\n allowedTypes?: string[];\n maxSizeBytes?: number;\n}\n\nexport function createUploadHandler(config: UploadHandlerConfig) {\n const { json, requireAuth, requireEntityPermission, storage, localUploadDir = 'public/uploads', allowedTypes, maxSizeBytes = 10 * 1024 * 1024 } = config;\n const allowed = allowedTypes ?? ['image/jpeg', 'image/png', 'image/gif', 'image/webp', 'application/pdf', 'text/plain'];\n return async function POST(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'upload', 'create');\n if (pe) return pe;\n }\n try {\n const formData = await req.formData();\n const file = formData.get('file') as File | null;\n if (!file) return json({ error: 'No file uploaded' }, { status: 400 });\n if (!allowed.includes(file.type)) return json({ error: 'File type not allowed' }, { status: 400 });\n if (file.size > maxSizeBytes) return json({ error: 'File size exceeds limit' }, { status: 400 });\n const buffer = Buffer.from(await file.arrayBuffer());\n const fileName = `${Date.now()}-${file.name}`;\n const contentType = file.type || 'application/octet-stream';\n const raw = typeof storage === 'function' ? storage() : storage;\n const storageService = raw instanceof Promise ? await raw : raw;\n if (storageService) {\n const fileUrl = await storageService.upload(buffer, `uploads/${fileName}`, contentType);\n return json({ filePath: fileUrl });\n }\n const fs = await import('fs/promises');\n const path = await import('path');\n const dir = path.join(process.cwd(), localUploadDir);\n await fs.mkdir(dir, { recursive: true });\n const filePath = path.join(dir, fileName);\n await fs.writeFile(filePath, buffer);\n return json({ filePath: `/${localUploadDir.replace(/^\\/+/, '').replace(/\\\\/g, '/')}/${fileName}` });\n } catch (err) {\n return json({ error: 'File upload failed' }, { status: 500 });\n }\n };\n}\n\nexport interface BlogBySlugConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\nexport function createBlogBySlugHandler(config: BlogBySlugConfig) {\n const { dataSource, entityMap, json } = config;\n return async function GET(_req: Request, slug: string): Promise<Response> {\n try {\n const blogRepo = dataSource.getRepository(entityMap.blogs);\n const blog = await blogRepo.findOne({\n where: { slug, published: true },\n relations: ['author', 'category', 'tags', 'seo'],\n });\n if (!blog) return json({ error: 'Blog not found' }, { status: 404 });\n return json(blog);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormBySlugConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\nexport function createFormBySlugHandler(config: FormBySlugConfig) {\n const { dataSource, entityMap, json } = config;\n return async function GET(_req: Request, slug: string): Promise<Response> {\n try {\n const formRepo = dataSource.getRepository(entityMap.forms);\n const form = await formRepo.findOne({\n where: { slug, published: true, deleted: false },\n relations: ['fields'],\n order: { fields: { order: 'ASC' } },\n });\n if (!form) return json({ error: 'Form not found' }, { status: 404 });\n const out = form as { fields?: unknown[] };\n if (Array.isArray(out.fields)) out.fields = out.fields.filter((f) => !(f as { deleted?: boolean }).deleted);\n return json(form);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormSaveHandlersConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\n/** Normalize a field from payload for DB: only include columns that exist on form_fields. */\nfunction normalizeFieldRow(f: Record<string, unknown>, formId: number): Record<string, unknown> {\n const order = typeof f.order === 'number' ? f.order : Number(f.order) || 0;\n const groupId = typeof f.groupId === 'number' ? f.groupId : Number(f.groupId) || 1;\n const columnWidth = typeof f.columnWidth === 'number' ? f.columnWidth : Number(f.columnWidth) || 12;\n return {\n formId,\n label: f.label != null ? String(f.label) : '',\n type: f.type != null ? String(f.type) : 'text',\n placeholder: f.placeholder != null ? String(f.placeholder) : null,\n options: f.options != null ? (typeof f.options === 'string' ? f.options : JSON.stringify(f.options)) : null,\n required: Boolean(f.required),\n validation: f.validation != null ? (typeof f.validation === 'string' ? f.validation : JSON.stringify(f.validation)) : null,\n order,\n groupId,\n columnWidth,\n };\n}\n\nexport function createFormSaveHandlers(config: FormSaveHandlersConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n const formRepo = () => dataSource.getRepository(entityMap.forms);\n const fieldRepo = () => dataSource.getRepository(entityMap.form_fields);\n\n return {\n async GET(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'read');\n if (pe) return pe;\n }\n try {\n const formId = Number(id);\n if (!Number.isInteger(formId) || formId <= 0) return json({ error: 'Invalid form id' }, { status: 400 });\n const form = await formRepo().findOne({\n where: { id: formId },\n relations: ['fields'],\n order: { fields: { order: 'ASC' } },\n });\n if (!form) return json({ message: 'Not found' }, { status: 404 });\n const out = form as { fields?: { deleted?: boolean }[] };\n if (Array.isArray(out.fields)) out.fields = out.fields.filter((f) => !f.deleted);\n return json(form);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async POST(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'create');\n if (pe) return pe;\n }\n try {\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid request payload' }, { status: 400 });\n const fields = Array.isArray(body.fields) ? body.fields : [];\n const { fields: _f, ...formRow } = body;\n const form = await formRepo().save(formRepo().create(formRow as object));\n for (let i = 0; i < fields.length; i++) {\n const row = normalizeFieldRow(fields[i] as Record<string, unknown>, form.id);\n (row as Record<string, unknown>).order = i + 1;\n await fieldRepo().save(fieldRepo().create(row as object));\n }\n const saved = await formRepo().findOne({ where: { id: form.id }, relations: ['fields'], order: { fields: { order: 'ASC' } } });\n return json(saved ?? form, { status: 201 });\n } catch (e) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async PUT(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'update');\n if (pe) return pe;\n }\n try {\n const formId = Number(id);\n if (!Number.isInteger(formId) || formId <= 0) return json({ error: 'Invalid form id' }, { status: 400 });\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid request payload' }, { status: 400 });\n const existing = await formRepo().findOne({ where: { id: formId } });\n if (!existing) return json({ message: 'Not found' }, { status: 404 });\n const fields = Array.isArray(body.fields) ? body.fields : [];\n const formRow: Record<string, unknown> = {};\n for (const key of ['name', 'description', 'campaign', 'slug', 'published']) {\n if (body[key] !== undefined) formRow[key] = body[key];\n }\n if (Object.keys(formRow).length > 0) await formRepo().update(formId, formRow as object);\n await fieldRepo().delete({ formId });\n for (let i = 0; i < fields.length; i++) {\n const row = normalizeFieldRow(fields[i] as Record<string, unknown>, formId);\n (row as Record<string, unknown>).order = i + 1;\n await fieldRepo().save(fieldRepo().create(row as object));\n }\n const saved = await formRepo().findOne({ where: { id: formId }, relations: ['fields'], order: { fields: { order: 'ASC' } } });\n return saved ? json(saved) : json({ message: 'Not found' }, { status: 404 });\n } catch (e) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface FormSubmissionHandlerConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n /** When set, form submission notification email is queued (CRM recipient). */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n getRecipientForChannel?: (channel: 'crm' | 'sales' | 'fulfilment') => Promise<string | null>;\n}\n\nasync function isErpIntegrationEnabled(dataSource: DataSource, entityMap: EntityMap): Promise<boolean> {\n const repo = dataSource.getRepository(entityMap.configs);\n const rows = await repo.find({ where: { settings: 'erp', deleted: false } as object });\n for (const row of rows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled') return r.value !== 'false';\n }\n return true;\n}\n\n/** Form IDs that use `form.submitted` (opportunity). `null` = config key absent (treat as no IDs → all submissions use lead). */\nasync function getErpOpportunityFormIds(dataSource: DataSource, entityMap: EntityMap): Promise<number[] | null> {\n const repo = dataSource.getRepository(entityMap.configs);\n const row = await repo.findOne({\n where: { settings: 'erp', key: 'opportunityFormIds', deleted: false } as object,\n });\n if (!row) return null;\n const raw = ((row as { value: string }).value ?? '').trim();\n if (!raw) return [];\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (!Array.isArray(parsed)) return [];\n const ids = parsed\n .map((x) => (typeof x === 'number' ? x : Number(x)))\n .filter((n): n is number => Number.isInteger(n) && n > 0);\n return [...new Set(ids)];\n } catch {\n return [];\n }\n}\n\nexport interface FormSubmissionGetByIdConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n}\n\nexport function createFormSubmissionGetByIdHandler(config: FormSubmissionGetByIdConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n return async function GET(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'form_submissions', 'read');\n if (pe) return pe;\n }\n try {\n const submissionId = Number(id);\n if (!Number.isInteger(submissionId) || submissionId <= 0) return json({ error: 'Invalid id' }, { status: 400 });\n const repo = dataSource.getRepository(entityMap.form_submissions);\n const submission = await repo.findOne({\n where: { id: submissionId },\n relations: ['form', 'contact'],\n });\n if (!submission) return json({ message: 'Not found' }, { status: 404 });\n const form = submission as { form?: { id: number; fields?: unknown[] } };\n if (form.form?.id) {\n const formRepo = dataSource.getRepository(entityMap.forms);\n const formWithFields = await formRepo.findOne({\n where: { id: form.form.id },\n relations: ['fields'],\n order: { fields: { order: 'ASC' as const } },\n });\n if (formWithFields) (submission as { form?: unknown }).form = formWithFields;\n }\n return json(submission);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormSubmissionListConfig extends FormSubmissionGetByIdConfig {}\n\nexport function createFormSubmissionListHandler(config: FormSubmissionListConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n return async function GET(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'form_submissions', 'read');\n if (pe) return pe;\n }\n try {\n const repo = dataSource.getRepository(entityMap.form_submissions);\n const { searchParams } = new URL(req.url);\n const page = Number(searchParams.get('page')) || 1;\n const limit = Math.min(100, Number(searchParams.get('limit')) || 10);\n const skip = (page - 1) * limit;\n const sortField = searchParams.get('sortField') || 'createdAt';\n const sortOrder = searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const [data, total] = await repo.findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n relations: ['form', 'contact'],\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nfunction formatSubmissionFieldValue(raw: unknown): string {\n if (raw == null || raw === '') return '—';\n if (typeof raw === 'object') return JSON.stringify(raw);\n return String(raw);\n}\n\nfunction pickContactFromSubmission(\n fields: { id: number; type: string; label: string }[],\n data: Record<string, unknown>\n): { name: string; email: string; phone: string | null } | null {\n let email: string | null = null;\n let name: string | null = null;\n let phone: string | null = null;\n for (const f of fields) {\n const val = data[String(f.id)];\n if (val == null || val === '') continue;\n const str = String(val).trim();\n if (f.type === 'email' || (f.label && f.label.toLowerCase().includes('email'))) {\n if (str && !email) email = str;\n } else if (f.type === 'phone' || (f.label && f.label.toLowerCase().includes('phone'))) {\n if (str && !phone) phone = str;\n } else if (f.label && f.label.toLowerCase().includes('name') && (f.type === 'text' || !f.type)) {\n if (str && !name) name = str;\n }\n }\n if (!email) return null;\n return { name: name || email, email, phone: phone || null };\n}\n\nexport function createFormSubmissionHandler(config: FormSubmissionHandlerConfig) {\n const { dataSource, entityMap, json, getCms } = config;\n return async function POST(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') {\n return json({ error: 'Invalid request payload' }, { status: 400 });\n }\n const captchaErr = await assertCaptchaOk(getCms, body, req, json);\n if (captchaErr) return captchaErr;\n const formId = typeof body.formId === 'number' ? body.formId : Number(body.formId);\n if (!Number.isInteger(formId) || formId <= 0) {\n return json({ error: 'formId is required and must be a positive integer' }, { status: 400 });\n }\n const data = body.data;\n if (!data || typeof data !== 'object' || Array.isArray(data)) {\n return json({ error: 'data is required and must be an object' }, { status: 400 });\n }\n const formRepo = dataSource.getRepository(entityMap.forms);\n const form = await formRepo.findOne({\n where: { id: formId, published: true, deleted: false },\n relations: ['fields'],\n });\n if (!form) {\n return json({ error: 'Form not found' }, { status: 404 });\n }\n const fields = (form as { fields?: { id: number; type: string; label: string; deleted?: boolean }[] }).fields ?? [];\n const activeFields = fields.filter((f) => !f.deleted);\n\n let contactId: number | null =\n body.contactId != null && body.contactId !== ''\n ? (typeof body.contactId === 'number' ? body.contactId : Number(body.contactId))\n : null;\n\n if (!contactId) {\n const contactData = pickContactFromSubmission(activeFields, data as Record<string, unknown>);\n if (contactData) {\n const contactRepo = dataSource.getRepository(entityMap.contacts);\n let contact = await contactRepo.findOne({ where: { email: contactData.email } });\n if (!contact) {\n contact = await contactRepo.save(\n contactRepo.create({\n name: contactData.name,\n email: contactData.email,\n phone: contactData.phone,\n })\n );\n }\n contactId = contact.id;\n }\n }\n\n const ipAddress = (req.headers.get('x-forwarded-for') ?? req.headers.get('x-real-ip') ?? null) as string | null;\n const userAgent = req.headers.get('user-agent') ?? null;\n const submissionRepo = dataSource.getRepository(entityMap.form_submissions);\n const created = await submissionRepo.save(\n submissionRepo.create({\n formId,\n contactId: Number.isInteger(contactId) ? contactId : null,\n data: data as Record<string, unknown>,\n ipAddress: ipAddress?.slice(0, 255) ?? null,\n userAgent: userAgent?.slice(0, 500) ?? null,\n })\n );\n\n const formWithName = form as { name?: string };\n const formName = formWithName.name ?? 'Form';\n let contactName = 'Unknown';\n let contactEmail = '';\n if (Number.isInteger(contactId)) {\n const contactRepo = dataSource.getRepository(entityMap.contacts);\n const contact = await contactRepo.findOne({ where: { id: contactId }, select: ['name', 'email'] });\n if (contact) {\n contactName = (contact as { name: string }).name ?? contactName;\n contactEmail = (contact as { email: string }).email ?? contactEmail;\n }\n } else {\n const contactData = pickContactFromSubmission(activeFields, data as Record<string, unknown>);\n if (contactData) {\n contactName = contactData.name;\n contactEmail = contactData.email;\n }\n }\n\n if (config.getCms) {\n try {\n const cms = await config.getCms();\n if (config.getCompanyDetails && config.getRecipientForChannel) {\n const to = await config.getRecipientForChannel('crm');\n if (to) {\n const companyDetails = await config.getCompanyDetails();\n const formFieldRows = activeFields.map((f) => ({\n label: (f.label && String(f.label).trim()) || `Field ${f.id}`,\n value: formatSubmissionFieldValue(data[String(f.id)]),\n }));\n await queueEmail(cms, {\n to,\n templateName: 'formSubmission',\n ctx: {\n formName,\n contactName,\n contactEmail,\n formData: data,\n formFieldRows,\n companyDetails: companyDetails ?? {},\n },\n });\n }\n }\n if (await isErpIntegrationEnabled(dataSource, entityMap)) {\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (erp) {\n const contact = erp.submission.extractContactData(data as Record<string, unknown>, activeFields);\n if (contact?.email?.trim()) {\n const opportunityFormIds = await getErpOpportunityFormIds(dataSource, entityMap);\n const asOpportunity =\n opportunityFormIds != null &&\n opportunityFormIds.length > 0 &&\n opportunityFormIds.includes(formId);\n await queueErp(\n cms,\n asOpportunity ? { kind: 'formOpportunity', contact } : { kind: 'lead', contact }\n );\n }\n }\n }\n } catch {\n // do not fail the submission if notification / ERP queue fails\n }\n }\n\n return json(created, { status: 201 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface UsersApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n baseUrl: string;\n /** When set with email queue/plugin, invite emails are sent on user create and regenerate-invite. */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n}\n\nexport function createUsersApiHandlers(config: UsersApiConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission, baseUrl, getCms, getCompanyDetails } = config;\n\n async function trySendInviteEmail(toEmail: string, inviteLink: string, inviteeName: string): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n await queueEmail(cms, {\n to: toEmail,\n templateName: 'invite',\n ctx: {\n inviteLink,\n email: toEmail,\n inviteeName: inviteeName.trim(),\n companyDetails: companyDetails ?? {},\n },\n });\n } catch {\n // do not fail user APIs if email fails\n }\n }\n const userRepo = () => dataSource.getRepository(entityMap.users);\n return {\n async list(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'read');\n if (pe) return pe;\n }\n try {\n const url = new URL(req.url);\n const page = Math.max(1, parseInt(url.searchParams.get('page') || '1', 10));\n const limit = Math.min(100, parseInt(url.searchParams.get('limit') || '10', 10));\n const skip = (page - 1) * limit;\n const sortField = url.searchParams.get('sortField') || 'createdAt';\n const sortOrder = url.searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const search = url.searchParams.get('search');\n const where = search ? [{ name: ILike(`%${search}%`) }, { email: ILike(`%${search}%`) }] : {};\n const [data, total] = await userRepo().findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n where,\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async create(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'create');\n if (pe) return pe;\n }\n try {\n const body = await req.json() as Record<string, unknown>;\n if (!body?.name || !body?.email) return json({ error: 'Name and email are required' }, { status: 400 });\n const existing = await userRepo().findOne({ where: { email: body.email as string } });\n if (existing) return json({ error: 'User with this email already exists' }, { status: 400 });\n const groupRepo = dataSource.getRepository(entityMap.user_groups);\n const customerG = await groupRepo.findOne({ where: { name: 'Customer', deleted: false } });\n const gid = (body.groupId as number) ?? null;\n const isCustomer = !!(customerG && gid === customerG.id);\n const adminAccess = isCustomer ? false : body.adminAccess === false ? false : true;\n const newUser = await userRepo().save(\n userRepo().create({\n name: body.name,\n email: body.email,\n password: null,\n blocked: true,\n groupId: gid,\n adminAccess,\n })\n );\n if (entityMap.contacts) {\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, newUser.id, newUser.email as string);\n }\n const emailToken = Buffer.from(newUser.email).toString('base64');\n const inviteLink = `${baseUrl}/admin/invite?token=${emailToken}`;\n await trySendInviteEmail(\n newUser.email as string,\n inviteLink,\n (newUser.name as string) ?? ''\n );\n return json({ message: 'User created successfully (blocked until password is set)', user: newUser, inviteLink }, { status: 201 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async getById(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'read');\n if (pe) return pe;\n }\n try {\n const user = await userRepo().findOne({\n where: { id: parseInt(id, 10) },\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n return json(user);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async update(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'update');\n if (pe) return pe;\n }\n try {\n const body = await req.json() as Record<string, unknown>;\n const { password: _p, ...safe } = body;\n await userRepo().update(parseInt(id, 10), safe as object);\n const updated = await userRepo().findOne({\n where: { id: parseInt(id, 10) },\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n return updated ? json(updated) : json({ error: 'Not found' }, { status: 404 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async delete(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'delete');\n if (pe) return pe;\n }\n try {\n const r = await userRepo().delete(parseInt(id, 10));\n if (r.affected === 0) return json({ error: 'User not found' }, { status: 404 });\n return json({ message: 'User deleted successfully' });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async regenerateInvite(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'update');\n if (pe) return pe;\n }\n try {\n const user = await userRepo().findOne({ where: { id: parseInt(id, 10) }, select: ['email', 'name'] });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n const emailToken = Buffer.from(user.email).toString('base64');\n const inviteLink = `${baseUrl}/admin/invite?token=${emailToken}`;\n await trySendInviteEmail(user.email as string, inviteLink, (user.name as string) ?? '');\n return json({ message: 'New invite link generated successfully', inviteLink });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface UserAvatarConfig extends CmsHandlersBase {\n getSession: () => Promise<{ user?: { email?: string | null } } | null>;\n /** Save avatar (buffer, fileName) => publicUrl. Default: local public/uploads/avatars */\n saveAvatar?: (buffer: Buffer, fileName: string) => Promise<string>;\n}\n\nexport function createUserAvatarHandler(config: UserAvatarConfig) {\n const { json, getSession, saveAvatar } = config;\n return async function POST(req: Request): Promise<Response> {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n try {\n const formData = await req.formData();\n const file = formData.get('avatar') as File | null;\n if (!file) return json({ error: 'No file uploaded' }, { status: 400 });\n if (!file.type.startsWith('image/')) return json({ error: 'File must be an image' }, { status: 400 });\n if (file.size > 5 * 1024 * 1024) return json({ error: 'File size must be less than 5MB' }, { status: 400 });\n const buffer = Buffer.from(await file.arrayBuffer());\n const ext = file.name.split('.').pop() || 'jpg';\n const fileName = `avatar_${session.user.email}_${Date.now()}.${ext}`;\n const avatarUrl = saveAvatar\n ? await saveAvatar(buffer, fileName)\n : await (async () => {\n const fs = await import('fs/promises');\n const path = await import('path');\n const dir = path.join(process.cwd(), 'public', 'uploads', 'avatars');\n await fs.mkdir(dir, { recursive: true });\n await fs.writeFile(path.join(dir, fileName), buffer);\n return `/uploads/avatars/${fileName}`;\n })();\n return json({ message: 'Avatar uploaded successfully', avatarUrl });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface UserProfileConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n getSession: () => Promise<{ user?: { email?: string | null; name?: string | null } } | null>;\n}\n\nexport function createUserProfileHandler(config: UserProfileConfig) {\n const { dataSource, entityMap, json, getSession } = config;\n return async function PUT(req: Request): Promise<Response> {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n try {\n const body = await req.json() as { name?: string };\n if (!body?.name) return json({ error: 'Name is required' }, { status: 400 });\n const userRepo = dataSource.getRepository(entityMap.users);\n await userRepo.update({ email: session.user.email }, { name: body.name, updatedAt: new Date() });\n const updated = await userRepo.findOne({ where: { email: session.user.email }, select: ['id', 'name', 'email'] });\n if (!updated) return json({ error: 'Not found' }, { status: 404 });\n return json({ message: 'Profile updated successfully', user: { id: updated.id, name: updated.name, email: updated.email } });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface SettingsApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n encryptionKey?: string;\n /** Groups in this list are readable without auth (GET returns all keys for the group). */\n publicGetGroups?: string[];\n}\n\nfunction simpleEncrypt(text: string, key: string): string {\n const buf = Buffer.from(text, 'utf8');\n const keyBuf = Buffer.from(key.padEnd(32, '0').slice(0, 32), 'utf8');\n const out = Buffer.alloc(buf.length);\n for (let i = 0; i < buf.length; i++) out[i] = buf[i] ^ keyBuf[i % keyBuf.length];\n return out.toString('base64');\n}\n\nfunction simpleDecrypt(encoded: string, key: string): string {\n const buf = Buffer.from(encoded, 'base64');\n const keyBuf = Buffer.from(key.padEnd(32, '0').slice(0, 32), 'utf8');\n const out = Buffer.alloc(buf.length);\n for (let i = 0; i < buf.length; i++) out[i] = buf[i] ^ keyBuf[i % keyBuf.length];\n return out.toString('utf8');\n}\n\n/**\n * Structural types so apps with their own `typeorm` install (e.g. npm link) typecheck without duplicate-package errors.\n */\nexport type GetPublicSettingsGroupDataSource = {\n getRepository(entity: unknown): {\n find(options: object): Promise<unknown[]>;\n };\n};\n\nexport interface GetPublicSettingsGroupConfig {\n dataSource: GetPublicSettingsGroupDataSource;\n entityMap: Record<string, unknown>;\n encryptionKey?: string;\n}\n\n/** Same rows as unauthenticated GET /api/settings/:group when the group is in publicGetGroups. */\nexport async function getPublicSettingsGroup(\n config: GetPublicSettingsGroupConfig,\n group: string\n): Promise<Record<string, string>> {\n const { dataSource, entityMap, encryptionKey } = config;\n const repo = dataSource.getRepository(entityMap.configs);\n const rows = await repo.find({ where: { settings: group, deleted: false } });\n const result: Record<string, string> = {};\n for (const row of rows) {\n const r = row as { key: string; value: string; encrypted?: boolean };\n let val = r.value;\n if (r.encrypted && encryptionKey) {\n try {\n val = simpleDecrypt(val, encryptionKey);\n } catch {\n /* keep raw if decrypt fails */\n }\n }\n result[r.key] = val;\n }\n return result;\n}\n\nexport function createSettingsApiHandlers(config: SettingsApiConfig) {\n const { dataSource, entityMap, json, requireAuth, encryptionKey, publicGetGroups } = config;\n const configRepo = () => dataSource.getRepository(entityMap.configs);\n\n return {\n async GET(req: Request, group: string): Promise<Response> {\n const isPublicGroup = publicGetGroups?.includes(group);\n const authErr = isPublicGroup ? null : await requireAuth(req);\n const isAuthed = !authErr;\n\n try {\n if (isPublicGroup) {\n const result = await getPublicSettingsGroup(\n { dataSource, entityMap, encryptionKey },\n group\n );\n return json(result);\n }\n\n const where: Record<string, unknown> = { settings: group, deleted: false };\n if (!isAuthed && !isPublicGroup) where.type = 'public';\n\n const rows = await configRepo().find({ where });\n const result: Record<string, unknown> = {};\n for (const row of rows) {\n const r = row as { key: string; value: string; encrypted?: boolean; type?: string };\n let val = r.value;\n if (r.encrypted && encryptionKey) {\n try { val = simpleDecrypt(val, encryptionKey); } catch { /* return raw if decrypt fails */ }\n }\n result[r.key] = val;\n }\n return json(result);\n } catch {\n return json({ error: 'Failed to fetch settings' }, { status: 500 });\n }\n },\n\n async PUT(req: Request, group: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n\n try {\n const body = await req.json() as Record<string, { value: string; type?: 'public' | 'private'; encrypted?: boolean }>;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid payload' }, { status: 400 });\n\n const repo = configRepo();\n for (const [key, entry] of Object.entries(body)) {\n const val = typeof entry === 'string' ? entry : entry.value;\n const type = (typeof entry === 'object' && entry.type) || 'private';\n const encrypted = !!(typeof entry === 'object' && entry.encrypted);\n\n let storedValue = val;\n if (encrypted && encryptionKey) {\n storedValue = simpleEncrypt(val, encryptionKey);\n }\n\n const existing = await repo.findOne({ where: { settings: group, key } });\n if (existing) {\n await repo.update(existing.id, { value: storedValue, type, encrypted, updatedAt: new Date() } as object);\n } else {\n await repo.save(repo.create({ settings: group, key, value: storedValue, type, encrypted } as object));\n }\n }\n return json({ message: 'Settings saved' });\n } catch {\n return json({ error: 'Failed to save settings' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface ChatApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n getCms: () => Promise<{ getPlugin: (name: string) => unknown }>;\n}\n\nconst KB_CHUNK_LIMIT = 10;\nconst KB_CONTEXT_MAX_CHARS = 4000;\n\nfunction getQueryTerms(message: string): string[] {\n return message\n .replace(/[^\\w\\s]/g, ' ')\n .split(/\\s+/)\n .filter((w) => w.length > 2)\n .slice(0, 6);\n}\n\nexport function createChatHandlers(config: ChatApiConfig) {\n const { dataSource, entityMap, json, getCms } = config;\n const contactRepo = () => dataSource.getRepository(entityMap.contacts);\n const convRepo = () => dataSource.getRepository(entityMap.chat_conversations);\n const msgRepo = () => dataSource.getRepository(entityMap.chat_messages);\n const chunkRepo = () => dataSource.getRepository(entityMap.knowledge_base_chunks);\n\n return {\n async identify(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as { name?: string; email?: string; phone?: string };\n const name = body?.name?.trim();\n const email = body?.email?.trim();\n if (!name || !email) return json({ error: 'name and email required' }, { status: 400 });\n const repo = contactRepo();\n let contact = await repo.findOne({ where: { email, deleted: false } });\n if (!contact) {\n const created = repo.create({ name, email, phone: body.phone?.trim() || null } as object);\n contact = await repo.save(created);\n }\n const convRepoInst = convRepo();\n const conv = await convRepoInst.save(convRepoInst.create({ contactId: (contact as { id: number }).id } as object));\n return json({\n contactId: (contact as { id: number }).id,\n conversationId: (conv as { id: number }).id,\n });\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Failed to identify';\n return json({ error: 'Failed to identify', detail: message }, { status: 500 });\n }\n },\n async getMessages(req: Request, conversationId: string): Promise<Response> {\n try {\n const conv = await convRepo().findOne({\n where: { id: parseInt(conversationId, 10) },\n relations: ['messages'],\n });\n if (!conv) return json({ error: 'Conversation not found' }, { status: 404 });\n const messages = ((conv as { messages?: Array<{ role: string; content: string; createdAt: string }> }).messages ?? [])\n .sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime())\n .map((m) => ({ role: m.role, content: m.content }));\n return json({ messages });\n } catch {\n return json({ error: 'Failed to fetch messages' }, { status: 500 });\n }\n },\n async postMessage(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as { conversationId?: number; message?: string };\n const conversationId = body?.conversationId;\n const message = body?.message?.trim();\n if (!conversationId || !message) return json({ error: 'conversationId and message required' }, { status: 400 });\n const conv = await convRepo().findOne({\n where: { id: conversationId },\n relations: ['messages'],\n });\n if (!conv) return json({ error: 'Conversation not found' }, { status: 404 });\n const msgRepoInst = msgRepo();\n await msgRepoInst.save(msgRepoInst.create({ conversationId, role: 'user', content: message } as object));\n const cms = await getCms();\n const llm = cms.getPlugin('llm') as {\n chat: (messages: Array<{ role: string; content: string }>, opts?: object) => Promise<{ content: string }>;\n embed?: (text: string) => Promise<number[]>;\n } | undefined;\n if (!llm?.chat) return json({ error: 'LLM not configured' }, { status: 503 });\n let contextParts: string[] = [];\n const queryEmbedding = llm.embed ? await llm.embed(message) : null;\n if (queryEmbedding && queryEmbedding.length > 0) {\n const vectorStr = '[' + queryEmbedding.join(',') + ']';\n try {\n const rows = (await dataSource.query(\n `SELECT id, content FROM knowledge_base_chunks WHERE embedding IS NOT NULL ORDER BY embedding <=> $1::vector LIMIT $2`,\n [vectorStr, KB_CHUNK_LIMIT]\n )) as Array<{ id: number; content: string }>;\n let totalLen = 0;\n for (const r of rows) {\n const text = (r.content ?? '').trim();\n if (!text || totalLen + text.length > KB_CONTEXT_MAX_CHARS) continue;\n contextParts.push(text);\n totalLen += text.length;\n }\n } catch {\n // pgvector not available or query failed; fall back to keyword\n }\n }\n if (contextParts.length === 0) {\n const terms = getQueryTerms(message);\n if (terms.length > 0) {\n const conditions = terms.map((t) => ({ content: ILike(`%${t}%`) }));\n const chunks = await chunkRepo().find({\n where: conditions,\n take: KB_CHUNK_LIMIT,\n order: { id: 'ASC' },\n });\n const seen = new Set<string>();\n let totalLen = 0;\n for (const c of chunks as Array<{ content: string }>) {\n const text = c.content.trim();\n if (seen.has(text) || totalLen + text.length > KB_CONTEXT_MAX_CHARS) continue;\n seen.add(text);\n contextParts.push(text);\n totalLen += text.length;\n }\n }\n }\n const history = ((conv as { messages?: Array<{ role: string; content: string; createdAt?: string }> }).messages ?? [])\n .sort((a, b) => new Date(a.createdAt ?? 0).getTime() - new Date(b.createdAt ?? 0).getTime())\n .map((m) => ({ role: m.role as 'user' | 'assistant' | 'system', content: m.content }));\n const systemContent = contextParts.length > 0\n ? `Use the following context about the company and its products to answer. If the answer is not in the context, say so.\\n\\nContext:\\n${contextParts.join('\\n\\n')}`\n : 'You are a helpful assistant for the company. If you do not have specific information, say so.';\n const messages = [\n { role: 'system' as const, content: systemContent },\n ...history,\n { role: 'user' as const, content: message },\n ];\n const { content } = await llm.chat(messages);\n await msgRepoInst.save(msgRepoInst.create({ conversationId, role: 'assistant', content } as object));\n return json({ content });\n } catch (err) {\n const msg = err instanceof Error ? err.message : '';\n return json({ error: msg || 'Failed to send message' }, { status: 500 });\n }\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport { SMS_MESSAGE_TEMPLATE_DEFAULTS, getSmsTemplateDefault } from '../message-templates/sms-defaults';\nimport type { RequireEntityPermissionFn } from './cms-handlers';\n\nexport interface MessageTemplateAdminHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n requireAuth: (req: Request) => Promise<Response | null>;\n requireEntityPermission?: RequireEntityPermissionFn;\n}\n\nexport function createSmsMessageTemplateHandlers(config: MessageTemplateAdminHandlersConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n\n const repo = () => dataSource.getRepository(entityMap.message_templates);\n\n async function requireSettingsRead(req: Request): Promise<Response | null> {\n const a = await requireAuth(req);\n if (a) return a;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'settings', 'read');\n if (pe) return pe;\n }\n return null;\n }\n\n async function requireSettingsUpdate(req: Request): Promise<Response | null> {\n const a = await requireAuth(req);\n if (a) return a;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'settings', 'update');\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request): Promise<Response> {\n const err = await requireSettingsRead(req);\n if (err) return err;\n try {\n const rows = await repo().find({ where: { channel: 'sms', deleted: false } as object });\n const byKey = new Map(rows.map((r) => [r.templateKey, r]));\n const items = SMS_MESSAGE_TEMPLATE_DEFAULTS.map((def) => {\n const row = byKey.get(def.templateKey);\n return {\n templateKey: def.templateKey,\n name: def.name,\n defaultBody: def.body,\n body: row?.body?.trim() ? row.body : def.body,\n externalTemplateRef: row?.externalTemplateRef?.trim() ?? '',\n otpVarKey:\n row?.providerMeta && typeof row.providerMeta.otpVarKey === 'string'\n ? String(row.providerMeta.otpVarKey)\n : def.providerMeta?.otpVarKey ?? 'var1',\n enabled: row ? row.enabled : false,\n dbId: row?.id ?? null,\n };\n });\n return json({ items });\n } catch {\n return json({ error: 'Failed to load templates' }, { status: 500 });\n }\n },\n\n async PUT(req: Request): Promise<Response> {\n const err = await requireSettingsUpdate(req);\n if (err) return err;\n try {\n const raw = (await req.json().catch(() => null)) as {\n items?: Array<{\n templateKey?: string;\n body?: string;\n externalTemplateRef?: string;\n otpVarKey?: string;\n enabled?: boolean;\n }>;\n } | null;\n if (!raw?.items || !Array.isArray(raw.items)) {\n return json({ error: 'Invalid payload' }, { status: 400 });\n }\n\n for (const item of raw.items) {\n const templateKey = typeof item.templateKey === 'string' ? item.templateKey.trim() : '';\n if (!getSmsTemplateDefault(templateKey)) continue;\n\n const body = typeof item.body === 'string' ? item.body : '';\n const externalTemplateRef =\n typeof item.externalTemplateRef === 'string' ? item.externalTemplateRef.trim() : '';\n const otpVarKey =\n typeof item.otpVarKey === 'string' && item.otpVarKey.trim()\n ? item.otpVarKey.trim()\n : 'var1';\n const enabled = item.enabled !== false;\n\n const existing = await repo().findOne({\n where: { channel: 'sms', templateKey, deleted: false } as object,\n });\n const def = getSmsTemplateDefault(templateKey)!;\n const providerMeta = { otpVarKey };\n\n if (existing) {\n await repo().update(existing.id, {\n name: def.name,\n body,\n externalTemplateRef: externalTemplateRef || null,\n providerMeta,\n enabled,\n updatedAt: new Date(),\n } as object);\n } else {\n await repo().save(\n repo().create({\n channel: 'sms',\n templateKey,\n name: def.name,\n subject: null,\n body,\n externalTemplateRef: externalTemplateRef || null,\n providerMeta,\n enabled,\n deleted: false,\n } as object)\n );\n }\n }\n return json({ ok: true });\n } catch {\n return json({ error: 'Failed to save templates' }, { status: 500 });\n }\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport { getPermissionableEntityKeys, isSuperAdminGroupName } from '../auth/permission-entities';\nimport type { SessionUser } from '../auth/helpers';\nimport { canManageRoles } from '../auth/helpers';\n\nexport type GetSessionUser = () => Promise<SessionUser | null>;\n\nexport interface AdminRolesHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n getSessionUser: GetSessionUser;\n}\n\nexport function createAdminRolesHandlers(config: AdminRolesHandlersConfig) {\n const { dataSource, entityMap, json, getSessionUser } = config;\n const baseEntities = getPermissionableEntityKeys(entityMap as Record<string, unknown>);\n const allowEntities = new Set([...baseEntities, 'users']);\n const groupRepo = () => dataSource.getRepository(entityMap.user_groups);\n const permRepo = () => dataSource.getRepository(entityMap.permissions);\n const userRepo = () => dataSource.getRepository(entityMap.users);\n\n async function gate(): Promise<Response | null> {\n const u = await getSessionUser();\n if (!u?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n if (!canManageRoles(u)) return json({ error: 'Forbidden' }, { status: 403 });\n return null;\n }\n\n return {\n async list(): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const groups = await groupRepo().find({\n where: { deleted: false },\n order: { id: 'ASC' },\n relations: ['permissions'],\n });\n const entities = [...allowEntities].sort();\n return json({\n entities,\n groups: groups.map((g) => ({\n id: g.id,\n name: g.name,\n permissions: (g.permissions ?? [])\n .filter((p) => !p.deleted)\n .map((p) => ({\n entity: p.entity,\n canCreate: p.canCreate,\n canRead: p.canRead,\n canUpdate: p.canUpdate,\n canDelete: p.canDelete,\n })),\n })),\n });\n },\n\n async createGroup(req: Request): Promise<Response> {\n const err = await gate();\n if (err) return err;\n try {\n const body = (await req.json()) as { name?: string };\n const name = body?.name?.trim();\n if (!name) return json({ error: 'Name is required' }, { status: 400 });\n const repo = groupRepo();\n const existing = await repo.findOne({ where: { name } });\n if (existing) return json({ error: 'Group name already exists' }, { status: 400 });\n const g = await repo.save(repo.create({ name }));\n return json({ id: g.id, name: g.name, permissions: [] }, { status: 201 });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n\n async patchGroup(req: Request, idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const id = parseInt(idStr, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n try {\n const body = (await req.json()) as { name?: string };\n const name = body?.name?.trim();\n if (!name) return json({ error: 'Name is required' }, { status: 400 });\n const repo = groupRepo();\n const g = await repo.findOne({ where: { id, deleted: false } });\n if (!g) return json({ error: 'Not found' }, { status: 404 });\n if (isSuperAdminGroupName(g.name) && !isSuperAdminGroupName(name)) {\n return json({ error: 'Cannot rename the administrator group' }, { status: 400 });\n }\n const dup = await repo.findOne({ where: { name } });\n if (dup && dup.id !== id) return json({ error: 'Name already in use' }, { status: 400 });\n g.name = name;\n await repo.save(g);\n return json({ id: g.id, name: g.name });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n\n async deleteGroup(idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const id = parseInt(idStr, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const repo = groupRepo();\n const g = await repo.findOne({ where: { id, deleted: false } });\n if (!g) return json({ error: 'Not found' }, { status: 404 });\n if (isSuperAdminGroupName(g.name)) return json({ error: 'Cannot delete the administrator group' }, { status: 400 });\n const userCount = await userRepo().count({ where: { groupId: id } });\n if (userCount > 0) return json({ error: 'Reassign users before deleting this group' }, { status: 409 });\n await permRepo().delete({ groupId: id });\n await repo.update(id, { deleted: true, deletedAt: new Date() } as object);\n return json({ ok: true });\n },\n\n async putPermissions(req: Request, idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const groupId = parseInt(idStr, 10);\n if (!Number.isFinite(groupId)) return json({ error: 'Invalid id' }, { status: 400 });\n const groupRepository = groupRepo();\n const g = await groupRepository.findOne({ where: { id: groupId, deleted: false } });\n if (!g) return json({ error: 'Group not found' }, { status: 404 });\n try {\n const body = (await req.json()) as {\n permissions?: Array<{\n entity: string;\n canCreate?: boolean;\n canRead?: boolean;\n canUpdate?: boolean;\n canDelete?: boolean;\n }>;\n };\n const rows = body?.permissions;\n if (!Array.isArray(rows)) return json({ error: 'permissions array required' }, { status: 400 });\n for (const r of rows) {\n if (!r?.entity || !allowEntities.has(r.entity)) {\n return json({ error: `Invalid entity: ${r?.entity ?? ''}` }, { status: 400 });\n }\n }\n await dataSource.transaction(async (em) => {\n await em.getRepository(entityMap.permissions).delete({ groupId });\n for (const r of rows) {\n await em.getRepository(entityMap.permissions).save(\n em.getRepository(entityMap.permissions).create({\n groupId,\n entity: r.entity,\n canCreate: !!r.canCreate,\n canRead: !!r.canRead,\n canUpdate: !!r.canUpdate,\n canDelete: !!r.canDelete,\n })\n );\n }\n });\n const updated = await groupRepository.findOne({\n where: { id: groupId },\n relations: ['permissions'],\n });\n return json({\n id: groupId,\n permissions: (updated?.permissions ?? []).map((p) => ({\n entity: p.entity,\n canCreate: p.canCreate,\n canRead: p.canRead,\n canUpdate: p.canUpdate,\n canDelete: p.canDelete,\n })),\n });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n };\n}\n","/**\n * Single CMS API handler: dashboard, analytics, upload, blog/form by slug, users API, user auth, CRUD. Mount once (e.g. app/api/[[...path]]/route.ts).\n */\nimport type { DataSource } from 'typeorm';\nimport { createCrudHandler, createCrudByIdHandler } from './crud';\nimport type { CrudHandlerOptions, EntityMap } from './crud';\nimport { createUserAuthApiRouter } from './auth-handlers';\nimport type { UserAuthApiConfig } from './auth-handlers';\nimport {\n createDashboardStatsHandler,\n createAnalyticsHandlers,\n createUploadHandler,\n createBlogBySlugHandler,\n createFormBySlugHandler,\n createFormSaveHandlers,\n createFormSubmissionHandler,\n createFormSubmissionGetByIdHandler,\n createFormSubmissionListHandler,\n createUsersApiHandlers,\n createUserAvatarHandler,\n createUserProfileHandler,\n createSettingsApiHandlers,\n createChatHandlers,\n} from './cms-handlers';\nimport { createSmsMessageTemplateHandlers } from './message-template-admin-handlers';\nimport type {\n DashboardStatsConfig,\n AnalyticsHandlerConfig,\n UploadHandlerConfig,\n BlogBySlugConfig,\n FormBySlugConfig,\n FormSaveHandlersConfig,\n FormSubmissionHandlerConfig,\n FormSubmissionGetByIdConfig,\n UsersApiConfig,\n UserAvatarConfig,\n UserProfileConfig,\n SettingsApiConfig,\n ChatApiConfig,\n} from './cms-handlers';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport type { SessionUser } from '../auth/helpers';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { createAdminRolesHandlers } from './admin-roles-handlers';\n\n/** CMS instance with getPlugin; when provided, analytics and userAuth.sendEmail can be resolved from plugins when not passed. */\nexport type CmsGetter = () => Promise<{ getPlugin: (name: string) => unknown }>;\n\nexport interface CmsApiHandlerConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n requireAuth: (req: Request) => Promise<Response | null>;\n json: (body: unknown, init?: { status?: number }) => Response;\n pathToModel?: (segment: string) => string;\n crudResources?: string[];\n /** When set, analytics and userAuth.sendEmail can be derived from getPlugin('analytics') and getPlugin('email') when not provided. */\n getCms?: CmsGetter;\n /** Optional: used when deriving userAuth.sendEmail to pass company details into email templates (e.g. from settings). */\n getCompanyDetails?: () => Promise<CompanyDetails>;\n /** Optional: used for form submission and other channel-based email recipients (from settings group \"email\"). */\n getRecipientForChannel?: (channel: 'crm' | 'sales' | 'fulfilment') => Promise<string | null>;\n userAuth?: UserAuthApiConfig;\n /** GET /api/dashboard/stats */\n dashboard?: DashboardStatsConfig;\n /** GET /api/analytics. If omitted and getCms is set, uses getCms().getPlugin('analytics'). */\n analytics?: AnalyticsHandlerConfig;\n /** POST /api/upload (S3 or local) */\n upload?: UploadHandlerConfig;\n /** GET /api/blogs/slug/:slug (public) */\n blogBySlug?: BlogBySlugConfig;\n /** GET /api/forms/slug/:slug (public) */\n formBySlug?: FormBySlugConfig;\n /** POST/PUT /api/forms (save form + fields; when set, overrides CRUD for forms) */\n formSave?: FormSaveHandlersConfig;\n /** POST /api/form-submissions (public, no auth) */\n formSubmission?: FormSubmissionHandlerConfig;\n /** GET /api/form-submissions/:id (auth) with form + contact relations */\n formSubmissionGetById?: FormSubmissionGetByIdConfig;\n /** GET/POST /api/users, GET/PUT/DELETE /api/users/:id, POST /api/users/:id/regenerate-invite */\n usersApi?: UsersApiConfig;\n /** POST /api/users/avatar */\n userAvatar?: UserAvatarConfig;\n /** PUT /api/users/profile */\n userProfile?: UserProfileConfig;\n /** GET/PUT /api/settings/:group */\n settings?: SettingsApiConfig;\n /** POST /api/chat/identify, GET /api/chat/conversations/:id/messages, POST /api/chat/messages */\n chat?: ChatApiConfig;\n /** When set, CRUD and admin routes enforce entity-level permissions from session */\n requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n /** Required for GET/POST/PATCH/DELETE /api/admin/roles */\n getSessionUser?: () => Promise<SessionUser | null>;\n}\n\nconst DEFAULT_EXCLUDE = new Set([\n 'users',\n 'password_reset_tokens',\n 'user_groups',\n 'permissions',\n 'comments',\n 'form_fields',\n 'configs',\n 'carts',\n 'cart_items',\n 'wishlists',\n 'wishlist_items',\n 'message_templates',\n]);\n\nexport function createCmsApiHandler(config: CmsApiHandlerConfig) {\n const {\n dataSource,\n entityMap,\n pathToModel = (s) => s,\n crudResources = Object.keys(entityMap).filter((k) => !DEFAULT_EXCLUDE.has(k)),\n getCms,\n userAuth: userAuthConfig,\n dashboard,\n analytics: analyticsConfig,\n upload,\n blogBySlug,\n formBySlug,\n formSave: formSaveConfig,\n formSubmission: formSubmissionConfig,\n formSubmissionGetById: formSubmissionGetByIdConfig,\n usersApi,\n userAvatar,\n userProfile,\n settings: settingsConfig,\n chat: chatConfig,\n requireEntityPermission: reqEntityPerm,\n getSessionUser,\n } = config;\n\n const analytics: AnalyticsHandlerConfig | undefined =\n analyticsConfig ??\n (getCms\n ? {\n json: config.json,\n requireAuth: async () => null,\n getAnalyticsData: async (days) => {\n const cms = await getCms();\n const a = cms.getPlugin('analytics') as { getAnalyticsData?: (days: number) => Promise<unknown> } | undefined;\n if (!a?.getAnalyticsData) throw new Error('Analytics not configured');\n return a.getAnalyticsData(days);\n },\n getPropertyId: () => ({ currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID }),\n getPermissions: () => ({\n serviceAccountEmail: process.env.GOOGLE_ANALYTICS_CLIENT_EMAIL,\n currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID,\n }),\n }\n : undefined);\n\n const userAuth: UserAuthApiConfig | undefined =\n userAuthConfig && getCms && userAuthConfig.sendEmail === undefined\n ? {\n ...userAuthConfig,\n sendEmail: async (opts) => {\n const cms = await getCms();\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n const companyDetails = config.getCompanyDetails ? await config.getCompanyDetails() : {};\n const resetLink =\n (typeof opts.resetLink === 'string' && opts.resetLink.trim()) ||\n (typeof opts.text === 'string' && opts.text.trim()) ||\n (typeof opts.html === 'string' ? (opts.html.match(/href\\s*=\\s*[\"']([^\"']+)[\"']/)?.[1] ?? '') : '');\n const ctx = { resetLink, companyDetails };\n if (queue) {\n const { queueEmail } = await import('../plugins/email/email-queue');\n await queueEmail(cms as { getPlugin: (name: string) => unknown }, { to: opts.to, templateName: 'passwordReset', ctx });\n return;\n }\n const email = cms.getPlugin('email') as { send: (data: { subject: string; html: string; text?: string; to?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (!email?.send) return;\n const rendered = email.renderTemplate('passwordReset', ctx);\n await email.send({ subject: rendered.subject, html: rendered.html, text: rendered.text, to: opts.to });\n },\n }\n : userAuthConfig;\n\n const crudOpts: CrudHandlerOptions = {\n requireAuth: config.requireAuth,\n json: config.json,\n requireEntityPermission: reqEntityPerm,\n getCms,\n };\n const crud = createCrudHandler(dataSource, entityMap, crudOpts);\n const crudById = createCrudByIdHandler(dataSource, entityMap, crudOpts);\n\n const mergePerm = <T extends object>(c: T | undefined): T | undefined =>\n !c ? undefined : reqEntityPerm ? ({ ...c, requireEntityPermission: reqEntityPerm } as T) : c;\n\n const adminRoles =\n getSessionUser &&\n createAdminRolesHandlers({\n dataSource,\n entityMap,\n json: config.json,\n getSessionUser,\n });\n const userAuthRouter = userAuth ? createUserAuthApiRouter(userAuth) : null;\n\n const dashboardGet = dashboard ? createDashboardStatsHandler(mergePerm(dashboard) ?? dashboard) : null;\n const analyticsHandlers = analytics ? createAnalyticsHandlers(analytics) : null;\n const uploadPost = upload ? createUploadHandler(mergePerm(upload) ?? upload) : null;\n const blogBySlugGet = blogBySlug ? createBlogBySlugHandler(blogBySlug) : null;\n const formBySlugGet = formBySlug ? createFormBySlugHandler(formBySlug) : null;\n const formSaveHandlers = formSaveConfig ? createFormSaveHandlers(mergePerm(formSaveConfig) ?? formSaveConfig) : null;\n const formSubmissionPost = formSubmissionConfig ? createFormSubmissionHandler(formSubmissionConfig) : null;\n const formSubmissionGetById = formSubmissionGetByIdConfig\n ? createFormSubmissionGetByIdHandler(mergePerm(formSubmissionGetByIdConfig) ?? formSubmissionGetByIdConfig)\n : null;\n const formSubmissionList = formSubmissionGetByIdConfig\n ? createFormSubmissionListHandler(mergePerm(formSubmissionGetByIdConfig) ?? formSubmissionGetByIdConfig)\n : null;\n const usersApiMerged =\n usersApi && getCms\n ? {\n ...usersApi,\n getCms: usersApi.getCms ?? getCms,\n getCompanyDetails: usersApi.getCompanyDetails ?? config.getCompanyDetails,\n }\n : usersApi;\n const usersHandlers = usersApiMerged ? createUsersApiHandlers(mergePerm(usersApiMerged) ?? usersApiMerged) : null;\n const avatarPost = userAvatar ? createUserAvatarHandler(userAvatar) : null;\n const profilePut = userProfile ? createUserProfileHandler(userProfile) : null;\n const settingsHandlers = settingsConfig ? createSettingsApiHandlers(settingsConfig) : null;\n const smsMessageTemplateHandlers = createSmsMessageTemplateHandlers({\n dataSource,\n entityMap,\n json: config.json,\n requireAuth: config.requireAuth,\n requireEntityPermission: reqEntityPerm,\n });\n const chatHandlers = chatConfig ? createChatHandlers(chatConfig) : null;\n\n function resolveResource(segment: string): string {\n const model = pathToModel(segment);\n return crudResources.includes(model) ? model : segment;\n }\n\n return {\n async handle(method: string, path: string[], req: Request): Promise<Response> {\n const perm = reqEntityPerm;\n async function analyticsGate(): Promise<Response | null> {\n const a = await config.requireAuth(req);\n if (a) return a;\n if (perm) return perm(req, 'analytics', 'read');\n return null;\n }\n\n // Admin roles: [\"admin\", \"roles\"], [\"admin\", \"roles\", id], [\"admin\", \"roles\", id, \"permissions\"]\n if (path[0] === 'admin' && path[1] === 'roles') {\n if (!adminRoles) return config.json({ error: 'Not found' }, { status: 404 });\n if (path.length === 2 && method === 'GET') return adminRoles.list();\n if (path.length === 2 && method === 'POST') return adminRoles.createGroup(req);\n if (path.length === 3 && method === 'PATCH') return adminRoles.patchGroup(req, path[2]!);\n if (path.length === 3 && method === 'DELETE') return adminRoles.deleteGroup(path[2]!);\n if (path.length === 4 && path[3] === 'permissions' && method === 'PUT') return adminRoles.putPermissions(req, path[2]!);\n return config.json({ error: 'Not found' }, { status: 404 });\n }\n\n // Dashboard: [\"dashboard\", \"stats\"]\n if (path[0] === 'dashboard' && path[1] === 'stats' && path.length === 2 && method === 'GET' && dashboardGet) {\n return dashboardGet(req);\n }\n // Analytics: [\"analytics\"], [\"analytics\", \"property-id\"], [\"analytics\", \"permissions\"]\n if (path[0] === 'analytics' && analyticsHandlers) {\n if (path.length === 1 && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.GET(req);\n }\n if (path.length === 2 && path[1] === 'property-id' && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.propertyId();\n }\n if (path.length === 2 && path[1] === 'permissions' && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.permissions();\n }\n }\n // Upload: [\"upload\"]\n if (path[0] === 'upload' && path.length === 1 && method === 'POST' && uploadPost) return uploadPost(req);\n // Blog by slug: [\"blogs\", \"slug\", slug] (public)\n if (path[0] === 'blogs' && path[1] === 'slug' && path.length === 3 && method === 'GET' && blogBySlugGet) {\n return blogBySlugGet(req, path[2]);\n }\n // Form by slug: [\"forms\", \"slug\", slug] (public)\n if (path[0] === 'forms' && path[1] === 'slug' && path.length === 3 && method === 'GET' && formBySlugGet) {\n return formBySlugGet(req, path[2]);\n }\n // Form submission: [\"form-submissions\"] GET (auth, list with relations) or POST (public); [\"form-submissions\", id] GET (auth, with relations)\n if (path[0] === 'form-submissions') {\n if (path.length === 1) {\n if (method === 'GET' && formSubmissionList) return formSubmissionList(req);\n if (method === 'POST' && formSubmissionPost) return formSubmissionPost(req);\n }\n if (path.length === 2 && method === 'GET' && formSubmissionGetById) return formSubmissionGetById(req, path[1]);\n }\n // Form save: [\"forms\"] POST; [\"forms\", id] GET (with fields) or PUT/PATCH (saves form + fields)\n if (path[0] === 'forms' && formSaveHandlers) {\n if (path.length === 1 && method === 'POST') return formSaveHandlers.POST(req);\n if (path.length === 2) {\n if (method === 'GET') return formSaveHandlers.GET(req, path[1]);\n if (method === 'PUT' || method === 'PATCH') return formSaveHandlers.PUT(req, path[1]);\n }\n }\n // Users API\n if (path[0] === 'users' && usersHandlers) {\n if (path.length === 1) {\n if (method === 'GET') return usersHandlers.list(req);\n if (method === 'POST') return usersHandlers.create(req);\n }\n if (path.length === 2) {\n if (path[1] === 'avatar' && method === 'POST' && avatarPost) return avatarPost(req);\n if (path[1] === 'profile' && method === 'PUT' && profilePut) return profilePut(req);\n // users/:id\n const id = path[1];\n if (method === 'GET') return usersHandlers.getById(req, id);\n if (method === 'PUT' || method === 'PATCH') return usersHandlers.update(req, id);\n if (method === 'DELETE') return usersHandlers.delete(req, id);\n }\n if (path.length === 3 && path[2] === 'regenerate-invite' && method === 'POST') {\n return usersHandlers.regenerateInvite(req, path[1]);\n }\n }\n // User auth: [\"users\", \"forgot-password\"] etc.\n if (path[0] === 'users' && path.length === 2 && userAuthRouter && method === 'POST') {\n return userAuthRouter.POST(req, path[1]);\n }\n // Settings: [\"settings\", group]\n if (path[0] === 'settings' && path.length === 2 && settingsHandlers) {\n const group = path[1]!;\n const isPublic = settingsConfig?.publicGetGroups?.includes(group);\n if (method === 'GET') {\n if (!isPublic && perm) {\n const a = await config.requireAuth(req);\n if (a) return a;\n const pe = await perm(req, 'settings', 'read');\n if (pe) return pe;\n }\n return settingsHandlers.GET(req, group);\n }\n if (method === 'PUT') {\n if (perm) {\n const pe = await perm(req, 'settings', 'update');\n if (pe) return pe;\n }\n return settingsHandlers.PUT(req, group);\n }\n }\n // SMS message templates (plugins page only): [\"message-templates\", \"sms\"]\n if (path[0] === 'message-templates' && path[1] === 'sms' && path.length === 2) {\n if (method === 'GET') return smsMessageTemplateHandlers.GET(req);\n if (method === 'PUT') return smsMessageTemplateHandlers.PUT(req);\n }\n // Chat: [\"chat\", \"identify\"] POST; [\"chat\", \"conversations\", id, \"messages\"] GET; [\"chat\", \"messages\"] POST\n if (path[0] === 'chat' && chatHandlers) {\n if (path.length === 2 && path[1] === 'identify' && method === 'POST') return chatHandlers.identify(req);\n if (path.length === 4 && path[1] === 'conversations' && path[3] === 'messages' && method === 'GET') return chatHandlers.getMessages(req, path[2]);\n if (path.length === 2 && path[1] === 'messages' && method === 'POST') return chatHandlers.postMessage(req);\n }\n\n // Invoice PDF: [\"orders\", id, \"invoice\"] (admin auth + orders:read)\n if (path[0] === 'orders' && path.length === 3 && path[2] === 'invoice' && method === 'GET' && getCms) {\n const a = await config.requireAuth(req);\n if (a) return a;\n if (perm) {\n const pe = await perm(req, 'orders', 'read');\n if (pe) return pe;\n }\n const cms = await getCms();\n const { streamOrderInvoicePdf } = await import('../plugins/erp/erp-order-invoice');\n const oid = Number(path[1]);\n if (!Number.isFinite(oid)) return config.json({ error: 'Invalid id' }, { status: 400 });\n return streamOrderInvoicePdf(cms, dataSource, entityMap, oid, {});\n }\n\n // CRUD: [\"blogs\"] or [\"blogs\", \"123\"]\n if (path.length === 0) return config.json({ error: 'Not found' }, { status: 404 });\n const resource = resolveResource(path[0]);\n if (!crudResources.includes(resource)) return config.json({ error: 'Invalid resource' }, { status: 400 });\n\n // Bulk operations: [\"products\", \"metadata\"], [\"products\", \"bulk\"], [\"products\", \"export\"]\n if (path.length === 2) {\n if (path[1] === 'metadata' && method === 'GET') {\n return crud.GET_METADATA(req, resource);\n }\n if (path[1] === 'bulk' && method === 'POST') {\n return crud.BULK_POST(req, resource);\n }\n if (path[1] === 'export' && method === 'GET') {\n return crud.GET_EXPORT(req, resource);\n }\n }\n\n if (path.length === 1) {\n if (method === 'GET') return crud.GET(req, resource);\n if (method === 'POST') return crud.POST(req, resource);\n return config.json({ error: 'Method not allowed' }, { status: 405 });\n }\n if (path.length === 2) {\n const id = path[1];\n if (method === 'GET') return crudById.GET(req, resource, id);\n if (method === 'PUT' || method === 'PATCH') return crudById.PUT(req, resource, id);\n if (method === 'DELETE') return crudById.DELETE(req, resource, id);\n return config.json({ error: 'Method not allowed' }, { status: 405 });\n }\n return config.json({ error: 'Not found' }, { status: 404 });\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityTarget, ObjectLiteral } from 'typeorm';\nimport { In, IsNull } from 'typeorm';\nimport type { SessionUser } from '../auth/helpers';\nimport type { EntityMap } from './crud';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport { isValidSignupEmail } from '../lib/is-valid-signup-email';\nimport { queueEmail } from '../plugins/email/email-queue';\nimport { queueErpCreateContactIfEnabled } from '../plugins/erp/erp-contact-sync';\nimport { buildCanonicalOrderNumber, temporaryOrderNumberPlaceholder } from '../lib/order-number';\nimport { tryRefreshOrderFromErpForStorefront } from '../plugins/erp/erp-order-sync';\nimport { streamOrderInvoicePdf } from '../plugins/erp/erp-order-invoice';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { assertCaptchaOk } from '../plugins/captcha/assert';\nimport { queueSms } from '../plugins/sms/sms-queue';\nimport {\n createOtpChallenge,\n verifyAndConsumeOtpChallenge,\n generateNumericOtp,\n normalizePhoneE164,\n type OtpPurpose,\n} from '../lib/otp-challenge';\nconst GUEST_COOKIE = 'guest_id';\nconst ONE_YEAR = 60 * 60 * 24 * 365;\n\nexport interface StorefrontOtpFlags {\n login?: boolean;\n verifyEmail?: boolean;\n verifyPhone?: boolean;\n}\n\nfunction parseCookies(header: string | null): Record<string, string> {\n const out: Record<string, string> = {};\n if (!header) return out;\n for (const part of header.split(';')) {\n const i = part.indexOf('=');\n if (i === -1) continue;\n const k = part.slice(0, i).trim();\n const v = part.slice(i + 1).trim();\n out[k] = decodeURIComponent(v);\n }\n return out;\n}\n\nfunction guestCookieHeader(name: string, token: string): string {\n return `${name}=${encodeURIComponent(token)}; Path=/; HttpOnly; SameSite=Lax; Max-Age=${ONE_YEAR}`;\n}\n\nexport interface StorefrontApiConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number; headers?: HeadersInit }) => Response;\n getSessionUser: () => Promise<SessionUser | null>;\n guestCookieName?: string;\n /** Required for POST storefront/register */\n hashPassword?: (plain: string) => Promise<string>;\n /** When set, new registrations are blocked until email is verified and a signup email is sent. */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n /** Origin for verify links (e.g. process.env.NEXTAUTH_URL). Required with getCms for verification URLs. */\n publicSiteUrl?: string;\n /** When a flag is true, the corresponding OTP storefront route is enabled. */\n otpFlags?: StorefrontOtpFlags;\n /** Defaults to OTP_PEPPER or NEXTAUTH_SECRET. */\n otpPepper?: string;\n defaultPhoneCountryCode?: string;\n /** When false, login OTP send accepts email only (no phone). Default true. */\n otpAllowPhoneLogin?: boolean;\n}\n\nconst SIGNUP_VERIFY_EXPIRY_HOURS = 72;\n\nexport function createStorefrontApiHandler(config: StorefrontApiConfig) {\n const { dataSource, entityMap, json, getSessionUser, getCms, getCompanyDetails, publicSiteUrl } = config;\n const cookieName = config.guestCookieName ?? GUEST_COOKIE;\n const otpFlags = config.otpFlags;\n const otpPepper = config.otpPepper;\n const defaultPhoneCc = config.defaultPhoneCountryCode;\n const otpAllowPhoneLogin = config.otpAllowPhoneLogin !== false;\n\n function otpOff(key: keyof StorefrontOtpFlags) {\n return !otpFlags || otpFlags[key] !== true;\n }\n\n const cartRepo = () => dataSource.getRepository(entityMap.carts);\n const cartItemRepo = () => dataSource.getRepository(entityMap.cart_items);\n const productRepo = () => dataSource.getRepository(entityMap.products);\n const contactRepo = () => dataSource.getRepository(entityMap.contacts);\n const addressRepo = () => dataSource.getRepository(entityMap.addresses);\n const orderRepo = () => dataSource.getRepository(entityMap.orders);\n const orderItemRepo = () => dataSource.getRepository(entityMap.order_items);\n const wishlistRepo = () => dataSource.getRepository(entityMap.wishlists);\n const wishlistItemRepo = () => dataSource.getRepository(entityMap.wishlist_items);\n const userRepo = () => dataSource.getRepository(entityMap.users);\n const tokenRepo = () => dataSource.getRepository(entityMap.password_reset_tokens);\n const collectionRepo = () => dataSource.getRepository(entityMap.collections);\n const groupRepo = () => dataSource.getRepository(entityMap.user_groups);\n\n async function syncContactToErp(contact: ObjectLiteral): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n await queueErpCreateContactIfEnabled(cms, dataSource, entityMap, {\n name: String((contact as { name: string }).name ?? ''),\n email: String((contact as { email: string }).email ?? '').trim(),\n phone: (contact as { phone: string | null }).phone,\n type: (contact as { type: string | null }).type,\n company: (contact as { company: string | null }).company,\n notes: (contact as { notes: string | null }).notes,\n });\n } catch {\n /* ignore */\n }\n }\n\n async function ensureContactForUser(userId: number): Promise<{ id: number } | null> {\n let c = await contactRepo().findOne({ where: { userId, deleted: false } as ObjectLiteral });\n if (c) return c as { id: number };\n const u = await userRepo().findOne({ where: { id: userId } });\n if (!u) return null;\n const unclaimed = await contactRepo().findOne({\n where: { email: u.email, userId: IsNull(), deleted: false } as ObjectLiteral,\n });\n if (unclaimed) {\n await contactRepo().update((unclaimed as { id: number }).id, { userId });\n return { id: (unclaimed as { id: number }).id };\n }\n const created = await contactRepo().save(\n contactRepo().create({\n name: u.name,\n email: u.email,\n phone: null,\n userId,\n deleted: false,\n } as ObjectLiteral)\n );\n await syncContactToErp(created as ObjectLiteral);\n return { id: (created as { id: number }).id };\n }\n\n async function getOrCreateCart(req: Request): Promise<{ cart: ObjectLiteral; setCookie: string | null; err: Response | null }> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return { cart: {} as ObjectLiteral, setCookie: null, err: json({ error: 'User not found' }, { status: 400 }) };\n let cart = await cartRepo().findOne({\n where: { contactId: contact.id } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ contactId: contact.id, guestToken: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: null, err: null };\n }\n const cookies = parseCookies(req.headers.get('cookie'));\n let token = cookies[cookieName] || '';\n if (!token) {\n token = crypto.randomUUID();\n let cart = await cartRepo().findOne({\n where: { guestToken: token } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ guestToken: token, contactId: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: guestCookieHeader(cookieName, token), err: null };\n }\n let cart = await cartRepo().findOne({\n where: { guestToken: token } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ guestToken: token, contactId: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: null, err: null };\n }\n\n function primaryProductImageUrl(metadata: unknown): string | null {\n const meta = metadata as { images?: Array<{ url?: string; isDefault?: boolean }> } | null;\n const images = meta?.images;\n if (!Array.isArray(images) || !images.length) return null;\n const sorted = images.filter((i) => i?.url) as Array<{ url: string; isDefault?: boolean }>;\n if (!sorted.length) return null;\n const di = sorted.findIndex((i) => i.isDefault);\n if (di > 0) {\n const [d] = sorted.splice(di, 1);\n sorted.unshift(d!);\n }\n return sorted[0]!.url;\n }\n\n function serializeCart(cart: ObjectLiteral) {\n const items = (cart.items as ObjectLiteral[]) || [];\n return {\n id: cart.id,\n currency: cart.currency,\n items: items.map((it) => {\n const p = it.product as ObjectLiteral | undefined;\n return {\n id: it.id,\n productId: it.productId,\n quantity: it.quantity,\n metadata: it.metadata,\n product: p\n ? {\n id: p.id,\n name: p.name,\n slug: p.slug,\n price: p.price,\n sku: p.sku,\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n }),\n };\n }\n\n function serializeSeo(seo: unknown): ObjectLiteral | undefined {\n if (!seo || typeof seo !== 'object') return undefined;\n const s = seo as ObjectLiteral;\n return {\n title: s.title ?? null,\n description: s.description ?? null,\n keywords: s.keywords ?? null,\n ogTitle: s.ogTitle ?? null,\n ogDescription: s.ogDescription ?? null,\n ogImage: s.ogImage ?? null,\n slug: s.slug ?? null,\n };\n }\n\n function serializeProduct(p: ObjectLiteral) {\n const seo = serializeSeo(p.seo);\n return {\n id: p.id,\n name: p.name,\n slug: p.slug,\n sku: p.sku,\n hsn: p.hsn,\n price: p.price,\n compareAtPrice: p.compareAtPrice,\n status: p.status,\n collectionId: p.collectionId,\n metadata: p.metadata,\n ...(seo ? { seo } : {}),\n };\n }\n\n return {\n async handle(method: string, path: string[], req: Request): Promise<Response> {\n try {\n // --- products GET list ---\n if (path[0] === 'products' && path.length === 1 && method === 'GET') {\n const url = new URL(req.url || '', 'http://localhost');\n const collectionSlug = url.searchParams.get('collection')?.trim();\n const collectionId = url.searchParams.get('collectionId');\n const limit = Math.min(100, Math.max(1, parseInt(url.searchParams.get('limit') || '20', 10)));\n const offset = Math.max(0, parseInt(url.searchParams.get('offset') || '0', 10));\n const where: ObjectLiteral = { status: 'available', deleted: false };\n let collectionFilter: { name: string | null; slug: string | null } | null = null;\n if (collectionSlug) {\n let col: ObjectLiteral | null = null;\n if (/^\\d+$/.test(collectionSlug)) {\n col = (await collectionRepo().findOne({\n where: {\n id: parseInt(collectionSlug, 10),\n active: true,\n deleted: false,\n } as ObjectLiteral,\n })) as ObjectLiteral | null;\n } else {\n col = (await collectionRepo()\n .createQueryBuilder('c')\n .where('LOWER(c.slug) = LOWER(:slug)', { slug: collectionSlug })\n .andWhere('c.active = :a', { a: true })\n .andWhere('c.deleted = :d', { d: false })\n .getOne()) as ObjectLiteral | null;\n }\n if (!col) {\n return json({ products: [], total: 0, collection: null });\n }\n where.collectionId = col.id;\n collectionFilter = { name: col.name as string | null, slug: col.slug as string | null };\n } else if (collectionId) {\n const cid = parseInt(collectionId, 10);\n if (Number.isFinite(cid)) where.collectionId = cid;\n }\n const [items, total] = await productRepo().findAndCount({\n where,\n order: { id: 'ASC' },\n take: limit,\n skip: offset,\n });\n return json({\n products: items.map(serializeProduct),\n total,\n ...(collectionFilter && { collection: collectionFilter }),\n });\n }\n\n // --- products GET by id or slug ---\n if (path[0] === 'products' && path.length === 2 && method === 'GET') {\n const idOrSlug = path[1]!;\n const byId = /^\\d+$/.test(idOrSlug);\n const product = await productRepo().findOne({\n where: byId\n ? { id: parseInt(idOrSlug, 10), status: 'available', deleted: false }\n : { slug: idOrSlug, status: 'available', deleted: false },\n relations: ['attributes', 'attributes.attribute', 'seo'],\n } as ObjectLiteral);\n if (!product) return json({ error: 'Not found' }, { status: 404 });\n const p = product as ObjectLiteral;\n const attrRows = (p.attributes as ObjectLiteral[] | undefined) ?? [];\n const attributeTags = attrRows.map((pa) => ({\n name: ((pa.attribute as ObjectLiteral)?.name as string) ?? '',\n value: String(pa.value ?? ''),\n })).filter((t) => t.name || t.value);\n return json({ ...serializeProduct(p), attributes: attributeTags });\n }\n\n // --- collections GET list ---\n if (path[0] === 'collections' && path.length === 1 && method === 'GET') {\n const items = await collectionRepo().find({\n where: { active: true, deleted: false } as ObjectLiteral,\n order: { sortOrder: 'ASC', id: 'ASC' },\n });\n const ids = items.map((c) => (c as ObjectLiteral).id as number);\n const countByCollection: Record<number, number> = {};\n if (ids.length > 0) {\n const rows = await productRepo()\n .createQueryBuilder('p')\n .select('p.collectionId', 'collectionId')\n .addSelect('COUNT(p.id)', 'cnt')\n .where('p.collectionId IN (:...ids)', { ids })\n .andWhere('p.status = :status', { status: 'available' })\n .andWhere('p.deleted = :del', { del: false })\n .groupBy('p.collectionId')\n .getRawMany();\n for (const r of rows) {\n const cid = (r as ObjectLiteral).collectionId;\n if (cid != null) countByCollection[Number(cid)] = parseInt(String((r as ObjectLiteral).cnt), 10);\n }\n }\n return json({\n collections: items.map((c) => {\n const col = c as ObjectLiteral;\n const id = col.id as number;\n return {\n id,\n name: col.name,\n slug: col.slug,\n description: col.description,\n image: col.image,\n productCount: countByCollection[id] ?? 0,\n };\n }),\n });\n }\n\n // --- collections GET by id or slug ---\n if (path[0] === 'collections' && path.length === 2 && method === 'GET') {\n const idOrSlug = path[1]!;\n const byId = /^\\d+$/.test(idOrSlug);\n const collection = await collectionRepo().findOne({\n where: byId\n ? { id: parseInt(idOrSlug, 10), active: true, deleted: false }\n : { slug: idOrSlug, active: true, deleted: false },\n relations: ['seo'],\n } as ObjectLiteral);\n if (!collection) return json({ error: 'Not found' }, { status: 404 });\n const col = collection as ObjectLiteral;\n const products = await productRepo().find({\n where: { collectionId: col.id, status: 'available', deleted: false } as ObjectLiteral,\n order: { id: 'ASC' },\n });\n const colSeo = serializeSeo(col.seo);\n return json({\n id: col.id,\n name: col.name,\n slug: col.slug,\n description: col.description,\n image: col.image,\n ...(colSeo ? { seo: colSeo } : {}),\n products: products.map((p) => serializeProduct(p as ObjectLiteral)),\n });\n }\n\n // --- profile GET (auth) ---\n if (path[0] === 'profile' && path.length === 1 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const user = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n if (!user) return json({ error: 'Not found' }, { status: 404 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n return json({\n user: { id: (user as ObjectLiteral).id, name: (user as ObjectLiteral).name, email: (user as ObjectLiteral).email },\n contact: contact\n ? {\n id: (contact as ObjectLiteral).id,\n name: (contact as ObjectLiteral).name,\n email: (contact as ObjectLiteral).email,\n phone: (contact as ObjectLiteral).phone,\n }\n : null,\n });\n }\n\n // --- profile PUT (auth) ---\n if (path[0] === 'profile' && path.length === 1 && method === 'PUT') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const b = (await req.json().catch(() => ({}))) as { name?: string; phone?: string };\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (contact) {\n const updates: ObjectLiteral = {};\n if (typeof b.name === 'string' && b.name.trim()) updates.name = b.name.trim();\n if (b.phone !== undefined) updates.phone = b.phone === null || b.phone === '' ? null : String(b.phone);\n if (Object.keys(updates).length) await contactRepo().update((contact as { id: number }).id, updates);\n }\n const user = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n if (user && typeof b.name === 'string' && b.name.trim()) {\n await userRepo().update(uid, { name: b.name.trim() });\n }\n const updatedContact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (updatedContact) await syncContactToErp(updatedContact as ObjectLiteral);\n const updatedUser = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n return json({\n user: updatedUser ? { id: (updatedUser as ObjectLiteral).id, name: (updatedUser as ObjectLiteral).name, email: (updatedUser as ObjectLiteral).email } : null,\n contact: updatedContact\n ? {\n id: (updatedContact as ObjectLiteral).id,\n name: (updatedContact as ObjectLiteral).name,\n email: (updatedContact as ObjectLiteral).email,\n phone: (updatedContact as ObjectLiteral).phone,\n }\n : null,\n });\n }\n\n // --- addresses (auth: contact's addresses) ---\n async function getContactForAddresses(): Promise<{ contactId: number } | Response> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Contact not found' }, { status: 404 });\n return { contactId: (contact as { id: number }).id };\n }\n function serializeAddress(a: ObjectLiteral) {\n return {\n id: a.id,\n contactId: a.contactId,\n tag: a.tag,\n line1: a.line1,\n line2: a.line2,\n city: a.city,\n state: a.state,\n postalCode: a.postalCode,\n country: a.country,\n };\n }\n if (path[0] === 'addresses' && path.length === 1 && method === 'GET') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const list = await addressRepo().find({\n where: { contactId: contactOrErr.contactId },\n order: { id: 'ASC' },\n });\n return json({ addresses: list.map((a) => serializeAddress(a as ObjectLiteral)) });\n }\n if (path[0] === 'addresses' && path.length === 1 && method === 'POST') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const created = await addressRepo().save(\n addressRepo().create({\n contactId: contactOrErr.contactId,\n tag: typeof b.tag === 'string' ? b.tag.trim() || null : null,\n line1: typeof b.line1 === 'string' ? b.line1.trim() || null : null,\n line2: typeof b.line2 === 'string' ? b.line2.trim() || null : null,\n city: typeof b.city === 'string' ? b.city.trim() || null : null,\n state: typeof b.state === 'string' ? b.state.trim() || null : null,\n postalCode: typeof b.postalCode === 'string' ? b.postalCode.trim() || null : null,\n country: typeof b.country === 'string' ? b.country.trim() || null : null,\n } as ObjectLiteral)\n );\n return json(serializeAddress(created as ObjectLiteral));\n }\n if (path[0] === 'addresses' && path.length === 2 && (method === 'PATCH' || method === 'PUT')) {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const id = parseInt(path[1]!, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const existing = await addressRepo().findOne({ where: { id, contactId: contactOrErr.contactId } as ObjectLiteral });\n if (!existing) return json({ error: 'Not found' }, { status: 404 });\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const updates: ObjectLiteral = {};\n if (b.tag !== undefined) updates.tag = typeof b.tag === 'string' ? b.tag.trim() || null : null;\n if (b.line1 !== undefined) updates.line1 = typeof b.line1 === 'string' ? b.line1.trim() || null : null;\n if (b.line2 !== undefined) updates.line2 = typeof b.line2 === 'string' ? b.line2.trim() || null : null;\n if (b.city !== undefined) updates.city = typeof b.city === 'string' ? b.city.trim() || null : null;\n if (b.state !== undefined) updates.state = typeof b.state === 'string' ? b.state.trim() || null : null;\n if (b.postalCode !== undefined) updates.postalCode = typeof b.postalCode === 'string' ? b.postalCode.trim() || null : null;\n if (b.country !== undefined) updates.country = typeof b.country === 'string' ? b.country.trim() || null : null;\n if (Object.keys(updates).length) await addressRepo().update(id, updates);\n const updated = await addressRepo().findOne({ where: { id } as ObjectLiteral });\n return json(serializeAddress(updated as ObjectLiteral));\n }\n if (path[0] === 'addresses' && path.length === 2 && method === 'DELETE') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const id = parseInt(path[1]!, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const existing = await addressRepo().findOne({ where: { id, contactId: contactOrErr.contactId } as ObjectLiteral });\n if (!existing) return json({ error: 'Not found' }, { status: 404 });\n await addressRepo().delete(id);\n return json({ deleted: true });\n }\n\n // --- verify-email POST (public) ---\n if (path[0] === 'verify-email' && path.length === 1 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as { token?: string };\n const token = typeof b.token === 'string' ? b.token.trim() : '';\n if (!token) return json({ error: 'token is required' }, { status: 400 });\n const record = await tokenRepo().findOne({ where: { token } as ObjectLiteral });\n if (!record || (record as { expiresAt: Date }).expiresAt < new Date()) {\n return json({ error: 'Invalid or expired link. Please sign up again or contact support.' }, { status: 400 });\n }\n const email = (record as { email: string }).email;\n const user = await userRepo().findOne({ where: { email }, select: ['id', 'blocked'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n await userRepo().update((user as { id: number }).id, {\n blocked: false,\n emailVerifiedAt: new Date(),\n updatedAt: new Date(),\n } as ObjectLiteral);\n await tokenRepo().delete({ email } as ObjectLiteral);\n return json({ success: true, message: 'Email verified. You can sign in.' });\n }\n\n // --- auth/otp/send POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'send' && path.length === 3 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const purposeRaw = typeof b.purpose === 'string' ? b.purpose.trim() : '';\n const purpose = (purposeRaw === 'login' || purposeRaw === 'verify_email' || purposeRaw === 'verify_phone'\n ? purposeRaw\n : '') as OtpPurpose | '';\n if (!purpose) return json({ error: 'purpose must be login, verify_email, or verify_phone' }, { status: 400 });\n if (purpose === 'login' && otpOff('login')) return json({ error: 'otp_disabled' }, { status: 403 });\n if (purpose === 'verify_email' && otpOff('verifyEmail')) return json({ error: 'otp_disabled' }, { status: 403 });\n if (purpose === 'verify_phone' && otpOff('verifyPhone')) return json({ error: 'otp_disabled' }, { status: 403 });\n\n const capOtp = await assertCaptchaOk(getCms, b, req, json);\n if (capOtp) return capOtp;\n\n const emailIn = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const phoneIn = typeof b.phone === 'string' ? b.phone.trim() : '';\n let identifier: string;\n let channel: 'email' | 'sms';\n\n if (purpose === 'login') {\n if (emailIn) {\n identifier = emailIn;\n channel = 'email';\n } else if (phoneIn) {\n if (!otpAllowPhoneLogin) {\n return json({ error: 'Phone sign-in is not enabled' }, { status: 403 });\n }\n const p = normalizePhoneE164(phoneIn, defaultPhoneCc);\n if (!p) return json({ error: 'Invalid phone' }, { status: 400 });\n identifier = p;\n channel = 'sms';\n } else {\n return json({ error: 'email or phone required' }, { status: 400 });\n }\n const user =\n channel === 'email'\n ? await userRepo().findOne({ where: { email: identifier } as ObjectLiteral })\n : await userRepo().findOne({ where: { phone: identifier } as ObjectLiteral });\n if (!user || (user as { deleted?: boolean }).deleted || (user as { blocked?: boolean }).blocked) {\n return json({ ok: true });\n }\n } else if (purpose === 'verify_email') {\n if (!emailIn || !isValidSignupEmail(emailIn)) return json({ error: 'Valid email required' }, { status: 400 });\n identifier = emailIn;\n channel = 'email';\n const user = await userRepo().findOne({ where: { email: identifier } as ObjectLiteral });\n if (!user || (user as { deleted?: boolean }).deleted) return json({ ok: true });\n } else {\n const su = await getSessionUser();\n const uid = su?.id ? parseInt(String(su.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const p = normalizePhoneE164(phoneIn, defaultPhoneCc);\n if (!p) return json({ error: 'Valid phone required' }, { status: 400 });\n identifier = p;\n channel = 'sms';\n const taken = await userRepo().findOne({\n where: { phone: identifier } as ObjectLiteral,\n select: ['id'],\n });\n if (taken && (taken as { id: number }).id !== uid) {\n return json({ error: 'Phone already in use' }, { status: 400 });\n }\n }\n\n const code = generateNumericOtp(6);\n const created = await createOtpChallenge(dataSource, entityMap, {\n purpose,\n channel,\n identifier,\n code,\n pepper: otpPepper,\n });\n if (!created.ok) return json({ error: created.error }, { status: created.status });\n\n if (!getCms) return json({ error: 'OTP delivery not configured' }, { status: 503 });\n\n try {\n const cms = await getCms();\n if (channel === 'email') {\n if (!cms.getPlugin('email')) return json({ error: 'Email not configured' }, { status: 503 });\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n await queueEmail(cms, {\n to: identifier,\n templateName: 'otp',\n ctx: { code, companyDetails: companyDetails ?? {} },\n });\n } else {\n if (!cms.getPlugin('sms')) return json({ error: 'SMS not configured' }, { status: 503 });\n const templateKey = purpose === 'verify_phone' ? 'auth.otp_verify_phone' : 'auth.otp_login';\n await queueSms(cms, { to: identifier, templateKey, variables: { code } });\n }\n } catch {\n return json({ error: 'Failed to send code' }, { status: 500 });\n }\n return json({ ok: true });\n }\n\n // --- auth/otp/verify-email POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'verify-email' && path.length === 3 && method === 'POST') {\n if (otpOff('verifyEmail')) return json({ error: 'otp_disabled' }, { status: 403 });\n const b = (await req.json().catch(() => ({}))) as { email?: string; code?: string };\n const email = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const code = typeof b.code === 'string' ? b.code.trim() : '';\n if (!email || !code) return json({ error: 'email and code required' }, { status: 400 });\n const v = await verifyAndConsumeOtpChallenge(dataSource, entityMap, {\n purpose: 'verify_email',\n identifier: email,\n code,\n pepper: otpPepper,\n });\n if (!v.ok) return json({ error: v.error }, { status: v.status });\n const user = await userRepo().findOne({ where: { email } as ObjectLiteral });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n await userRepo().update((user as { id: number }).id, {\n blocked: false,\n emailVerifiedAt: new Date(),\n updatedAt: new Date(),\n } as ObjectLiteral);\n await tokenRepo().delete({ email } as ObjectLiteral);\n return json({ success: true, message: 'Email verified. You can sign in.' });\n }\n\n // --- auth/otp/verify-phone POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'verify-phone' && path.length === 3 && method === 'POST') {\n if (otpOff('verifyPhone')) return json({ error: 'otp_disabled' }, { status: 403 });\n const su = await getSessionUser();\n const uid = su?.id ? parseInt(String(su.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const b = (await req.json().catch(() => ({}))) as { phone?: string; code?: string };\n const phoneRaw = typeof b.phone === 'string' ? b.phone.trim() : '';\n const code = typeof b.code === 'string' ? b.code.trim() : '';\n const phone = normalizePhoneE164(phoneRaw, defaultPhoneCc);\n if (!phone || !code) return json({ error: 'phone and code required' }, { status: 400 });\n const v = await verifyAndConsumeOtpChallenge(dataSource, entityMap, {\n purpose: 'verify_phone',\n identifier: phone,\n code,\n pepper: otpPepper,\n });\n if (!v.ok) return json({ error: v.error }, { status: v.status });\n const taken = await userRepo().findOne({ where: { phone } as ObjectLiteral, select: ['id'] });\n if (taken && (taken as { id: number }).id !== uid) {\n return json({ error: 'Phone already in use' }, { status: 400 });\n }\n await userRepo().update(uid, { phone, phoneVerifiedAt: new Date(), updatedAt: new Date() } as ObjectLiteral);\n const contact = await ensureContactForUser(uid);\n if (contact) {\n await contactRepo().update(contact.id, { phone } as ObjectLiteral);\n }\n return json({ success: true });\n }\n\n // --- register POST ---\n if (path[0] === 'register' && path.length === 1 && method === 'POST') {\n if (!config.hashPassword) return json({ error: 'Registration not configured' }, { status: 501 });\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capReg = await assertCaptchaOk(getCms, b, req, json);\n if (capReg) return capReg;\n const name = typeof b.name === 'string' ? b.name.trim() : '';\n const email = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const password = typeof b.password === 'string' ? b.password : '';\n if (!name || !email || !password) return json({ error: 'name, email and password are required' }, { status: 400 });\n if (!isValidSignupEmail(email)) return json({ error: 'Invalid email address' }, { status: 400 });\n const existing = await userRepo().findOne({ where: { email } });\n if (existing) return json({ error: 'User with this email already exists' }, { status: 400 });\n const customerG = await groupRepo().findOne({ where: { name: 'Customer', deleted: false } as ObjectLiteral });\n const groupId = customerG ? (customerG as { id: number }).id : null;\n const hashed = await config.hashPassword(password);\n const requireEmailVerification = Boolean(getCms);\n const newUser = await userRepo().save(\n userRepo().create({\n name,\n email,\n password: hashed,\n blocked: requireEmailVerification,\n groupId,\n adminAccess: false,\n } as ObjectLiteral)\n );\n const userId = (newUser as { id: number }).id;\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, userId, email);\n\n let emailVerificationSent = false;\n if (requireEmailVerification && getCms) {\n try {\n const crypto = await import('crypto');\n const rawToken = crypto.randomBytes(32).toString('hex');\n const expiresAt = new Date(Date.now() + SIGNUP_VERIFY_EXPIRY_HOURS * 60 * 60 * 1000);\n await tokenRepo().save(\n tokenRepo().create({ email, token: rawToken, expiresAt } as ObjectLiteral)\n );\n const cms = await getCms();\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n const base = (publicSiteUrl || '').replace(/\\/$/, '').trim() || 'http://localhost:3000';\n const verifyEmailUrl = `${base}/verify-email?token=${encodeURIComponent(rawToken)}`;\n await queueEmail(cms, {\n to: email,\n templateName: 'signup',\n ctx: { name, verifyEmailUrl, companyDetails: companyDetails ?? {} },\n });\n emailVerificationSent = true;\n } catch {\n await userRepo().update(userId, { blocked: false, updatedAt: new Date() } as ObjectLiteral);\n }\n }\n\n return json({\n success: true,\n userId,\n emailVerificationSent,\n });\n }\n\n // --- cart GET ---\n if (path[0] === 'cart' && path.length === 1 && method === 'GET') {\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const body = serializeCart(cart);\n if (setCookie) return json(body, { headers: { 'Set-Cookie': setCookie } });\n return json(body);\n }\n\n // --- cart/items POST ---\n if (path[0] === 'cart' && path[1] === 'items' && path.length === 2 && method === 'POST') {\n const body = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capCart = await assertCaptchaOk(getCms, body, req, json);\n if (capCart) return capCart;\n const productId = Number(body.productId);\n const quantity = Math.max(1, Number(body.quantity) || 1);\n if (!Number.isFinite(productId)) return json({ error: 'productId required' }, { status: 400 });\n const product = await productRepo().findOne({ where: { id: productId, deleted: false } as ObjectLiteral });\n if (!product) return json({ error: 'Product not found' }, { status: 404 });\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const cartId = (cart as { id: number }).id;\n const existing = await cartItemRepo().findOne({ where: { cartId, productId } as ObjectLiteral });\n if (existing) {\n await cartItemRepo().update((existing as { id: number }).id, {\n quantity: (existing as { quantity: number }).quantity + quantity,\n });\n } else {\n await cartItemRepo().save(\n cartItemRepo().create({ cartId, productId, quantity } as ObjectLiteral)\n );\n }\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n\n // --- cart/items/:id PATCH DELETE ---\n if (path[0] === 'cart' && path[1] === 'items' && path.length === 3) {\n const itemId = parseInt(path[2]!, 10);\n if (!Number.isFinite(itemId)) return json({ error: 'Invalid item id' }, { status: 400 });\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const cartId = (cart as { id: number }).id;\n const item = await cartItemRepo().findOne({ where: { id: itemId, cartId } as ObjectLiteral });\n if (!item) return json({ error: 'Not found' }, { status: 404 });\n if (method === 'DELETE') {\n await cartItemRepo().delete(itemId);\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n if (method === 'PATCH') {\n const b = (await req.json().catch(() => ({}))) as { quantity?: number };\n const q = Math.max(0, Number(b.quantity) || 0);\n if (q === 0) await cartItemRepo().delete(itemId);\n else await cartItemRepo().update(itemId, { quantity: q });\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n }\n\n // --- cart/merge POST (auth) ---\n if (path[0] === 'cart' && path[1] === 'merge' && method === 'POST') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact not found' }, { status: 400 });\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ merged: false, message: 'No guest cart' });\n const guestCart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: ['items'],\n });\n if (!guestCart || !((guestCart as { items?: unknown[] }).items || []).length) {\n let uc = await cartRepo().findOne({\n where: { contactId: contact.id } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!uc) uc = { items: [] } as ObjectLiteral;\n return json(\n { merged: false, cart: serializeCart(uc) },\n { headers: { 'Set-Cookie': `${cookieName}=; Path=/; Max-Age=0` } }\n );\n }\n let userCart = await cartRepo().findOne({ where: { contactId: contact.id } as ObjectLiteral });\n if (!userCart) {\n userCart = await cartRepo().save(\n cartRepo().create({ contactId: contact.id, guestToken: null, currency: (guestCart as { currency: string }).currency } as ObjectLiteral)\n );\n }\n const uidCart = (userCart as { id: number }).id;\n const gItems = ((guestCart as { items: ObjectLiteral[] }).items) || [];\n for (const gi of gItems) {\n const existing = await cartItemRepo().findOne({\n where: { cartId: uidCart, productId: (gi as { productId: number }).productId } as ObjectLiteral,\n });\n if (existing) {\n await cartItemRepo().update((existing as { id: number }).id, {\n quantity: (existing as { quantity: number }).quantity + (gi as { quantity: number }).quantity,\n });\n } else {\n await cartItemRepo().save(\n cartItemRepo().create({\n cartId: uidCart,\n productId: (gi as { productId: number }).productId,\n quantity: (gi as { quantity: number }).quantity,\n metadata: (gi as { metadata: unknown }).metadata,\n } as ObjectLiteral)\n );\n }\n }\n await cartRepo().delete((guestCart as { id: number }).id);\n await cartRepo().update(uidCart, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: uidCart },\n relations: ['items', 'items.product'],\n });\n // Merge guest wishlist into contact wishlist (same guestId cookie)\n const guestWishlist = await wishlistRepo().findOne({\n where: { guestId: guestToken } as ObjectLiteral,\n relations: ['items'],\n });\n if (guestWishlist && ((guestWishlist as { items?: unknown[] }).items || []).length > 0) {\n const userWishlist = await getDefaultWishlist(contact.id);\n const gItems = ((guestWishlist as { items: ObjectLiteral[] }).items) || [];\n for (const gi of gItems) {\n const pid = (gi as { productId: number }).productId;\n const ex = await wishlistItemRepo().findOne({ where: { wishlistId: userWishlist.id, productId: pid } as ObjectLiteral });\n if (!ex) await wishlistItemRepo().save(wishlistItemRepo().create({ wishlistId: userWishlist.id, productId: pid } as ObjectLiteral));\n }\n await wishlistRepo().delete((guestWishlist as { id: number }).id);\n }\n return json({ merged: true, cart: serializeCart(fresh!) }, { headers: { 'Set-Cookie': `${cookieName}=; Path=/; Max-Age=0` } });\n }\n\n // --- wishlist (guest or auth: same guestId cookie as cart) ---\n async function getDefaultWishlist(contactId: number) {\n let w = await wishlistRepo().findOne({ where: { contactId, name: 'default' } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ contactId, guestId: null, name: 'default' } as ObjectLiteral));\n }\n return w as { id: number };\n }\n\n async function getOrCreateWishlist(req: Request): Promise<{ wishlist: ObjectLiteral; setCookie: string | null; err: Response | null }> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return { wishlist: {} as ObjectLiteral, setCookie: null, err: json({ error: 'User not found' }, { status: 400 }) };\n const w = await getDefaultWishlist(contact.id);\n const wishlist = await wishlistRepo().findOne({ where: { id: w.id } as ObjectLiteral });\n return { wishlist: wishlist!, setCookie: null, err: null };\n }\n const cookies = parseCookies(req.headers.get('cookie'));\n let token = cookies[cookieName] || '';\n if (!token) {\n token = crypto.randomUUID();\n let w = await wishlistRepo().findOne({ where: { guestId: token } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ guestId: token, contactId: null, name: 'default' } as ObjectLiteral));\n }\n return { wishlist: w!, setCookie: guestCookieHeader(cookieName, token), err: null };\n }\n let w = await wishlistRepo().findOne({ where: { guestId: token } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ guestId: token, contactId: null, name: 'default' } as ObjectLiteral));\n }\n return { wishlist: w!, setCookie: null, err: null };\n }\n\n if (path[0] === 'wishlist' && path.length === 1 && method === 'GET') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const items = await wishlistItemRepo().find({\n where: { wishlistId: (wishlist as { id: number }).id } as ObjectLiteral,\n relations: ['product'],\n });\n const body = {\n wishlistId: (wishlist as { id: number }).id,\n items: items.map((it) => {\n const p = (it as ObjectLiteral).product as ObjectLiteral | undefined;\n return {\n id: (it as ObjectLiteral).id,\n productId: (it as ObjectLiteral).productId,\n product: p\n ? {\n id: p.id,\n name: p.name,\n slug: p.slug,\n price: p.price,\n sku: p.sku,\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n }),\n };\n if (setCookie) return json(body, { headers: { 'Set-Cookie': setCookie } });\n return json(body);\n }\n\n if (path[0] === 'wishlist' && path[1] === 'items' && path.length === 2 && method === 'POST') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capWl = await assertCaptchaOk(getCms, b, req, json);\n if (capWl) return capWl;\n const productId = Number(b.productId);\n if (!Number.isFinite(productId)) return json({ error: 'productId required' }, { status: 400 });\n const wid = (wishlist as { id: number }).id;\n const ex = await wishlistItemRepo().findOne({ where: { wishlistId: wid, productId } as ObjectLiteral });\n if (!ex) await wishlistItemRepo().save(wishlistItemRepo().create({ wishlistId: wid, productId } as ObjectLiteral));\n if (setCookie) return json({ ok: true }, { headers: { 'Set-Cookie': setCookie } });\n return json({ ok: true });\n }\n\n if (path[0] === 'wishlist' && path[1] === 'items' && path.length === 3 && method === 'DELETE') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const productId = parseInt(path[2]!, 10);\n await wishlistItemRepo().delete({ wishlistId: (wishlist as { id: number }).id, productId } as ObjectLiteral);\n if (setCookie) return json({ ok: true }, { headers: { 'Set-Cookie': setCookie } });\n return json({ ok: true });\n }\n\n // --- checkout/order POST (create order only, for Razorpay; do not clear cart) ---\n if (path[0] === 'checkout' && path[1] === 'order' && path.length === 2 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capOrd = await assertCaptchaOk(getCms, b, req, json);\n if (capOrd) return capOrd;\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n let contactId: number;\n let cart: ObjectLiteral | null;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact required' }, { status: 400 });\n contactId = contact.id;\n cart = await cartRepo().findOne({\n where: { contactId } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n } else {\n const email = String(b.email ?? '').trim();\n const name = String(b.name ?? '').trim();\n if (!email || !name) return json({ error: 'name and email required for guest checkout' }, { status: 400 });\n let contact = await contactRepo().findOne({ where: { email, deleted: false } as ObjectLiteral });\n if (contact && (contact as { userId: number | null }).userId != null) {\n return json({ error: 'Please sign in to complete checkout' }, { status: 400 });\n }\n if (!contact) {\n contact = await contactRepo().save(\n contactRepo().create({\n name,\n email,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : null,\n userId: null,\n deleted: false,\n } as ObjectLiteral)\n );\n } else if (name)\n await contactRepo().update((contact as { id: number }).id, {\n name,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : (contact as { phone: string | null }).phone,\n });\n contactId = (contact as { id: number }).id;\n const guestForErp = await contactRepo().findOne({ where: { id: contactId } as ObjectLiteral });\n if (guestForErp) await syncContactToErp(guestForErp);\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ error: 'Cart not found' }, { status: 400 });\n cart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n }\n if (!cart || !((cart.items as ObjectLiteral[]) || []).length) {\n return json({ error: 'Cart is empty' }, { status: 400 });\n }\n let subtotal = 0;\n const lines: { productId: number; quantity: number; unitPrice: number; tax: number; total: number }[] = [];\n for (const it of (cart.items as ObjectLiteral[]) || []) {\n const p = it.product as ObjectLiteral | undefined;\n if (!p || p.deleted || p.status !== 'available') continue;\n const unit = Number(p.price);\n const qty = (it.quantity as number) || 1;\n const lineTotal = unit * qty;\n subtotal += lineTotal;\n lines.push({ productId: p.id as number, quantity: qty, unitPrice: unit, tax: 0, total: lineTotal });\n }\n if (!lines.length) return json({ error: 'No available items in cart' }, { status: 400 });\n const total = subtotal;\n const cartId = (cart as { id: number }).id;\n const ord = await orderRepo().save(\n orderRepo().create({\n orderNumber: temporaryOrderNumberPlaceholder(),\n orderKind: 'sale',\n parentOrderId: null,\n contactId,\n billingAddressId: typeof b.billingAddressId === 'number' ? b.billingAddressId : null,\n shippingAddressId: typeof b.shippingAddressId === 'number' ? b.shippingAddressId : null,\n status: 'pending',\n subtotal,\n tax: 0,\n discount: 0,\n total,\n currency: (cart.currency as string) || 'INR',\n metadata: { cartId },\n } as ObjectLiteral)\n );\n const oid = (ord as { id: number }).id;\n await orderRepo().update(oid, {\n orderNumber: buildCanonicalOrderNumber('sale', oid, (ord as { createdAt?: Date }).createdAt ?? new Date()),\n } as ObjectLiteral);\n for (const line of lines) {\n await orderItemRepo().save(\n orderItemRepo().create({\n orderId: oid,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n } as ObjectLiteral)\n );\n }\n return json({\n orderId: oid,\n orderNumber: (ord as { orderNumber: string }).orderNumber,\n total,\n currency: (cart.currency as string) || 'INR',\n });\n }\n\n // --- checkout POST ---\n if (path[0] === 'checkout' && path.length === 1 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capChk = await assertCaptchaOk(getCms, b, req, json);\n if (capChk) return capChk;\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n let contactId: number;\n let cart: ObjectLiteral | null;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact required' }, { status: 400 });\n contactId = contact.id;\n cart = await cartRepo().findOne({\n where: { contactId } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n } else {\n const email = String(b.email ?? '').trim();\n const name = String(b.name ?? '').trim();\n if (!email || !name) return json({ error: 'name and email required for guest checkout' }, { status: 400 });\n let contact = await contactRepo().findOne({ where: { email, deleted: false } as ObjectLiteral });\n if (contact && (contact as { userId: number | null }).userId != null) {\n return json({ error: 'Please sign in to complete checkout' }, { status: 400 });\n }\n if (!contact) {\n contact = await contactRepo().save(\n contactRepo().create({\n name,\n email,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : null,\n userId: null,\n deleted: false,\n } as ObjectLiteral)\n );\n } else if (name)\n await contactRepo().update((contact as { id: number }).id, {\n name,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : (contact as { phone: string | null }).phone,\n });\n contactId = (contact as { id: number }).id;\n const guestForErp2 = await contactRepo().findOne({ where: { id: contactId } as ObjectLiteral });\n if (guestForErp2) await syncContactToErp(guestForErp2);\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ error: 'Cart not found' }, { status: 400 });\n cart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n }\n if (!cart || !((cart.items as ObjectLiteral[]) || []).length) {\n return json({ error: 'Cart is empty' }, { status: 400 });\n }\n let subtotal = 0;\n const lines: { productId: number; quantity: number; unitPrice: number; tax: number; total: number }[] = [];\n for (const it of (cart.items as ObjectLiteral[]) || []) {\n const p = it.product as ObjectLiteral | undefined;\n if (!p || p.deleted || p.status !== 'available') continue;\n const unit = Number(p.price);\n const qty = (it.quantity as number) || 1;\n const lineTotal = unit * qty;\n subtotal += lineTotal;\n lines.push({ productId: p.id as number, quantity: qty, unitPrice: unit, tax: 0, total: lineTotal });\n }\n if (!lines.length) return json({ error: 'No available items in cart' }, { status: 400 });\n const total = subtotal;\n const ord = await orderRepo().save(\n orderRepo().create({\n orderNumber: temporaryOrderNumberPlaceholder(),\n orderKind: 'sale',\n parentOrderId: null,\n contactId,\n billingAddressId: typeof b.billingAddressId === 'number' ? b.billingAddressId : null,\n shippingAddressId: typeof b.shippingAddressId === 'number' ? b.shippingAddressId : null,\n status: 'pending',\n subtotal,\n tax: 0,\n discount: 0,\n total,\n currency: (cart.currency as string) || 'INR',\n } as ObjectLiteral)\n );\n const oid = (ord as { id: number }).id;\n await orderRepo().update(oid, {\n orderNumber: buildCanonicalOrderNumber('sale', oid, (ord as { createdAt?: Date }).createdAt ?? new Date()),\n } as ObjectLiteral);\n for (const line of lines) {\n await orderItemRepo().save(\n orderItemRepo().create({\n orderId: oid,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n } as ObjectLiteral)\n );\n }\n await cartItemRepo().delete({ cartId: (cart as { id: number }).id } as ObjectLiteral);\n await cartRepo().delete((cart as { id: number }).id);\n return json({\n orderId: oid,\n orderNumber: (ord as { orderNumber: string }).orderNumber,\n total,\n });\n }\n\n // --- orders GET ---\n if (path[0] === 'orders' && path.length === 1 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ orders: [] });\n const orders = await orderRepo().find({\n where: { contactId: (contact as { id: number }).id, deleted: false, orderKind: 'sale' } as ObjectLiteral,\n order: { createdAt: 'DESC' },\n take: 50,\n });\n const orderIds = orders.map((o) => (o as ObjectLiteral).id as number);\n const previewByOrder: Record<number, string[]> = {};\n if (orderIds.length) {\n const oItems = await orderItemRepo().find({\n where: { orderId: In(orderIds) },\n relations: ['product'],\n order: { id: 'ASC' },\n });\n for (const oi of oItems) {\n const oid = (oi as ObjectLiteral).orderId as number;\n if (!previewByOrder[oid]) previewByOrder[oid] = [];\n if (previewByOrder[oid].length >= 4) continue;\n const url = primaryProductImageUrl(((oi as ObjectLiteral).product as ObjectLiteral)?.metadata);\n if (url && !previewByOrder[oid].includes(url)) previewByOrder[oid].push(url);\n }\n }\n return json({\n orders: orders.map((o) => {\n const ol = o as ObjectLiteral;\n return {\n id: ol.id,\n orderNumber: ol.orderNumber,\n status: ol.status,\n total: ol.total,\n currency: ol.currency,\n createdAt: ol.createdAt,\n previewImages: previewByOrder[ol.id as number] ?? [],\n };\n }),\n });\n }\n\n if (path[0] === 'orders' && path.length === 3 && path[2] === 'invoice' && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n if (!getCms) return json({ error: 'Not found' }, { status: 404 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Not found' }, { status: 404 });\n const orderId = parseInt(path[1]!, 10);\n if (!Number.isFinite(orderId)) return json({ error: 'Invalid id' }, { status: 400 });\n const cms = await getCms();\n return streamOrderInvoicePdf(cms, dataSource, entityMap, orderId, {\n ownerContactId: (contact as { id: number }).id,\n });\n }\n\n if (path[0] === 'orders' && path.length === 2 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Not found' }, { status: 404 });\n const orderId = parseInt(path[1]!, 10);\n let order = await orderRepo().findOne({\n where: { id: orderId, contactId: (contact as { id: number }).id, deleted: false } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!order) return json({ error: 'Not found' }, { status: 404 });\n if (getCms) {\n const cms = await getCms();\n await tryRefreshOrderFromErpForStorefront(cms, dataSource, entityMap, order as ObjectLiteral);\n order = await orderRepo().findOne({\n where: { id: orderId, contactId: (contact as { id: number }).id, deleted: false } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n }\n if (!order) return json({ error: 'Not found' }, { status: 404 });\n const o = order as ObjectLiteral;\n const lines = ((o.items as ObjectLiteral[]) || []).map((line) => {\n const p = line.product as ObjectLiteral | undefined;\n return {\n id: line.id,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n product: p\n ? {\n name: p.name,\n slug: p.slug,\n sku: p.sku,\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n });\n const kind = (o.orderKind as string) || 'sale';\n let relatedOrders: ObjectLiteral[] = [];\n if (kind === 'sale') {\n relatedOrders = await orderRepo().find({\n where: { parentOrderId: orderId, deleted: false } as ObjectLiteral,\n order: { id: 'ASC' },\n });\n }\n const meta = o.metadata as Record<string, unknown> | null | undefined;\n const fulfillmentPreview =\n meta && typeof meta.fulfillment === 'object' && meta.fulfillment && 'status' in (meta.fulfillment as object)\n ? String((meta.fulfillment as { status?: string }).status ?? '')\n : '';\n return json({\n order: {\n id: o.id,\n orderNumber: o.orderNumber,\n orderKind: kind,\n parentOrderId: o.parentOrderId ?? null,\n status: o.status,\n subtotal: o.subtotal,\n tax: o.tax,\n discount: o.discount,\n total: o.total,\n currency: o.currency,\n createdAt: o.createdAt,\n metadata: o.metadata ?? null,\n items: lines,\n },\n relatedOrders: relatedOrders.map((r) => ({\n id: r.id,\n orderNumber: r.orderNumber,\n orderKind: r.orderKind ?? 'return',\n status: r.status,\n createdAt: r.createdAt,\n fulfillmentStatus:\n r.metadata &&\n typeof r.metadata === 'object' &&\n (r.metadata as { fulfillment?: { status?: string } }).fulfillment?.status,\n })),\n fulfillmentPreview: fulfillmentPreview || undefined,\n });\n }\n\n return json({ error: 'Not found' }, { status: 404 });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n };\n}\n","/** Max lengths per RFC 5321 / 5322 (practical subset). */\nconst MAX_EMAIL = 254;\nconst MAX_LOCAL = 64;\n\n/**\n * Rejects obviously invalid signup addresses (format, length, missing TLD).\n * Used by storefront register so junk strings cannot create accounts.\n */\nexport function isValidSignupEmail(email: string): boolean {\n if (!email || email.length > MAX_EMAIL) return false;\n const at = email.indexOf('@');\n if (at <= 0 || at !== email.lastIndexOf('@')) return false;\n const local = email.slice(0, at);\n const domain = email.slice(at + 1);\n if (!local || local.length > MAX_LOCAL || !domain || domain.length > 253) return false;\n if (local.startsWith('.') || local.endsWith('.') || local.includes('..')) return false;\n if (domain.startsWith('.') || domain.endsWith('.') || domain.includes('..')) return false;\n if (!/^[a-z0-9._%+-]+$/i.test(local)) return false;\n if (!/^[a-z0-9](?:[a-z0-9-]*[a-z0-9])?(?:\\.[a-z0-9](?:[a-z0-9-]*[a-z0-9])?)+$/i.test(domain)) return false;\n const tld = domain.split('.').pop()!;\n return tld.length >= 2;\n}\n","export interface AdminNavItem {\n href: string;\n label: string;\n icon?: string;\n}\n\nexport const DEFAULT_ADMIN_NAV: AdminNavItem[] = [\n { href: '/admin/dashboard', label: 'Dashboard' },\n { href: '/admin/contacts', label: 'Contacts' },\n { href: '/admin/blogs', label: 'Blogs' },\n { href: '/admin/users', label: 'Users' },\n { href: '/admin/roles', label: 'Roles' },\n { href: '/admin/forms', label: 'Forms' },\n { href: '/admin/submissions', label: 'Submissions' },\n { href: '/admin/pages', label: 'Pages' },\n];\n"],"mappings":"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","names":["pickInvoiceId","Stripe","Razorpay","crypto","pick","escapeHtml","subject","escapeHtml","bodyHtml","text","html","render","escapeHtml","render","escapeHtml","escapeHtml","render","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","render","nodemailer","resolveSmsTemplateForSend","applyTemplateVars","StripePaymentService","RazorpayPaymentService","Redis","fastq","import_crypto","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","isPublicMethod","_CredentialsProvider","import_typeorm","repo","sortField","total","data","typeFilter","updatePayload","updated","crypto","import_typeorm","isErpIntegrationEnabled","result","queueEmail","streamOrderInvoicePdf","import_typeorm","cart","serializeAddress","crypto","gItems","req","w"]}
1
+ {"version":3,"sources":["../src/plugins/erp/erp-queue.ts","../src/plugins/erp/paid-order-erp.ts","../src/plugins/erp/erp-response-map.ts","../src/plugins/erp/erp-config-enabled.ts","../src/plugins/erp/erp-order-invoice.ts","../src/plugins/email/email-queue.ts","../src/message-templates/sms-defaults.ts","../src/message-templates/interpolate.ts","../src/plugins/payment/stripe.ts","../src/plugins/payment/razorpay.ts","../src/index.ts","../src/plugins/registry.ts","../src/plugins/erp/erp-submission.ts","../src/plugins/erp/index.ts","../src/plugins/erp/erp-contact-sync.ts","../src/lib/order-number.ts","../src/lib/order-storefront-metadata.ts","../src/plugins/erp/erp-order-status-map.ts","../src/plugins/erp/erp-order-sync.ts","../src/plugins/erp/erp-product-sync.ts","../src/plugins/email/email-service.ts","../src/plugins/email/templates/layout.ts","../src/plugins/email/templates/types.ts","../src/plugins/email/templates/inline-cta.ts","../src/plugins/email/templates/signup.ts","../src/plugins/email/templates/passwordReset.ts","../src/plugins/email/templates/passwordChange.ts","../src/plugins/email/templates/orderPlaced.ts","../src/plugins/email/templates/returnInitiated.ts","../src/plugins/email/templates/shippingUpdate.ts","../src/plugins/email/templates/invite.ts","../src/plugins/email/templates/formSubmission.ts","../src/plugins/email/templates/otp.ts","../src/plugins/email/templates/index.ts","../src/plugins/email/renderer.ts","../src/plugins/email/index.ts","../src/plugins/analytics/analytics-service.ts","../src/plugins/analytics/index.ts","../src/plugins/sms/sms-service.ts","../src/plugins/sms/sms-queue.ts","../src/plugins/sms/index.ts","../src/plugins/payment/index.ts","../src/plugins/storage/s3.ts","../src/plugins/storage/local.ts","../src/plugins/llm/llm-service.ts","../src/plugins/llm/index.ts","../src/plugins/cache/index.ts","../src/plugins/cache/memory-cache.ts","../src/plugins/cache/redis-cache.ts","../src/plugins/queue/bullmq-queue.ts","../src/plugins/queue/memory-queue.ts","../src/plugins/queue/index.ts","../src/plugins/captcha/captcha-service.ts","../src/plugins/captcha/captcha-plugin.ts","../src/plugins/captcha/assert.ts","../src/lib/utils.ts","../src/lib/seo-metadata.ts","../src/lib/link-contact-to-user.ts","../src/lib/email-recipients.ts","../src/lib/otp-challenge.ts","../src/entities/user.entity.ts","../src/entities/user-group.entity.ts","../src/entities/permission.entity.ts","../src/entities/otp-challenge.entity.ts","../src/entities/password-reset-token.entity.ts","../src/entities/blog.entity.ts","../src/entities/category.entity.ts","../src/entities/seo.entity.ts","../src/entities/comment.entity.ts","../src/entities/tag.entity.ts","../src/entities/contact.entity.ts","../src/entities/form-submission.entity.ts","../src/entities/form.entity.ts","../src/entities/form-field.entity.ts","../src/entities/address.entity.ts","../src/entities/order.entity.ts","../src/entities/payment.entity.ts","../src/entities/chat-conversation.entity.ts","../src/entities/chat-message.entity.ts","../src/entities/config.entity.ts","../src/entities/message-template.entity.ts","../src/entities/media.entity.ts","../src/entities/page.entity.ts","../src/entities/product-category.entity.ts","../src/entities/collection.entity.ts","../src/entities/brand.entity.ts","../src/entities/product.entity.ts","../src/entities/attribute.entity.ts","../src/entities/product-attribute.entity.ts","../src/entities/tax.entity.ts","../src/entities/product-tax.entity.ts","../src/entities/order-item.entity.ts","../src/entities/knowledge-base-document.entity.ts","../src/entities/knowledge-base-chunk.entity.ts","../src/entities/cart.entity.ts","../src/entities/cart-item.entity.ts","../src/entities/wishlist.entity.ts","../src/entities/wishlist-item.entity.ts","../src/entities/index.ts","../src/auth/permission-entities.ts","../src/auth/helpers.ts","../src/auth/seed-permissions.ts","../src/auth/middleware.ts","../src/auth/nextauth-options.ts","../src/api/crud.ts","../src/api/auth-handlers.ts","../src/api/cms-handlers.ts","../src/api/message-template-admin-handlers.ts","../src/api/admin-roles-handlers.ts","../src/api/cms-api-handler.ts","../src/api/storefront-handlers.ts","../src/lib/is-valid-signup-email.ts","../src/admin/config.ts"],"sourcesContent":["import type { ERPSubmissionService } from './erp-submission';\nimport type { ContactFormData, ErpCreateContactPayload } from './erp-submission';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\nconst ERP_QUEUE_NAME = 'erp';\n\nexport type ErpJobPayload =\n | { kind: 'lead'; contact: ContactFormData }\n | { kind: 'formOpportunity'; contact: ContactFormData }\n | { kind: 'createContact'; contact: ErpCreateContactPayload }\n | { kind: 'order'; order: Record<string, unknown> }\n | { kind: 'productUpsert'; product: Record<string, unknown> };\n\nexport async function queueErp(cms: CmsAppLike, payload: ErpJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n if (!queue) return;\n await queue.add(ERP_QUEUE_NAME, payload as object);\n}\n\nexport function registerErpQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as\n | { registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void }\n | undefined;\n if (!queue) return;\n queue.registerProcessor(ERP_QUEUE_NAME, async (data: object) => {\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp) return;\n const payload = data as ErpJobPayload;\n if (payload.kind === 'lead') {\n await erp.submission.submitContact(payload.contact);\n } else if (payload.kind === 'formOpportunity') {\n await erp.submission.submitFormOpportunity(payload.contact);\n } else if (payload.kind === 'createContact') {\n await erp.submission.submitCreateContact(payload.contact);\n } else if (payload.kind === 'order') {\n await erp.submission.submitOrder(payload.order);\n } else if (payload.kind === 'productUpsert') {\n await erp.submission.submitProductUpsert(payload.product);\n }\n });\n}\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp, type CmsAppLike } from './erp-queue';\n\n/**\n * Duck-typed DataSource so host apps (e.g. Next) can pass their own TypeORM instance\n * without duplicate-package type errors vs core’s node_modules/typeorm.\n */\nexport type ErpPaidOrderDataSource = {\n getRepository(entity: unknown): {\n find(options?: object): Promise<unknown[]>;\n findOne(options?: object): Promise<unknown | null>;\n };\n};\n\n/**\n * Entity map with `configs` and `orders` entries (e.g. ENTITY_MAP).\n * Typed as `Record<string, unknown>` so apps that cast their map to `Record<string, EntityTarget<…>>` still pass.\n */\nexport type ErpPaidOrderEntityMap = Record<string, unknown>;\n\nfunction roundMoney(major: number): number {\n return Math.round(major * 100) / 100;\n}\n\nfunction addressToWebhookDto(a: ObjectLiteral | null | undefined): Record<string, unknown> {\n if (!a) return {};\n return {\n line1: a.line1 ?? '',\n line2: a.line2 ?? '',\n city: a.city ?? '',\n state: a.state ?? '',\n postalCode: a.postalCode ?? '',\n country: a.country ?? '',\n };\n}\n\nfunction orderStatusLabel(status: string | undefined): string {\n const s = (status || '').toLowerCase();\n if (s === 'confirmed') return 'Confirmed';\n if (s === 'pending') return 'Pending';\n if (!status) return 'Pending';\n return status.charAt(0).toUpperCase() + status.slice(1);\n}\n\nfunction paymentRowToWebhookDto(p: ObjectLiteral, amountMajorOverride?: number): Record<string, unknown> {\n const currency = String(p.currency || 'INR');\n const amountMajor =\n amountMajorOverride != null && Number.isFinite(amountMajorOverride)\n ? amountMajorOverride\n : Number(p.amount);\n const meta = { ...((p.metadata as Record<string, unknown>) || {}) };\n delete meta.amount;\n delete meta.currency;\n return {\n id: String(p.externalReference || `payment_${p.id}`),\n amount: roundMoney(amountMajor),\n currency_code: currency,\n captured_at: p.paidAt\n ? new Date(p.paidAt as Date | string).toISOString()\n : new Date().toISOString(),\n provider_id: String(p.method || 'unknown'),\n data: { status: 'captured', ...meta },\n };\n}\n\n/**\n * Enqueues ERP `order.created` only when the order has at least one **completed** payment.\n * Include `payments` in the payload.\n */\nexport async function queueErpPaidOrderForOrderId(\n cms: CmsAppLike,\n dataSource: ErpPaidOrderDataSource,\n entityMap: ErpPaidOrderEntityMap,\n orderId: number\n): Promise<void> {\n try {\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return;\n }\n if (!cms.getPlugin('erp')) return;\n\n const orderRepo = dataSource.getRepository(entityMap.orders);\n const ord = await orderRepo.findOne({\n where: { id: orderId } as ObjectLiteral,\n relations: ['items', 'items.product', 'contact', 'billingAddress', 'shippingAddress', 'payments'],\n });\n if (!ord) return;\n const o = ord as ObjectLiteral;\n const okKind = o.orderKind === undefined || o.orderKind === null || o.orderKind === 'sale';\n if (!okKind) return;\n const rawPayments = (o.payments as ObjectLiteral[]) ?? [];\n const completedPayments = rawPayments.filter((pay) => pay.status === 'completed' && pay.deleted !== true);\n if (!completedPayments.length) return;\n\n const rawItems = (o.items as ObjectLiteral[]) ?? [];\n const lines = rawItems\n .filter((it) => it.product)\n .map((it) => {\n const p = it.product as ObjectLiteral;\n const sku = (p.sku as string) || `SKU-${p.id}`;\n const itemType =\n typeof it.productType === 'string' && it.productType.trim()\n ? String(it.productType).trim()\n : p.type === 'service'\n ? 'service'\n : 'product';\n return {\n sku,\n quantity: Number(it.quantity) || 1,\n unitPrice: Number(it.unitPrice),\n title: (p.name as string) || sku,\n discount: 0,\n tax: Number(it.tax) || 0,\n uom: (typeof it.uom === 'string' && it.uom.trim() ? it.uom : p.uom) || undefined,\n tax_code:\n typeof it.taxCode === 'string' && it.taxCode.trim() ? String(it.taxCode).trim() : undefined,\n hsn_number:\n (typeof it.hsn === 'string' && it.hsn.trim() ? it.hsn : p.hsn) || undefined,\n type: itemType,\n };\n });\n if (!lines.length) return;\n\n const contact = o.contact as ObjectLiteral | undefined;\n const orderTotalMajor = Number(o.total);\n const paymentDtos =\n completedPayments.length === 1 && Number.isFinite(orderTotalMajor)\n ? [paymentRowToWebhookDto(completedPayments[0]!, orderTotalMajor)]\n : completedPayments.map((pay) => paymentRowToWebhookDto(pay));\n const baseMeta =\n o.metadata && typeof o.metadata === 'object' && !Array.isArray(o.metadata)\n ? { ...(o.metadata as Record<string, unknown>) }\n : {};\n\n const orderDto: Record<string, unknown> = {\n platformType: 'website',\n platformOrderId: String(o.orderNumber),\n platformOrderNumber: String(o.orderNumber),\n order_date: o.createdAt ? new Date(o.createdAt as Date | string).toISOString() : undefined,\n status: orderStatusLabel(o.status as string | undefined),\n customer: {\n name: (contact?.name as string) || '',\n email: (contact?.email as string) || '',\n phone: (contact?.phone as string) || '',\n },\n shippingAddress: addressToWebhookDto(o.shippingAddress as ObjectLiteral),\n billingAddress: addressToWebhookDto(o.billingAddress as ObjectLiteral),\n items: lines,\n payments: paymentDtos,\n metadata: { ...baseMeta, source: 'storefront' },\n };\n\n await queueErp(cms, { kind: 'order', order: orderDto });\n } catch {\n /* non-fatal */\n }\n}\n","import type { OrderFulfillmentEvent, OrderFulfillmentMetadata } from '../../lib/order-storefront-metadata';\n\nfunction pickString(o: Record<string, unknown>, keys: string[]): string | undefined {\n for (const k of keys) {\n const v = o[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\nexport function unwrapErpReadData(json: unknown): Record<string, unknown> | null {\n if (!json || typeof json !== 'object') return null;\n const o = json as Record<string, unknown>;\n const d = o.data;\n if (d && typeof d === 'object' && !Array.isArray(d)) return d as Record<string, unknown>;\n return o;\n}\n\nfunction firstObject(\n data: Record<string, unknown>,\n keys: string[]\n): Record<string, unknown> | null {\n for (const k of keys) {\n const v = data[k];\n if (v && typeof v === 'object' && !Array.isArray(v)) return v as Record<string, unknown>;\n }\n return null;\n}\n\nfunction extractEvents(src: Record<string, unknown>): OrderFulfillmentEvent[] | undefined {\n const timeline = src.timeline ?? src.events ?? src.history ?? src.trackingEvents;\n if (!Array.isArray(timeline) || !timeline.length) return undefined;\n const events: OrderFulfillmentEvent[] = [];\n for (const row of timeline) {\n if (!row || typeof row !== 'object') continue;\n const r = row as Record<string, unknown>;\n const at =\n pickString(r, ['at', 'timestamp', 'date', 'occurredAt']) ??\n (r.at instanceof Date ? (r.at as Date).toISOString() : undefined);\n const label = pickString(r, ['label', 'status', 'title', 'message', 'description']);\n const detail = pickString(r, ['detail', 'notes', 'description']);\n if (at || label || detail) events.push({ at, label, detail });\n }\n return events.length ? events : undefined;\n}\n\nexport function mapErpPayloadToFulfillment(data: Record<string, unknown>): OrderFulfillmentMetadata | undefined {\n const nested = firstObject(data, ['fulfillment', 'packaging', 'shipment', 'shipping', 'delivery']);\n const src = nested || data;\n const status = pickString(src, ['status', 'fulfillmentStatus', 'state', 'label', 'packagingStatus']);\n const trackingId = pickString(src, ['trackingId', 'tracking_id', 'trackingNumber', 'awb', 'trackingUrl']);\n const events = extractEvents(src);\n if (!status && !trackingId && !(events && events.length)) return undefined;\n return { status, trackingId, events };\n}\n\nexport function mapErpPayloadToInvoiceNumber(data: Record<string, unknown>): string | undefined {\n const nested = firstObject(data, ['invoice', 'latestInvoice', 'postedInvoice']);\n const src = nested || data;\n return pickString(src, ['invoiceNumber', 'invoice_number', 'number', 'name', 'id']);\n}\n\nexport type ErpChildOrderRef = { ref: string; orderKind: 'return' | 'replacement' };\n\nexport function extractChildOrderRefsFromSalePayload(data: Record<string, unknown>): ErpChildOrderRef[] {\n const lists = [data.returns, data.returnOrders, data.relatedReturns, data.childOrders, data.children];\n const seen = new Set<string>();\n const out: ErpChildOrderRef[] = [];\n for (const list of lists) {\n if (!Array.isArray(list)) continue;\n for (const item of list) {\n if (!item || typeof item !== 'object') continue;\n const o = item as Record<string, unknown>;\n const ref =\n pickString(o, ['platformReturnId', 'platform_return_id', 'refId', 'ref_id']) ??\n (typeof o.id === 'string' ? o.id : undefined);\n if (!ref || seen.has(ref)) continue;\n seen.add(ref);\n const t = (pickString(o, ['kind', 'type', 'orderKind']) || '').toLowerCase();\n const orderKind: 'return' | 'replacement' = /replac/.test(t) ? 'replacement' : 'return';\n out.push({ ref, orderKind });\n }\n }\n return out;\n}\n","import type { ObjectLiteral } from 'typeorm';\n\nexport type CmsAppLike = { getPlugin(name: string): unknown };\n\nexport type ErpConfigDataSource = {\n getRepository(entity: unknown): {\n find(options?: object): Promise<unknown[]>;\n };\n};\n\n/** True when ERP plugin is loaded and config `erp.enabled` is not `'false'`. */\nexport async function isErpIntegrationEnabled(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>\n): Promise<boolean> {\n if (!cms.getPlugin('erp')) return false;\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return false;\n }\n return true;\n}\n","import type { ObjectLiteral, Repository } from 'typeorm';\nimport { unwrapErpReadData } from './erp-response-map';\nimport type { ERPSubmissionService } from './erp-submission';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nfunction pickInvoiceId(data: Record<string, unknown>): string | undefined {\n const nested =\n data.invoice && typeof data.invoice === 'object' && !Array.isArray(data.invoice)\n ? (data.invoice as Record<string, unknown>)\n : null;\n const src = nested || data;\n for (const k of ['invoiceId', 'invoice_id', 'id']) {\n const v = src[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\n/**\n * Streams invoice PDF for a **sale** order. Resolves `invoiceId` from order metadata or `get-invoice`.\n */\nexport async function streamOrderInvoicePdf(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n orderId: number,\n options: { ownerContactId?: number }\n): Promise<Response> {\n const jsonErr = (msg: string, status: number) =>\n new Response(JSON.stringify({ error: msg }), {\n status,\n headers: { 'Content-Type': 'application/json' },\n });\n\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return jsonErr('Invoice not available', 503);\n\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp?.submission) return jsonErr('Invoice not available', 503);\n\n const orderRepo = dataSource.getRepository(entityMap.orders) as Repository<ObjectLiteral>;\n const order = await orderRepo.findOne({ where: { id: orderId, deleted: false } as ObjectLiteral });\n if (!order) return jsonErr('Not found', 404);\n\n const kind = (order.orderKind as string) || 'sale';\n if (kind !== 'sale') return jsonErr('Invoice only for sale orders', 400);\n\n if (options.ownerContactId != null && order.contactId !== options.ownerContactId) {\n return jsonErr('Not found', 404);\n }\n\n const meta =\n order.metadata && typeof order.metadata === 'object' && !Array.isArray(order.metadata)\n ? (order.metadata as Record<string, unknown>)\n : {};\n const inv = meta.invoice && typeof meta.invoice === 'object' && !Array.isArray(meta.invoice)\n ? (meta.invoice as Record<string, unknown>)\n : {};\n let invoiceId = typeof inv.invoiceId === 'string' ? inv.invoiceId.trim() : '';\n\n if (!invoiceId) {\n const refId = String(order.orderNumber || '');\n const r = await erp.submission.postErpReadAction('get-invoice', { platformOrderId: refId });\n const d = r.ok ? unwrapErpReadData(r.json) : null;\n invoiceId = d ? pickInvoiceId(d) || '' : '';\n }\n\n if (!invoiceId) return jsonErr('Invoice not ready', 404);\n\n const pdf = await erp.submission.fetchInvoicePdf(invoiceId);\n if (!pdf.ok || !pdf.buffer) return jsonErr(pdf.error || 'PDF fetch failed', 502);\n\n const filename = `invoice-${orderId}.pdf`;\n return new Response(pdf.buffer, {\n status: 200,\n headers: {\n 'Content-Type': pdf.contentType || 'application/pdf',\n 'Content-Disposition': `attachment; filename=\"${filename}\"`,\n },\n });\n}\n","import type { CompanyDetails, EmailTemplateName, OrderPlacedLineItem } from './templates/types';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\n/** Context for template rendering: at least companyDetails; template-specific fields (formName, etc.) are allowed. */\nexport interface EmailJobPayload {\n to: string;\n templateName?: EmailTemplateName;\n ctx?: Record<string, unknown> & { companyDetails: CompanyDetails };\n subject?: string;\n html?: string;\n text?: string;\n}\n\nconst EMAIL_QUEUE_NAME = 'email';\n\nexport function registerEmailQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void>; registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void } | undefined;\n const email = cms.getPlugin('email') as { send: (opts: { to: string; subject: string; html: string; text?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (!queue || !email) return;\n queue.registerProcessor(EMAIL_QUEUE_NAME, async (data: object) => {\n const payload = data as EmailJobPayload;\n const { to, templateName, ctx, subject, html, text } = payload;\n if (!to) return;\n if (templateName && ctx) {\n const rendered = email.renderTemplate(templateName, ctx);\n await email.send({ to, subject: rendered.subject, html: rendered.html, text: rendered.text });\n } else if (subject != null && html != null) {\n await email.send({ to, subject, html, text });\n }\n });\n}\n\nexport async function queueEmail(cms: CmsAppLike, payload: EmailJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n if (queue) {\n await queue.add(EMAIL_QUEUE_NAME, payload);\n return;\n }\n const email = cms.getPlugin('email') as { send: (opts: { to: string; subject: string; html: string; text?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (email && payload.templateName && payload.ctx) {\n const rendered = email.renderTemplate(payload.templateName, payload.ctx);\n await email.send({ to: payload.to, subject: rendered.subject, html: rendered.html, text: rendered.text });\n } else if (email && payload.subject != null && payload.html != null) {\n await email.send({ to: payload.to, subject: payload.subject, html: payload.html, text: payload.text });\n }\n}\n\nexport interface OrderPlacedEmailPayload {\n orderNumber: string;\n total?: string | number;\n subtotal?: string | number;\n tax?: string | number;\n currency?: string;\n customerName?: string;\n /** Customer receipt; omitted if empty */\n customerEmail?: string | null;\n /** Parsed list from email plugin “Sales team” settings */\n salesTeamEmails: string[];\n companyDetails: CompanyDetails;\n lineItems: OrderPlacedLineItem[];\n billingAddress?: Record<string, string>;\n shippingAddress?: Record<string, string>;\n}\n\n/** Queues one `orderPlaced` email per recipient (customer + each unique sales address; skips sales if same as customer). */\nexport async function queueOrderPlacedEmails(cms: CmsAppLike, payload: OrderPlacedEmailPayload): Promise<void> {\n const {\n orderNumber,\n total,\n subtotal,\n tax,\n currency,\n customerName,\n customerEmail,\n salesTeamEmails,\n companyDetails,\n lineItems,\n billingAddress,\n shippingAddress,\n } = payload;\n const base = {\n orderNumber,\n total: total != null ? String(total) : undefined,\n subtotal: subtotal != null ? String(subtotal) : undefined,\n tax: tax != null ? String(tax) : undefined,\n currency,\n customerName,\n companyDetails: companyDetails ?? {},\n lineItems: lineItems ?? [],\n billingAddress,\n shippingAddress,\n };\n const customerLower = customerEmail?.trim().toLowerCase() ?? '';\n const jobs: Promise<void>[] = [];\n if (customerEmail?.trim()) {\n jobs.push(\n queueEmail(cms, {\n to: customerEmail.trim(),\n templateName: 'orderPlaced',\n ctx: { ...base, audience: 'customer' as const },\n })\n );\n }\n const seen = new Set<string>();\n for (const raw of salesTeamEmails) {\n const to = raw.trim();\n if (!to) continue;\n const key = to.toLowerCase();\n if (seen.has(key)) continue;\n seen.add(key);\n if (customerLower && key === customerLower) continue;\n jobs.push(\n queueEmail(cms, {\n to,\n templateName: 'orderPlaced',\n ctx: {\n ...base,\n audience: 'sales' as const,\n internalCustomerEmail: customerEmail?.trim() || undefined,\n },\n })\n );\n }\n await Promise.all(jobs);\n}\n","export type SmsMessageTemplateDefault = {\n templateKey: string;\n name: string;\n body: string;\n externalTemplateRef?: string;\n providerMeta?: { otpVarKey?: string };\n};\n\nexport const SMS_MESSAGE_TEMPLATE_DEFAULTS: SmsMessageTemplateDefault[] = [\n {\n templateKey: 'auth.otp_login',\n name: 'Sign-in OTP (SMS)',\n body: 'Your sign-in code is {{code}}. Valid 10 minutes.',\n providerMeta: { otpVarKey: 'var1' },\n },\n {\n templateKey: 'auth.otp_verify_phone',\n name: 'Verify phone OTP (SMS)',\n body: 'Your verification code is {{code}}. Valid 10 minutes.',\n providerMeta: { otpVarKey: 'var1' },\n },\n];\n\nexport function getSmsTemplateDefault(templateKey: string): SmsMessageTemplateDefault | undefined {\n return SMS_MESSAGE_TEMPLATE_DEFAULTS.find((d) => d.templateKey === templateKey);\n}\n\nexport type ResolvedSmsTemplate = {\n body: string;\n externalTemplateRef: string | undefined;\n otpVarKey: string;\n};\n\nexport async function resolveSmsTemplateForSend(\n templateKey: string,\n getRow: (channel: string, key: string) => Promise<{\n body: string;\n externalTemplateRef: string | null;\n providerMeta: Record<string, unknown> | null;\n enabled: boolean;\n } | null>\n): Promise<ResolvedSmsTemplate | null> {\n const def = getSmsTemplateDefault(templateKey);\n if (!def) return null;\n\n const row = await getRow('sms', templateKey);\n let body = def.body;\n let ref = def.externalTemplateRef;\n let otpVarKey = def.providerMeta?.otpVarKey ?? 'var1';\n\n if (row && row.enabled !== false) {\n if (row.body.trim()) body = row.body;\n if (row.externalTemplateRef?.trim()) ref = row.externalTemplateRef.trim();\n const pm = row.providerMeta;\n const k = pm && typeof pm.otpVarKey === 'string' ? pm.otpVarKey.trim() : '';\n if (k) otpVarKey = k;\n }\n\n return {\n body,\n externalTemplateRef: ref?.trim() || undefined,\n otpVarKey: otpVarKey || 'var1',\n };\n}\n","/** Replace `{{key}}` placeholders (also accepts legacy `{{otp}}` as alias for `code` when code is set). */\nexport function applyTemplateVars(template: string, variables: Record<string, string>): string {\n let out = template;\n for (const [k, v] of Object.entries(variables)) {\n out = out.split(`{{${k}}}`).join(v);\n }\n if (variables.code !== undefined) {\n out = out.split('{{otp}}').join(variables.code);\n }\n return out;\n}\n","import Stripe from 'stripe';\nimport type { PaymentServiceInterface, PaymentIntent } from './index';\n\nexport interface StripeServiceConfig {\n secretKey: string;\n webhookSecret?: string;\n}\n\nexport class StripePaymentService implements PaymentServiceInterface {\n private stripe: Stripe;\n private webhookSecret?: string;\n\n constructor(config: StripeServiceConfig) {\n this.stripe = new Stripe(config.secretKey, { apiVersion: '2024-06-20' });\n this.webhookSecret = config.webhookSecret;\n }\n\n async createPaymentIntent(\n amount: number,\n currency: string,\n metadata?: Record<string, string>\n ): Promise<PaymentIntent> {\n const intent = await this.stripe.paymentIntents.create({\n amount: Math.round(amount * 100),\n currency: currency.toLowerCase(),\n metadata,\n });\n\n return {\n id: intent.id,\n clientSecret: intent.client_secret || '',\n amount,\n currency,\n status: intent.status,\n };\n }\n\n async capturePayment(paymentId: string, amount?: number): Promise<boolean> {\n try {\n await this.stripe.paymentIntents.capture(paymentId, {\n amount_to_capture: amount ? Math.round(amount * 100) : undefined,\n });\n return true;\n } catch {\n return false;\n }\n }\n\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean {\n if (!this.webhookSecret) return false;\n try {\n this.stripe.webhooks.constructEvent(payload, signature, this.webhookSecret);\n return true;\n } catch {\n return false;\n }\n }\n}\n","import Razorpay from 'razorpay';\nimport crypto from 'crypto';\nimport type { PaymentServiceInterface, PaymentIntent } from './index';\n\nexport interface RazorpayServiceConfig {\n keyId: string;\n keySecret: string;\n webhookSecret?: string;\n}\n\nexport class RazorpayPaymentService implements PaymentServiceInterface {\n private razorpay: Razorpay;\n private keySecret: string;\n private webhookSecret?: string;\n\n constructor(config: RazorpayServiceConfig) {\n this.razorpay = new Razorpay({\n key_id: config.keyId,\n key_secret: config.keySecret,\n });\n this.keySecret = config.keySecret;\n this.webhookSecret = config.webhookSecret;\n }\n\n async createPaymentIntent(\n amount: number,\n currency: string,\n metadata?: Record<string, string>\n ): Promise<PaymentIntent> {\n const order = await this.razorpay.orders.create({\n amount: Math.round(amount * 100),\n currency: currency.toUpperCase(),\n notes: metadata,\n });\n\n return {\n id: order.id,\n clientSecret: order.id,\n amount,\n currency,\n status: order.status,\n };\n }\n\n async capturePayment(paymentId: string, amount?: number): Promise<boolean> {\n try {\n await this.razorpay.payments.capture(paymentId, amount ? Math.round(amount * 100) : 0, 'INR');\n return true;\n } catch {\n return false;\n }\n }\n\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean {\n const secret = this.webhookSecret || this.keySecret;\n const expectedSignature = crypto\n .createHmac('sha256', secret)\n .update(typeof payload === 'string' ? payload : payload.toString('utf8'))\n .digest('hex');\n return crypto.timingSafeEqual(Buffer.from(signature), Buffer.from(expectedSignature));\n }\n}\n","export * from './plugins/types';\nexport * from './plugins/registry';\nexport {\n erpPlugin,\n ERPSubmissionService,\n registerErpQueueProcessor,\n queueErp,\n queueErpPaidOrderForOrderId,\n queueErpCreateContactIfEnabled,\n type ErpPaidOrderDataSource,\n type ErpPaidOrderEntityMap,\n type ErpContactSyncInput,\n} from './plugins/erp';\nexport type {\n ERPPluginConfig,\n ERPPluginInstance,\n ErpJobPayload,\n ErpCreateContactPayload,\n PipelineNames,\n} from './plugins/erp';\nexport { emailPlugin, EmailService, emailTemplates, renderEmail, getCompanyDetailsFromSettings, mergeEmailLayoutCompanyDetails, renderLayout, registerEmailQueueProcessor, queueEmail, queueOrderPlacedEmails } from './plugins/email';\nexport type {\n EmailPluginConfig,\n EmailData,\n EmailServiceInterface,\n RenderedEmail,\n CompanyDetails,\n SocialLinkItem,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n RenderEmailOptions,\n EmailJobPayload,\n OrderPlacedEmailPayload,\n OrderPlacedLineItem,\n} from './plugins/email';\nexport { analyticsPlugin } from './plugins/analytics';\nexport type { AnalyticsPluginConfig } from './plugins/analytics';\nexport { smsPlugin, registerSmsQueueProcessor, queueSms } from './plugins/sms';\nexport type {\n SmsPluginConfig,\n SmsServiceInterface,\n SmsJobPayload,\n SmsProviderId,\n SmsProviderChoice,\n SmsServiceConfig,\n} from './plugins/sms';\nexport { paymentPlugin } from './plugins/payment';\nexport type { PaymentPluginConfig, PaymentServiceInterface, PaymentIntent } from './plugins/payment';\nexport { s3StoragePlugin, localStoragePlugin } from './plugins/storage';\nexport type { StorageService, S3StoragePluginConfig, LocalStoragePluginConfig } from './plugins/storage';\nexport { llmPlugin, LlmService } from './plugins/llm';\nexport type { LlmPluginConfig, LlmServiceInterface, LlmMessage, LlmChatOptions } from './plugins/llm';\nexport { cachePlugin } from './plugins/cache';\nexport type { CacheService, CachePluginConfig } from './plugins/cache';\nexport { queuePlugin } from './plugins/queue';\nexport type { QueueService, QueuePluginConfig } from './plugins/queue';\nexport { captchaPlugin, CaptchaService, buildCaptchaPublicConfig, assertCaptchaOk } from './plugins/captcha';\nexport type { CaptchaPublicConfig, CaptchaProviderId, CaptchaVerifyResult } from './plugins/captcha';\nexport * from './lib';\nexport * from './entities';\nexport * from './auth';\nexport * from './api';\nexport { DEFAULT_ADMIN_NAV } from './admin/config';\nexport type { AdminNavItem } from './admin/config';\n","import type { CmsPlugin, PluginContext } from './types';\n\nexport interface CreateCmsAppOptions {\n dataSource: unknown;\n config?: Record<string, string>;\n plugins?: CmsPlugin[];\n logger?: PluginContext['logger'];\n}\n\nconst noopLogger: PluginContext['logger'] = {\n info: () => {},\n warn: () => {},\n error: () => {},\n};\n\nexport interface CmsApp {\n dataSource: unknown;\n getPlugin<T = unknown>(name: string): T | undefined;\n}\n\nexport async function createCmsApp(options: CreateCmsAppOptions): Promise<CmsApp> {\n const { dataSource, config = {}, plugins = [], logger = noopLogger } = options;\n const context: PluginContext = { dataSource, config, logger };\n const registry = new Map<string, unknown>();\n\n for (const plugin of plugins) {\n try {\n const instance = await plugin.init(context);\n registry.set(plugin.name, instance !== undefined ? instance : plugin);\n } catch (err) {\n logger.warn(`Plugin \"${plugin.name}\" failed to init: ${err instanceof Error ? err.message : err}`);\n }\n }\n\n return {\n dataSource,\n getPlugin<T>(name: string): T | undefined {\n return registry.get(name) as T | undefined;\n },\n };\n}\n","export interface ContactFormData {\n firstName: string;\n lastName: string;\n email: string;\n phone?: string;\n industry?: string;\n message?: string;\n}\n\nexport interface ERPSubmissionResult {\n success: boolean;\n error?: string;\n status?: number;\n}\n\nexport type PipelineNames = { pipelineName: string; pipelineStageName: string };\n\n/** Payload for ERP `create-contact` (§7c) — CRM contact upsert, no lead/opportunity row. */\nexport interface ErpCreateContactPayload {\n email: string;\n firstName: string;\n lastName: string;\n phone?: string;\n companyName?: string;\n type?: string;\n notes?: string;\n tags?: string[];\n}\n\nexport class ERPSubmissionService {\n private webhookUrl: string;\n private webhookJwt: string;\n private getPipelineNames?: () => Promise<PipelineNames>;\n\n constructor(config: {\n webhookUrl: string;\n webhookJwt: string;\n getPipelineNames?: () => Promise<PipelineNames>;\n }) {\n this.webhookUrl = config.webhookUrl.replace(/\\/$/, '');\n this.webhookJwt = config.webhookJwt;\n this.getPipelineNames = config.getPipelineNames;\n }\n\n /** Replace trailing path segment of webhook URL with `action` (e.g. `get-order-status`, `invoice-pdf`). */\n resolveErpActionUrl(action: string): string {\n const base = this.webhookUrl.replace(/\\/$/, '');\n const i = base.lastIndexOf('/');\n if (i === -1) return `${base}/${action}`;\n return `${base.slice(0, i + 1)}${action}`;\n }\n\n /**\n * Synchronous ERP read (§7d): POST JSON to `.../generic-webhook/<action>`.\n * Body is sent as-is (or wrap with envelope if ERP expects `event_type` — caller may pass envelope).\n */\n async postErpReadAction(\n action: string,\n body: Record<string, unknown>\n ): Promise<{ ok: boolean; status?: number; json?: unknown; error?: string }> {\n const url = this.resolveErpActionUrl(action);\n try {\n const res = await fetch(url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-External-Token': this.webhookJwt,\n },\n body: JSON.stringify(body),\n });\n const text = await res.text();\n let parsed: unknown;\n try {\n parsed = text ? JSON.parse(text) : null;\n } catch {\n parsed = null;\n }\n if (!res.ok) {\n return { ok: false, status: res.status, error: `${res.status} ${text.slice(0, 500)}` };\n }\n return { ok: true, status: res.status, json: parsed };\n } catch (e: unknown) {\n return { ok: false, error: e instanceof Error ? e.message : 'ERP read request failed' };\n }\n }\n\n /** GET PDF bytes for `invoice-pdf` action (§7d). */\n async fetchInvoicePdf(invoiceId: string): Promise<{\n ok: boolean;\n buffer?: ArrayBuffer;\n contentType?: string;\n error?: string;\n }> {\n const base = this.resolveErpActionUrl('invoice-pdf');\n let u: URL;\n try {\n u = new URL(base);\n } catch {\n return { ok: false, error: 'Invalid ERP webhook URL' };\n }\n u.searchParams.set('invoiceId', invoiceId);\n u.searchParams.set('token', this.webhookJwt);\n try {\n const res = await fetch(u.toString(), { method: 'GET' });\n if (!res.ok) {\n const t = await res.text();\n return { ok: false, error: `${res.status} ${t.slice(0, 200)}` };\n }\n const buffer = await res.arrayBuffer();\n const contentType = res.headers.get('content-type') || 'application/pdf';\n return { ok: true, buffer, contentType };\n } catch (e: unknown) {\n return { ok: false, error: e instanceof Error ? e.message : 'invoice PDF fetch failed' };\n }\n }\n\n /** `product.updated` envelope → `update-product` (§7e). */\n async submitProductUpsert(productData: Record<string, unknown>): Promise<ERPSubmissionResult> {\n const envelope = {\n event_type: 'product.updated',\n timestamp: new Date().toISOString(),\n data: productData,\n };\n return this.postWebhookJson(envelope);\n }\n\n private async postWebhookJson(body: unknown): Promise<ERPSubmissionResult> {\n try {\n const res = await fetch(this.webhookUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-External-Token': this.webhookJwt,\n },\n body: JSON.stringify(body),\n });\n if (res.ok) return { success: true, status: res.status };\n const text = await res.text();\n return { success: false, error: `${res.status} ${text.slice(0, 500)}`, status: res.status };\n } catch (e: unknown) {\n const message = e instanceof Error ? e.message : 'ERP webhook request failed';\n return { success: false, error: message };\n }\n }\n\n private async resolvePipelineNames(): Promise<PipelineNames> {\n let pipelineName = '';\n let pipelineStageName = '';\n if (this.getPipelineNames) {\n try {\n const n = await this.getPipelineNames();\n pipelineName = (n.pipelineName ?? '').trim();\n pipelineStageName = (n.pipelineStageName ?? '').trim();\n } catch {\n /* ignore */\n }\n }\n return { pipelineName, pipelineStageName };\n }\n\n /** Generic webhook POST (e.g. order.created envelope). */\n async postWebhook(body: unknown): Promise<ERPSubmissionResult> {\n return this.postWebhookJson(body);\n }\n\n /** Shared CRM inbound fields (§7a/b in ERP-plugin.md). */\n private async buildCrmInboundData(formData: ContactFormData): Promise<{ error?: string; data?: Record<string, unknown> }> {\n const email = (formData.email ?? '').trim();\n if (!email) {\n return { error: 'email is required' };\n }\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n if (!emailRegex.test(email)) {\n return { error: 'Invalid email format' };\n }\n\n const firstName = (formData.firstName ?? '').trim() || email.split('@')[0] || 'Contact';\n const lastName = (formData.lastName ?? '').trim() || '';\n\n const { pipelineName, pipelineStageName } = await this.resolvePipelineNames();\n\n const data: Record<string, unknown> = {\n email,\n firstName,\n lastName,\n phone: formData.phone?.trim() || undefined,\n subject: formData.industry?.trim() || undefined,\n message: formData.message?.trim() || undefined,\n };\n if (pipelineName) data.pipelineName = pipelineName;\n if (pipelineStageName) data.pipelineStageName = pipelineStageName;\n return { data };\n }\n\n /** `lead.created` → ERP create-lead (CRM leads table). */\n async submitContact(formData: ContactFormData): Promise<ERPSubmissionResult> {\n const built = await this.buildCrmInboundData(formData);\n if (built.error || !built.data) {\n return { success: false, error: built.error ?? 'invalid payload' };\n }\n built.data.leadSource = 'Website';\n\n const envelope = {\n event_type: 'lead.created',\n timestamp: new Date().toISOString(),\n data: built.data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /** `create-contact` → ERP ContactService upsert (§7c). */\n async submitCreateContact(payload: ErpCreateContactPayload): Promise<ERPSubmissionResult> {\n const email = (payload.email ?? '').trim();\n if (!email) {\n return { success: false, error: 'email is required' };\n }\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n if (!emailRegex.test(email)) {\n return { success: false, error: 'Invalid email format' };\n }\n const firstName = (payload.firstName ?? '').trim() || email.split('@')[0] || 'Contact';\n const lastName = (payload.lastName ?? '').trim() || '';\n\n const data: Record<string, unknown> = {\n email,\n firstName,\n lastName,\n phone: payload.phone?.trim() || undefined,\n companyName: payload.companyName?.trim() || undefined,\n type: payload.type?.trim() || undefined,\n message: payload.notes?.trim() || undefined,\n metadata: { source: 'cms' },\n };\n if (payload.tags?.length) {\n data.tags = payload.tags;\n }\n\n const envelope = {\n event_type: 'create-contact',\n timestamp: new Date().toISOString(),\n data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /** `form.submitted` → ERP create-opportunity (pipeline). */\n async submitFormOpportunity(formData: ContactFormData): Promise<ERPSubmissionResult> {\n const built = await this.buildCrmInboundData(formData);\n if (built.error || !built.data) {\n return { success: false, error: built.error ?? 'invalid payload' };\n }\n\n const envelope = {\n event_type: 'form.submitted',\n timestamp: new Date().toISOString(),\n data: built.data,\n };\n return this.postWebhookJson(envelope);\n }\n\n /**\n * Order payload aligned with ERP generic webhook create-order (§6).\n */\n async submitOrder(orderDto: Record<string, unknown>): Promise<ERPSubmissionResult> {\n const envelope = {\n event_type: 'order.created',\n timestamp: new Date().toISOString(),\n data: orderDto,\n };\n return this.postWebhookJson(envelope);\n }\n\n extractContactData(formData: Record<string, unknown>, formFields: { id: string | number; type: string; label: string }[]): ContactFormData | null {\n const contactData: ContactFormData = {\n firstName: '',\n lastName: '',\n email: '',\n phone: '',\n industry: '',\n message: '',\n };\n let hasEmail = false;\n\n for (const field of formFields) {\n const fieldValue = formData[field.id.toString()];\n if (fieldValue == null) continue;\n const label = field.label.toLowerCase();\n const value = String(fieldValue).trim();\n\n if (field.type === 'email') {\n contactData.email = value;\n hasEmail = true;\n } else if (field.type === 'phone') {\n contactData.phone = value;\n } else if (field.type === 'text' || field.type === 'textarea') {\n if (label.includes('first name') || label.includes('firstname')) {\n contactData.firstName = value;\n } else if (label.includes('last name') || label.includes('lastname')) {\n contactData.lastName = value;\n } else if (label.includes('name') && !contactData.firstName) {\n const nameParts = value.split(' ');\n if (nameParts.length >= 2) {\n contactData.firstName = nameParts[0]!;\n contactData.lastName = nameParts.slice(1).join(' ');\n } else {\n contactData.firstName = value;\n }\n } else if (label.includes('industry')) {\n contactData.industry = value;\n } else if (label.includes('message') || label.includes('comment') || label.includes('description') || label.includes('inquiry')) {\n contactData.message = value;\n }\n }\n }\n\n if (!hasEmail && !contactData.firstName) return null;\n if (!contactData.firstName && contactData.email) {\n contactData.firstName = contactData.email.split('@')[0] ?? '';\n contactData.lastName = '';\n }\n return contactData;\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { ERPSubmissionService } from './erp-submission';\nimport type { PipelineNames } from './erp-submission';\n\nexport interface ERPPluginConfig {\n webhookUrl: string;\n webhookJwt: string;\n getPipelineNames?: () => Promise<PipelineNames>;\n}\n\nexport interface ERPPluginInstance {\n submission: ERPSubmissionService;\n}\n\nexport function erpPlugin(config: ERPPluginConfig): CmsPlugin<ERPPluginInstance> {\n return {\n name: 'erp',\n version: '2.0.0',\n async init() {\n const submission = new ERPSubmissionService({\n webhookUrl: config.webhookUrl,\n webhookJwt: config.webhookJwt,\n getPipelineNames: config.getPipelineNames,\n });\n return { submission };\n },\n };\n}\n\nexport { ERPSubmissionService } from './erp-submission';\nexport type { ContactFormData, ERPSubmissionResult, ErpCreateContactPayload, PipelineNames } from './erp-submission';\nexport { registerErpQueueProcessor, queueErp } from './erp-queue';\nexport type { ErpJobPayload } from './erp-queue';\nexport { queueErpPaidOrderForOrderId, type ErpPaidOrderDataSource, type ErpPaidOrderEntityMap } from './paid-order-erp';\nexport { queueErpCreateContactIfEnabled, type ErpContactSyncInput } from './erp-contact-sync';\nexport { tryRefreshOrderFromErpForStorefront, refreshOrderFromErp } from './erp-order-sync';\nexport { streamOrderInvoicePdf } from './erp-order-invoice';\nexport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\nexport { queueErpProductUpsertIfEnabled } from './erp-product-sync';\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp, type CmsAppLike } from './erp-queue';\nimport type { ErpPaidOrderDataSource, ErpPaidOrderEntityMap } from './paid-order-erp';\n\nexport interface ErpContactSyncInput {\n name: string;\n email: string;\n phone?: string | null;\n type?: string | null;\n company?: string | null;\n notes?: string | null;\n /** Passed to ERP as `tags` when non-empty (§7c). */\n tags?: string[];\n}\n\nfunction splitName(full: string): { firstName: string; lastName: string } {\n const t = (full || '').trim();\n if (!t) return { firstName: 'Contact', lastName: '' };\n const parts = t.split(/\\s+/);\n if (parts.length === 1) return { firstName: parts[0]!, lastName: '' };\n return { firstName: parts[0]!, lastName: parts.slice(1).join(' ') };\n}\n\n/**\n * When ERP is enabled and plugin loaded, enqueue `create-contact` (non-fatal on failure).\n * Used for admin CRUD and storefront contact paths — not form submissions (those use lead/opportunity).\n */\nexport async function queueErpCreateContactIfEnabled(\n cms: CmsAppLike,\n dataSource: ErpPaidOrderDataSource,\n entityMap: ErpPaidOrderEntityMap,\n input: ErpContactSyncInput\n): Promise<void> {\n try {\n const configRepo = dataSource.getRepository(entityMap.configs);\n const cfgRows = await configRepo.find({ where: { settings: 'erp', deleted: false } as ObjectLiteral });\n for (const row of cfgRows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled' && r.value === 'false') return;\n }\n if (!cms.getPlugin('erp')) return;\n\n const email = (input.email ?? '').trim();\n if (!email) return;\n\n const { firstName, lastName } = splitName(input.name);\n\n await queueErp(cms, {\n kind: 'createContact',\n contact: {\n email,\n firstName,\n lastName,\n phone: input.phone?.trim() || undefined,\n companyName: input.company?.trim() || undefined,\n type: input.type?.trim() || undefined,\n notes: input.notes?.trim() || undefined,\n tags: input.tags?.length ? [...input.tags] : undefined,\n },\n });\n } catch {\n /* non-fatal */\n }\n}\n","export type OrderKind = 'sale' | 'return' | 'replacement';\n\nconst KIND_PREFIX: Record<OrderKind, string> = {\n sale: 'OSL',\n return: 'ORT',\n replacement: 'ORP',\n};\n\n/** YYMM in UTC (two-digit year + month). */\nexport function orderNumberYymmUtc(at: Date): string {\n const yy = String(at.getUTCFullYear()).slice(-2);\n const mm = String(at.getUTCMonth() + 1).padStart(2, '0');\n return yy + mm;\n}\n\n/** Stable 8-char mask from numeric id (not reversible as plain decimal). */\nexport function maskOrderIdSegment(id: number): string {\n let x = (id >>> 0) ^ 0xa5a5a5a5;\n x = Math.imul(x, 2654435761) >>> 0;\n return x.toString(36).toUpperCase().padStart(8, '0').slice(-8);\n}\n\nexport function buildCanonicalOrderNumber(kind: OrderKind, id: number, at: Date): string {\n return KIND_PREFIX[kind] + orderNumberYymmUtc(at) + maskOrderIdSegment(id);\n}\n\nexport function temporaryOrderNumberPlaceholder(): string {\n return `TMP${Date.now().toString(36)}${Math.random().toString(36).slice(2, 10)}`.toUpperCase();\n}\n","export interface OrderFulfillmentEvent {\n at?: string;\n label?: string;\n detail?: string;\n}\n\nexport interface OrderFulfillmentMetadata {\n status?: string;\n trackingId?: string;\n events?: OrderFulfillmentEvent[];\n}\n\nexport interface OrderInvoiceMetadata {\n invoiceNumber?: string;\n /** ERP invoice UUID for PDF fetch (no JWT). */\n invoiceId?: string;\n /** App path for invoice PDF (storefront or site-relative). */\n link?: string;\n}\n\nexport interface OrderStorefrontMetadataShape {\n fulfillment?: OrderFulfillmentMetadata;\n invoice?: OrderInvoiceMetadata;\n cartId?: number;\n platformRef?: string;\n [key: string]: unknown;\n}\n\nexport function mergeOrderMetadataPatch(\n existing: Record<string, unknown> | null | undefined,\n patch: { fulfillment?: OrderFulfillmentMetadata | null; invoice?: OrderInvoiceMetadata | null }\n): Record<string, unknown> {\n const base =\n existing && typeof existing === 'object' && !Array.isArray(existing) ? { ...existing } : {};\n if (patch.fulfillment !== undefined) {\n if (patch.fulfillment === null) delete base.fulfillment;\n else base.fulfillment = patch.fulfillment;\n }\n if (patch.invoice !== undefined) {\n if (patch.invoice === null) delete base.invoice;\n else base.invoice = patch.invoice;\n }\n return base;\n}\n","import type { Order } from '../../entities/order.entity';\n\n/** Map ERP sale-order status labels into CMS `Order.status`. Unknown → undefined (caller keeps existing). */\nexport function mapErpSaleStatusToOrderStatus(erpLabel: string | undefined): Order['status'] | undefined {\n if (!erpLabel || typeof erpLabel !== 'string') return undefined;\n const k = erpLabel.trim().toLowerCase().replace(/\\s+/g, '_');\n const map: Record<string, Order['status']> = {\n draft: 'pending',\n pending: 'pending',\n open: 'pending',\n new: 'pending',\n unconfirmed: 'pending',\n confirmed: 'confirmed',\n processing: 'processing',\n packed: 'processing',\n shipped: 'processing',\n in_transit: 'processing',\n out_for_delivery: 'processing',\n delivered: 'completed',\n completed: 'completed',\n closed: 'completed',\n fulfilled: 'completed',\n cancelled: 'cancelled',\n canceled: 'cancelled',\n void: 'cancelled',\n };\n return map[k];\n}\n","import type { ObjectLiteral, Repository } from 'typeorm';\nimport { buildCanonicalOrderNumber, temporaryOrderNumberPlaceholder, type OrderKind } from '../../lib/order-number';\nimport { mergeOrderMetadataPatch, type OrderFulfillmentMetadata, type OrderInvoiceMetadata } from '../../lib/order-storefront-metadata';\nimport { mapErpSaleStatusToOrderStatus } from './erp-order-status-map';\nimport {\n extractChildOrderRefsFromSalePayload,\n mapErpPayloadToFulfillment,\n mapErpPayloadToInvoiceNumber,\n unwrapErpReadData,\n} from './erp-response-map';\nimport type { ERPSubmissionService } from './erp-submission';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nfunction pickInvoiceId(data: Record<string, unknown>): string | undefined {\n const nested =\n data.invoice && typeof data.invoice === 'object' && !Array.isArray(data.invoice)\n ? (data.invoice as Record<string, unknown>)\n : null;\n const src = nested || data;\n for (const k of ['invoiceId', 'invoice_id', 'id']) {\n const v = src[k];\n if (typeof v === 'string' && v.trim()) return v.trim();\n }\n return undefined;\n}\n\nasync function ensureChildOrdersFromRefs(\n orderRepo: Repository<ObjectLiteral>,\n parent: ObjectLiteral,\n refs: ReturnType<typeof extractChildOrderRefsFromSalePayload>,\n contactId: number,\n currency: string\n): Promise<void> {\n for (const { ref, orderKind } of refs) {\n const existing = await orderRepo\n .createQueryBuilder('o')\n .where('o.parentOrderId = :pid', { pid: parent.id as number })\n .andWhere('o.deleted = :d', { d: false })\n .andWhere(\"o.metadata->>'platformRef' = :ref\", { ref })\n .getOne();\n if (existing) continue;\n const tmp = temporaryOrderNumberPlaceholder();\n const row = await orderRepo.save(\n orderRepo.create({\n orderNumber: tmp,\n orderKind,\n parentOrderId: parent.id as number,\n contactId,\n billingAddressId: null,\n shippingAddressId: null,\n status: 'pending',\n subtotal: 0,\n tax: 0,\n discount: 0,\n total: 0,\n currency,\n metadata: { platformRef: ref },\n deleted: false,\n } as ObjectLiteral)\n );\n const r = row as { id: number; createdAt: Date };\n await orderRepo.update(r.id, {\n orderNumber: buildCanonicalOrderNumber(orderKind as OrderKind, r.id, r.createdAt ?? new Date()),\n } as ObjectLiteral);\n }\n}\n\nfunction deepMergeFulfillment(\n a: OrderFulfillmentMetadata | undefined,\n b: OrderFulfillmentMetadata | undefined\n): OrderFulfillmentMetadata | undefined {\n if (!a) return b;\n if (!b) return a;\n return {\n ...a,\n ...b,\n events: b.events?.length ? b.events : a.events,\n };\n}\n\n/**\n * Calls ERP §7d reads for the given order row, updates `status` + `metadata` (fulfillment / invoice),\n * and upserts child return/replacement rows when the sale payload lists them.\n */\nexport async function refreshOrderFromErp(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n submission: ERPSubmissionService,\n order: ObjectLiteral\n): Promise<void> {\n const orderRepo = dataSource.getRepository(entityMap.orders) as Repository<ObjectLiteral>;\n\n const kind = (order.orderKind as string) || 'sale';\n const meta =\n order.metadata && typeof order.metadata === 'object' && !Array.isArray(order.metadata)\n ? { ...(order.metadata as Record<string, unknown>) }\n : {};\n\n if (kind === 'sale') {\n const refId = String(order.orderNumber || '');\n let fulfillment: OrderFulfillmentMetadata | undefined;\n let invoiceNumber: string | undefined;\n let invoiceId: string | undefined;\n let newStatus: string | undefined;\n\n const r1 = await submission.postErpReadAction('get-order-status', { platformOrderId: refId });\n const d1 = r1.ok ? unwrapErpReadData(r1.json) : null;\n if (d1) {\n const mapped = mapErpSaleStatusToOrderStatus(\n typeof d1.status === 'string'\n ? d1.status\n : typeof d1.orderStatus === 'string'\n ? d1.orderStatus\n : typeof d1.state === 'string'\n ? d1.state\n : undefined\n );\n if (mapped) newStatus = mapped;\n fulfillment = mapErpPayloadToFulfillment(d1);\n const refs = extractChildOrderRefsFromSalePayload(d1);\n if (refs.length) {\n await ensureChildOrdersFromRefs(\n orderRepo,\n order,\n refs,\n order.contactId as number,\n String(order.currency || 'INR')\n );\n }\n }\n\n const r2 = await submission.postErpReadAction('get-fulfillment-status', { platformOrderId: refId });\n const d2 = r2.ok ? unwrapErpReadData(r2.json) : null;\n if (d2) {\n fulfillment = deepMergeFulfillment(fulfillment, mapErpPayloadToFulfillment(d2));\n }\n\n const r3 = await submission.postErpReadAction('get-invoice', { platformOrderId: refId });\n const d3 = r3.ok ? unwrapErpReadData(r3.json) : null;\n if (d3) {\n invoiceNumber = mapErpPayloadToInvoiceNumber(d3);\n invoiceId = pickInvoiceId(d3);\n }\n\n const oid = order.id as number;\n const prevInv =\n meta.invoice && typeof meta.invoice === 'object' && !Array.isArray(meta.invoice)\n ? { ...(meta.invoice as Record<string, unknown>) }\n : {};\n const nextInvoice: OrderInvoiceMetadata = {\n ...prevInv,\n link: `/api/storefront/orders/${oid}/invoice`,\n ...(invoiceNumber ? { invoiceNumber } : {}),\n ...(invoiceId ? { invoiceId } : {}),\n };\n\n const patch: {\n fulfillment?: OrderFulfillmentMetadata | null;\n invoice?: OrderInvoiceMetadata | null;\n } = { invoice: nextInvoice };\n if (fulfillment !== undefined) patch.fulfillment = fulfillment;\n\n const nextMeta = mergeOrderMetadataPatch(meta, patch);\n\n await orderRepo.update(oid, {\n ...(newStatus ? { status: newStatus } : {}),\n metadata: nextMeta,\n updatedAt: new Date(),\n } as ObjectLiteral);\n return;\n }\n\n if (kind === 'return' || kind === 'replacement') {\n const platformReturnId = String(order.orderNumber || '');\n const r = await submission.postErpReadAction('get-return-status', { platformReturnId });\n const d = r.ok ? unwrapErpReadData(r.json) : null;\n if (!d) return;\n const mapped = mapErpSaleStatusToOrderStatus(\n typeof d.status === 'string' ? d.status : typeof d.returnStatus === 'string' ? d.returnStatus : undefined\n );\n const fulfillment = mapErpPayloadToFulfillment(d);\n const patch: { fulfillment?: OrderFulfillmentMetadata | null } = {};\n if (fulfillment !== undefined) patch.fulfillment = fulfillment;\n const nextMeta = Object.keys(patch).length ? mergeOrderMetadataPatch(meta, patch) : meta;\n await orderRepo.update(order.id as number, {\n ...(mapped ? { status: mapped } : {}),\n metadata: nextMeta,\n updatedAt: new Date(),\n } as ObjectLiteral);\n }\n}\n\nexport async function tryRefreshOrderFromErpForStorefront(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n order: ObjectLiteral\n): Promise<void> {\n try {\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return;\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (!erp?.submission) return;\n await refreshOrderFromErp(cms, dataSource, entityMap, erp.submission, order);\n } catch {\n /* non-fatal */\n }\n}\n","import type { ObjectLiteral } from 'typeorm';\nimport { queueErp } from './erp-queue';\nimport { isErpIntegrationEnabled, type CmsAppLike, type ErpConfigDataSource } from './erp-config-enabled';\n\nexport async function queueErpProductUpsertIfEnabled(\n cms: CmsAppLike,\n dataSource: ErpConfigDataSource,\n entityMap: Record<string, unknown>,\n product: ObjectLiteral\n): Promise<void> {\n try {\n const sku = typeof product.sku === 'string' ? product.sku.trim() : '';\n if (!sku) return;\n const on = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (!on) return;\n const rawMeta = product.metadata;\n let metadata: Record<string, unknown> | undefined;\n if (rawMeta && typeof rawMeta === 'object' && !Array.isArray(rawMeta)) {\n const { description: _d, ...rest } = rawMeta as Record<string, unknown>;\n metadata = Object.keys(rest).length ? rest : undefined;\n }\n const payload: Record<string, unknown> = {\n sku,\n title: (product.name as string) || sku,\n name: product.name,\n hsn_number: product.hsn,\n uom: product.uom != null && String(product.uom).trim() ? String(product.uom).trim() : undefined,\n type: product.type === 'service' ? 'service' : 'product',\n is_active: product.status === 'available',\n metadata,\n };\n await queueErp(cms, { kind: 'productUpsert', product: payload });\n } catch {\n /* non-fatal */\n }\n}\n","import { SESClient, SendEmailCommand } from '@aws-sdk/client-ses';\nimport nodemailer from 'nodemailer';\nimport type { EmailTemplateName, TemplateContext } from './templates/types';\nimport type { RenderEmailOptions } from './renderer';\nimport { renderEmail } from './renderer';\n\nexport interface EmailPluginConfig {\n type: 'AWS' | 'SMTP' | 'GMAIL' | 'SENDGRID';\n user?: string;\n password?: string;\n from: string;\n to: string;\n /** When type is SMTP, defaults to env SMTP_HOST or undefined (set host for custom relay). */\n host?: string;\n port?: number;\n secure?: boolean;\n region?: string;\n accessKeyId?: string;\n secretAccessKey?: string;\n templateOptions?: RenderEmailOptions;\n}\n\nexport interface EmailData {\n subject: string;\n html: string;\n text?: string;\n to?: string;\n from?: string;\n}\n\nexport interface EmailServiceInterface {\n send(emailData: EmailData): Promise<boolean>;\n}\n\nexport interface RenderedEmail {\n subject: string;\n html: string;\n text?: string;\n}\n\nexport class EmailService implements EmailServiceInterface {\n private config: EmailPluginConfig;\n private templateOptions?: RenderEmailOptions;\n private sesClient?: SESClient;\n private transporter?: nodemailer.Transporter;\n\n constructor(config: EmailPluginConfig) {\n this.config = config;\n this.templateOptions = config.templateOptions;\n if (config.type === 'AWS') {\n if (!config.region || !config.accessKeyId || !config.secretAccessKey) {\n throw new Error('AWS SES configuration incomplete');\n }\n this.sesClient = new SESClient({\n region: config.region,\n credentials: { accessKeyId: config.accessKeyId, secretAccessKey: config.secretAccessKey },\n });\n } else if (config.type === 'SMTP' || config.type === 'GMAIL') {\n if (!config.user || !config.password) throw new Error('SMTP configuration incomplete');\n const host =\n config.type === 'GMAIL' ? 'smtp.gmail.com' : config.host || undefined;\n const port = config.port ?? 587;\n const secure = config.secure ?? false;\n this.transporter = nodemailer.createTransport({\n ...(host ? { host } : {}),\n port,\n secure,\n auth: { user: config.user, pass: config.password },\n });\n } else {\n throw new Error(`Unsupported email type: ${config.type}`);\n }\n }\n\n async send(emailData: EmailData): Promise<boolean> {\n try {\n if (this.config.type === 'AWS' && this.sesClient) {\n await this.sesClient.send(\n new SendEmailCommand({\n Source: emailData.from || this.config.from,\n Destination: { ToAddresses: [emailData.to || this.config.to] },\n Message: {\n Subject: { Data: emailData.subject, Charset: 'UTF-8' },\n Body: {\n Html: { Data: emailData.html, Charset: 'UTF-8' },\n ...(emailData.text && { Text: { Data: emailData.text, Charset: 'UTF-8' } }),\n },\n },\n })\n );\n return true;\n }\n if ((this.config.type === 'SMTP' || this.config.type === 'GMAIL') && this.transporter) {\n await this.transporter.sendMail({\n from: emailData.from || this.config.from,\n to: emailData.to || this.config.to,\n subject: emailData.subject,\n html: emailData.html,\n text: emailData.text,\n });\n return true;\n }\n return false;\n } catch (error) {\n console.error('Email sending failed:', error);\n return false;\n }\n }\n\n renderTemplate(templateName: EmailTemplateName, ctx: TemplateContext<unknown>): RenderedEmail {\n return renderEmail(templateName, ctx, this.templateOptions);\n }\n}\n\nexport const emailTemplates = {\n formSubmission: (data: { formName: string; contactName: string; contactEmail: string; formData: unknown }) => ({\n subject: `New Form Submission: ${data.formName}`,\n html: `<h2>New Form Submission</h2><p><strong>Form:</strong> ${data.formName}</p><p><strong>Contact:</strong> ${data.contactName} (${data.contactEmail})</p><pre>${JSON.stringify(data.formData, null, 2)}</pre>`,\n text: `New Form Submission\\nForm: ${data.formName}\\nContact: ${data.contactName} (${data.contactEmail})\\n${JSON.stringify(data.formData, null, 2)}`,\n }),\n contactSubmission: (data: { name: string; email: string; phone?: string; message?: string }) => ({\n subject: `New Contact Form Submission from ${data.name}`,\n html: `<h2>New Contact Form Submission</h2><p><strong>Name:</strong> ${data.name}</p><p><strong>Email:</strong> ${data.email}</p>${data.phone ? `<p><strong>Phone:</strong> ${data.phone}</p>` : ''}${data.message ? `<p><strong>Message:</strong></p><p>${data.message}</p>` : ''}`,\n text: `New Contact Form Submission\\nName: ${data.name}\\nEmail: ${data.email}\\n${data.phone ? `Phone: ${data.phone}\\n` : ''}${data.message ? `Message: ${data.message}` : ''}`,\n }),\n passwordReset: (data: { resetLink: string }) => ({\n subject: 'Reset your password',\n html: `<h2>Reset your password</h2><p>Click the link below to set a new password. This link expires in 1 hour.</p><p><a href=\"${data.resetLink}\">${data.resetLink}</a></p>`,\n text: `Reset your password: ${data.resetLink}\\n\\nThis link expires in 1 hour.`,\n }),\n};\n","import type { CompanyDetails, SocialLinkItem } from './types';\n\nfunction escapeHtml(s: string): string {\n return s\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;');\n}\n\nfunction socialLinkInnerHtml(s: SocialLinkItem): string {\n if (s.iconUrl?.trim()) {\n return `<img src=\"${escapeHtml(s.iconUrl.trim())}\" alt=\"\" width=\"20\" height=\"20\" style=\"display:inline-block;vertical-align:middle;border:0;\" />`;\n }\n const label = (s.icon && s.icon.trim()) || '🔗';\n return escapeHtml(label);\n}\n\nfunction supportLinesHtml(supportEmail?: string, supportPhone?: string, opts?: { tightTop?: boolean }): string {\n const parts: string[] = [];\n const top = opts?.tightTop ? 'margin-top:0;' : 'margin-top:4px;';\n if (supportEmail) {\n parts.push(`<div style=\"font-size:13px;color:#444;${top}\">&#128231; ${escapeHtml(supportEmail)}</div>`);\n }\n if (supportPhone) {\n parts.push(`<div style=\"font-size:13px;color:#444;margin-top:2px;\">&#128222; ${escapeHtml(supportPhone)}</div>`);\n }\n return parts.join('');\n}\n\nexport function renderLayout(options: {\n bodyHtml: string;\n companyDetails: CompanyDetails;\n}): string {\n const { bodyHtml, companyDetails } = options;\n const {\n logoUrl,\n companyName,\n supportEmail,\n supportPhone,\n socialLinks,\n footerDisclaimer,\n followUsTitle = 'Follow Us',\n } = companyDetails;\n\n const supportFooterHtml = supportLinesHtml(supportEmail, supportPhone, { tightTop: true });\n\n const hasSocial = Boolean(socialLinks?.length);\n const brandFooterInner: string[] = [];\n if (logoUrl) {\n brandFooterInner.push(\n `<img src=\"${escapeHtml(logoUrl)}\" alt=\"\" style=\"max-height:28px;max-width:160px;height:auto;width:auto;display:inline-block;vertical-align:middle;margin-right:10px;border:0;\" />`\n );\n }\n if (companyName) {\n brandFooterInner.push(\n `<span style=\"font-size:14px;font-weight:600;color:#111;vertical-align:middle;\">${escapeHtml(companyName)}</span>`\n );\n }\n const brandFooterHtml = brandFooterInner.length\n ? `<div style=\"line-height:1.4;\">${brandFooterInner.join('')}</div>`\n : '';\n\n const followTitleHtml =\n hasSocial && followUsTitle\n ? `<div class=\"footer-follow-title\" style=\"font-size:13px;font-weight:600;color:#555;margin:0;\">${escapeHtml(followUsTitle)}</div>`\n : '';\n\n const socialRowHtml =\n hasSocial && socialLinks\n ? `<div class=\"footer-social-icons\" style=\"margin-top:6px;text-align:left;font-size:0;line-height:0;\">${socialLinks\n .map(\n (s) =>\n `<a href=\"${escapeHtml(s.url)}\" style=\"display:inline-block;margin-right:12px;text-decoration:none;color:#333;font-size:18px;line-height:1;vertical-align:middle;\">${socialLinkInnerHtml(s)}</a>`\n )\n .join('')}</div>`\n : '';\n\n const rowSupport = supportFooterHtml\n ? `<tr><td class=\"footer-support-cell\" colspan=\"2\" valign=\"top\" style=\"padding:0 0 10px 0;\">${supportFooterHtml}</td></tr>`\n : '';\n\n let footerBlock = '';\n if (brandFooterHtml || followTitleHtml || socialRowHtml || footerDisclaimer || rowSupport) {\n const rowBrand = brandFooterHtml\n ? `<tr><td class=\"footer-brand-cell\" colspan=\"2\" valign=\"top\" style=\"padding:0 0 10px 0;\">${brandFooterHtml}</td></tr>`\n : '';\n const rowFollowTitle = followTitleHtml\n ? `<tr><td class=\"footer-follow-cell\" colspan=\"2\" valign=\"top\" style=\"padding:14px 0 0 0;text-align:left;\">${followTitleHtml}</td></tr>`\n : '';\n const rowSocial = socialRowHtml\n ? `<tr><td class=\"footer-social-cell\" colspan=\"2\" style=\"padding:0;text-align:left;\">${socialRowHtml}</td></tr>`\n : '';\n footerBlock = `<table role=\"presentation\" class=\"footer-main\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"margin-top:28px;padding-top:16px;\">${rowBrand}${rowSupport}${rowFollowTitle}${rowSocial}</table>`;\n }\n\n const disclaimerBlock = footerDisclaimer\n ? `<div style=\"margin-top:20px;padding-top:12px;font-size:11px;line-height:1.5;color:#888;\">${escapeHtml(footerDisclaimer).replace(/\\n/g, '<br/>')}</div>`\n : '';\n\n const responsiveCss = `\n@media only screen and (max-width: 600px) {\n .email-wrap { padding-left: 12px !important; padding-right: 12px !important; }\n .footer-follow-cell { padding-top: 12px !important; }\n}\n`;\n\n return `<!DOCTYPE html>\n<html>\n<head>\n<meta charset=\"utf-8\">\n<meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n<style type=\"text/css\">${responsiveCss}</style>\n</head>\n<body style=\"margin:0;padding:0;font-family:-apple-system,BlinkMacSystemFont,'Segoe UI',Roboto,sans-serif;font-size:14px;line-height:1.5;color:#333;background:#fff;\">\n <table role=\"presentation\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"background:#fff;\">\n <tr><td class=\"email-wrap\" style=\"padding:16px;\">\n <div>\n ${bodyHtml}\n </div>\n ${footerBlock}\n ${disclaimerBlock}\n </td></tr>\n </table>\n</body>\n</html>`;\n}\n","/** Social entry: use `iconUrl` for an image, or `icon` (emoji/text) as fallback when no image. */\nexport interface SocialLinkItem {\n url: string;\n iconUrl?: string;\n icon?: string;\n}\n\nexport interface CompanyDetails {\n logoUrl?: string;\n companyName?: string;\n supportEmail?: string;\n supportPhone?: string;\n socialLinks?: SocialLinkItem[];\n /** Shown below footer rows; from email plugin settings */\n footerDisclaimer?: string;\n /** Heading above social icons (default \"Follow Us\") */\n followUsTitle?: string;\n}\n\n/** One row in order-placed transactional emails */\nexport interface OrderPlacedLineItem {\n productName: string;\n quantity: number;\n unitPrice: number | string;\n lineTotal: number | string;\n sku?: string | null;\n tax?: number | string | null;\n hsn?: string | null;\n}\n\nexport function normalizeSocialLinkItem(o: {\n icon?: string;\n iconUrl?: string;\n icon_image?: string;\n url?: string;\n}): SocialLinkItem | null {\n const url = String(o.url ?? '').trim();\n if (!url) return null;\n let iconUrl = String(o.iconUrl ?? o.icon_image ?? '').trim();\n let icon = String(o.icon ?? '').trim();\n if (!iconUrl && /^https?:\\/\\//i.test(icon)) {\n iconUrl = icon;\n icon = '';\n }\n const item: SocialLinkItem = { url };\n if (iconUrl) item.iconUrl = iconUrl;\n if (icon) item.icon = icon;\n return item;\n}\n\nexport function parseSocialLinksJson(raw: string | undefined | null): SocialLinkItem[] | undefined {\n if (raw == null || raw.trim() === '') return undefined;\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (!Array.isArray(parsed)) return undefined;\n const out: SocialLinkItem[] = [];\n for (const item of parsed) {\n if (item && typeof item === 'object' && 'url' in item) {\n const n = normalizeSocialLinkItem(item as { icon?: string; iconUrl?: string; url?: string });\n if (n) out.push(n);\n }\n }\n return out.length ? out : undefined;\n } catch {\n return undefined;\n }\n}\n\n/** Merge branding + email plugin settings for layout (email overrides when set). */\nexport function mergeEmailLayoutCompanyDetails(\n branding: Record<string, string>,\n emailSettings: Record<string, string>\n): CompanyDetails {\n const fromBranding = getCompanyDetailsFromSettings(branding);\n const pick = (emailVal: string | undefined, fallback?: string) => {\n const t = emailVal?.trim();\n return t || fallback?.trim() || undefined;\n };\n const logoUrl = pick(emailSettings.logoUrl ?? emailSettings.emailLogoUrl, fromBranding.logoUrl);\n const companyName = pick(emailSettings.companyName ?? emailSettings.emailCompanyName, fromBranding.companyName);\n const supportEmail = pick(emailSettings.supportEmail ?? emailSettings.emailSupportEmail, fromBranding.supportEmail);\n const supportPhone = pick(emailSettings.supportPhone, undefined);\n const footerDisclaimer = pick(emailSettings.footerDisclaimer, undefined);\n const followUsTitle = pick(emailSettings.followUsTitle, 'Follow Us') || 'Follow Us';\n const socialFromEmail = parseSocialLinksJson(emailSettings.socialLinks);\n const socialLinks = socialFromEmail?.length ? socialFromEmail : fromBranding.socialLinks;\n return {\n logoUrl,\n companyName,\n supportEmail,\n supportPhone,\n socialLinks,\n footerDisclaimer,\n followUsTitle,\n };\n}\n\nexport interface EmailTemplateResult {\n subject: string;\n bodyHtml: string;\n text?: string;\n}\n\nexport type TemplateContext<T = unknown> = T & { companyDetails: CompanyDetails };\n\nexport const EMAIL_TEMPLATE_NAMES = [\n 'signup',\n 'passwordReset',\n 'passwordChange',\n 'orderPlaced',\n 'returnInitiated',\n 'shippingUpdate',\n 'invite',\n 'formSubmission',\n 'otp',\n] as const;\n\nexport type EmailTemplateName = (typeof EMAIL_TEMPLATE_NAMES)[number];\n\nexport function getCompanyDetailsFromSettings(\n settingsGroup: Record<string, string>\n): CompanyDetails {\n const logoUrl = settingsGroup.logo ?? settingsGroup.logoUrl ?? '';\n const companyName = settingsGroup.companyName ?? settingsGroup.company_name ?? '';\n const supportEmail = settingsGroup.supportEmail ?? settingsGroup.support_email ?? '';\n let socialLinks: SocialLinkItem[] = [];\n const raw = settingsGroup.socialLinks ?? settingsGroup.social_links;\n if (typeof raw === 'string') {\n try {\n const arr = JSON.parse(raw) as unknown[];\n if (Array.isArray(arr)) {\n for (const item of arr) {\n if (item && typeof item === 'object') {\n const n = normalizeSocialLinkItem(item as { icon?: string; iconUrl?: string; url?: string });\n if (n) socialLinks.push(n);\n }\n }\n }\n } catch {\n // ignore\n }\n }\n return { logoUrl: logoUrl || undefined, companyName: companyName || undefined, supportEmail: supportEmail || undefined, socialLinks: socialLinks.length ? socialLinks : undefined };\n}\n","/** Email-safe primary button (table-based). */\n\nexport function escapeHtml(s: string): string {\n return s\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;')\n .replace(/\"/g, '&quot;');\n}\n\n/** href for use inside double-quoted HTML attribute */\nexport function escapeAttr(s: string): string {\n return escapeHtml(s);\n}\n\nexport function primaryCtaButton(href: string, label: string): string {\n return `<table role=\"presentation\" cellpadding=\"0\" cellspacing=\"0\" style=\"margin:20px 0;\">\n<tr><td style=\"border-radius:6px;background:#1a1a1a;\">\n<a href=\"${escapeAttr(href)}\" style=\"display:inline-block;padding:12px 22px;font-size:14px;font-weight:600;color:#ffffff;text-decoration:none;\">${escapeHtml(label)}</a>\n</td></tr>\n</table>`;\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(\n ctx: TemplateContext<{ name?: string; loginUrl?: string; verifyEmailUrl?: string }>\n) {\n const { name, loginUrl, verifyEmailUrl, companyDetails } = ctx;\n\n if (verifyEmailUrl) {\n const subject = 'Confirm your email';\n const greeting =\n name && name.trim() ? `Hi ${escapeHtml(name.trim())},` : 'Hi,';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${greeting}</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Thanks for signing up. Please confirm your email address to activate your account. Until you confirm, you won’t be able to sign in.</p>\n${primaryCtaButton(verifyEmailUrl, 'Confirm email address')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">This link expires in a few days. If you didn’t create an account, you can ignore this message.<br/><span style=\"word-break:break-all;\">${escapeHtml(verifyEmailUrl)}</span></p>`;\n const text = [\n greeting,\n '',\n 'Confirm your email to activate your account:',\n verifyEmailUrl,\n '',\n 'You cannot sign in until your email is confirmed.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n }\n\n const subject = 'Welcome';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Welcome${name ? `, ${escapeHtml(name)}` : ''}.</p><p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Your account has been created.</p>${\n loginUrl ? `${primaryCtaButton(loginUrl, 'Sign in')}` : ''\n }`;\n const text = `Welcome${name ? `, ${name}` : ''}. Your account has been created.${loginUrl ? ` Sign in: ${loginUrl}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(ctx: TemplateContext<{ resetLink: string }>) {\n const { resetLink, companyDetails } = ctx;\n const subject = 'Reset your password';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">Hello,</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">We received a request to reset the password for your account. If you made this request, use the button below to choose a new password.</p>\n<p style=\"margin:0 0 8px 0;font-size:14px;line-height:1.5;color:#555;\">For your security, this link will stop working after a short time. If it expires, request a new reset from the sign-in page.</p>\n${primaryCtaButton(resetLink, 'Reset password')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If you did not request a password reset, you can safely ignore this email—your password will stay the same.</p>\n<p style=\"margin:12px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If the button does not work, copy and paste this URL into your browser:<br/><span style=\"word-break:break-all;\">${escapeHtml(resetLink)}</span></p>`;\n const text = [\n 'Hello,',\n '',\n 'We received a request to reset your password. Open the link below to set a new password:',\n resetLink,\n '',\n 'This link expires after a limited time.',\n '',\n 'If you did not request this, you can ignore this email.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(ctx: TemplateContext<{ name?: string }>) {\n const { name, companyDetails } = ctx;\n const subject = 'Password changed';\n const bodyHtml = `<h2>Password changed</h2><p>Your password has been updated successfully${name ? `, ${escapeHtml(name)}` : ''}.</p>`;\n const text = `Your password has been updated successfully${name ? `, ${name}` : ''}.`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext, OrderPlacedLineItem } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml } from './inline-cta';\n\nfunction formatMoney(amount: number | string, currency?: string): string {\n const v = typeof amount === 'string' ? parseFloat(amount) : amount;\n const num = Number.isFinite(v) ? Number(v).toFixed(2) : String(amount);\n return currency ? `${num} ${currency}` : num;\n}\n\nfunction renderLineItemsHtml(items: OrderPlacedLineItem[], currency?: string): string {\n if (!items.length) {\n return '<p style=\"margin:12px 0 0 0;font-size:14px;color:#666;\">No line items.</p>';\n }\n const rows = items\n .map((it) => {\n const name = escapeHtml(it.productName);\n const sku =\n it.sku && String(it.sku).trim()\n ? `<span style=\"font-size:12px;color:#888;\"> (${escapeHtml(String(it.sku).trim())})</span>`\n : '';\n const hsn =\n it.hsn && String(it.hsn).trim()\n ? `<br/><span style=\"font-size:11px;color:#888;\">HSN: ${escapeHtml(String(it.hsn).trim())}</span>`\n : '';\n const taxNote =\n it.tax != null && String(it.tax).trim() && Number(it.tax) !== 0\n ? `<br/><span style=\"font-size:11px;color:#888;\">Tax: ${escapeHtml(formatMoney(it.tax, currency))}</span>`\n : '';\n return `<tr>\n<td style=\"padding:10px 8px 10px 0;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;\">${name}${sku}${hsn}${taxNote}</td>\n<td align=\"right\" style=\"padding:10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(String(it.quantity))}</td>\n<td align=\"right\" style=\"padding:10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(formatMoney(it.unitPrice, currency))}</td>\n<td align=\"right\" style=\"padding:10px 0 10px 8px;border-bottom:1px solid #eee;vertical-align:top;font-size:14px;color:#333;white-space:nowrap;\">${escapeHtml(formatMoney(it.lineTotal, currency))}</td>\n</tr>`;\n })\n .join('');\n return `<p style=\"margin:16px 0 8px 0;font-size:14px;font-weight:600;color:#111;\">Order items</p>\n<table role=\"presentation\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\" style=\"border-collapse:collapse;\">\n<tr>\n<td style=\"padding:0 8px 8px 0;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;letter-spacing:0.02em;\">Item</td>\n<td align=\"right\" style=\"padding:0 8px 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Qty</td>\n<td align=\"right\" style=\"padding:0 8px 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Price</td>\n<td align=\"right\" style=\"padding:0 0 8px 8px;font-size:12px;font-weight:600;color:#555;text-transform:uppercase;\">Total</td>\n</tr>\n${rows}\n</table>`;\n}\n\nfunction renderLineItemsText(items: OrderPlacedLineItem[], currency?: string): string {\n if (!items.length) return '';\n const lines = items.map((it) => {\n let s = `- ${it.productName} × ${it.quantity} @ ${formatMoney(it.unitPrice, currency)} = ${formatMoney(it.lineTotal, currency)}`;\n if (it.sku) s += ` [${it.sku}]`;\n if (it.hsn && String(it.hsn).trim()) s += ` HSN:${it.hsn}`;\n if (it.tax != null && Number(it.tax) !== 0) s += ` tax:${formatMoney(it.tax, currency)}`;\n return s;\n });\n return ['Items:', ...lines].join('\\n');\n}\n\nfunction formatAddressLines(addr: Record<string, string> | undefined): string[] {\n if (!addr || typeof addr !== 'object') return [];\n const parts = [addr.line1, addr.line2, [addr.city, addr.state].filter(Boolean).join(', '), addr.postalCode, addr.country].filter(\n (x) => x != null && String(x).trim() !== ''\n ) as string[];\n return parts.map((p) => String(p).trim()).filter(Boolean);\n}\n\nfunction renderAddressesHtml(billing?: Record<string, string>, shipping?: Record<string, string>): string {\n const bLines = formatAddressLines(billing);\n const sLines = formatAddressLines(shipping);\n if (!bLines.length && !sLines.length) return '';\n let html = '<p style=\"margin:16px 0 8px 0;font-size:14px;font-weight:600;color:#111;\">Addresses</p>';\n if (bLines.length) {\n html += `<p style=\"margin:0 0 4px 0;font-size:12px;font-weight:600;color:#555;\">Billing</p><p style=\"margin:0 0 12px 0;font-size:14px;line-height:1.5;color:#333;\">${bLines.map((l) => escapeHtml(l)).join('<br/>')}</p>`;\n }\n if (sLines.length) {\n html += `<p style=\"margin:0 0 4px 0;font-size:12px;font-weight:600;color:#555;\">Shipping</p><p style=\"margin:0 0 0 0;font-size:14px;line-height:1.5;color:#333;\">${sLines.map((l) => escapeHtml(l)).join('<br/>')}</p>`;\n }\n return html;\n}\n\nfunction renderAddressesText(billing?: Record<string, string>, shipping?: Record<string, string>): string {\n const bLines = formatAddressLines(billing);\n const sLines = formatAddressLines(shipping);\n const out: string[] = [];\n if (bLines.length) out.push('Billing:', ...bLines.map((l) => ` ${l}`));\n if (sLines.length) out.push('Shipping:', ...sLines.map((l) => ` ${l}`));\n return out.length ? out.join('\\n') : '';\n}\n\nexport function render(\n ctx: TemplateContext<{\n orderNumber: string;\n total?: string;\n subtotal?: string;\n tax?: string;\n currency?: string;\n customerName?: string;\n audience?: 'customer' | 'sales';\n internalCustomerEmail?: string;\n lineItems?: OrderPlacedLineItem[];\n billingAddress?: Record<string, string>;\n shippingAddress?: Record<string, string>;\n }>\n) {\n const {\n orderNumber,\n total,\n subtotal,\n tax,\n currency,\n customerName,\n companyDetails,\n audience = 'customer',\n internalCustomerEmail,\n lineItems = [],\n billingAddress,\n shippingAddress,\n } = ctx;\n\n const itemsHtml = renderLineItemsHtml(lineItems, currency);\n const itemsText = renderLineItemsText(lineItems, currency);\n const addrHtml = renderAddressesHtml(billingAddress, shippingAddress);\n const addrText = renderAddressesText(billingAddress, shippingAddress);\n\n const subtotalLine =\n subtotal != null && String(subtotal).trim() !== ''\n ? `<p style=\"margin:8px 0 0 0;font-size:14px;line-height:1.5;color:#333;\"><strong>Subtotal:</strong> ${escapeHtml(String(subtotal))}${currency ? ` ${escapeHtml(currency)}` : ''}</p>`\n : '';\n const taxLine =\n tax != null && String(tax).trim() !== '' && Number(tax) !== 0\n ? `<p style=\"margin:4px 0 0 0;font-size:14px;line-height:1.5;color:#333;\"><strong>Tax:</strong> ${escapeHtml(String(tax))}${currency ? ` ${escapeHtml(currency)}` : ''}</p>`\n : '';\n const totalLine =\n total != null && String(total).trim() !== ''\n ? `<p style=\"margin:12px 0 0 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order total:</strong> ${escapeHtml(String(total))}${currency ? ` ${escapeHtml(currency)}` : ''}</p>`\n : '';\n\n let subject: string;\n let bodyHtml: string;\n let text: string;\n\n if (audience === 'sales') {\n subject = `New order #${orderNumber}`;\n const who =\n customerName || internalCustomerEmail\n ? `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Customer:</strong> ${escapeHtml(customerName || '—')}${\n internalCustomerEmail\n ? ` <span style=\"color:#555;\">(${escapeHtml(internalCustomerEmail)})</span>`\n : ''\n }</p>`\n : '';\n bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">A new order has been placed and payment completed.</p>\n<p style=\"margin:0 0 8px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order number:</strong> ${escapeHtml(orderNumber)}</p>\n${who}\n${addrHtml}\n${itemsHtml}\n${subtotalLine}${taxLine}${totalLine}`;\n text = [\n `New order #${orderNumber}`,\n customerName ? `Customer: ${customerName}` : '',\n internalCustomerEmail ? `Email: ${internalCustomerEmail}` : '',\n addrText,\n itemsText,\n subtotal != null ? `Subtotal: ${subtotal}${currency ? ` ${currency}` : ''}` : '',\n tax != null && Number(tax) !== 0 ? `Tax: ${tax}${currency ? ` ${currency}` : ''}` : '',\n total != null ? `Order total: ${total}${currency ? ` ${currency}` : ''}` : '',\n ]\n .filter(Boolean)\n .join('\\n\\n');\n } else {\n subject = `Order confirmed #${orderNumber}`;\n const thanksPlain =\n customerName && customerName.trim()\n ? `Thank you, ${customerName.trim()}.`\n : 'Thank you for your order.';\n const thanksHtml = `${escapeHtml(thanksPlain)} We’ve received your order and will process it shortly.`;\n bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${thanksHtml}</p>\n<p style=\"margin:0 0 8px 0;font-size:15px;line-height:1.5;color:#333;\"><strong>Order number:</strong> ${escapeHtml(orderNumber)}</p>\n${addrHtml}\n${itemsHtml}\n${subtotalLine}${taxLine}${totalLine}\n<p style=\"margin:16px 0 0 0;font-size:13px;line-height:1.5;color:#666;\">If you have questions, reply to this email or contact us using the details below.</p>`;\n text = [\n `Order confirmed #${orderNumber}`,\n `${thanksPlain} We’ve received your order and will process it shortly.`,\n '',\n addrText,\n itemsText,\n subtotal != null ? `Subtotal: ${subtotal}${currency ? ` ${currency}` : ''}` : '',\n tax != null && Number(tax) !== 0 ? `Tax: ${tax}${currency ? ` ${currency}` : ''}` : '',\n total != null ? `Order total: ${total}${currency ? ` ${currency}` : ''}` : '',\n '',\n 'We will process your order shortly.',\n ]\n .filter((line, i, arr) => !(line === '' && arr[i - 1] === ''))\n .join('\\n');\n }\n\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{ returnId?: string; orderNumber?: string }>\n) {\n const { returnId, orderNumber, companyDetails } = ctx;\n const subject = 'Return initiated';\n const bodyHtml = `<h2>Return initiated</h2><p>Your return request has been received.</p>${orderNumber ? `<p><strong>Order:</strong> ${escapeHtml(orderNumber)}</p>` : ''}${returnId ? `<p><strong>Return ID:</strong> ${escapeHtml(returnId)}</p>` : ''}`;\n const text = `Return initiated.${orderNumber ? ` Order: ${orderNumber}` : ''}${returnId ? ` Return ID: ${returnId}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{\n orderNumber?: string;\n status?: string;\n trackingUrl?: string;\n }>\n) {\n const { orderNumber, status, trackingUrl, companyDetails } = ctx;\n const subject = status ? `Shipping update: ${status}` : 'Shipping update';\n const bodyHtml = `<h2>Shipping update</h2>${status ? `<p><strong>Status:</strong> ${escapeHtml(status)}</p>` : ''}${orderNumber ? `<p><strong>Order:</strong> ${escapeHtml(orderNumber)}</p>` : ''}${trackingUrl ? `<p><a href=\"${escapeHtml(trackingUrl)}\">Track your order</a></p>` : ''}`;\n const text = `Shipping update.${status ? ` Status: ${status}` : ''}${orderNumber ? ` Order: ${orderNumber}` : ''}${trackingUrl ? ` Track: ${trackingUrl}` : ''}`;\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\nimport { escapeHtml, primaryCtaButton } from './inline-cta';\n\nexport function render(\n ctx: TemplateContext<{ inviteLink: string; email: string; inviteeName?: string }>\n) {\n const { inviteLink, email, inviteeName, companyDetails } = ctx;\n const subject = \"You're invited\";\n const greeting =\n inviteeName && inviteeName.trim()\n ? `Hello ${escapeHtml(inviteeName.trim())},`\n : 'Hello,';\n const bodyHtml = `<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">${greeting}</p>\n<p style=\"margin:0 0 12px 0;font-size:15px;line-height:1.5;color:#333;\">You have been invited to create your account for <strong>${escapeHtml(email)}</strong>. Use the secure link below to choose your password and activate access.</p>\n<p style=\"margin:0 0 8px 0;font-size:14px;line-height:1.5;color:#555;\">This link is personal to you. If you did not expect this invitation, you can ignore this message.</p>\n${primaryCtaButton(inviteLink, 'Accept invitation & set password')}\n<p style=\"margin:16px 0 0 0;font-size:12px;line-height:1.5;color:#888;\">If the button does not work, copy and paste this URL into your browser:<br/><span style=\"word-break:break-all;\">${escapeHtml(inviteLink)}</span></p>`;\n const text = [\n inviteeName?.trim() ? `Hello ${inviteeName.trim()},` : 'Hello,',\n '',\n `You have been invited to create your account (${email}).`,\n 'Open this link to set your password:',\n inviteLink,\n '',\n 'If you did not expect this invitation, you can ignore this email.',\n ].join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport function render(\n ctx: TemplateContext<{\n formName: string;\n contactName: string;\n contactEmail: string;\n formData: unknown;\n formFieldRows?: { label: string; value: string }[];\n }>\n) {\n const { formName, contactName, contactEmail, formData, formFieldRows, companyDetails } = ctx;\n const subject = `New Form Submission: ${formName}`;\n\n const fieldsBlock =\n formFieldRows && formFieldRows.length > 0\n ? formFieldRows\n .map(\n (r) =>\n `<p style=\"margin:10px 0 0 0;line-height:1.5;\"><strong>${escapeHtml(r.label)}</strong><br/>${escapeHtml(r.value)}</p>`\n )\n .join('')\n : `<p style=\"margin:10px 0 0 0;font-size:13px;white-space:pre-wrap;\">${escapeHtml(JSON.stringify(formData, null, 2))}</p>`;\n\n const bodyHtml = `<p style=\"margin:0 0 6px 0;font-size:18px;font-weight:600;color:#111;\">New Form Submission</p>\n<p style=\"margin:0 0 4px 0;\"><strong>Form:</strong> ${escapeHtml(formName)}</p>\n<p style=\"margin:0 0 8px 0;\"><strong>Contact:</strong> ${escapeHtml(contactName)}${contactEmail ? ` (${escapeHtml(contactEmail)})` : ''}</p>\n${fieldsBlock}`;\n\n const textLines = [\n 'New Form Submission',\n `Form: ${formName}`,\n `Contact: ${contactName}${contactEmail ? ` (${contactEmail})` : ''}`,\n '',\n ...(formFieldRows?.length\n ? formFieldRows.map((r) => `${r.label}: ${r.value}`)\n : [JSON.stringify(formData, null, 2)]),\n ];\n const text = textLines.join('\\n');\n const html = renderLayout({ bodyHtml, companyDetails });\n return { subject, html, text };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","import type { TemplateContext } from './types';\nimport { renderLayout } from './layout';\n\nexport type OtpTemplateCtx = TemplateContext<{\n code: string;\n}>;\n\nexport function render(ctx: OtpTemplateCtx) {\n const { code, companyDetails } = ctx;\n const name = companyDetails.companyName?.trim() || 'Our store';\n const bodyHtml = `\n <p>Your verification code is:</p>\n <p style=\"font-size:24px;font-weight:bold;letter-spacing:4px;\">${escapeHtml(code)}</p>\n <p style=\"color:#666;font-size:14px;\">This code expires in 10 minutes. If you did not request it, you can ignore this email.</p>\n `;\n const html = renderLayout({ bodyHtml, companyDetails });\n return {\n subject: `${name}: your sign-in code`,\n html,\n text: `Your code is ${code}. It expires in 10 minutes.`,\n };\n}\n\nfunction escapeHtml(s: string): string {\n return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/\"/g, '&quot;');\n}\n","export * from './types';\nexport { renderLayout } from './layout';\nimport type { EmailTemplateName, TemplateContext } from './types';\nimport { render as renderSignup } from './signup';\nimport { render as renderPasswordReset } from './passwordReset';\nimport { render as renderPasswordChange } from './passwordChange';\nimport { render as renderOrderPlaced } from './orderPlaced';\nimport { render as renderReturnInitiated } from './returnInitiated';\nimport { render as renderShippingUpdate } from './shippingUpdate';\nimport { render as renderInvite } from './invite';\nimport { render as renderFormSubmission } from './formSubmission';\nimport { render as renderOtp } from './otp';\n\nexport { renderSignup, renderPasswordReset, renderPasswordChange, renderOrderPlaced, renderReturnInitiated, renderShippingUpdate, renderInvite, renderFormSubmission, renderOtp };\n\nexport type EmailRenderResult = { subject: string; html: string; text?: string };\n\nconst templateRenderMap: Record<\n EmailTemplateName,\n (ctx: TemplateContext<unknown>) => EmailRenderResult\n> = {\n signup: renderSignup as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n passwordReset: renderPasswordReset as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n passwordChange: renderPasswordChange as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n orderPlaced: renderOrderPlaced as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n returnInitiated: renderReturnInitiated as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n shippingUpdate: renderShippingUpdate as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n invite: renderInvite as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n formSubmission: renderFormSubmission as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n otp: renderOtp as (ctx: TemplateContext<unknown>) => EmailRenderResult,\n};\n\nexport function getTemplateRenderer(name: EmailTemplateName) {\n return templateRenderMap[name];\n}\n","import type {\n CompanyDetails,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n} from './templates/types';\nimport { renderLayout } from './templates/layout';\nimport { getTemplateRenderer } from './templates/index';\n\nexport interface RenderEmailOptions {\n renderLayout?: (opts: { bodyHtml: string; companyDetails: CompanyDetails }) => string;\n getBody?: (templateName: string, ctx: TemplateContext<unknown>) => EmailTemplateResult | null;\n}\n\nexport interface RenderedEmail {\n subject: string;\n html: string;\n text?: string;\n}\n\nexport function renderEmail(\n templateName: EmailTemplateName,\n ctx: TemplateContext<unknown>,\n options?: RenderEmailOptions\n): RenderedEmail {\n const layoutFn = options?.renderLayout ?? renderLayout;\n const getBody = options?.getBody;\n\n if (getBody) {\n const custom = getBody(templateName, ctx);\n if (custom != null) {\n const html = layoutFn({ bodyHtml: custom.bodyHtml, companyDetails: ctx.companyDetails });\n return { subject: custom.subject, html, text: custom.text };\n }\n }\n\n const render = getTemplateRenderer(templateName);\n if (!render) {\n throw new Error(`Unknown email template: ${templateName}`);\n }\n return render(ctx);\n}\n","import type { CmsPlugin } from '../types';\nimport { EmailService } from './email-service';\nimport type { RenderEmailOptions } from './renderer';\n\nexport interface EmailPluginConfig {\n type: 'AWS' | 'SMTP' | 'GMAIL' | 'SENDGRID';\n user?: string;\n password?: string;\n from: string;\n to: string;\n host?: string;\n port?: number;\n secure?: boolean;\n region?: string;\n accessKeyId?: string;\n secretAccessKey?: string;\n templateOptions?: RenderEmailOptions;\n}\n\nexport function emailPlugin(config: EmailPluginConfig): CmsPlugin<EmailService> {\n return {\n name: 'email',\n version: '1.0.0',\n async init(context) {\n const from = config.from || context.config.SMTP_FROM || 'no-reply@example.com';\n const to = config.to || context.config.SMTP_TO || 'info@example.com';\n const type = (config.type || context.config.SMTP_TYPE) as EmailPluginConfig['type'] || 'SMTP';\n const portEnv = context.config.SMTP_PORT;\n const portParsed = portEnv ? parseInt(portEnv, 10) : undefined;\n const merged: EmailPluginConfig = {\n ...config,\n from,\n to,\n type,\n user: config.user ?? context.config.SMTP_USER,\n password: config.password ?? context.config.SMTP_PASSWORD,\n host: config.host ?? context.config.SMTP_HOST,\n port: config.port ?? (Number.isFinite(portParsed) ? portParsed : undefined),\n secure: config.secure ?? context.config.SMTP_SECURE === 'true',\n region: config.region ?? context.config.AWS_REGION,\n accessKeyId: config.accessKeyId ?? context.config.AWS_ACCESS_KEY_ID,\n secretAccessKey: config.secretAccessKey ?? context.config.AWS_SECRET_ACCESS_KEY,\n };\n if (type === 'AWS' && (!merged.region || !merged.accessKeyId || !merged.secretAccessKey)) {\n context.logger.warn('Email plugin skipped: AWS SES configuration incomplete');\n return undefined;\n }\n if ((type === 'SMTP' || type === 'GMAIL') && (!merged.user || !merged.password)) {\n context.logger.warn('Email plugin skipped: SMTP credentials not configured');\n return undefined;\n }\n return new EmailService(merged);\n },\n };\n}\n\nexport { EmailService, emailTemplates } from './email-service';\nexport type { EmailData, EmailPluginConfig as EmailConfig, EmailServiceInterface, RenderedEmail } from './email-service';\nexport { renderEmail } from './renderer';\nexport { getCompanyDetailsFromSettings, mergeEmailLayoutCompanyDetails } from './templates/types';\nexport { renderLayout } from './templates/layout';\nexport { registerEmailQueueProcessor, queueEmail, queueOrderPlacedEmails } from './email-queue';\nexport type { EmailJobPayload, OrderPlacedEmailPayload } from './email-queue';\nexport type {\n CompanyDetails,\n SocialLinkItem,\n EmailTemplateName,\n EmailTemplateResult,\n TemplateContext,\n OrderPlacedLineItem,\n} from './templates/types';\nexport type { RenderEmailOptions } from './renderer';\n","import { google } from 'googleapis';\n\nexport interface AnalyticsData {\n visitors: number;\n pageViews: number;\n bounceRate: number;\n avgSessionDuration: number;\n topPages: Array<{ page: string; views: number }>;\n trafficSources: Array<{ source: string; sessions: number }>;\n geographicData: Array<{ country: string; sessions: number }>;\n dailyUsers: Array<{ date: string; users: number }>;\n}\n\nexport interface AnalyticsPluginConfig {\n privateKey: string;\n clientEmail: string;\n viewId: string;\n}\n\nexport class AnalyticsService {\n private analytics: ReturnType<typeof google.analyticsdata>;\n private viewId: string;\n\n constructor(config: AnalyticsPluginConfig) {\n const privateKey = config.privateKey.replace(/\\\\n/g, '\\n');\n const auth = new google.auth.JWT({\n email: config.clientEmail,\n key: privateKey,\n scopes: ['https://www.googleapis.com/auth/analytics.readonly'],\n });\n this.analytics = google.analyticsdata({ version: 'v1beta', auth });\n this.viewId = config.viewId;\n }\n\n async getAnalyticsData(days = 30): Promise<AnalyticsData> {\n const endDate = new Date();\n const startDate = new Date();\n startDate.setDate(startDate.getDate() - days);\n\n const [visitors, pageViews, bounceRate, sessionDuration, topPages, trafficSources, geographicData, dailyUsers] =\n await Promise.all([\n this.getVisitors(startDate, endDate),\n this.getPageViews(startDate, endDate),\n this.getBounceRate(startDate, endDate),\n this.getAvgSessionDuration(startDate, endDate),\n this.getTopPages(startDate, endDate),\n this.getTrafficSources(startDate, endDate),\n this.getGeographicData(startDate, endDate),\n this.getDailyUsers(startDate, endDate),\n ]);\n\n return {\n visitors,\n pageViews,\n bounceRate,\n avgSessionDuration: sessionDuration,\n topPages,\n trafficSources,\n geographicData,\n dailyUsers,\n };\n }\n\n private async runReport(requestBody: Record<string, unknown>) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const response = await (this.analytics.properties.runReport as any)({\n property: `properties/${this.viewId}`,\n requestBody,\n });\n return response?.data as { rows?: Array<{ dimensionValues: { value: string }[]; metricValues: { value: string }[] }> };\n }\n\n private async getVisitors(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'totalUsers' }],\n });\n return parseInt(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getPageViews(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'screenPageViews' }],\n });\n return parseInt(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getBounceRate(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'bounceRate' }],\n });\n return parseFloat(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getAvgSessionDuration(startDate: Date, endDate: Date): Promise<number> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n metrics: [{ name: 'averageSessionDuration' }],\n });\n return parseFloat(data.rows?.[0]?.metricValues?.[0]?.value || '0');\n }\n\n private async getTopPages(startDate: Date, endDate: Date): Promise<Array<{ page: string; views: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'pagePath' }],\n metrics: [{ name: 'screenPageViews' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ page: row.dimensionValues[0].value, views: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getTrafficSources(\n startDate: Date,\n endDate: Date\n ): Promise<Array<{ source: string; sessions: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'sessionSource' }],\n metrics: [{ name: 'sessions' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ source: row.dimensionValues[0].value, sessions: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getGeographicData(\n startDate: Date,\n endDate: Date\n ): Promise<Array<{ country: string; sessions: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'country' }],\n metrics: [{ name: 'sessions' }],\n limit: 10,\n });\n return data.rows?.map((row) => ({ country: row.dimensionValues[0].value, sessions: parseInt(row.metricValues[0].value) })) || [];\n }\n\n private async getDailyUsers(startDate: Date, endDate: Date): Promise<Array<{ date: string; users: number }>> {\n const data = await this.runReport({\n dateRanges: [{ startDate: startDate.toISOString().split('T')[0], endDate: endDate.toISOString().split('T')[0] }],\n dimensions: [{ name: 'date' }],\n metrics: [{ name: 'totalUsers' }],\n orderBys: [{ dimension: { dimensionName: 'date' } }],\n });\n return (\n data.rows?.map((row) => {\n const dateStr = row.dimensionValues[0].value;\n return { date: `${dateStr.substring(4, 6)}/${dateStr.substring(6, 8)}`, users: parseInt(row.metricValues[0].value) };\n }) || []\n );\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { AnalyticsService } from './analytics-service';\n\nexport interface AnalyticsPluginConfig {\n privateKey?: string;\n clientEmail?: string;\n viewId?: string;\n}\n\nexport function analyticsPlugin(config: AnalyticsPluginConfig = {}): CmsPlugin<AnalyticsService> {\n return {\n name: 'analytics',\n version: '1.0.0',\n async init(context) {\n const privateKey = config.privateKey ?? context.config.GOOGLE_ANALYTICS_PRIVATE_KEY;\n const clientEmail = config.clientEmail ?? context.config.GOOGLE_ANALYTICS_CLIENT_EMAIL;\n const viewId = config.viewId ?? context.config.GOOGLE_ANALYTICS_VIEW_ID ?? '';\n if (!privateKey || !clientEmail || !viewId) {\n throw new Error('Google Analytics credentials not configured');\n }\n return new AnalyticsService({ privateKey, clientEmail, viewId });\n },\n };\n}\n\nexport { AnalyticsService } from './analytics-service';\nexport type { AnalyticsData } from './analytics-service';\n","/**\n * India (TRAI / DLT): promotional and service SMS must use templates registered on operator DLT\n * (e.g. Jio, Airtel, BSNL/VIL portals) and linked to your Principal Entity ID and Header (sender).\n * MSG91 maps your DLT-approved template to a Flow / template_id used by their API. Arbitrary free-text\n * SMS APIs are often rejected (e.g. “template id missing”). Prefer Flow with `template_id` from the\n * `message_templates` row (`externalTemplateRef`) or optional env `MSG91_TEMPLATE_ID`, and variables that\n * match your DLT template placeholders (commonly var1…).\n *\n * References: TRAI TCCCPR-2018; MSG91 DLT / Flow docs; operator DLT portals for template approval.\n */\n\nexport interface SmsSendOptions {\n to: string;\n /** Plain body (Twilio/webhook/sendhttp). With `templateKey`, filled after resolving the template. */\n body?: string;\n /** Resolve copy + Flow ref from DB `message_templates` (channel sms) with code defaults. */\n templateKey?: string;\n /** For MSG91 Flow: value injected into the OTP variable (see template `providerMeta.otpVarKey`). */\n otpCode?: string;\n /** Template variables (e.g. `code`) and extra Flow recipient fields. */\n variables?: Record<string, string>;\n}\n\nexport type MessageTemplateRowLoader = (\n channel: string,\n templateKey: string\n) => Promise<{\n body: string;\n externalTemplateRef: string | null;\n providerMeta: Record<string, unknown> | null;\n enabled: boolean;\n} | null>;\n\n/** Which transport to use for outbound SMS. */\nexport type SmsProviderId = 'msg91' | 'twilio' | 'webhook';\n\n/** Stored in admin `smsProvider` or env `SMS_PROVIDER`. `auto` picks first with credentials: msg91 → twilio → webhook. */\nexport type SmsProviderChoice = SmsProviderId | 'auto';\n\n/** MSG91: use DLT Flow API vs legacy sendhttp (non‑India / legacy only). */\nexport type Msg91ApiMode = 'flow' | 'sendhttp' | 'auto';\n\nexport interface SmsServiceConfig {\n provider?: SmsProviderChoice;\n /** Admin settings key `smsProvider` (msg91 | twilio | webhook | auto). */\n smsProvider?: string;\n /** MSG91: auth key */\n msg91AuthKey?: string;\n /** DLT-approved sender ID (required for sendhttp; Flow uses template/header mapped in MSG91 panel). */\n msg91SenderId?: string;\n /** Transactional route; default 4 (sendhttp only). */\n msg91Route?: string;\n /**\n * MSG91 Flow ID / template_id shown in panel after mapping DLT template (India).\n * Required for `flow` mode.\n */\n msg91TemplateId?: string;\n /** `flow` = POST control.msg91.com/api/v5/flow/ (DLT). `sendhttp` = legacy GET API. `auto` = flow if template id set, else sendhttp. */\n msg91ApiMode?: Msg91ApiMode;\n /** Recipient JSON key for OTP value in Flow API (must match MSG91 template variables). Default var1. */\n msg91OtpVarKey?: string;\n /** Flow API short_url flag; default 0. */\n msg91FlowShortUrl?: string;\n accountSid?: string;\n authToken?: string;\n from?: string;\n webhookUrl?: string;\n webhookSecret?: string;\n}\n\nfunction norm(v: string | undefined): string | undefined {\n const t = v?.trim();\n return t || undefined;\n}\n\nfunction pick(\n plugin: Partial<SmsServiceConfig>,\n db: Record<string, string>,\n env: Record<string, string>,\n ...keys: string[]\n): string | undefined {\n for (const key of keys) {\n const pk = key as keyof SmsServiceConfig;\n if (pk in plugin && plugin[pk] !== undefined && String(plugin[pk]).trim() !== '') {\n const v = norm(String(plugin[pk]));\n if (v) return v;\n }\n if (norm(db[key])) return norm(db[key]);\n if (norm(env[key])) return norm(env[key]);\n }\n return undefined;\n}\n\nfunction parseMsg91ApiMode(raw: string | undefined): Msg91ApiMode {\n const x = (raw || 'auto').toLowerCase().trim();\n if (x === 'flow' || x === 'sendhttp' || x === 'auto') return x;\n return 'auto';\n}\n\nexport function mergeSmsConfigLayers(\n env: Record<string, string>,\n db: Record<string, string>,\n plugin: Partial<SmsServiceConfig>\n): SmsServiceConfig {\n const pRaw = pick(plugin, db, env, 'smsProvider', 'SMS_PROVIDER', 'provider')?.toLowerCase();\n const provider: SmsProviderChoice | undefined =\n pRaw === 'msg91' || pRaw === 'twilio' || pRaw === 'webhook' || pRaw === 'auto' ? (pRaw as SmsProviderChoice) : undefined;\n\n return {\n provider: provider ?? 'auto',\n msg91AuthKey: pick(plugin, db, env, 'msg91AuthKey', 'MSG91_AUTH_KEY'),\n msg91SenderId: pick(plugin, db, env, 'msg91SenderId', 'MSG91_SENDER_ID'),\n msg91Route: pick(plugin, db, env, 'msg91Route', 'MSG91_ROUTE') ?? '4',\n msg91TemplateId: pick(plugin, db, env, 'msg91TemplateId', 'MSG91_TEMPLATE_ID'),\n msg91ApiMode: parseMsg91ApiMode(pick(plugin, db, env, 'msg91ApiMode', 'MSG91_API_MODE')),\n msg91OtpVarKey: pick(plugin, db, env, 'msg91OtpVarKey', 'MSG91_OTP_VAR_KEY') ?? 'var1',\n msg91FlowShortUrl: pick(plugin, db, env, 'msg91FlowShortUrl', 'MSG91_FLOW_SHORT_URL') ?? '0',\n accountSid: pick(plugin, db, env, 'accountSid', 'TWILIO_ACCOUNT_SID'),\n authToken: pick(plugin, db, env, 'authToken', 'TWILIO_AUTH_TOKEN'),\n from: pick(plugin, db, env, 'from', 'TWILIO_FROM_NUMBER', 'TWILIO_MESSAGING_SERVICE_SID'),\n webhookUrl: pick(plugin, db, env, 'webhookUrl', 'SMS_WEBHOOK_URL'),\n webhookSecret: pick(plugin, db, env, 'webhookSecret', 'SMS_WEBHOOK_SECRET'),\n };\n}\n\n/** True when Flow can run: auth key and a template id (per-send or global/env). */\nexport function msg91FlowConfigured(cfg: SmsServiceConfig, flowTemplateIdForSend?: string): boolean {\n const tid = (flowTemplateIdForSend || cfg.msg91TemplateId || '').trim();\n return Boolean(cfg.msg91AuthKey && tid);\n}\n\nexport function msg91SendhttpConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.msg91AuthKey && cfg.msg91SenderId);\n}\n\n/** MSG91 is usable with auth key (Flow id may come per message from `message_templates`). */\nexport function msg91Configured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.msg91AuthKey);\n}\n\nexport function resolveMsg91SendMode(cfg: SmsServiceConfig, flowTemplateIdForSend?: string): 'flow' | 'sendhttp' | null {\n const tid = (flowTemplateIdForSend || cfg.msg91TemplateId || '').trim();\n const mode = cfg.msg91ApiMode ?? 'auto';\n if (mode === 'flow') {\n if (!cfg.msg91AuthKey || !tid) return null;\n return 'flow';\n }\n if (mode === 'sendhttp') return msg91SendhttpConfigured(cfg) ? 'sendhttp' : null;\n if (cfg.msg91AuthKey && tid) return 'flow';\n if (msg91SendhttpConfigured(cfg)) return 'sendhttp';\n return null;\n}\n\nexport function twilioConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.accountSid && cfg.authToken && cfg.from);\n}\n\nexport function webhookConfigured(cfg: SmsServiceConfig): boolean {\n return Boolean(cfg.webhookUrl);\n}\n\nexport function resolveSmsProvider(cfg: SmsServiceConfig): SmsProviderId | null {\n const want = cfg.provider ?? 'auto';\n if (want === 'msg91' && msg91Configured(cfg)) return 'msg91';\n if (want === 'twilio' && twilioConfigured(cfg)) return 'twilio';\n if (want === 'webhook' && webhookConfigured(cfg)) return 'webhook';\n if (want !== 'auto') return null;\n\n if (msg91Configured(cfg)) return 'msg91';\n if (twilioConfigured(cfg)) return 'twilio';\n if (webhookConfigured(cfg)) return 'webhook';\n return null;\n}\n\nexport function anySmsProviderConfigured(cfg: SmsServiceConfig): boolean {\n return resolveSmsProvider(cfg) !== null;\n}\n\nconst MSG91_FLOW_URL = 'https://control.msg91.com/api/v5/flow/';\n\nasync function sendViaMsg91Flow(\n cfg: SmsServiceConfig,\n to: string,\n body: string,\n opts: Pick<SmsSendOptions, 'otpCode' | 'variables'>,\n overrides?: { templateId?: string; otpVarKey?: string }\n): Promise<boolean> {\n const authkey = cfg.msg91AuthKey!;\n const templateId = (overrides?.templateId || cfg.msg91TemplateId || '').trim();\n if (!templateId) return false;\n const mobile = to.replace(/^\\+/, '').replace(/\\s/g, '');\n if (!mobile) return false;\n\n const varKey = (overrides?.otpVarKey || cfg.msg91OtpVarKey || 'var1').trim() || 'var1';\n const recipient: Record<string, string> = { mobiles: mobile };\n const vars: Record<string, string> = { ...(opts.variables ?? {}) };\n if (opts.otpCode) vars[varKey] = opts.otpCode;\n else if (!(varKey in vars)) {\n const m = body.match(/\\d{4,8}/);\n if (m) vars[varKey] = m[0];\n }\n if (Object.keys(vars).length === 0) return false;\n\n for (const [k, v] of Object.entries(vars)) {\n recipient[k] = v;\n }\n\n const payload = {\n template_id: templateId,\n short_url: cfg.msg91FlowShortUrl ?? '0',\n recipients: [recipient],\n };\n\n try {\n const res = await fetch(MSG91_FLOW_URL, {\n method: 'POST',\n headers: {\n accept: 'application/json',\n 'content-type': 'application/json',\n authkey,\n },\n body: JSON.stringify(payload),\n });\n const text = await res.text();\n if (!res.ok) return false;\n try {\n const j = JSON.parse(text) as { type?: string; message?: string };\n if (j.type === 'error' || (typeof j.message === 'string' && /error|fail/i.test(j.message))) return false;\n } catch {\n const lower = text.toLowerCase();\n if (lower.includes('error') || lower.includes('invalid') || lower.includes('fail')) return false;\n }\n return true;\n } catch {\n return false;\n }\n}\n\nasync function sendViaMsg91Sendhttp(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const authkey = cfg.msg91AuthKey;\n const sender = cfg.msg91SenderId;\n if (!authkey || !sender) return false;\n const route = cfg.msg91Route?.trim() || '4';\n const mobile = to.replace(/^\\+/, '').replace(/\\s/g, '');\n if (!mobile) return false;\n const params = new URLSearchParams({\n authkey,\n mobiles: mobile,\n message: body,\n sender,\n route,\n country: '0',\n });\n const url = `https://api.msg91.com/api/sendhttp.php?${params.toString()}`;\n try {\n const res = await fetch(url, { method: 'GET' });\n const text = (await res.text()).trim();\n if (!res.ok) return false;\n const lower = text.toLowerCase();\n if (lower.includes('error') || lower.includes('invalid') || lower.includes('failed')) return false;\n return true;\n } catch {\n return false;\n }\n}\n\nasync function sendViaMsg91(\n cfg: SmsServiceConfig,\n to: string,\n body: string,\n opts: SmsSendOptions,\n flowOverrides?: { templateId?: string; otpVarKey?: string }\n): Promise<boolean> {\n const flowTid = (flowOverrides?.templateId || cfg.msg91TemplateId || '').trim();\n const mode = resolveMsg91SendMode(cfg, flowTid);\n if (mode === 'flow') {\n return sendViaMsg91Flow(cfg, to, body, { otpCode: opts.otpCode, variables: opts.variables }, flowOverrides);\n }\n if (mode === 'sendhttp') {\n return sendViaMsg91Sendhttp(cfg, to, body);\n }\n return false;\n}\n\nasync function sendViaTwilio(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const { accountSid, authToken, from } = cfg;\n if (!accountSid || !authToken || !from) return false;\n const url = `https://api.twilio.com/2010-04-01/Accounts/${accountSid}/Messages.json`;\n const auth = Buffer.from(`${accountSid}:${authToken}`).toString('base64');\n const params = new URLSearchParams({ To: to, From: from, Body: body });\n const res = await fetch(url, {\n method: 'POST',\n headers: { Authorization: `Basic ${auth}`, 'Content-Type': 'application/x-www-form-urlencoded' },\n body: params.toString(),\n });\n return res.ok;\n}\n\nasync function sendViaWebhook(cfg: SmsServiceConfig, to: string, body: string): Promise<boolean> {\n const { webhookUrl, webhookSecret } = cfg;\n if (!webhookUrl) return false;\n const headers: Record<string, string> = { 'Content-Type': 'application/json' };\n if (webhookSecret) headers.Authorization = `Bearer ${webhookSecret}`;\n const res = await fetch(webhookUrl, {\n method: 'POST',\n headers,\n body: JSON.stringify({ to, message: body }),\n });\n return res.ok;\n}\n\nexport type SmsSettingsLoader = () => Promise<Record<string, string>>;\n\nexport class SmsService {\n private env: Record<string, string>;\n private plugin: Partial<SmsServiceConfig>;\n private getSmsSettings?: SmsSettingsLoader;\n private getMessageTemplateRow?: MessageTemplateRowLoader;\n\n constructor(\n env: Record<string, string>,\n plugin: Partial<SmsServiceConfig> = {},\n getSmsSettings?: SmsSettingsLoader,\n getMessageTemplateRow?: MessageTemplateRowLoader\n ) {\n this.env = env;\n this.plugin = plugin;\n this.getSmsSettings = getSmsSettings;\n this.getMessageTemplateRow = getMessageTemplateRow;\n }\n\n private async merged(): Promise<SmsServiceConfig> {\n const db = (await this.getSmsSettings?.()) ?? {};\n return mergeSmsConfigLayers(this.env, db, this.plugin);\n }\n\n async send(opts: SmsSendOptions): Promise<boolean> {\n try {\n const dbEarly = (await this.getSmsSettings?.()) ?? {};\n if (dbEarly.enabled === 'false') return false;\n let effective: SmsSendOptions = { ...opts };\n let flowOverrides: { templateId?: string; otpVarKey?: string } | undefined;\n\n if (opts.templateKey?.trim()) {\n if (!this.getMessageTemplateRow) return false;\n const { resolveSmsTemplateForSend } = await import('../../message-templates/sms-defaults');\n const { applyTemplateVars } = await import('../../message-templates/interpolate');\n const resolved = await resolveSmsTemplateForSend(opts.templateKey.trim(), this.getMessageTemplateRow);\n if (!resolved) return false;\n const vars = { ...(opts.variables ?? {}) };\n const code = vars.code ?? vars.otp ?? opts.otpCode ?? '';\n if (code) {\n vars.code = code;\n delete vars.otp;\n }\n effective.body = applyTemplateVars(resolved.body, vars);\n effective.otpCode = code || effective.otpCode;\n effective.variables = vars;\n flowOverrides = {\n templateId: resolved.externalTemplateRef,\n otpVarKey: resolved.otpVarKey,\n };\n }\n\n const body = (effective.body ?? '').trim();\n if (!body) return false;\n\n const cfg = await this.merged();\n const p = resolveSmsProvider(cfg);\n if (!p) return false;\n if (p === 'msg91') return sendViaMsg91(cfg, opts.to, body, effective, flowOverrides);\n if (p === 'twilio') return sendViaTwilio(cfg, opts.to, body);\n return sendViaWebhook(cfg, opts.to, body);\n } catch {\n return false;\n }\n }\n}\n","const SMS_QUEUE_NAME = 'sms';\n\nexport interface CmsAppLike {\n getPlugin(name: string): unknown;\n}\n\nexport interface SmsJobPayload {\n to: string;\n /** Legacy / Twilio / sendhttp plain text. */\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n}\n\nexport function registerSmsQueueProcessor(cms: CmsAppLike): void {\n const queue = cms.getPlugin('queue') as\n | { add: (name: string, data: object) => Promise<void>; registerProcessor: (name: string, fn: (data: object) => Promise<void>) => void }\n | undefined;\n const sms = cms.getPlugin('sms') as\n | {\n send: (opts: {\n to: string;\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n }) => Promise<boolean>;\n }\n | undefined;\n if (!queue || !sms || typeof sms.send !== 'function') return;\n queue.registerProcessor(SMS_QUEUE_NAME, async (data: object) => {\n const payload = data as SmsJobPayload;\n if (!payload.to) return;\n if (payload.templateKey?.trim()) {\n await sms.send({\n to: payload.to,\n templateKey: payload.templateKey.trim(),\n variables: payload.variables,\n otpCode: payload.otpCode,\n });\n return;\n }\n if (!payload.body?.trim()) return;\n await sms.send({\n to: payload.to,\n body: payload.body,\n otpCode: payload.otpCode,\n variables: payload.variables,\n });\n });\n}\n\nexport async function queueSms(cms: CmsAppLike, payload: SmsJobPayload): Promise<void> {\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n const sms = cms.getPlugin('sms') as\n | {\n send: (opts: {\n to: string;\n body?: string;\n templateKey?: string;\n variables?: Record<string, string>;\n otpCode?: string;\n }) => Promise<boolean>;\n }\n | undefined;\n if (queue) {\n await queue.add(SMS_QUEUE_NAME, payload);\n return;\n }\n if (sms && typeof sms.send === 'function') {\n if (payload.templateKey?.trim()) {\n await sms.send({\n to: payload.to,\n templateKey: payload.templateKey.trim(),\n variables: payload.variables,\n otpCode: payload.otpCode,\n });\n return;\n }\n if (payload.body?.trim()) {\n await sms.send({\n to: payload.to,\n body: payload.body,\n otpCode: payload.otpCode,\n variables: payload.variables,\n });\n }\n }\n}\n","import type { CmsPlugin, PluginContext } from '../types';\nimport {\n SmsService,\n type SmsServiceConfig,\n type MessageTemplateRowLoader,\n mergeSmsConfigLayers,\n anySmsProviderConfigured,\n} from './sms-service';\n\nexport interface SmsPluginConfig extends Partial<SmsServiceConfig> {\n /** Load admin settings group `sms` (enabled, smsProvider). Secrets typically stay in env. */\n getSmsSettings?: () => Promise<Record<string, string>>;\n /** Resolve `message_templates` rows for SMS (required for `templateKey` sends). */\n getMessageTemplateRow?: MessageTemplateRowLoader;\n}\n\nexport interface SmsServiceInterface {\n send(opts: {\n to: string;\n body?: string;\n templateKey?: string;\n otpCode?: string;\n variables?: Record<string, string>;\n }): Promise<boolean>;\n}\n\nexport function smsPlugin(config: SmsPluginConfig = {}): CmsPlugin<SmsServiceInterface | null> {\n return {\n name: 'sms',\n version: '1.0.0',\n async init(context: PluginContext) {\n const env = context.config as Record<string, string>;\n const { getSmsSettings, getMessageTemplateRow, ...staticRest } = config;\n let db: Record<string, string> = {};\n try {\n db = (await getSmsSettings?.()) ?? {};\n } catch {\n /* ignore */\n }\n const merged = mergeSmsConfigLayers(env, db, staticRest);\n if (!anySmsProviderConfigured(merged)) {\n context.logger.warn(\n 'SMS plugin skipped: set MSG91_AUTH_KEY (Flow template id per Plugins → SMS templates or MSG91_TEMPLATE_ID; or sendhttp: MSG91_SENDER_ID), or TWILIO_*, or SMS_WEBHOOK_URL'\n );\n return null;\n }\n const svc = new SmsService(env, staticRest, getSmsSettings, getMessageTemplateRow);\n return {\n send: (opts) => svc.send(opts),\n };\n },\n };\n}\n\nexport {\n mergeSmsConfigLayers,\n resolveSmsProvider,\n anySmsProviderConfigured,\n msg91Configured,\n msg91FlowConfigured,\n msg91SendhttpConfigured,\n resolveMsg91SendMode,\n twilioConfigured,\n webhookConfigured,\n} from './sms-service';\nexport type {\n SmsProviderId,\n SmsProviderChoice,\n SmsServiceConfig,\n SmsSendOptions,\n SmsSettingsLoader,\n Msg91ApiMode,\n MessageTemplateRowLoader,\n} from './sms-service';\n\nexport { registerSmsQueueProcessor, queueSms } from './sms-queue';\nexport type { SmsJobPayload } from './sms-queue';\n","import type { CmsPlugin, PluginContext } from '../types';\n\nexport interface PaymentIntent {\n id: string;\n clientSecret: string;\n amount: number;\n currency: string;\n status: string;\n}\n\nexport interface PaymentServiceInterface {\n createPaymentIntent(amount: number, currency: string, metadata?: Record<string, string>): Promise<PaymentIntent>;\n capturePayment(paymentId: string, amount?: number): Promise<boolean>;\n verifyWebhookSignature(payload: string | Buffer, signature: string): boolean;\n}\n\nexport interface StripeConfig {\n provider: 'stripe';\n secretKey: string;\n webhookSecret?: string;\n}\n\nexport interface RazorpayConfig {\n provider: 'razorpay';\n keyId: string;\n keySecret: string;\n webhookSecret?: string;\n}\n\nexport type PaymentPluginConfig = StripeConfig | RazorpayConfig;\n\nexport function paymentPlugin(config: PaymentPluginConfig): CmsPlugin<PaymentServiceInterface | undefined> {\n return {\n name: 'payment',\n version: '1.0.0',\n async init(context: PluginContext) {\n if (config.provider === 'stripe') {\n const { StripePaymentService } = await import('./stripe');\n const secretKey = config.secretKey || context.config.STRIPE_SECRET_KEY;\n if (!secretKey) {\n context.logger.warn('Payment plugin skipped: Stripe secret key not configured');\n return undefined;\n }\n return new StripePaymentService({\n secretKey,\n webhookSecret: config.webhookSecret || context.config.STRIPE_WEBHOOK_SECRET,\n });\n }\n\n if (config.provider === 'razorpay') {\n const { RazorpayPaymentService } = await import('./razorpay');\n const keyId = config.keyId || context.config.RAZORPAY_KEY_ID;\n const keySecret = config.keySecret || context.config.RAZORPAY_KEY_SECRET;\n if (!keyId || !keySecret) {\n context.logger.warn('Payment plugin skipped: Razorpay credentials not configured');\n return undefined;\n }\n return new RazorpayPaymentService({\n keyId,\n keySecret,\n webhookSecret: config.webhookSecret || context.config.RAZORPAY_WEBHOOK_SECRET,\n });\n }\n\n context.logger.warn(`Payment plugin skipped: unknown provider \"${(config as { provider: string }).provider}\"`);\n return undefined;\n },\n };\n}\n","import type { CmsPlugin } from '../types';\nimport type { StorageService } from './types';\n\nexport interface S3StoragePluginConfig {\n region: string;\n accessKeyId: string;\n secretAccessKey: string;\n bucket: string;\n baseUrl?: string;\n prefix?: string;\n}\n\nfunction getS3Storage(config: S3StoragePluginConfig): StorageService {\n const prefix = (config.prefix ?? 'uploads').replace(/\\/$/, '');\n const keyPrefix = prefix ? `${prefix}/` : '';\n\n return {\n async upload(buffer: Buffer, key: string, contentType: string): Promise<string> {\n const { S3Client, PutObjectCommand } = await import('@aws-sdk/client-s3');\n const client = new S3Client({\n region: config.region,\n credentials: {\n accessKeyId: config.accessKeyId,\n secretAccessKey: config.secretAccessKey,\n },\n });\n const fullKey = keyPrefix + key.replace(/^\\/+/, '');\n await client.send(\n new PutObjectCommand({\n Bucket: config.bucket,\n Key: fullKey,\n Body: buffer,\n ContentType: contentType,\n })\n );\n if (config.baseUrl) {\n return `${config.baseUrl.replace(/\\/$/, '')}/${fullKey}`;\n }\n return `https://s3.${config.region}.amazonaws.com/${config.bucket}/${fullKey}`;\n },\n };\n}\n\nexport function s3StoragePlugin(\n config: Partial<S3StoragePluginConfig> & { bucket: string }\n): CmsPlugin<StorageService> {\n return {\n name: 'storage',\n version: '1.0.0',\n async init(context) {\n const region = config.region ?? context.config.AWS_REGION ?? '';\n const accessKeyId = config.accessKeyId ?? context.config.AWS_ACCESS_KEY_ID ?? '';\n const secretAccessKey = config.secretAccessKey ?? context.config.AWS_SECRET_ACCESS_KEY ?? '';\n const bucket = config.bucket ?? context.config.AWS_BUCKET_NAME ?? '';\n const baseUrl = config.baseUrl ?? context.config.S3_MEDIA_URL ?? context.config.S3_CUSTOM_DOMAIN;\n const resolvedBaseUrl =\n typeof baseUrl === 'string' && baseUrl\n ? baseUrl.startsWith('http')\n ? baseUrl\n : `https://${baseUrl}`\n : undefined;\n if (!region || !accessKeyId || !secretAccessKey || !bucket) {\n throw new Error('S3 storage plugin: missing region, accessKeyId, secretAccessKey or bucket');\n }\n return getS3Storage({\n region,\n accessKeyId,\n secretAccessKey,\n bucket,\n baseUrl: resolvedBaseUrl,\n prefix: config.prefix ?? 'uploads',\n });\n },\n };\n}\n","import type { CmsPlugin } from '../types';\nimport type { StorageService } from './types';\n\nexport interface LocalStoragePluginConfig {\n /** Directory relative to process.cwd() (e.g. \"public/uploads\"). */\n dir?: string;\n /** Public URL prefix for returned URLs (e.g. \"\" for \"/uploads/...\"). */\n publicPath?: string;\n}\n\nfunction getLocalStorage(config: LocalStoragePluginConfig): StorageService {\n const dir = (config.dir ?? 'public/uploads').replace(/\\/$/, '');\n const publicPath =\n config.publicPath ??\n (dir === 'public/uploads' ? '/uploads' : `/${dir.replace(/^\\/+/, '').replace(/\\\\/g, '/')}`);\n\n return {\n async upload(buffer: Buffer, key: string, _contentType: string): Promise<string> {\n const fs = await import('fs/promises');\n const path = await import('path');\n const normalizedKey = key.replace(/^\\/+/, '').replace(/\\.\\./g, '');\n const fileName = normalizedKey.replace(/^uploads\\//, '');\n const fullDir = path.join(process.cwd(), dir);\n await fs.mkdir(fullDir, { recursive: true });\n const filePath = path.join(fullDir, fileName);\n const fileDir = path.dirname(filePath);\n await fs.mkdir(fileDir, { recursive: true });\n await fs.writeFile(filePath, buffer);\n return `${publicPath.replace(/\\/$/, '')}/${fileName}`;\n },\n };\n}\n\nexport function localStoragePlugin(config: LocalStoragePluginConfig = {}): CmsPlugin<StorageService> {\n return {\n name: 'storage',\n version: '1.0.0',\n async init(context) {\n const dir = config.dir ?? context.config.UPLOAD_DIR ?? 'public/uploads';\n return getLocalStorage({ ...config, dir });\n },\n };\n}\n","export interface LlmMessage {\n role: 'user' | 'assistant' | 'system';\n content: string;\n}\n\nexport interface LlmChatOptions {\n model?: string;\n temperature?: number;\n max_tokens?: number;\n}\n\nexport interface LlmServiceInterface {\n chat(messages: LlmMessage[], options?: LlmChatOptions): Promise<{ content: string }>;\n streamChat(messages: LlmMessage[], options?: LlmChatOptions): AsyncIterable<string>;\n /** OpenAI-compatible embeddings; returns [] if gateway does not support /v1/embeddings */\n embed?(text: string, options?: { model?: string }): Promise<number[]>;\n}\n\nexport class LlmService implements LlmServiceInterface {\n private static embedWarned = false;\n\n constructor(\n private readonly baseURL: string,\n private readonly apiKey: string,\n private readonly defaultEmbeddingModel?: string\n ) {}\n\n private get base(): string {\n return this.baseURL.replace(/\\/$/, '');\n }\n\n private get url(): string {\n return `${this.base}/v1/chat/completions`;\n }\n\n /**\n * OpenAI-compatible embeddings. Returns [] if endpoint not available or unexpected response.\n * Server must implement: POST /v1/embeddings\n * Body: { model: string, input: string }\n * Response: { data: [ { embedding: number[] } ] } or { embedding: number[] }\n */\n async embed(text: string, options: { model?: string } = {}): Promise<number[]> {\n const model = options.model ?? this.defaultEmbeddingModel ?? 'text-embedding-3-small';\n const input = text.slice(0, 8000);\n const res = await fetch(`${this.base}/v1/embeddings`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model,\n input,\n }),\n });\n if (!res.ok) {\n const body = await res.text();\n if (!LlmService.embedWarned) {\n LlmService.embedWarned = true;\n console.warn(`[LLM embed] ${res.status} ${res.statusText}: ${body.slice(0, 400)}`);\n }\n return [];\n }\n let data: Record<string, unknown>;\n try {\n data = (await res.json()) as Record<string, unknown>;\n } catch {\n console.warn('[LLM embed] Response is not JSON');\n return [];\n }\n // OpenAI shape: { data: [ { embedding: number[] } ] }\n let embedding = (data?.data as Array<{ embedding?: number[] }>)?.[0]?.embedding;\n if (!Array.isArray(embedding) && Array.isArray(data?.embedding)) {\n embedding = data.embedding as number[];\n }\n if (!Array.isArray(embedding) && Array.isArray((data?.data as unknown[])?.[0])) {\n const first = (data.data as unknown[])[0];\n embedding = Array.isArray(first) ? first : (first as { embedding?: number[] })?.embedding;\n }\n if (!Array.isArray(embedding) && !LlmService.embedWarned) {\n LlmService.embedWarned = true;\n console.warn('[LLM embed] Unexpected response shape. Keys:', Object.keys(data), 'data[0] type:', Array.isArray(data?.data) ? typeof (data.data as unknown[])[0] : 'n/a');\n }\n return Array.isArray(embedding) ? embedding : [];\n }\n\n async chat(messages: LlmMessage[], options: LlmChatOptions = {}): Promise<{ content: string }> {\n const res = await fetch(this.url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model: options.model ?? 'qwen2.5:3b-instruct',\n messages,\n stream: false,\n temperature: options.temperature,\n max_tokens: options.max_tokens,\n }),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`LLM gateway error: ${res.status} ${text}`);\n }\n const data = (await res.json()) as { choices?: Array<{ message?: { content?: string } }> };\n const content = data.choices?.[0]?.message?.content ?? '';\n return { content };\n }\n\n async *streamChat(messages: LlmMessage[], options: LlmChatOptions = {}): AsyncIterable<string> {\n const res = await fetch(this.url, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n Authorization: `Bearer ${this.apiKey}`,\n },\n body: JSON.stringify({\n model: options.model ?? 'qwen2.5:3b-instruct',\n messages,\n stream: true,\n temperature: options.temperature,\n max_tokens: options.max_tokens,\n }),\n });\n if (!res.ok) {\n const text = await res.text();\n throw new Error(`LLM gateway error: ${res.status} ${text}`);\n }\n const reader = res.body?.getReader();\n if (!reader) return;\n const decoder = new TextDecoder();\n let buffer = '';\n while (true) {\n const { value, done } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() ?? '';\n for (const line of lines) {\n if (line.startsWith('data: ') && line !== 'data: [DONE]') {\n try {\n const json = JSON.parse(line.slice(6)) as { choices?: Array<{ delta?: { content?: string } }> };\n const content = json.choices?.[0]?.delta?.content;\n if (content) yield content;\n } catch {\n // skip malformed chunk\n }\n }\n }\n }\n }\n}\n","import type { CmsPlugin } from '../types';\nimport { LlmService } from './llm-service';\n\nexport interface LlmPluginConfig {\n baseURL?: string;\n apiKey?: string;\n /** Default model for embed(); also from env EMBEDDING_MODEL (e.g. for Qwen use your server's embedding model name). */\n embeddingModel?: string;\n}\n\nexport function llmPlugin(config: LlmPluginConfig = {}): CmsPlugin<LlmService | undefined> {\n return {\n name: 'llm',\n version: '1.0.0',\n async init(context) {\n const baseURL = config.baseURL ?? context.config.LLM_GATEWAY_URL;\n const apiKey = config.apiKey ?? context.config.LLM_API_KEY;\n if (!baseURL || !apiKey) {\n context.logger.warn('LLM plugin skipped: LLM_GATEWAY_URL or LLM_API_KEY not set');\n return undefined;\n }\n const embeddingModel = config.embeddingModel ?? context.config.EMBEDDING_MODEL;\n return new LlmService(baseURL.replace(/\\/$/, ''), apiKey, embeddingModel);\n },\n };\n}\n\nexport { LlmService } from './llm-service';\nexport type { LlmServiceInterface, LlmMessage, LlmChatOptions } from './llm-service';\n","import Redis from 'ioredis';\nimport type { CmsPlugin } from '../types';\nimport type { CacheService } from './types';\nimport { MemoryCache } from './memory-cache';\nimport { RedisCache } from './redis-cache';\n\nexport type { CacheService } from './types';\nexport { MemoryCache } from './memory-cache';\nexport { RedisCache } from './redis-cache';\n\nexport interface CachePluginConfig {\n redisUrl?: string;\n}\n\nexport function cachePlugin(config?: CachePluginConfig): CmsPlugin<CacheService> {\n return {\n name: 'cache',\n version: '1.0.0',\n async init(context) {\n const redisUrl = config?.redisUrl ?? context.config.REDIS_URL ?? '';\n const url = typeof redisUrl === 'string' ? redisUrl.trim() : '';\n if (url) {\n const client = new Redis(url);\n return new RedisCache(client);\n }\n return new MemoryCache();\n },\n };\n}\n","import { LRUCache } from 'lru-cache';\nimport type { CacheService } from './types';\n\nconst DEFAULT_MAX = 500;\nconst DEFAULT_TTL_MS = 60 * 1000;\n\nexport class MemoryCache implements CacheService {\n private readonly cache: LRUCache<string, string>;\n\n constructor(options?: { max?: number; ttlMs?: number }) {\n const max = options?.max ?? DEFAULT_MAX;\n const ttlMs = options?.ttlMs ?? DEFAULT_TTL_MS;\n this.cache = new LRUCache<string, string>({ max, ttl: ttlMs });\n }\n\n async get(key: string): Promise<string | null> {\n const value = this.cache.get(key);\n return value ?? null;\n }\n\n async set(key: string, value: string, ttlSeconds?: number): Promise<void> {\n const ttlMs = ttlSeconds != null && ttlSeconds > 0 ? ttlSeconds * 1000 : undefined;\n this.cache.set(key, value, { ttl: ttlMs });\n }\n\n async del(key: string): Promise<void> {\n this.cache.delete(key);\n }\n}\n","import type Redis from 'ioredis';\nimport type { CacheService } from './types';\n\nexport class RedisCache implements CacheService {\n constructor(private readonly client: Redis) {}\n\n async get(key: string): Promise<string | null> {\n return this.client.get(key);\n }\n\n async set(key: string, value: string, ttlSeconds?: number): Promise<void> {\n if (ttlSeconds != null && ttlSeconds > 0) {\n await this.client.set(key, value, 'EX', ttlSeconds);\n } else {\n await this.client.set(key, value);\n }\n }\n\n async del(key: string): Promise<void> {\n await this.client.del(key);\n }\n}\n","import { Queue, Worker } from 'bullmq';\nimport type { Job } from 'bullmq';\nimport type { QueueService } from './types';\n\nfunction connectionFromUrl(redisUrl: string): { host: string; port: number; username?: string; password?: string; maxRetriesPerRequest: number | null } {\n const u = new URL(redisUrl);\n return {\n host: u.hostname,\n port: u.port ? parseInt(u.port, 10) : 6379,\n ...(u.username && { username: u.username }),\n ...(u.password && { password: u.password }),\n maxRetriesPerRequest: null,\n };\n}\n\nexport class BullMQQueue implements QueueService {\n private readonly connectionOptions: ReturnType<typeof connectionFromUrl>;\n private readonly queues = new Map<string, Queue>();\n private readonly workers = new Map<string, Worker>();\n\n constructor(redisUrl: string) {\n this.connectionOptions = connectionFromUrl(redisUrl);\n }\n\n async add(queueName: string, data: object): Promise<void> {\n const queue = this.getOrCreateQueue(queueName);\n await queue.add(queueName, data);\n }\n\n registerProcessor(queueName: string, processor: (data: object) => Promise<void>): void {\n if (this.workers.has(queueName)) return;\n const worker = new Worker(\n queueName,\n async (job: Job) => {\n await processor(job.data as object);\n },\n { connection: this.connectionOptions }\n );\n this.workers.set(queueName, worker);\n }\n\n private getOrCreateQueue(queueName: string): Queue {\n let queue = this.queues.get(queueName);\n if (!queue) {\n queue = new Queue(queueName, { connection: this.connectionOptions });\n this.queues.set(queueName, queue);\n }\n return queue;\n }\n\n async destroy(): Promise<void> {\n for (const w of this.workers.values()) await w.close();\n this.workers.clear();\n for (const q of this.queues.values()) await q.close();\n this.queues.clear();\n }\n}\n","import fastq from 'fastq';\nimport type { QueueService } from './types';\n\ntype Task = object;\n\nexport class MemoryQueue implements QueueService {\n private readonly queues = new Map<string, fastq.queueAsPromised<Task>>();\n\n async add(queueName: string, data: object): Promise<void> {\n const q = this.queues.get(queueName);\n if (!q) throw new Error(`No processor registered for queue: ${queueName}`);\n // fastq.promise push resolves when the task finishes — do not await or HTTP handlers block on SMTP etc.\n void Promise.resolve(q.push(data as Task)).catch(() => undefined);\n }\n\n registerProcessor(queueName: string, processor: (data: object) => Promise<void>): void {\n if (this.queues.has(queueName)) return;\n const worker = async (task: Task) => processor(task);\n const q = fastq.promise(worker, 1);\n this.queues.set(queueName, q);\n }\n}\n","import type { CmsPlugin } from '../types';\nimport type { QueueService } from './types';\nimport { BullMQQueue } from './bullmq-queue';\nimport { MemoryQueue } from './memory-queue';\n\nexport type { QueueService } from './types';\nexport { BullMQQueue } from './bullmq-queue';\nexport { MemoryQueue } from './memory-queue';\n\nexport interface QueuePluginConfig {\n redisUrl?: string;\n}\n\nexport function queuePlugin(config?: QueuePluginConfig): CmsPlugin<QueueService> {\n let instance: BullMQQueue | MemoryQueue | undefined;\n return {\n name: 'queue',\n version: '1.0.0',\n async init(context) {\n const redisUrl = config?.redisUrl ?? context.config.REDIS_URL ?? '';\n const url = typeof redisUrl === 'string' ? redisUrl.trim() : '';\n if (url) {\n instance = new BullMQQueue(url);\n return instance;\n }\n instance = new MemoryQueue();\n return instance;\n },\n async destroy() {\n if (instance && 'destroy' in instance && typeof instance.destroy === 'function') {\n await instance.destroy();\n }\n },\n };\n}\n","export type CaptchaProviderId = 'turnstile' | 'recaptcha_v3';\n\nexport interface CaptchaPublicConfig {\n enabled: boolean;\n activeProvider: CaptchaProviderId | null;\n siteKey: string | null;\n availableProviders: Array<{ id: CaptchaProviderId; siteKey: string }>;\n multipleProviders: boolean;\n}\n\nexport type CaptchaVerifyResult =\n | { ok: true }\n | { ok: false; status: number; message: string };\n\ninterface ResolvedKeys {\n turnstile: { site: string; secret: string } | null;\n recaptcha: { site: string; secret: string } | null;\n /** When both providers configured, set only if CAPTCHA_PROVIDER env is set */\n envDefaultProvider: CaptchaProviderId | null;\n minScore: number;\n}\n\nfunction resolveKeys(config: Record<string, string>): ResolvedKeys {\n const turnstile =\n config.TURNSTILE_SITE_KEY?.trim() && config.TURNSTILE_SECRET_KEY?.trim()\n ? { site: config.TURNSTILE_SITE_KEY.trim(), secret: config.TURNSTILE_SECRET_KEY.trim() }\n : null;\n const recaptcha =\n config.RECAPTCHA_SITE_KEY?.trim() && config.RECAPTCHA_SECRET_KEY?.trim()\n ? { site: config.RECAPTCHA_SITE_KEY.trim(), secret: config.RECAPTCHA_SECRET_KEY.trim() }\n : null;\n const raw = config.CAPTCHA_PROVIDER?.trim().toLowerCase();\n let envDefaultProvider: CaptchaProviderId | null = null;\n if (raw === 'turnstile') envDefaultProvider = 'turnstile';\n else if (raw === 'recaptcha_v3' || raw === 'recaptcha') envDefaultProvider = 'recaptcha_v3';\n\n const minScore = parseFloat(config.RECAPTCHA_MIN_SCORE ?? '0.5');\n return {\n turnstile,\n recaptcha,\n envDefaultProvider,\n minScore: Number.isFinite(minScore) ? minScore : 0.5,\n };\n}\n\nexport function buildCaptchaPublicConfig(config: Record<string, string>): CaptchaPublicConfig {\n const r = resolveKeys(config);\n const available: Array<{ id: CaptchaProviderId; siteKey: string }> = [];\n if (r.turnstile) available.push({ id: 'turnstile', siteKey: r.turnstile.site });\n if (r.recaptcha) available.push({ id: 'recaptcha_v3', siteKey: r.recaptcha.site });\n const enabled = available.length > 0;\n const multipleProviders = available.length > 1;\n let activeProvider: CaptchaProviderId | null = null;\n if (r.turnstile && !r.recaptcha) activeProvider = 'turnstile';\n else if (r.recaptcha && !r.turnstile) activeProvider = 'recaptcha_v3';\n else if (multipleProviders) {\n activeProvider = r.envDefaultProvider ?? available[0]!.id;\n }\n let siteKey: string | null = null;\n if (activeProvider === 'turnstile' && r.turnstile) siteKey = r.turnstile.site;\n else if (activeProvider === 'recaptcha_v3' && r.recaptcha) siteKey = r.recaptcha.site;\n if (!siteKey && available.length === 1) siteKey = available[0]!.siteKey;\n return {\n enabled,\n activeProvider,\n siteKey,\n availableProviders: available,\n multipleProviders,\n };\n}\n\nfunction clientIp(req: Request): string | undefined {\n const cf = req.headers.get('cf-connecting-ip')?.trim();\n if (cf) return cf;\n const xff = req.headers.get('x-forwarded-for')?.trim();\n if (xff) return xff.split(',')[0]!.trim();\n const real = req.headers.get('x-real-ip')?.trim();\n if (real) return real;\n return undefined;\n}\n\nfunction normalizeBodyProvider(v: unknown): CaptchaProviderId | null {\n if (v === 'turnstile') return 'turnstile';\n if (v === 'recaptcha_v3' || v === 'recaptcha') return 'recaptcha_v3';\n return null;\n}\n\nfunction pickVerifyProvider(\n body: Record<string, unknown>,\n r: ResolvedKeys\n): { provider: CaptchaProviderId; secret: string } | { error: string } {\n const bodyP = normalizeBodyProvider(body.captchaProvider);\n const hasT = !!r.turnstile;\n const hasR = !!r.recaptcha;\n\n if (hasT && !hasR) {\n if (bodyP && bodyP !== 'turnstile') return { error: 'Invalid captchaProvider' };\n return { provider: 'turnstile', secret: r.turnstile!.secret };\n }\n if (hasR && !hasT) {\n if (bodyP && bodyP !== 'recaptcha_v3') return { error: 'Invalid captchaProvider' };\n return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n }\n if (!hasT && !hasR) return { error: 'Captcha not configured' };\n\n if (bodyP === 'turnstile' && hasT) return { provider: 'turnstile', secret: r.turnstile!.secret };\n if (bodyP === 'recaptcha_v3' && hasR) return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n if (r.envDefaultProvider === 'turnstile' && hasT) return { provider: 'turnstile', secret: r.turnstile!.secret };\n if (r.envDefaultProvider === 'recaptcha_v3' && hasR) return { provider: 'recaptcha_v3', secret: r.recaptcha!.secret };\n return { error: 'captchaProvider required (turnstile or recaptcha_v3)' };\n}\n\nasync function verifyTurnstile(secret: string, token: string, remoteip?: string): Promise<boolean> {\n const body = new URLSearchParams({ secret, response: token });\n if (remoteip) body.set('remoteip', remoteip);\n const res = await fetch('https://challenges.cloudflare.com/turnstile/v0/siteverify', {\n method: 'POST',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body: body.toString(),\n });\n if (!res.ok) return false;\n const data = (await res.json()) as { success?: boolean };\n return data.success === true;\n}\n\nasync function verifyRecaptcha(\n secret: string,\n token: string,\n remoteip: string | undefined,\n minScore: number\n): Promise<{ ok: boolean; score?: number }> {\n const body = new URLSearchParams({ secret, response: token });\n if (remoteip) body.set('remoteip', remoteip);\n const res = await fetch('https://www.google.com/recaptcha/api/siteverify', {\n method: 'POST',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body: body.toString(),\n });\n if (!res.ok) return { ok: false };\n const data = (await res.json()) as { success?: boolean; score?: number };\n if (data.success !== true) return { ok: false };\n const score = typeof data.score === 'number' ? data.score : 0;\n return { ok: score >= minScore, score };\n}\n\nexport class CaptchaService {\n constructor(private readonly env: Record<string, string>) {}\n\n isEnforced(): boolean {\n return buildCaptchaPublicConfig(this.env).enabled;\n }\n\n getPublicConfig(): CaptchaPublicConfig {\n return buildCaptchaPublicConfig(this.env);\n }\n\n async verify(body: Record<string, unknown>, req: Request): Promise<CaptchaVerifyResult> {\n if (!this.isEnforced()) return { ok: true };\n const token = typeof body.captchaToken === 'string' ? body.captchaToken.trim() : '';\n if (!token) return { ok: false, status: 400, message: 'captchaToken required' };\n\n const r = resolveKeys(this.env);\n const picked = pickVerifyProvider(body, r);\n if ('error' in picked) return { ok: false, status: 400, message: picked.error };\n\n const ip = clientIp(req);\n if (picked.provider === 'turnstile') {\n const ok = await verifyTurnstile(picked.secret, token, ip);\n if (!ok) return { ok: false, status: 400, message: 'Captcha verification failed' };\n return { ok: true };\n }\n const vr = await verifyRecaptcha(picked.secret, token, ip, r.minScore);\n if (!vr.ok) return { ok: false, status: 400, message: 'Captcha verification failed or score too low' };\n return { ok: true };\n }\n}\n","import type { CmsPlugin, PluginContext } from '../types';\nimport { CaptchaService } from './captcha-service';\n\nexport function captchaPlugin(): CmsPlugin<CaptchaService | void> {\n return {\n name: 'captcha',\n version: '1.0.0',\n async init(ctx: PluginContext) {\n const s = new CaptchaService(ctx.config);\n return s.isEnforced() ? s : undefined;\n },\n };\n}\n","import type { CaptchaService } from './captcha-service';\n\ntype JsonResponse = (body: unknown, init?: { status?: number; headers?: HeadersInit }) => Response;\n\nexport async function assertCaptchaOk(\n getCms: (() => Promise<{ getPlugin: (name: string) => unknown }>) | undefined,\n body: Record<string, unknown>,\n req: Request,\n json: JsonResponse\n): Promise<Response | null> {\n if (!getCms) return null;\n let cms: { getPlugin: (name: string) => unknown };\n try {\n cms = await getCms();\n } catch {\n return null;\n }\n const svc = cms.getPlugin('captcha') as CaptchaService | undefined;\n if (!svc || typeof (svc as CaptchaService).verify !== 'function') return null;\n const result = await svc.verify(body, req);\n if (result.ok) return null;\n return json({ error: result.message }, { status: result.status });\n}\n","import { type ClassValue, clsx } from 'clsx';\nimport { twMerge } from 'tailwind-merge';\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n\nexport function generateSlug(title: string): string {\n return title\n .toLowerCase()\n .replace(/[^a-z0-9]+/g, '-')\n .replace(/(^-|-$)/g, '');\n}\n\nexport function validateSlug(slug: string): boolean {\n const slugRegex = /^[a-z0-9]+(?:-[a-z0-9]+)*$/;\n return slugRegex.test(slug) && slug.length >= 3 && slug.length <= 50;\n}\n\nexport function formatDate(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });\n}\n\nexport function formatDateTime(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', {\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n });\n}\n\nexport function formatDateOnly(date: Date | string): string {\n const d = new Date(date);\n return d.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });\n}\n\nexport function truncateText(text: string, maxLength: number): string {\n if (text.length <= maxLength) return text;\n return text.substring(0, maxLength).trim() + '...';\n}\n","import type { DataSource } from 'typeorm';\nimport type { Metadata } from 'next';\nimport type { EntityMap } from '../api/crud';\n\n/** Minimal SEO row shape (DB or API). */\nexport type SeoLike = {\n title?: string | null;\n description?: string | null;\n keywords?: string | null;\n ogTitle?: string | null;\n ogDescription?: string | null;\n ogImage?: string | null;\n};\n\nexport type SeoMetadataOverrides = {\n title?: string;\n description?: string;\n keywords?: string;\n ogTitle?: string;\n ogDescription?: string;\n ogImage?: string;\n};\n\nfunction pickStr(\n override: string | undefined,\n seoPrimary: string | null | undefined,\n seoSecondary: string | null | undefined,\n fallback: string | undefined,\n): string | undefined {\n const o = override?.trim();\n if (o) return o;\n const a = seoPrimary?.trim();\n if (a) return a;\n const b = seoSecondary?.trim();\n if (b) return b;\n return fallback?.trim() || undefined;\n}\n\nfunction pickSeo(row: unknown): SeoLike | null {\n if (!row || typeof row !== 'object') return null;\n const r = row as Record<string, unknown>;\n return {\n title: (r.title as string) ?? null,\n description: (r.description as string) ?? null,\n keywords: (r.keywords as string) ?? null,\n ogTitle: (r.ogTitle as string) ?? null,\n ogDescription: (r.ogDescription as string) ?? null,\n ogImage: (r.ogImage as string) ?? null,\n };\n}\n\n/** Per-field: join row wins over slug row. */\nexport function mergeSeoBySlug(join: SeoLike | null | undefined, bySlug: SeoLike | null): SeoLike | null {\n const a = join ? pickSeo(join) : null;\n const b = bySlug ? pickSeo(bySlug) : null;\n if (!a && !b) return null;\n return {\n title: a?.title ?? b?.title ?? null,\n description: a?.description ?? b?.description ?? null,\n keywords: a?.keywords ?? b?.keywords ?? null,\n ogTitle: a?.ogTitle ?? b?.ogTitle ?? null,\n ogDescription: a?.ogDescription ?? b?.ogDescription ?? null,\n ogImage: a?.ogImage ?? b?.ogImage ?? null,\n };\n}\n\nexport async function fetchSeoBySlug(\n dataSource: DataSource,\n entityMap: EntityMap,\n slug: string,\n): Promise<SeoLike | null> {\n const trimmed = slug?.trim();\n if (!trimmed) return null;\n const SeoEntity = entityMap.seos;\n if (!SeoEntity) return null;\n const repo = dataSource.getRepository(SeoEntity);\n const row = await repo.findOne({\n where: { slug: trimmed, deleted: false } as import('typeorm').ObjectLiteral,\n });\n return row ? pickSeo(row) : null;\n}\n\nfunction toAbsoluteImage(url: string, metadataBase: URL | undefined): string {\n const u = url.trim();\n if (!u) return u;\n if (u.startsWith('http://') || u.startsWith('https://')) return u;\n if (metadataBase) {\n try {\n const path = u.startsWith('/') ? u : `/${u}`;\n return new URL(path, metadataBase).toString();\n } catch {\n return u;\n }\n }\n return u;\n}\n\n/**\n * Build Next.js Metadata from merged SEO + optional layers.\n * `overrides` win over SEO; `fallbacks` apply when SEO (and overrides) omit a field.\n * Returns a partial object; omit fields so the root layout defaults apply.\n */\nexport async function resolvePublicMetadata(args: {\n dataSource: DataSource;\n entityMap: EntityMap;\n slug: string;\n seoFromJoin?: SeoLike | null;\n overrides?: SeoMetadataOverrides;\n fallbacks?: SeoMetadataOverrides;\n canonicalPath?: string;\n metadataBase?: URL;\n}): Promise<Metadata> {\n const { dataSource, entityMap, slug, seoFromJoin, overrides, fallbacks, canonicalPath, metadataBase } = args;\n const bySlug = await fetchSeoBySlug(dataSource, entityMap, slug);\n const seo = mergeSeoBySlug(seoFromJoin, bySlug);\n const o = overrides ?? {};\n const f = fallbacks ?? {};\n\n const title = pickStr(o.title, seo?.title, seo?.ogTitle, f.title);\n\n const description = pickStr(o.description, seo?.description, seo?.ogDescription, f.description);\n\n const keywordsRaw =\n o.keywords?.trim() || seo?.keywords?.trim() || f.keywords?.trim() || undefined;\n const keywords = keywordsRaw\n ? keywordsRaw.split(',').map((k) => k.trim()).filter(Boolean)\n : undefined;\n\n const ogTitle =\n o.ogTitle?.trim() ||\n seo?.ogTitle?.trim() ||\n seo?.title?.trim() ||\n o.title?.trim() ||\n f.ogTitle?.trim() ||\n title;\n\n const ogDescription =\n o.ogDescription?.trim() ||\n seo?.ogDescription?.trim() ||\n seo?.description?.trim() ||\n o.description?.trim() ||\n f.ogDescription?.trim() ||\n description;\n\n const ogImageRaw =\n o.ogImage?.trim() || seo?.ogImage?.trim() || f.ogImage?.trim() || undefined;\n\n const out: Metadata = {};\n\n if (title) out.title = title;\n if (description) out.description = description;\n if (keywords?.length) out.keywords = keywords;\n\n if (ogTitle || ogDescription || ogImageRaw) {\n const images = ogImageRaw\n ? [{ url: toAbsoluteImage(ogImageRaw, metadataBase), alt: ogTitle || title || '' }]\n : undefined;\n out.openGraph = {\n ...(ogTitle && { title: ogTitle }),\n ...(ogDescription && { description: ogDescription }),\n ...(images && { images }),\n };\n out.twitter = {\n card: 'summary_large_image',\n ...(ogTitle && { title: ogTitle }),\n ...(ogDescription && { description: ogDescription }),\n ...(images?.[0] && { images: [images[0].url] }),\n };\n }\n\n if (canonicalPath?.trim() && metadataBase) {\n try {\n const path = canonicalPath.startsWith('/') ? canonicalPath : `/${canonicalPath}`;\n out.alternates = { canonical: new URL(path, metadataBase).toString() };\n } catch {\n /* ignore */\n }\n }\n\n return out;\n}\n","import type { DataSource, EntityTarget, ObjectLiteral } from 'typeorm';\nimport { IsNull } from 'typeorm';\n\n/** Links an unclaimed contact (same email, no userId) to the new user after signup or invite. */\nexport async function linkUnclaimedContactToUser(\n dataSource: DataSource,\n contactsEntity: EntityTarget<ObjectLiteral>,\n userId: number,\n email: string\n): Promise<void> {\n const repo = dataSource.getRepository(contactsEntity);\n const found = await repo.findOne({\n where: { email, userId: IsNull(), deleted: false } as ObjectLiteral,\n });\n if (found) await repo.update((found as { id: number }).id, { userId });\n}\n","/** Parse stored config value into a list of emails (JSON array, or comma/semicolon-separated string). */\nexport function parseEmailRecipientsFromConfig(raw: string | undefined | null): string[] {\n if (raw == null || raw === '') return [];\n const trimmed = raw.trim();\n if (trimmed.startsWith('[')) {\n try {\n const parsed = JSON.parse(trimmed) as unknown;\n if (Array.isArray(parsed)) {\n return parsed.map((e) => String(e).trim()).filter(Boolean);\n }\n } catch {\n // fall through\n }\n }\n return trimmed\n .split(/[,;]+/)\n .map((s) => s.trim())\n .filter(Boolean);\n}\n\n/** Serialize email list for config storage (JSON array in DB). */\nexport function serializeEmailRecipients(emails: string[]): string {\n return JSON.stringify(emails);\n}\n\n/** Join for SMTP `to` header (multiple recipients). */\nexport function joinRecipientsForSend(emails: string[]): string | null {\n if (!emails.length) return null;\n return emails.join(', ');\n}\n","import { createHmac, randomInt, timingSafeEqual } from 'crypto';\nimport type { DataSource } from 'typeorm';\nimport type { EntityTarget, ObjectLiteral } from 'typeorm';\nimport { IsNull, MoreThan } from 'typeorm';\n\nexport type OtpPurpose = 'login' | 'verify_email' | 'verify_phone';\nexport type OtpChannel = 'email' | 'sms';\n\nconst OTP_TTL_MS = 10 * 60 * 1000;\nconst MAX_SENDS_PER_HOUR = 5;\nconst MAX_VERIFY_ATTEMPTS = 8;\n\nfunction getPepper(explicit?: string): string {\n return (explicit || process.env.OTP_PEPPER || process.env.NEXTAUTH_SECRET || 'dev-otp-pepper').trim();\n}\n\nexport function hashOtpCode(code: string, purpose: string, identifier: string, pepper?: string): string {\n return createHmac('sha256', getPepper(pepper)).update(`${purpose}|${identifier}|${code}`).digest('hex');\n}\n\nexport function verifyOtpCodeHash(code: string, storedHash: string, purpose: string, identifier: string, pepper?: string): boolean {\n const h = hashOtpCode(code, purpose, identifier, pepper);\n try {\n return timingSafeEqual(Buffer.from(h, 'utf8'), Buffer.from(storedHash, 'utf8'));\n } catch {\n return false;\n }\n}\n\nexport function generateNumericOtp(length = 6): string {\n const max = 10 ** length;\n return randomInt(0, max).toString().padStart(length, '0');\n}\n\n/** Normalize to E.164-like +digits */\nexport function normalizePhoneE164(raw: string, defaultCountryCode?: string): string | null {\n const t = raw.trim();\n const digitsOnly = t.replace(/\\D/g, '');\n if (digitsOnly.length < 10) return null;\n if (t.startsWith('+')) return `+${digitsOnly}`;\n const cc = (defaultCountryCode || process.env.DEFAULT_PHONE_COUNTRY_CODE || '91').replace(/\\D/g, '');\n if (digitsOnly.length > 10) return `+${digitsOnly}`;\n return `+${cc}${digitsOnly}`;\n}\n\ntype EntityMap = Record<string, EntityTarget<ObjectLiteral>>;\n\nexport async function countRecentOtpSends(\n dataSource: DataSource,\n entityMap: EntityMap,\n purpose: OtpPurpose,\n identifier: string,\n since: Date\n): Promise<number> {\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n return repo.count({\n where: { purpose, identifier, createdAt: MoreThan(since) } as ObjectLiteral,\n });\n}\n\nexport async function createOtpChallenge(\n dataSource: DataSource,\n entityMap: EntityMap,\n input: { purpose: OtpPurpose; channel: OtpChannel; identifier: string; code: string; pepper?: string }\n): Promise<{ ok: true } | { ok: false; error: string; status: number }> {\n const { purpose, channel, identifier, code, pepper } = input;\n const since = new Date(Date.now() - 60 * 60 * 1000);\n const recent = await countRecentOtpSends(dataSource, entityMap, purpose, identifier, since);\n if (recent >= MAX_SENDS_PER_HOUR) {\n return { ok: false, error: 'Too many codes sent. Try again later.', status: 429 };\n }\n\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n await repo.delete({\n purpose,\n identifier,\n consumedAt: IsNull(),\n } as ObjectLiteral);\n\n const expiresAt = new Date(Date.now() + OTP_TTL_MS);\n const codeHash = hashOtpCode(code, purpose, identifier, pepper);\n await repo.save(\n repo.create({\n purpose,\n channel,\n identifier,\n codeHash,\n expiresAt,\n attempts: 0,\n consumedAt: null,\n } as ObjectLiteral)\n );\n return { ok: true };\n}\n\nexport async function verifyAndConsumeOtpChallenge(\n dataSource: DataSource,\n entityMap: EntityMap,\n input: { purpose: OtpPurpose; identifier: string; code: string; pepper?: string }\n): Promise<{ ok: true } | { ok: false; error: string; status: number }> {\n const { purpose, identifier, code, pepper } = input;\n const repo = dataSource.getRepository(entityMap.otp_challenges);\n const row = await repo.findOne({\n where: { purpose, identifier, consumedAt: IsNull() } as ObjectLiteral,\n order: { id: 'DESC' },\n });\n if (!row) {\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n const r = row as ObjectLiteral;\n if (new Date(r.expiresAt as Date) < new Date()) {\n await repo.delete((row as { id: number }).id);\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n const attempts = (r.attempts as number) || 0;\n if (attempts >= MAX_VERIFY_ATTEMPTS) {\n await repo.delete((row as { id: number }).id);\n return { ok: false, error: 'Too many attempts', status: 400 };\n }\n\n const valid = verifyOtpCodeHash(code, r.codeHash as string, purpose, identifier, pepper);\n if (!valid) {\n await repo.update((row as { id: number }).id, { attempts: attempts + 1 } as ObjectLiteral);\n return { ok: false, error: 'Invalid or expired code', status: 400 };\n }\n\n await repo.update((row as { id: number }).id, { consumedAt: new Date(), attempts: attempts + 1 } as ObjectLiteral);\n return { ok: true };\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { UserGroup } from './user-group.entity';\n\n@Entity('users')\nexport class User {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n email: string;\n\n /** E.164 when set; unique among non-null values (partial index in DB). */\n @Column('varchar', { nullable: true })\n phone: string | null;\n\n @Column({ type: 'timestamp', nullable: true })\n phoneVerifiedAt: Date | null;\n\n @Column({ type: 'timestamp', nullable: true })\n emailVerifiedAt: Date | null;\n\n @Column('varchar', { nullable: true })\n password: string | null;\n\n @Column('boolean', { default: false })\n blocked: boolean;\n\n @Column('boolean', { default: false })\n adminAccess: boolean;\n\n @Column('int', { nullable: true })\n groupId: number | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => UserGroup, (g) => g.users, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'groupId' })\n group: UserGroup | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { Permission } from './permission.entity';\nimport { User } from './user.entity';\n\n@Entity('user_groups')\nexport class UserGroup {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => Permission, (p) => p.group)\n permissions: Permission[];\n\n @OneToMany(() => User, (u) => u.group)\n users: User[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { UserGroup } from './user-group.entity';\n\n@Entity('permissions')\nexport class Permission {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n groupId: number;\n\n @Column('varchar')\n entity: string;\n\n @Column('boolean', { default: false })\n canCreate: boolean;\n\n @Column('boolean', { default: false })\n canRead: boolean;\n\n @Column('boolean', { default: false })\n canUpdate: boolean;\n\n @Column('boolean', { default: false })\n canDelete: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => UserGroup, (g) => g.permissions, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'groupId' })\n group: UserGroup;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';\n\n@Entity('otp_challenges')\n@Index(['purpose', 'identifier'])\nexport class OtpChallenge {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n purpose: string;\n\n @Column('varchar')\n channel: string;\n\n @Column('varchar')\n identifier: string;\n\n @Column('varchar')\n codeHash: string;\n\n @Column({ type: 'timestamp' })\n expiresAt: Date;\n\n @Column('int', { default: 0 })\n attempts: number;\n\n @Column({ type: 'timestamp', nullable: true })\n consumedAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('password_reset_tokens')\nexport class PasswordResetToken {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n email: string;\n\n @Column('varchar', { unique: true })\n token: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n expiresAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n}\n","import {\n Entity,\n PrimaryGeneratedColumn,\n Column,\n ManyToOne,\n OneToMany,\n ManyToMany,\n JoinTable,\n JoinColumn,\n} from 'typeorm';\nimport { User } from './user.entity';\nimport { Category } from './category.entity';\nimport { Seo } from './seo.entity';\nimport { Comment } from './comment.entity';\nimport { Tag } from './tag.entity';\n\n@Entity('blogs')\nexport class Blog {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n title: string;\n\n @Column('text')\n content: string;\n\n @Column('varchar', { nullable: true })\n coverImage: string | null;\n\n @Column('int')\n authorId: number;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @ManyToOne(() => User, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'authorId' })\n author: User;\n\n @ManyToOne(() => Category, (c) => c.blogs, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: Category | null;\n\n @ManyToOne(() => Seo, (s) => s.blogs, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @OneToMany(() => Comment, (c) => c.blog)\n comments: Comment[];\n\n @ManyToMany(() => Tag, (t) => t.blogs)\n @JoinTable({\n name: 'blog_tags',\n joinColumn: { name: 'blogId', referencedColumnName: 'id' },\n inverseJoinColumn: { name: 'tagId', referencedColumnName: 'id' },\n })\n tags: Tag[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport type { Blog } from './blog.entity';\n\n@Entity('categories')\nexport class Category {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany('Blog', 'category')\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { Blog } from './blog.entity';\n\n@Entity('seos')\nexport class Seo {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n title: string | null;\n\n @Column('varchar', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n keywords: string | null;\n\n @Column('varchar', { nullable: true })\n ogTitle: string | null;\n\n @Column('varchar', { nullable: true })\n ogDescription: string | null;\n\n @Column('varchar', { nullable: true })\n ogImage: string | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => Blog, (blog) => blog.seo)\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { User } from './user.entity';\nimport { Blog } from './blog.entity';\n\n@Entity('comments')\nexport class Comment {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('text')\n content: string;\n\n @Column('int')\n blogId: number;\n\n @Column('int')\n authorId: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => User, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'authorId' })\n author: User;\n\n @ManyToOne(() => Blog, (b) => b.comments, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'blogId' })\n blog: Blog;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToMany } from 'typeorm';\nimport { Blog } from './blog.entity';\n\n@Entity('tags')\nexport class Tag {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToMany(() => Blog, (blog) => blog.tags)\n blogs: Blog[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne, JoinColumn } from 'typeorm';\nimport { User } from './user.entity';\nimport { FormSubmission } from './form-submission.entity';\nimport { Address } from './address.entity';\nimport { Order } from './order.entity';\nimport { Payment } from './payment.entity';\nimport { ChatConversation } from './chat-conversation.entity';\n\n@Entity('contacts')\nexport class Contact {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n email: string;\n\n @Column('varchar', { nullable: true })\n phone: string | null;\n\n @Column('varchar', { nullable: true })\n type: string | null;\n\n @Column('varchar', { nullable: true })\n company: string | null;\n\n @Column('varchar', { nullable: true })\n taxId: string | null;\n\n @Column('text', { nullable: true })\n notes: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n userId: number | null;\n\n @ManyToOne(() => User, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'userId' })\n user: User | null;\n\n @OneToMany(() => FormSubmission, (fs) => fs.contact)\n form_submissions: FormSubmission[];\n\n @OneToMany(() => Address, (a) => a.contact)\n addresses: Address[];\n\n @OneToMany(() => Order, (o) => o.contact)\n orders: Order[];\n\n @OneToMany(() => Payment, (p) => p.contact)\n payments: Payment[];\n\n @OneToMany(() => ChatConversation, (c) => c.contact)\n chatConversations: ChatConversation[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Form } from './form.entity';\nimport { Contact } from './contact.entity';\n\n@Entity('form_submissions')\nexport class FormSubmission {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n formId: number;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('jsonb')\n data: Record<string, unknown>;\n\n @Column('varchar', { nullable: true })\n ipAddress: string | null;\n\n @Column('varchar', { nullable: true })\n userAgent: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Form, (f) => f.submissions, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'formId' })\n form: Form;\n\n @ManyToOne(() => Contact, (c) => c.form_submissions, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { FormField } from './form-field.entity';\nimport { FormSubmission } from './form-submission.entity';\n\n@Entity('forms')\nexport class Form {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n campaign: string | null;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @OneToMany(() => FormField, (f) => f.form)\n fields: FormField[];\n\n @OneToMany(() => FormSubmission, (s) => s.form)\n submissions: FormSubmission[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Form } from './form.entity';\n\n@Entity('form_fields')\nexport class FormField {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n formId: number;\n\n @Column('varchar')\n label: string;\n\n @Column('varchar')\n type: string;\n\n @Column('varchar', { nullable: true })\n placeholder: string | null;\n\n @Column('varchar', { nullable: true })\n options: string | null;\n\n @Column('boolean', { default: false })\n required: boolean;\n\n @Column('varchar', { nullable: true })\n validation: string | null;\n\n @Column('int')\n order: number;\n\n @Column('int', { default: 1 })\n groupId: number;\n\n @Column('int', { default: 12 })\n columnWidth: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Form, (f) => f.fields, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'formId' })\n form: Form;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\n\n@Entity('addresses')\nexport class Address {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n contactId: number;\n\n @Column('varchar', { nullable: true })\n tag: string | null;\n\n @Column('varchar', { nullable: true })\n line1: string | null;\n\n @Column('varchar', { nullable: true })\n line2: string | null;\n\n @Column('varchar', { nullable: true })\n city: string | null;\n\n @Column('varchar', { nullable: true })\n state: string | null;\n\n @Column('varchar', { nullable: true })\n postalCode: string | null;\n\n @Column('varchar', { nullable: true })\n country: string | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, (c) => c.addresses, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport { Address } from './address.entity';\nimport type { OrderItem } from './order-item.entity';\nimport type { Payment } from './payment.entity';\n\nexport type OrderKind = 'sale' | 'return' | 'replacement';\n\n@Entity('orders')\nexport class Order {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { unique: true })\n orderNumber: string;\n\n @Column('varchar', { default: 'sale' })\n orderKind: OrderKind;\n\n @Column('int', { nullable: true })\n parentOrderId: number | null;\n\n @Column('int')\n contactId: number;\n\n @Column('int', { nullable: true })\n billingAddressId: number | null;\n\n @Column('int', { nullable: true })\n shippingAddressId: number | null;\n\n @Column('varchar', { default: 'pending' })\n status: 'pending' | 'confirmed' | 'processing' | 'completed' | 'cancelled';\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n subtotal: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n tax: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n discount: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n total: number;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Order, (o) => o.children, { nullable: true, onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentOrderId' })\n parentOrder: Order | null;\n\n @OneToMany(() => Order, (o) => o.parentOrder)\n children: Order[];\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n\n @ManyToOne(() => Address, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'billingAddressId' })\n billingAddress: Address | null;\n\n @ManyToOne(() => Address, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'shippingAddressId' })\n shippingAddress: Address | null;\n\n @OneToMany('OrderItem', 'order')\n items: OrderItem[];\n\n @OneToMany('Payment', 'order')\n payments: Payment[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Order } from './order.entity';\nimport { Contact } from './contact.entity';\n\n@Entity('payments')\nexport class Payment {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n orderId: number;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('decimal', { precision: 12, scale: 2 })\n amount: number;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column('varchar', { default: 'pending' })\n status: 'pending' | 'processing' | 'completed' | 'failed' | 'refunded';\n\n @Column('varchar', { nullable: true })\n method: string | null;\n\n @Column('varchar', { nullable: true })\n externalReference: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', nullable: true })\n paidAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => Order, (o) => o.payments, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'orderId' })\n order: Order;\n\n @ManyToOne(() => Contact, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport { ChatMessage } from './chat-message.entity';\n\n@Entity('chat_conversations')\nexport class ChatConversation {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n contactId: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, (c) => c.chatConversations, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact;\n\n @OneToMany(() => ChatMessage, (m) => m.conversation)\n messages: ChatMessage[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { ChatConversation } from './chat-conversation.entity';\n\n@Entity('chat_messages')\nexport class ChatMessage {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n conversationId: number;\n\n @Column('varchar')\n role: 'user' | 'assistant' | 'system';\n\n @Column('text')\n content: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @ManyToOne(() => ChatConversation, (c) => c.messages, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'conversationId' })\n conversation: ChatConversation;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, Unique } from 'typeorm';\n\n@Entity('configs')\n@Unique(['settings', 'key'])\nexport class Config {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n settings: string;\n\n @Column('varchar')\n key: string;\n\n @Column('varchar')\n value: string;\n\n @Column('varchar', { default: 'private' })\n type: 'public' | 'private';\n\n @Column('boolean', { default: false })\n encrypted: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('message_templates')\nexport class MessageTemplate {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n channel: string;\n\n @Column('varchar', { name: 'template_key' })\n templateKey: string;\n\n @Column('varchar', { nullable: true })\n name: string | null;\n\n @Column('varchar', { nullable: true })\n subject: string | null;\n\n @Column('text', { default: '' })\n body: string;\n\n @Column('varchar', { name: 'external_template_ref', nullable: true })\n externalTemplateRef: string | null;\n\n @Column({ type: 'jsonb', nullable: true })\n providerMeta: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n enabled: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('media')\nexport class Media {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n filename: string;\n\n @Column('varchar')\n url: string;\n\n @Column('varchar')\n mimeType: string;\n\n @Column('int', { default: 0 })\n size: number;\n\n @Column('varchar', { nullable: true })\n alt: string | null;\n\n @Column('boolean', { default: false })\n isPublic: boolean;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Seo } from './seo.entity';\n\n@Entity('pages')\nexport class Page {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n title: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column({ type: 'jsonb', default: {} })\n content: object;\n\n @Column('boolean', { default: false })\n published: boolean;\n\n @Column('varchar', { default: 'default' })\n theme: string;\n\n @Column('int', { nullable: true })\n parentId: number | null;\n\n @ManyToOne(() => Page, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentId' })\n parent: Page | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport type { Product } from './product.entity';\nimport type { Collection } from './collection.entity';\n\n@Entity('product_categories')\nexport class ProductCategory {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('int', { nullable: true })\n parentId: number | null;\n\n @Column('varchar', { nullable: true })\n image: string | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.children, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'parentId' })\n parent: ProductCategory | null;\n\n @OneToMany(() => ProductCategory, (c) => c.parent)\n children: ProductCategory[];\n\n @OneToMany('Product', 'category')\n products: Product[];\n\n @OneToMany('Collection', 'category')\n collections: Collection[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { ProductCategory } from './product-category.entity';\nimport { Brand } from './brand.entity';\nimport { Seo } from './seo.entity';\nimport type { Product } from './product.entity';\n\n@Entity('collections')\nexport class Collection {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('int', { nullable: true })\n brandId: number | null;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { nullable: true })\n hsn: string | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('varchar', { nullable: true })\n image: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.collections, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: ProductCategory | null;\n\n @ManyToOne(() => Brand, (b) => b.collections, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'brandId' })\n brand: Brand | null;\n\n @OneToMany('Product', 'collection')\n products: Product[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany, ManyToOne, JoinColumn } from 'typeorm';\nimport { Seo } from './seo.entity';\nimport type { Product } from './product.entity';\nimport type { Collection } from './collection.entity';\n\n@Entity('brands')\nexport class Brand {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { nullable: true })\n logo: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @OneToMany('Product', 'brand')\n products: Product[];\n\n @OneToMany('Collection', 'brand')\n collections: Collection[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Collection } from './collection.entity';\nimport { Brand } from './brand.entity';\nimport { ProductCategory } from './product-category.entity';\nimport { Seo } from './seo.entity';\nimport type { ProductAttribute } from './product-attribute.entity';\nimport type { ProductTax } from './product-tax.entity';\n\n@Entity('products')\nexport class Product {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int', { nullable: true })\n collectionId: number | null;\n\n @Column('int', { nullable: true })\n brandId: number | null;\n\n @Column('int', { nullable: true })\n categoryId: number | null;\n\n @Column('varchar', { nullable: true })\n sku: string | null;\n\n @Column('varchar', { nullable: true })\n hsn: string | null;\n\n /** Unit of measure, e.g. pcs, kg, hrs */\n @Column('varchar', { nullable: true })\n uom: string | null;\n\n @Column('varchar', { default: 'product' })\n type: 'product' | 'service';\n\n @Column('varchar', { unique: true, nullable: true })\n slug: string | null;\n\n @Column('varchar', { nullable: true })\n name: string | null;\n\n @Column('decimal', { precision: 12, scale: 2 })\n price: number;\n\n @Column('decimal', { precision: 12, scale: 2, nullable: true })\n compareAtPrice: number | null;\n\n @Column('int', { default: 0 })\n quantity: number;\n\n @Column('varchar', { default: 'draft' })\n status: 'draft' | 'available' | 'reserved' | 'sold';\n\n @Column('boolean', { default: false })\n featured: boolean;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n\n @Column('int', { nullable: true })\n seoId: number | null;\n\n @ManyToOne(() => Seo, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'seoId' })\n seo: Seo | null;\n\n @ManyToOne(() => Collection, (c) => c.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'collectionId' })\n collection: Collection | null;\n\n @ManyToOne(() => Brand, (b) => b.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'brandId' })\n brand: Brand | null;\n\n @ManyToOne(() => ProductCategory, (c) => c.products, { onDelete: 'SET NULL' })\n @JoinColumn({ name: 'categoryId' })\n category: ProductCategory | null;\n\n @OneToMany('ProductAttribute', 'product')\n attributes: ProductAttribute[];\n\n @OneToMany('ProductTax', 'product')\n taxes: ProductTax[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('attributes')\nexport class Attribute {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('varchar', { default: 'text' })\n type: 'text' | 'number' | 'select' | 'boolean';\n\n @Column('jsonb', { nullable: true })\n options: string[] | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('int', { default: 0 })\n sortOrder: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Product } from './product.entity';\nimport { Attribute } from './attribute.entity';\n\n@Entity('product_attributes')\nexport class ProductAttribute {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n productId: number;\n\n @Column('int')\n attributeId: number;\n\n @Column('varchar')\n value: string;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Product, (p) => p.attributes, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n\n @ManyToOne(() => Attribute, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'attributeId' })\n attribute: Attribute;\n}\n","import { Entity, PrimaryGeneratedColumn, Column } from 'typeorm';\n\n@Entity('taxes')\nexport class Tax {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { unique: true })\n slug: string;\n\n @Column('decimal', { precision: 5, scale: 2 })\n rate: number;\n\n @Column('boolean', { default: false })\n isDefault: boolean;\n\n @Column('text', { nullable: true })\n description: string | null;\n\n @Column('boolean', { default: true })\n active: boolean;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @Column({ type: 'timestamp', nullable: true })\n deletedAt: Date | null;\n\n @Column('boolean', { default: false })\n deleted: boolean;\n\n @Column('int', { nullable: true })\n createdBy: number | null;\n\n @Column('int', { nullable: true })\n updatedBy: number | null;\n\n @Column('int', { nullable: true })\n deletedBy: number | null;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Product } from './product.entity';\nimport { Tax } from './tax.entity';\n\n@Entity('product_taxes')\nexport class ProductTax {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n productId: number;\n\n @Column('int')\n taxId: number;\n\n @Column('decimal', { precision: 5, scale: 2, nullable: true })\n rate: number | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Product, (p) => p.taxes, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n\n @ManyToOne(() => Tax, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'taxId' })\n tax: Tax;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Order } from './order.entity';\nimport { Product } from './product.entity';\n\n@Entity('order_items')\nexport class OrderItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n orderId: number;\n\n @Column('int')\n productId: number;\n\n @Column('int', { default: 1 })\n quantity: number;\n\n @Column('decimal', { precision: 12, scale: 2 })\n unitPrice: number;\n\n @Column('decimal', { precision: 12, scale: 2, default: 0 })\n tax: number;\n\n @Column('decimal', { precision: 12, scale: 2 })\n total: number;\n\n /** Snapshot at order time */\n @Column('varchar', { nullable: true })\n hsn: string | null;\n\n @Column('varchar', { nullable: true })\n uom: string | null;\n\n @Column('varchar', { nullable: true })\n productType: string | null;\n\n @Column('decimal', { precision: 5, scale: 2, nullable: true })\n taxRate: number | null;\n\n /** Tax slug snapshot (from taxes.slug) */\n @Column('varchar', { nullable: true })\n taxCode: string | null;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Order, (o) => o.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'orderId' })\n order: Order;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from 'typeorm';\nimport { KnowledgeBaseChunk } from './knowledge-base-chunk.entity';\n\n@Entity('knowledge_base_documents')\nexport class KnowledgeBaseDocument {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar')\n name: string;\n\n @Column('varchar', { nullable: true })\n sourceUrl: string | null;\n\n @Column('text')\n content: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @OneToMany(() => KnowledgeBaseChunk, (c) => c.document)\n chunks: KnowledgeBaseChunk[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { KnowledgeBaseDocument } from './knowledge-base-document.entity';\n\n@Entity('knowledge_base_chunks')\nexport class KnowledgeBaseChunk {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n documentId: number;\n\n @Column('text')\n content: string;\n\n @Column('int', { default: 0 })\n chunkIndex: number;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @ManyToOne(() => KnowledgeBaseDocument, (d) => d.chunks, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'documentId' })\n document: KnowledgeBaseDocument;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport type { CartItem } from './cart-item.entity';\n\n@Entity('carts')\nexport class Cart {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n guestToken: string | null;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('varchar', { default: 'INR' })\n currency: string;\n\n @Column({ type: 'timestamp', nullable: true })\n expiresAt: Date | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n\n @OneToMany('CartItem', 'cart')\n items: CartItem[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Cart } from './cart.entity';\nimport { Product } from './product.entity';\n\n@Entity('cart_items')\nexport class CartItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n cartId: number;\n\n @Column('int')\n productId: number;\n\n @Column('int', { default: 1 })\n quantity: number;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Cart, (c) => c.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'cartId' })\n cart: Cart;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, OneToMany, JoinColumn } from 'typeorm';\nimport { Contact } from './contact.entity';\nimport type { WishlistItem } from './wishlist-item.entity';\n\n@Entity('wishlists')\nexport class Wishlist {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('varchar', { nullable: true })\n guestId: string | null;\n\n @Column('int', { nullable: true })\n contactId: number | null;\n\n @Column('varchar', { default: 'default' })\n name: string;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Contact, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'contactId' })\n contact: Contact | null;\n\n @OneToMany('WishlistItem', 'wishlist')\n items: WishlistItem[];\n}\n","import { Entity, PrimaryGeneratedColumn, Column, ManyToOne, JoinColumn } from 'typeorm';\nimport { Wishlist } from './wishlist.entity';\nimport { Product } from './product.entity';\n\n@Entity('wishlist_items')\nexport class WishlistItem {\n @PrimaryGeneratedColumn()\n id: number;\n\n @Column('int')\n wishlistId: number;\n\n @Column('int')\n productId: number;\n\n @Column('jsonb', { nullable: true })\n metadata: Record<string, unknown> | null;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n createdAt: Date;\n\n @Column({ type: 'timestamp', default: () => 'CURRENT_TIMESTAMP' })\n updatedAt: Date;\n\n @ManyToOne(() => Wishlist, (w) => w.items, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'wishlistId' })\n wishlist: Wishlist;\n\n @ManyToOne(() => Product, { onDelete: 'CASCADE' })\n @JoinColumn({ name: 'productId' })\n product: Product;\n}\n","import type { EntityTarget } from 'typeorm';\nimport { User } from './user.entity';\nimport { OtpChallenge } from './otp-challenge.entity';\nimport { PasswordResetToken } from './password-reset-token.entity';\nimport { UserGroup } from './user-group.entity';\nimport { Permission } from './permission.entity';\nimport { Blog } from './blog.entity';\nimport { Tag } from './tag.entity';\nimport { Category } from './category.entity';\nimport { Comment } from './comment.entity';\nimport { Contact } from './contact.entity';\nimport { Address } from './address.entity';\nimport { Form } from './form.entity';\nimport { FormField } from './form-field.entity';\nimport { FormSubmission } from './form-submission.entity';\nimport { Seo } from './seo.entity';\nimport { Config } from './config.entity';\nimport { MessageTemplate } from './message-template.entity';\nimport { Media } from './media.entity';\nimport { Page } from './page.entity';\nimport { ProductCategory } from './product-category.entity';\nimport { Collection } from './collection.entity';\nimport { Product } from './product.entity';\nimport { Attribute } from './attribute.entity';\nimport { ProductAttribute } from './product-attribute.entity';\nimport { Tax } from './tax.entity';\nimport { ProductTax } from './product-tax.entity';\nimport { Order } from './order.entity';\nimport { OrderItem } from './order-item.entity';\nimport { Payment } from './payment.entity';\nimport { Brand } from './brand.entity';\nimport { KnowledgeBaseDocument } from './knowledge-base-document.entity';\nimport { KnowledgeBaseChunk } from './knowledge-base-chunk.entity';\nimport { ChatConversation } from './chat-conversation.entity';\nimport { ChatMessage } from './chat-message.entity';\nimport { Cart } from './cart.entity';\nimport { CartItem } from './cart-item.entity';\nimport { Wishlist } from './wishlist.entity';\nimport { WishlistItem } from './wishlist-item.entity';\n\nexport {\n User,\n OtpChallenge,\n PasswordResetToken,\n UserGroup,\n Permission,\n Blog,\n Tag,\n Category,\n Comment,\n Contact,\n Form,\n FormField,\n FormSubmission,\n Seo,\n Config,\n MessageTemplate,\n Media,\n Page,\n ProductCategory,\n Collection,\n Product,\n Attribute,\n ProductAttribute,\n Tax,\n ProductTax,\n Order,\n OrderItem,\n Payment,\n Brand,\n KnowledgeBaseDocument,\n KnowledgeBaseChunk,\n ChatConversation,\n ChatMessage,\n Cart,\n CartItem,\n Wishlist,\n WishlistItem,\n};\n\n/** Map API resource segment (e.g. \"blogs\", \"form_submissions\") to entity. Used by CRUD handler. */\nexport const CMS_ENTITY_MAP: Record<string, EntityTarget<import('typeorm').ObjectLiteral>> = {\n users: User,\n otp_challenges: OtpChallenge,\n password_reset_tokens: PasswordResetToken,\n user_groups: UserGroup,\n permissions: Permission,\n blogs: Blog,\n tags: Tag,\n categories: Category,\n comments: Comment,\n contacts: Contact,\n addresses: Address,\n forms: Form,\n form_fields: FormField,\n form_submissions: FormSubmission,\n seos: Seo,\n configs: Config,\n message_templates: MessageTemplate,\n media: Media,\n pages: Page,\n product_categories: ProductCategory,\n collections: Collection,\n products: Product,\n attributes: Attribute,\n product_attributes: ProductAttribute,\n taxes: Tax,\n product_taxes: ProductTax,\n orders: Order,\n order_items: OrderItem,\n payments: Payment,\n brands: Brand,\n knowledge_base_documents: KnowledgeBaseDocument,\n knowledge_base_chunks: KnowledgeBaseChunk,\n chat_conversations: ChatConversation,\n chat_messages: ChatMessage,\n carts: Cart,\n cart_items: CartItem,\n wishlists: Wishlist,\n wishlist_items: WishlistItem,\n};\n","/** API resource keys excluded from the permission matrix (internal / not admin-CRUD). */\nexport const PERMISSION_ENTITY_INTERNAL_EXCLUDE = new Set([\n 'users',\n 'password_reset_tokens',\n 'user_groups',\n 'permissions',\n 'comments',\n 'form_fields',\n 'configs',\n 'knowledge_base_chunks',\n 'carts',\n 'cart_items',\n 'wishlists',\n 'wishlist_items',\n 'message_templates',\n]);\n\n/** Non-CRUD admin surfaces mapped to entity keys for RBAC. */\nexport const PERMISSION_LOGICAL_ENTITIES = [\n 'users',\n 'forms',\n 'form_submissions',\n 'dashboard',\n 'upload',\n 'settings',\n 'analytics',\n 'chat',\n] as const;\n\n/** Canonical name for new installs / migrations */\nexport const ADMIN_GROUP_NAME = 'Administrator';\n\n/** System administrator group (roles UI + users / user_groups / permissions bypass). */\nexport function isSuperAdminGroupName(name: string | null | undefined): boolean {\n return name === ADMIN_GROUP_NAME;\n}\n\nexport type EntityCrudAction = 'create' | 'read' | 'update' | 'delete';\n\nexport type EntityPermissionFlags = { c: boolean; r: boolean; u: boolean; d: boolean };\n\nexport function getPermissionableEntityKeys(entityMap: Record<string, unknown>): string[] {\n const fromMap = Object.keys(entityMap).filter((k) => !PERMISSION_ENTITY_INTERNAL_EXCLUDE.has(k));\n const logical = PERMISSION_LOGICAL_ENTITIES.filter((k) => !fromMap.includes(k));\n return [...fromMap.sort(), ...logical].filter((k, i, a) => a.indexOf(k) === i);\n}\n\nexport function permissionRowsToRecord(\n rows: Array<{ entity: string; canCreate: boolean; canRead: boolean; canUpdate: boolean; canDelete: boolean }> | undefined\n): Record<string, EntityPermissionFlags> {\n const out: Record<string, EntityPermissionFlags> = {};\n if (!rows?.length) return out;\n for (const p of rows) {\n out[p.entity] = {\n c: !!p.canCreate,\n r: !!p.canRead,\n u: !!p.canUpdate,\n d: !!p.canDelete,\n };\n }\n return out;\n}\n\nexport function hasEntityPermission(\n record: Record<string, EntityPermissionFlags> | undefined,\n entity: string,\n action: EntityCrudAction\n): boolean {\n const p = record?.[entity];\n if (!p) return false;\n if (action === 'create') return p.c;\n if (action === 'read') return p.r;\n if (action === 'update') return p.u;\n return p.d;\n}\n","import type { EntityCrudAction, EntityPermissionFlags } from './permission-entities';\nimport { hasEntityPermission } from './permission-entities';\n\n/** isRBACAdmin bypasses entity checks only for these (users / roles plumbing). */\nexport const RBAC_ADMIN_ONLY_ENTITIES = new Set(['users', 'user_groups', 'permissions']);\n\nexport interface SessionUser {\n id?: string;\n email?: string | null;\n name?: string | null;\n groupId?: number;\n /** @deprecated use entityPerms / isRBACAdmin */\n permissions?: string[];\n /** Administrator group: full access only for users, user_groups, permissions */\n isRBACAdmin?: boolean;\n entityPerms?: Record<string, EntityPermissionFlags>;\n /** When false and not isRBACAdmin, admin API/UI is denied. */\n adminAccess?: boolean;\n}\n\nexport function sessionHasEntityAccess(\n user: SessionUser | null | undefined,\n entity: string,\n action: EntityCrudAction\n): boolean {\n if (!user?.email) return false;\n if (user.isRBACAdmin && RBAC_ADMIN_ONLY_ENTITIES.has(entity)) return true;\n return hasEntityPermission(user.entityPerms, entity, action);\n}\n\nexport function canManageRoles(user: SessionUser | null | undefined): boolean {\n return !!(user?.email && user.isRBACAdmin);\n}\n\nexport type GetSession = () => Promise<{ user?: SessionUser } | null>;\n\nexport const OPEN_ENDPOINTS: Array<Record<string, string[]>> = [\n { '/api/contacts': ['POST'] },\n { '/api/form-submissions': ['POST'] },\n { '/api/blogs': ['GET'] },\n];\n\nexport const PERMISSION_REQUIRED_ENDPOINTS: Record<string, string[]> = {};\n\nexport function isOpenEndpoint(pathname: string): boolean {\n return OPEN_ENDPOINTS.some((endpoint) => pathname.startsWith(Object.keys(endpoint)[0]));\n}\n\nexport function getRequiredPermission(pathname: string): string[] | null {\n return null;\n}\n\nexport function isPublicMethod(pathname: string, method: string): boolean {\n for (const endpoint of OPEN_ENDPOINTS) {\n const key = Object.keys(endpoint)[0];\n if (pathname.startsWith(key) && endpoint[key].includes(method)) return true;\n }\n return false;\n}\n\nexport interface AuthHelpers {\n requireAuth(req: Request): Promise<Response | null>;\n requirePermission(req: Request, permission: string): Promise<Response | null>;\n requireEntityPermission(req: Request, entity: string, action: EntityCrudAction): Promise<Response | null>;\n requireAdminAccess(req: Request): Promise<Response | null>;\n getAuthenticatedUser(): Promise<SessionUser | null>;\n}\n\nexport function createAuthHelpers(getSession: GetSession, NextResponse: { json: (body: unknown, init?: { status?: number }) => Response }): AuthHelpers {\n return {\n async requireAuth() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n return null;\n },\n async requirePermission() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n return null;\n },\n async requireEntityPermission(_req: Request, entity: string, action: EntityCrudAction) {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n const u = session.user as SessionUser;\n if (sessionHasEntityAccess(u, entity, action)) return null;\n return NextResponse.json({ error: 'Forbidden', entity, action }, { status: 403 });\n },\n async requireAdminAccess() {\n const session = await getSession();\n if (!session?.user?.email) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });\n }\n const u = session.user as SessionUser;\n if (u.isRBACAdmin) return null;\n if (u.adminAccess === false) return NextResponse.json({ error: 'Forbidden', reason: 'admin_access' }, { status: 403 });\n return null;\n },\n async getAuthenticatedUser() {\n const session = await getSession();\n return (session?.user as SessionUser) ?? null;\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityTarget } from 'typeorm';\nimport type { ObjectLiteral } from 'typeorm';\nimport { getPermissionableEntityKeys, ADMIN_GROUP_NAME } from './permission-entities';\n\n/**\n * Ensures the Administrator group has full CRUD on every permissionable entity\n * for the given entity map. Idempotent: upserts per entity.\n */\nexport async function seedAdministratorPermissions(\n dataSource: DataSource,\n entityMap: Record<string, unknown>\n): Promise<void> {\n const entities = getPermissionableEntityKeys(entityMap);\n const groupRepo = dataSource.getRepository(entityMap.user_groups as EntityTarget<ObjectLiteral>);\n const permRepo = dataSource.getRepository(entityMap.permissions as EntityTarget<ObjectLiteral>);\n\n const adminGroup = await groupRepo.findOne({ where: { name: ADMIN_GROUP_NAME, deleted: false } });\n if (!adminGroup) return;\n\n const fullCrud = { canCreate: true, canRead: true, canUpdate: true, canDelete: true };\n\n for (const entity of entities) {\n const existing = await permRepo.findOne({\n where: { groupId: adminGroup.id, entity },\n });\n if (existing) {\n existing.canCreate = true;\n existing.canRead = true;\n existing.canUpdate = true;\n existing.canDelete = true;\n await permRepo.save(existing);\n } else {\n await permRepo.save(\n permRepo.create({\n groupId: adminGroup.id,\n entity,\n ...fullCrud,\n })\n );\n }\n }\n}\n","export interface CmsMiddlewareConfig {\n publicAdminPaths?: string[];\n publicApiPaths?: string[];\n /** path -> allowed methods */\n publicApiMethods?: Record<string, string[]>;\n signInPath?: string;\n getSessionToken?: (request: { cookies: { get: (name: string) => { value?: string } | undefined } }) => string | undefined;\n}\n\n/** Default public API paths (no auth). Sites should extend this with their own routes. */\nexport const defaultPublicApiMethods: Record<string, string[]> = {\n '/api/contacts': ['POST'],\n '/api/form-submissions': ['POST'],\n '/api/blogs': ['GET'],\n '/api/forms': ['GET'],\n '/api/auth': ['GET', 'POST'],\n '/api/health': ['GET'],\n '/api/users/forgot-password': ['POST'],\n '/api/users/set-password': ['POST'],\n '/api/users/invite': ['POST'],\n};\n\nfunction defaultGetSessionToken(request: { cookies: { get: (name: string) => { value?: string } | undefined } }): string | undefined {\n return (\n request.cookies.get('__Secure-next-auth.session-token')?.value ??\n request.cookies.get('next-auth.session-token')?.value\n );\n}\n\nfunction isPublicMethod(pathname: string, method: string, publicApiMethods: Record<string, string[]>): boolean {\n for (const [endpoint, methods] of Object.entries(publicApiMethods)) {\n if (pathname.startsWith(endpoint) && methods.includes(method)) return true;\n }\n return false;\n}\n\n/**\n * Returns middleware logic. Use from Next.js middleware:\n * import { createCmsMiddleware } from '@infuro/cms-core';\n * export const middleware = createCmsMiddleware({ ... });\n * export const config = { matcher: ['/admin/:path*', '/api/:path*'] };\n */\nexport function createCmsMiddleware(config: CmsMiddlewareConfig = {}) {\n const {\n publicAdminPaths = ['/admin/signin', '/admin/forgot-password', '/admin/reset-password', '/admin/invite'],\n publicApiMethods = defaultPublicApiMethods,\n signInPath = '/admin/signin',\n getSessionToken = defaultGetSessionToken,\n } = config;\n\n return function cmsMiddleware(request: {\n nextUrl: { pathname: string };\n url: string;\n method: string;\n cookies: { get: (name: string) => { value?: string } | undefined };\n }): { type: 'next' } | { type: 'redirect'; url: string } | { type: 'json'; status: number; body: unknown } {\n const pathname = request.nextUrl.pathname;\n const method = request.method;\n\n if (publicAdminPaths.some((p) => pathname === p || pathname.startsWith(p + '/'))) {\n return { type: 'next' };\n }\n\n if (pathname.startsWith('/admin')) {\n const token = getSessionToken(request);\n if (!token) {\n return { type: 'redirect', url: new URL(signInPath, request.url).toString() };\n }\n }\n\n if (pathname.startsWith('/api')) {\n if (isPublicMethod(pathname, method, publicApiMethods)) {\n return { type: 'next' };\n }\n const token = getSessionToken(request);\n if (!token) {\n return { type: 'json', status: 401, body: { error: 'Unauthorized' } };\n }\n }\n\n return { type: 'next' };\n };\n}\n","/**\n * Build NextAuth options for credentials auth. App can extend/override via extend().\n */\nimport type { NextAuthOptions } from 'next-auth';\nimport _CredentialsProvider from 'next-auth/providers/credentials';\nconst CredentialsProvider = (_CredentialsProvider as unknown as { default: typeof _CredentialsProvider }).default ?? _CredentialsProvider;\nimport { isSuperAdminGroupName, permissionRowsToRecord } from './permission-entities';\n\nexport interface NextAuthUser {\n id: number;\n email: string;\n name: string | null;\n password: string | null;\n blocked?: boolean;\n deleted?: boolean;\n groupId?: number | null;\n adminAccess?: boolean;\n group?: {\n name?: string;\n permissions?: Array<{\n entity: string;\n canCreate: boolean;\n canRead: boolean;\n canUpdate: boolean;\n canDelete: boolean;\n }>;\n };\n}\n\nexport interface AuthorizeOtpInput {\n identifier: string;\n channel: 'email' | 'sms';\n code: string;\n}\n\nexport interface NextAuthOptionsConfig {\n /** Resolve user by email (e.g. from TypeORM). Return null if not found. */\n getUserByEmail: (email: string) => Promise<NextAuthUser | null>;\n comparePassword: (plain: string, hash: string) => Promise<boolean>;\n signInPage?: string;\n secret?: string;\n extend?: (options: NextAuthOptions) => NextAuthOptions;\n /** When false, password CredentialsProvider is omitted. Default true. */\n enablePasswordLogin?: boolean;\n /** When true, registers CredentialsProvider id `otp`. Requires authorizeOtp. Default false. */\n enableOtpLogin?: boolean;\n /** Validate OTP and return user (same shape as password flow) or null. */\n authorizeOtp?: (input: AuthorizeOtpInput) => Promise<NextAuthUser | null>;\n}\n\nfunction sessionUserFromNextAuthUser(user: NextAuthUser) {\n const g = user.group;\n const isRBACAdmin = isSuperAdminGroupName(g?.name);\n const entityPerms = permissionRowsToRecord(g?.permissions);\n const adminAccess = (user as NextAuthUser & { adminAccess?: boolean }).adminAccess === true;\n return {\n id: user.id.toString(),\n email: user.email,\n name: user.name,\n groupId: user.groupId ?? undefined,\n isRBACAdmin,\n entityPerms,\n adminAccess,\n };\n}\n\nexport function getNextAuthOptions(config: NextAuthOptionsConfig): NextAuthOptions {\n const {\n getUserByEmail,\n comparePassword,\n signInPage = '/admin/signin',\n secret,\n extend,\n enablePasswordLogin = true,\n enableOtpLogin = false,\n authorizeOtp,\n } = config;\n\n const providers: NextAuthOptions['providers'] = [];\n\n if (enablePasswordLogin) {\n providers.push(\n CredentialsProvider({\n name: 'credentials',\n credentials: {\n email: { label: 'Email', type: 'email' },\n password: { label: 'Password', type: 'password' },\n },\n async authorize(credentials) {\n if (!credentials?.email || !credentials?.password) return null;\n try {\n const user = await getUserByEmail(credentials.email);\n if (!user || user.blocked || (user as { deleted?: boolean }).deleted || !user.password) return null;\n const valid = await comparePassword(credentials.password, user.password);\n if (!valid) return null;\n return sessionUserFromNextAuthUser(user);\n } catch {\n return null;\n }\n },\n })\n );\n }\n\n if (enableOtpLogin && authorizeOtp) {\n providers.push(\n CredentialsProvider({\n id: 'otp',\n name: 'otp',\n credentials: {\n identifier: { label: 'Email or phone', type: 'text' },\n code: { label: 'Code', type: 'text' },\n channel: { label: 'Channel', type: 'text' },\n },\n async authorize(credentials) {\n const identifier = typeof credentials?.identifier === 'string' ? credentials.identifier.trim() : '';\n const code = typeof credentials?.code === 'string' ? credentials.code.trim() : '';\n const ch = credentials?.channel === 'sms' ? 'sms' : 'email';\n if (!identifier || !code) return null;\n try {\n const user = await authorizeOtp({ identifier, channel: ch, code });\n if (!user || user.blocked || (user as { deleted?: boolean }).deleted) return null;\n return sessionUserFromNextAuthUser(user);\n } catch {\n return null;\n }\n },\n })\n );\n }\n\n const options: NextAuthOptions = {\n secret: secret ?? process.env.NEXTAUTH_SECRET,\n providers,\n session: { strategy: 'jwt' },\n pages: { signIn: signInPage },\n cookies: {\n sessionToken: {\n name: process.env.NEXTAUTH_URL?.startsWith('https')\n ? '__Secure-next-auth.session-token'\n : 'next-auth.session-token',\n options: {\n httpOnly: true,\n sameSite: 'lax',\n path: '/',\n secure: process.env.NEXTAUTH_URL?.startsWith('https') ?? false,\n },\n },\n },\n callbacks: {\n async jwt({ token, user }) {\n if (user) {\n const u = user as unknown as Record<string, unknown>;\n (token as Record<string, unknown>).id = u.id;\n (token as Record<string, unknown>).groupId = u.groupId;\n (token as Record<string, unknown>).isRBACAdmin = u.isRBACAdmin;\n (token as Record<string, unknown>).entityPerms = u.entityPerms;\n (token as Record<string, unknown>).adminAccess = u.adminAccess;\n }\n return token;\n },\n async session({ session, token }) {\n if (session.user) {\n const t = token as Record<string, unknown>;\n (session.user as Record<string, unknown>).id = t.id;\n (session.user as Record<string, unknown>).groupId = t.groupId;\n (session.user as Record<string, unknown>).isRBACAdmin = t.isRBACAdmin;\n (session.user as Record<string, unknown>).entityPerms = t.entityPerms;\n (session.user as Record<string, unknown>).adminAccess = t.adminAccess;\n }\n return session;\n },\n },\n };\n\n return extend ? extend(options) : options;\n}\n","import type { DataSource } from 'typeorm';\nimport { ILike, Like, MoreThan } from 'typeorm';\nimport type { Repository } from 'typeorm';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport { queueErpCreateContactIfEnabled } from '../plugins/erp/erp-contact-sync';\nimport { queueErpProductUpsertIfEnabled } from '../plugins/erp/erp-product-sync';\n\nexport type EntityMap = Record<string, import('typeorm').EntityTarget<import('typeorm').ObjectLiteral>>;\n\nexport interface CrudHandlerOptions {\n requireAuth: (req: Request) => Promise<Response | null>;\n json: (body: unknown, init?: { status?: number }) => Response;\n requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n /** When set, contact create/update enqueues ERP `create-contact` (non-fatal). */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n}\n\nconst DATE_COLUMN_TYPES = new Set([\n 'date', 'datetime', 'datetime2', 'timestamp', 'timestamptz', 'timetz', 'smalldatetime', 'timestamp with time zone', 'timestamp without time zone',\n]);\n\nconst TIMESTAMP_PROP_NAMES = new Set(['createdAt', 'updatedAt', 'deletedAt']);\n\nfunction isInvalidDateValue(v: unknown): boolean {\n if (v === '' || v == null) return true;\n if (typeof v === 'string') return isNaN(Date.parse(v)) || /NaN|Invalid/i.test(v);\n if (v instanceof Date) return isNaN(v.getTime());\n return false;\n}\n\n/** Strip empty/invalid values for boolean, number, and date columns so DB defaults apply. */\nfunction sanitizeBodyForEntity(repo: Repository<import('typeorm').ObjectLiteral>, body: Record<string, unknown>): void {\n const meta = repo.metadata;\n for (const col of meta.columns) {\n if (!(col.propertyName in body)) continue;\n const v = body[col.propertyName];\n const t = typeof col.type === 'string' ? col.type : (col.type as Function)?.name ?? '';\n const isBoolean = t === 'boolean' || t === 'bool' || col.type === Boolean;\n const isNumber = ['int', 'integer', 'int2', 'int4', 'int8', 'smallint', 'bigint', 'number', 'Number'].includes(t) || col.type === Number;\n const isDate = DATE_COLUMN_TYPES.has(t) || col.type === Date || TIMESTAMP_PROP_NAMES.has(col.propertyName);\n if (v === '' && (isBoolean || isNumber)) {\n delete body[col.propertyName];\n } else if (isDate && isInvalidDateValue(v)) {\n delete body[col.propertyName];\n }\n }\n}\n\n/** Keep only scalar columns (excludes relations like tags) so repo.update() does not throw. */\nfunction pickColumnUpdates(\n repo: Repository<import('typeorm').ObjectLiteral>,\n body: Record<string, unknown>\n): Record<string, unknown> {\n const cols = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const out: Record<string, unknown> = {};\n for (const k of Object.keys(body)) {\n if (cols.has(k)) out[k] = body[k];\n }\n return out;\n}\n\n/** OR search on columns that exist on the entity (avoids Tag/Category etc. without `title`). */\nfunction buildSearchWhereClause(\n repo: Repository<import('typeorm').ObjectLiteral>,\n search: string\n): Record<string, unknown>[] | Record<string, unknown> {\n const cols = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const term = ILike(`%${search}%`);\n const ors: Record<string, unknown>[] = [];\n for (const field of ['name', 'title', 'slug', 'email', 'filename'] as const) {\n if (cols.has(field)) ors.push({ [field]: term });\n }\n if (ors.length === 0) return {};\n return ors.length === 1 ? ors[0]! : ors;\n}\n\nfunction makeContactErpSync(\n dataSource: DataSource,\n entityMap: EntityMap,\n getCms: CrudHandlerOptions['getCms']\n): (row: import('typeorm').ObjectLiteral) => Promise<void> {\n return async function syncContactRowToErp(row: import('typeorm').ObjectLiteral): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n const c = row as {\n name: string;\n email: string;\n phone?: string | null;\n type?: string | null;\n company?: string | null;\n notes?: string | null;\n };\n await queueErpCreateContactIfEnabled(cms, dataSource, entityMap, {\n name: c.name,\n email: c.email,\n phone: c.phone,\n type: c.type,\n company: c.company,\n notes: c.notes,\n });\n } catch {\n /* ignore */\n }\n };\n}\n\nexport function createCrudHandler(\n dataSource: DataSource,\n entityMap: EntityMap,\n options: CrudHandlerOptions\n) {\n const { requireAuth, json, requireEntityPermission: reqPerm, getCms } = options;\n const syncContactRowToErp = makeContactErpSync(dataSource, entityMap, getCms);\n\n async function authz(req: Request, resource: string, action: EntityCrudAction): Promise<Response | null> {\n const authError = await requireAuth(req);\n if (authError) return authError;\n if (reqPerm) {\n const pe = await reqPerm(req, resource, action);\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const { searchParams } = new URL(req.url);\n const page = Number(searchParams.get('page')) || 1;\n const limit = Math.min(Number(searchParams.get('limit')) || 10, 100);\n const skip = (page - 1) * limit;\n const sortFieldRaw = searchParams.get('sortField') || 'createdAt';\n const sortOrder = searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const search = searchParams.get('search');\n\n // Orders list: include contact + itemsSummary; search (order# + customer), status, date, paymentRef\n if (resource === 'orders') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'orderNumber', 'contactId', 'status', 'total', 'currency', 'createdAt', 'updatedAt'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderOrders = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const statusFilter = searchParams.get('status')?.trim();\n const dateFrom = searchParams.get('dateFrom')?.trim();\n const dateTo = searchParams.get('dateTo')?.trim();\n const paymentRef = searchParams.get('paymentRef')?.trim();\n\n let orderIdsFromPayment: number[] | null = null;\n if (paymentRef && entityMap['payments']) {\n const paymentRepo = dataSource.getRepository(entityMap['payments']);\n const payments = await paymentRepo\n .createQueryBuilder('p')\n .select('p.orderId')\n .where('p.externalReference = :ref', { ref: paymentRef })\n .orWhere(\"p.metadata->>'razorpayPaymentId' = :ref\", { ref: paymentRef })\n .getRawMany<{ orderId: number }>();\n orderIdsFromPayment = payments.map((r) => r.orderId);\n if (orderIdsFromPayment.length === 0) {\n return json({ total: 0, page, limit, totalPages: 0, data: [] });\n }\n }\n\n const qb = repo\n .createQueryBuilder('order')\n .leftJoinAndSelect('order.contact', 'contact')\n .leftJoinAndSelect('order.items', 'items')\n .leftJoinAndSelect('items.product', 'product')\n .leftJoinAndSelect('product.collection', 'collection')\n .orderBy(`order.${sortField}`, sortOrderOrders)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere(\n '(order.orderNumber ILIKE :term OR contact.name ILIKE :term OR contact.email ILIKE :term)',\n { term }\n );\n }\n if (statusFilter) qb.andWhere('order.status = :status', { status: statusFilter });\n if (dateFrom) qb.andWhere('order.createdAt >= :dateFrom', { dateFrom: new Date(dateFrom + 'T00:00:00.000Z') });\n if (dateTo) qb.andWhere('order.createdAt <= :dateTo', { dateTo: new Date(dateTo + 'T23:59:59.999Z') });\n if (orderIdsFromPayment && orderIdsFromPayment.length) qb.andWhere('order.id IN (:...orderIds)', { orderIds: orderIdsFromPayment });\n\n const [rows, total] = await qb.getManyAndCount();\n const data = (rows as Record<string, unknown>[]).map((order: Record<string, unknown>) => {\n const contact = order.contact as Record<string, unknown> | undefined;\n const items = (order.items as Array<{ product?: { name?: string; collection?: { name?: string } }; quantity: number }>) ?? [];\n const itemsSummary = items\n .map((i) => {\n const label = i.product?.collection?.name ?? i.product?.name ?? 'Product';\n return `${label} × ${i.quantity}`;\n })\n .join(', ') || '—';\n return {\n ...order,\n contact: contact ? { id: contact.id, name: contact.name, email: contact.email, phone: contact.phone } : null,\n itemsSummary,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Payments list: include order + contact; search (customer), status, date, method, orderNumber\n if (resource === 'payments') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'orderId', 'amount', 'currency', 'status', 'method', 'paidAt', 'createdAt', 'updatedAt'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderPayments = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const statusFilter = searchParams.get('status')?.trim();\n const dateFrom = searchParams.get('dateFrom')?.trim();\n const dateTo = searchParams.get('dateTo')?.trim();\n const methodFilter = searchParams.get('method')?.trim();\n const orderNumberParam = searchParams.get('orderNumber')?.trim();\n\n const qb = repo\n .createQueryBuilder('payment')\n .leftJoinAndSelect('payment.order', 'ord')\n .leftJoinAndSelect('ord.contact', 'orderContact')\n .leftJoinAndSelect('payment.contact', 'contact')\n .orderBy(`payment.${sortField}`, sortOrderPayments)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere(\n '(orderContact.name ILIKE :term OR orderContact.email ILIKE :term OR contact.name ILIKE :term OR contact.email ILIKE :term)',\n { term }\n );\n }\n if (statusFilter) qb.andWhere('payment.status = :status', { status: statusFilter });\n if (dateFrom) qb.andWhere('payment.createdAt >= :dateFrom', { dateFrom: new Date(dateFrom + 'T00:00:00.000Z') });\n if (dateTo) qb.andWhere('payment.createdAt <= :dateTo', { dateTo: new Date(dateTo + 'T23:59:59.999Z') });\n if (methodFilter) qb.andWhere('payment.method = :method', { method: methodFilter });\n if (orderNumberParam) qb.andWhere('ord.orderNumber ILIKE :orderNumber', { orderNumber: `%${orderNumberParam}%` });\n\n const [rows, total] = await qb.getManyAndCount();\n const data = (rows as Record<string, unknown>[]).map((payment: Record<string, unknown>) => {\n const order = payment.order as Record<string, unknown> | undefined;\n const orderContact = order?.contact as Record<string, unknown> | undefined;\n const contact = payment.contact as Record<string, unknown> | undefined;\n const customer = orderContact ?? contact;\n return {\n ...payment,\n order: order ? { id: order.id, orderNumber: order.orderNumber, contact: orderContact ? { name: orderContact.name, email: orderContact.email } : null } : null,\n contact: customer ? { id: customer.id, name: customer.name, email: customer.email } : null,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Products list: status and inventory filters\n if (resource === 'products') {\n const repo = dataSource.getRepository(entity);\n const statusFilter = searchParams.get('status')?.trim();\n const inventory = searchParams.get('inventory')?.trim();\n const productWhere: Record<string, unknown> = {};\n if (statusFilter) productWhere.status = statusFilter;\n if (inventory === 'in_stock') productWhere.quantity = MoreThan(0);\n if (inventory === 'out_of_stock') productWhere.quantity = 0;\n if (search && typeof search === 'string' && search.trim()) {\n productWhere.name = ILike(`%${search.trim()}%`);\n }\n const [data, total] = await repo.findAndCount({\n where: Object.keys(productWhere).length ? productWhere : undefined,\n skip,\n take: limit,\n order: { [sortFieldRaw]: sortOrder },\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n // Contacts list: filters (type, orderId), optional includeSummary (orderCount, totalPaid)\n if (resource === 'contacts') {\n const repo = dataSource.getRepository(entity);\n const allowedSort = ['id', 'name', 'email', 'createdAt', 'type'];\n const sortField = allowedSort.includes(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n const sortOrderContacts = searchParams.get('sortOrder') === 'asc' ? 'ASC' : 'DESC';\n const typeFilter = searchParams.get('type')?.trim();\n const orderIdParam = searchParams.get('orderId')?.trim();\n const includeSummary = searchParams.get('includeSummary') === '1';\n\n const qb = repo\n .createQueryBuilder('contact')\n .orderBy(`contact.${sortField}`, sortOrderContacts)\n .skip(skip)\n .take(limit);\n\n if (search && typeof search === 'string' && search.trim()) {\n const term = `%${search.trim()}%`;\n qb.andWhere('(contact.name ILIKE :term OR contact.email ILIKE :term OR contact.phone ILIKE :term)', { term });\n }\n if (typeFilter) qb.andWhere('contact.type = :type', { type: typeFilter });\n if (orderIdParam) {\n const orderId = Number(orderIdParam);\n if (!Number.isNaN(orderId)) {\n qb.andWhere('contact.id IN (SELECT \"contactId\" FROM orders WHERE id = :orderId)', { orderId });\n }\n }\n\n if (includeSummary && entityMap['orders'] && entityMap['payments']) {\n qb.loadRelationCountAndMap('contact._orderCount', 'contact.orders');\n const [rows, total] = await qb.getManyAndCount();\n const contactIds = (rows as { id: number }[]).map((c) => c.id);\n const paymentRepo = dataSource.getRepository(entityMap['payments']);\n const paidByContact = await paymentRepo\n .createQueryBuilder('p')\n .select('p.contactId', 'contactId')\n .addSelect('COALESCE(SUM(CAST(p.amount AS DECIMAL)), 0)', 'total')\n .where('p.contactId IN (:...ids)', { ids: contactIds.length ? contactIds : [0] })\n .andWhere('p.status = :status', { status: 'completed' })\n .groupBy('p.contactId')\n .getRawMany<{ contactId: number; total: string }>();\n const totalPaidMap = new Map<number, number>(paidByContact.map((r) => [r.contactId, Number(r.total)]));\n const data = (rows as Record<string, unknown>[]).map((c) => {\n const { _orderCount, ...rest } = c as { _orderCount?: number; id: number };\n return {\n ...rest,\n orderCount: _orderCount ?? 0,\n totalPaid: totalPaidMap.get((rest as { id: number }).id) ?? 0,\n };\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n const [data, total] = await qb.getManyAndCount();\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n }\n\n const repo = dataSource.getRepository(entity);\n const typeFilter = searchParams.get('type');\n const columnNames = new Set(repo.metadata.columns.map((c) => c.propertyName));\n const sortField = columnNames.has(sortFieldRaw) ? sortFieldRaw : 'createdAt';\n let where: Record<string, unknown>[] | Record<string, unknown> = {};\n if (resource === 'media') {\n const mediaWhere: Record<string, unknown> = {};\n if (search) mediaWhere.filename = ILike(`%${search}%`);\n if (typeFilter) mediaWhere.mimeType = Like(`${typeFilter}/%`);\n where = Object.keys(mediaWhere).length > 0 ? mediaWhere : {};\n } else if (search) {\n where = buildSearchWhereClause(repo, search);\n }\n // Exact-match filters for common FK query params (e.g. product_attributes?productId=1)\n const intFilterKeys = ['productId', 'attributeId', 'taxId'] as const;\n const extraWhere: Record<string, unknown> = {};\n for (const key of intFilterKeys) {\n const v = searchParams.get(key);\n if (v != null && v !== '' && columnNames.has(key)) {\n const n = Number(v);\n if (Number.isFinite(n)) extraWhere[key] = n;\n }\n }\n if (Object.keys(extraWhere).length > 0) {\n if (Array.isArray(where)) {\n where = where.map((w) => ({ ...w, ...extraWhere }));\n } else if (where && typeof where === 'object' && Object.keys(where).length > 0) {\n where = { ...where, ...extraWhere };\n } else {\n where = extraWhere;\n }\n }\n const [data, total] = await repo.findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n where,\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n },\n\n async POST(req: Request, resource: string) {\n const authError = await authz(req, resource, 'create');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const body = await req.json();\n if (!body || typeof body !== 'object' || Object.keys(body).length === 0) {\n return json({ error: 'Invalid request payload' }, { status: 400 });\n }\n const repo = dataSource.getRepository(entity);\n sanitizeBodyForEntity(repo, body as Record<string, unknown>);\n const created = await repo.save(repo.create(body as object));\n if (resource === 'contacts') {\n await syncContactRowToErp(created as import('typeorm').ObjectLiteral);\n }\n if (resource === 'products' && getCms) {\n const cms = await getCms();\n await queueErpProductUpsertIfEnabled(cms, dataSource, entityMap, created as import('typeorm').ObjectLiteral);\n }\n return json(created, { status: 201 });\n },\n\n async GET_METADATA(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const repo = dataSource.getRepository(entity);\n const meta = repo.metadata;\n\n // Collect unique column names from indices\n const uniqueFromIndices = new Set<string>();\n for (const idx of meta.indices) {\n if (idx.isUnique && idx.columns.length === 1) {\n uniqueFromIndices.add(idx.columns[0].propertyName);\n }\n }\n for (const uniq of meta.uniques) {\n if (uniq.columns.length === 1) {\n uniqueFromIndices.add(uniq.columns[0].propertyName);\n }\n }\n\n const columns = meta.columns.map((col) => ({\n name: col.propertyName,\n type: typeof col.type === 'string' ? col.type : (col.type as { name?: string })?.name ?? 'unknown',\n nullable: col.isNullable,\n isUnique: uniqueFromIndices.has(col.propertyName),\n isPrimary: col.isPrimary,\n default: col.default,\n }));\n\n const uniqueColumns = [...uniqueFromIndices];\n\n return json({ columns, uniqueColumns });\n },\n\n async BULK_POST(req: Request, resource: string) {\n const authError = await authz(req, resource, 'update');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n const body = await req.json();\n const { records, upsertKey = 'id' } = body as { records: object[]; upsertKey?: string };\n\n if (!Array.isArray(records) || records.length === 0) {\n return json({ error: 'Records array is required' }, { status: 400 });\n }\n\n const repo = dataSource.getRepository(entity);\n\n // Sanitize each record\n for (const record of records) {\n sanitizeBodyForEntity(repo, record as Record<string, unknown>);\n }\n\n try {\n const result = await repo.upsert(records, {\n conflictPaths: [upsertKey],\n skipUpdateIfNoValuesChanged: true,\n });\n return json({\n success: true,\n imported: result.identifiers.length,\n identifiers: result.identifiers,\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Bulk import failed';\n return json({ error: message }, { status: 400 });\n }\n },\n\n async GET_EXPORT(req: Request, resource: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!resource || !entity) {\n return json({ error: 'Invalid resource' }, { status: 400 });\n }\n\n const { searchParams } = new URL(req.url);\n const format = searchParams.get('format') || 'csv';\n\n const repo = dataSource.getRepository(entity);\n const meta = repo.metadata;\n\n // Check if entity has 'deleted' column\n const hasDeleted = meta.columns.some((c) => c.propertyName === 'deleted');\n const where = hasDeleted ? { deleted: false } : {};\n\n const data = await repo.find({ where });\n\n // Get exportable columns (exclude soft-delete related)\n const excludeCols = new Set(['deletedAt', 'deletedBy', 'deleted']);\n const columns = meta.columns\n .filter((c) => !excludeCols.has(c.propertyName))\n .map((c) => c.propertyName);\n\n if (format === 'json') {\n return json(data);\n }\n\n // CSV format\n const escapeCSV = (val: unknown): string => {\n if (val === null || val === undefined) return '';\n const str = typeof val === 'object' ? JSON.stringify(val) : String(val);\n if (str.includes(',') || str.includes('\"') || str.includes('\\n')) {\n return `\"${str.replace(/\"/g, '\"\"')}\"`;\n }\n return str;\n };\n\n const header = columns.join(',');\n const rows = data.map((row) =>\n columns.map((col) => escapeCSV((row as Record<string, unknown>)[col])).join(',')\n );\n const csv = [header, ...rows].join('\\n');\n\n return new Response(csv, {\n headers: {\n 'Content-Type': 'text/csv; charset=utf-8',\n 'Content-Disposition': `attachment; filename=\"${resource}.csv\"`,\n },\n });\n },\n };\n}\n\nexport function createCrudByIdHandler(\n dataSource: DataSource,\n entityMap: EntityMap,\n options: CrudHandlerOptions\n) {\n const { requireAuth, json, requireEntityPermission: reqPerm, getCms } = options;\n const syncContactRowToErp = makeContactErpSync(dataSource, entityMap, getCms);\n\n async function authz(req: Request, resource: string, action: EntityCrudAction): Promise<Response | null> {\n const authError = await requireAuth(req);\n if (authError) return authError;\n if (reqPerm) {\n const pe = await reqPerm(req, resource, action);\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'read');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const repo = dataSource.getRepository(entity);\n\n if (resource === 'orders') {\n const order = await repo.findOne({\n where: { id: Number(id) },\n relations: ['contact', 'billingAddress', 'shippingAddress', 'items', 'items.product', 'items.product.collection', 'payments'],\n });\n if (!order) return json({ message: 'Not found' }, { status: 404 });\n const relatedOrders = await repo.find({\n where: { parentOrderId: Number(id), deleted: false } as import('typeorm').ObjectLiteral,\n order: { id: 'ASC' },\n });\n return json({ ...order, relatedOrders });\n }\n\n if (resource === 'contacts') {\n const contact = await repo.findOne({\n where: { id: Number(id) },\n relations: ['form_submissions', 'form_submissions.form', 'orders', 'payments', 'addresses'],\n });\n if (!contact) return json({ message: 'Not found' }, { status: 404 });\n const orders = (contact as { orders?: { total?: unknown; createdAt?: string }[] }).orders ?? [];\n const payments = (contact as { payments?: { status: string; amount?: number }[] }).payments ?? [];\n const totalPaid = payments\n .filter((p) => p.status === 'completed')\n .reduce((sum, p) => sum + Number(p.amount ?? 0), 0);\n const lastOrderAt =\n orders.length > 0\n ? orders.reduce((latest, o) => {\n const t = o.createdAt ? new Date(o.createdAt).getTime() : 0;\n return t > latest ? t : latest;\n }, 0)\n : null;\n return json({\n ...contact,\n summary: {\n totalOrders: orders.length,\n totalPaid,\n lastOrderAt: lastOrderAt ? new Date(lastOrderAt).toISOString() : null,\n },\n });\n }\n\n if (resource === 'payments') {\n const payment = await repo.findOne({\n where: { id: Number(id) },\n relations: ['order', 'order.contact', 'contact'],\n });\n if (!payment) return json({ message: 'Not found' }, { status: 404 });\n return json(payment);\n }\n\n if (resource === 'blogs') {\n const blog = await repo.findOne({\n where: { id: Number(id) },\n relations: ['category', 'seo', 'tags'],\n });\n return blog ? json(blog) : json({ message: 'Not found' }, { status: 404 });\n }\n\n const item = await repo.findOne({ where: { id: Number(id) } });\n return item ? json(item) : json({ message: 'Not found' }, { status: 404 });\n },\n\n async PUT(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'update');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const rawBody = (await req.json()) as Record<string, unknown> | null;\n const repo = dataSource.getRepository(entity);\n const numericId = Number(id);\n\n if (\n resource === 'blogs' &&\n rawBody &&\n typeof rawBody === 'object' &&\n entityMap.categories &&\n entityMap.seos &&\n entityMap.tags\n ) {\n const existing = await repo.findOne({ where: { id: numericId } });\n if (!existing) return json({ message: 'Not found' }, { status: 404 });\n\n const updatePayload = pickColumnUpdates(repo, rawBody);\n\n if ('category' in rawBody) {\n const c = rawBody.category;\n if (typeof c === 'string' && c.trim()) {\n const cat = await dataSource\n .getRepository(entityMap.categories)\n .findOne({ where: { name: c.trim() } });\n updatePayload.categoryId = cat?.id ?? null;\n } else {\n updatePayload.categoryId = null;\n }\n }\n\n const blogSlug =\n (typeof updatePayload.slug === 'string' && updatePayload.slug) ||\n (existing as { slug: string }).slug;\n const seoRepo = dataSource.getRepository(entityMap.seos);\n const seoField = (k: string): string | null | undefined => {\n if (!(k in rawBody)) return undefined;\n const v = rawBody[k];\n if (v == null || v === '') return null;\n return String(v);\n };\n if (\n 'metaTitle' in rawBody ||\n 'metaDescription' in rawBody ||\n 'metaKeywords' in rawBody ||\n 'ogImage' in rawBody\n ) {\n const title = seoField('metaTitle');\n const description = seoField('metaDescription');\n const keywords = seoField('metaKeywords');\n const ogImage = seoField('ogImage');\n const exSeoId = (existing as { seoId: number | null }).seoId;\n if (exSeoId) {\n const seo = await seoRepo.findOne({ where: { id: exSeoId } });\n if (seo) {\n const s = seo as Record<string, unknown>;\n if (title !== undefined) s.title = title;\n if (description !== undefined) s.description = description;\n if (keywords !== undefined) s.keywords = keywords;\n if (ogImage !== undefined) s.ogImage = ogImage;\n s.slug = blogSlug;\n await seoRepo.save(seo);\n }\n } else {\n let seoSlug = blogSlug;\n const taken = await seoRepo.findOne({ where: { slug: seoSlug } });\n if (taken) seoSlug = `blog-${numericId}-${blogSlug}`;\n const seo = await seoRepo.save(\n seoRepo.create({\n slug: seoSlug,\n title: title ?? null,\n description: description ?? null,\n keywords: keywords ?? null,\n ogImage: ogImage ?? null,\n })\n );\n updatePayload.seoId = (seo as { id: number }).id;\n }\n }\n\n sanitizeBodyForEntity(repo, updatePayload);\n await repo.update(numericId, updatePayload as object);\n\n if (Array.isArray(rawBody.tags)) {\n const tagNames = (rawBody.tags as unknown[]).map((t) => String(t).trim()).filter(Boolean);\n const tagRepo = dataSource.getRepository(entityMap.tags);\n const tagEntities: import('typeorm').ObjectLiteral[] = [];\n for (const name of tagNames) {\n let tag = await tagRepo.findOne({ where: { name } });\n if (!tag) tag = await tagRepo.save(tagRepo.create({ name }));\n tagEntities.push(tag);\n }\n const blog = await repo.findOne({ where: { id: numericId }, relations: ['tags'] });\n if (blog) {\n (blog as Record<string, unknown>).tags = tagEntities;\n await repo.save(blog);\n }\n }\n\n const updated = await repo.findOne({\n where: { id: numericId },\n relations: ['tags', 'category', 'seo'],\n });\n return updated ? json(updated) : json({ message: 'Not found' }, { status: 404 });\n }\n\n const updatePayload = rawBody && typeof rawBody === 'object' ? pickColumnUpdates(repo, rawBody) : {};\n if (Object.keys(updatePayload).length > 0) {\n sanitizeBodyForEntity(repo, updatePayload);\n await repo.update(numericId, updatePayload as object);\n }\n const updated = await repo.findOne({ where: { id: numericId } });\n if (resource === 'contacts' && updated) {\n await syncContactRowToErp(updated as import('typeorm').ObjectLiteral);\n }\n if (resource === 'products' && updated && getCms) {\n const cms = await getCms();\n await queueErpProductUpsertIfEnabled(cms, dataSource, entityMap, updated as import('typeorm').ObjectLiteral);\n }\n return updated ? json(updated) : json({ message: 'Not found' }, { status: 404 });\n },\n\n async DELETE(req: Request, resource: string, id: string) {\n const authError = await authz(req, resource, 'delete');\n if (authError) return authError;\n const entity = entityMap[resource];\n if (!entity) return json({ error: 'Invalid resource' }, { status: 400 });\n const repo = dataSource.getRepository(entity);\n const result = await repo.delete(Number(id));\n if (result.affected === 0) return json({ message: 'Not found' }, { status: 404 });\n return json({ message: 'Deleted successfully' }, { status: 200 });\n },\n };\n}\n","/**\n * Auth API handler factories. Inject dataSource + entityMap, sendEmail, hash/compare; optional hooks to customize.\n */\nimport type { DataSource } from 'typeorm';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport type { EntityMap } from './crud';\n\nexport interface AuthHandlersConfig {\n json: (body: unknown, init?: { status?: number }) => Response;\n baseUrl: string;\n hashPassword: (plain: string) => Promise<string>;\n comparePassword: (plain: string, hash: string) => Promise<boolean>;\n}\n\nexport interface ForgotPasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n sendEmail?: (opts: {\n to: string;\n subject: string;\n html: string;\n text?: string;\n /** Plain reset URL for templated emails (preferred over parsing html). */\n resetLink?: string;\n }) => Promise<void>;\n resetExpiryHours?: number;\n afterCreateToken?: (email: string, resetLink: string) => Promise<void>;\n}\n\nexport function createForgotPasswordHandler(config: ForgotPasswordConfig) {\n const { dataSource, entityMap, json, baseUrl, sendEmail, resetExpiryHours = 1, afterCreateToken } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { email?: string };\n const email = typeof body?.email === 'string' ? body.email.trim().toLowerCase() : '';\n if (!email) return json({ error: 'Email is required' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email }, select: ['email'] });\n const msg = 'If an account exists with this email, you will receive a reset link shortly.';\n if (!user) return json({ message: msg }, { status: 200 });\n\n const crypto = await import('crypto');\n const token = crypto.randomBytes(32).toString('hex');\n const expiresAt = new Date(Date.now() + (resetExpiryHours * 60 * 60 * 1000));\n const tokenRepo = dataSource.getRepository(entityMap.password_reset_tokens);\n await tokenRepo.save(tokenRepo.create({ email: user.email, token, expiresAt }));\n const resetLink = `${baseUrl}/admin/reset-password?token=${token}`;\n\n if (sendEmail)\n await sendEmail({\n to: user.email,\n subject: 'Password reset',\n html: `<a href=\"${resetLink}\">Reset password</a>`,\n text: resetLink,\n resetLink,\n });\n if (afterCreateToken) await afterCreateToken(user.email, resetLink);\n return json({ message: msg }, { status: 200 });\n } catch (err) {\n return json({ error: 'Something went wrong. Please try again.' }, { status: 500 });\n }\n };\n}\n\nexport interface SetPasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n minPasswordLength?: number;\n beforeUpdate?: (email: string, userId: number) => Promise<void>;\n}\n\nexport function createSetPasswordHandler(config: SetPasswordConfig) {\n const { dataSource, entityMap, json, hashPassword, minPasswordLength = 6, beforeUpdate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { token?: string; newPassword?: string };\n const { token, newPassword } = body;\n if (!token || !newPassword) return json({ error: 'Token and new password are required' }, { status: 400 });\n if (newPassword.length < minPasswordLength) return json({ error: 'Password must be at least 6 characters' }, { status: 400 });\n\n const tokenRepo = dataSource.getRepository(entityMap.password_reset_tokens);\n const record = await tokenRepo.findOne({ where: { token } });\n if (!record || record.expiresAt < new Date()) return json({ error: 'Invalid or expired reset link. Please request a new one.' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email: record.email }, select: ['id'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n\n if (beforeUpdate) await beforeUpdate(record.email, user.id);\n const hashedPassword = await hashPassword(newPassword);\n await userRepo.update(user.id, { password: hashedPassword, updatedAt: new Date() });\n await tokenRepo.delete({ email: record.email });\n return json({ message: 'Password updated successfully. You can now sign in.' });\n } catch {\n return json({ error: 'Something went wrong. Please try again.' }, { status: 500 });\n }\n };\n}\n\nexport interface InviteAcceptConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n beforeActivate?: (email: string, userId: number) => Promise<void>;\n}\n\n/** Decode invite token (base64 email) and set password + unblock user */\nexport function createInviteAcceptHandler(config: InviteAcceptConfig) {\n const { dataSource, entityMap, json, hashPassword, beforeActivate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const body = await request.json().catch(() => ({})) as { token?: string; password?: string };\n const { token, password } = body;\n if (!token || !password) return json({ error: 'Missing required fields: token, password' }, { status: 400 });\n\n let email: string;\n try {\n email = Buffer.from(token, 'base64').toString('utf8');\n } catch {\n return json({ error: 'Invalid or expired invite token' }, { status: 400 });\n }\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email }, select: ['id', 'blocked'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n if (!user.blocked) return json({ error: 'User is already active' }, { status: 400 });\n\n if (entityMap.contacts) {\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, user.id, email);\n }\n if (beforeActivate) await beforeActivate(email, user.id);\n const hashedPassword = await hashPassword(password);\n await userRepo.update(user.id, { password: hashedPassword, blocked: false });\n return json({ message: 'User account activated successfully' }, { status: 200 });\n } catch (err) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface ChangePasswordConfig extends AuthHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n getSession: () => Promise<{ user?: { email?: string | null } } | null>;\n minPasswordLength?: number;\n beforeUpdate?: (email: string) => Promise<void>;\n}\n\nexport function createChangePasswordHandler(config: ChangePasswordConfig) {\n const { dataSource, entityMap, json, comparePassword, hashPassword, getSession, minPasswordLength = 6, beforeUpdate } = config;\n return async function POST(request: Request): Promise<Response> {\n try {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n\n const body = await request.json().catch(() => ({})) as { currentPassword?: string; newPassword?: string };\n const { currentPassword, newPassword } = body;\n if (!currentPassword || !newPassword) return json({ error: 'Current password and new password are required' }, { status: 400 });\n if (newPassword.length < minPasswordLength) return json({ error: 'New password must be at least 6 characters long' }, { status: 400 });\n\n const userRepo = dataSource.getRepository(entityMap.users);\n const user = await userRepo.findOne({ where: { email: session.user.email }, select: ['password'] });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n if (!user.password) return json({ error: 'Current password is incorrect' }, { status: 400 });\n const valid = await comparePassword(currentPassword, user.password);\n if (!valid) return json({ error: 'Current password is incorrect' }, { status: 400 });\n\n if (beforeUpdate) await beforeUpdate(session.user.email);\n const hashedPassword = await hashPassword(newPassword);\n await userRepo.update({ email: session.user.email }, { password: hashedPassword, updatedAt: new Date() });\n return json({ message: 'Password updated successfully' });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface UserAuthApiConfig\n extends ForgotPasswordConfig,\n Omit<SetPasswordConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword'>,\n Omit<InviteAcceptConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword'>,\n Omit<ChangePasswordConfig, 'dataSource' | 'entityMap' | 'json' | 'baseUrl' | 'hashPassword' | 'comparePassword' | 'beforeUpdate' | 'getSession'> {\n getSession?: () => Promise<{ user?: { email?: string | null } } | null>;\n beforeChangePasswordUpdate?: (email: string) => Promise<void>;\n}\n\nconst USER_AUTH_PATHS = ['forgot-password', 'invite', 'set-password', 'reset-password'] as const;\n\n/**\n * Single router for all user-auth APIs. Mount in the app once.\n * Use when you have no users/[id] route (else Next.js gives [id] precedence and \"forgot-password\" would match as id).\n * Path is the segment after the mount (e.g. \"forgot-password\"). Returns 404 for unknown paths.\n */\nexport function createUserAuthApiRouter(config: UserAuthApiConfig) {\n const forgot = createForgotPasswordHandler(config);\n const setPass = createSetPasswordHandler(config);\n const invite = createInviteAcceptHandler(config);\n const changePass = config.getSession\n ? createChangePasswordHandler({\n ...config,\n getSession: config.getSession,\n beforeUpdate: config.beforeChangePasswordUpdate,\n })\n : null;\n\n return {\n async POST(req: Request, pathname: string): Promise<Response> {\n const path = pathname.replace(/\\/$/, '');\n if (!USER_AUTH_PATHS.includes(path as (typeof USER_AUTH_PATHS)[number])) {\n return config.json({ error: 'Not found' }, { status: 404 });\n }\n if (path === 'forgot-password') return forgot(req);\n if (path === 'set-password') return setPass(req);\n if (path === 'invite') return invite(req);\n if (path === 'reset-password' && changePass) return changePass(req);\n return config.json({ error: 'Not found' }, { status: 404 });\n },\n };\n}\n","/**\n * CMS API handlers: dashboard, analytics, upload, blog/form by slug, users (list/create/get/update/delete/regenerate-invite/avatar/profile).\n * All accept injectable deps; upload supports S3 or local.\n */\nimport type { DataSource } from 'typeorm';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport { MoreThanOrEqual, ILike } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport { queueEmail } from '../plugins/email/email-queue';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { queueErp } from '../plugins/erp/erp-queue';\nimport type { ERPSubmissionService } from '../plugins/erp/erp-submission';\nimport { assertCaptchaOk } from '../plugins/captcha/assert';\n\nexport type RequireEntityPermissionFn = (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n\nexport interface CmsHandlersBase {\n json: (body: unknown, init?: { status?: number }) => Response;\n requireAuth: (req: Request) => Promise<Response | null>;\n requireEntityPermission?: RequireEntityPermissionFn;\n}\n\nexport interface DashboardStatsConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n requirePermission?: (req: Request, permission: string) => Promise<Response | null>;\n}\n\nexport function createDashboardStatsHandler(config: DashboardStatsConfig) {\n const { dataSource, entityMap, json, requireAuth, requirePermission, requireEntityPermission } = config;\n return async function GET(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'dashboard', 'read');\n if (pe) return pe;\n }\n if (requirePermission) {\n const permErr = await requirePermission(req, 'view_dashboard');\n if (permErr) return permErr;\n }\n try {\n const sevenDaysAgo = new Date(Date.now() - 7 * 24 * 60 * 60 * 1000);\n const repo = (name: string) => entityMap[name] ? dataSource.getRepository(entityMap[name]) : undefined;\n const [contactsCount, formsCount, formSubmissionsCount, usersCount, blogsCount, recentContacts, recentSubmissions] = await Promise.all([\n repo('contacts')?.count() ?? 0,\n repo('forms')?.count({ where: { deleted: false } }) ?? 0,\n repo('form_submissions')?.count() ?? 0,\n repo('users')?.count({ where: { deleted: false } }) ?? 0,\n repo('blogs')?.count({ where: { deleted: false } }) ?? 0,\n repo('contacts')?.count({ where: { createdAt: MoreThanOrEqual(sevenDaysAgo) } }) ?? 0,\n repo('form_submissions')?.count({ where: { createdAt: MoreThanOrEqual(sevenDaysAgo) } }) ?? 0,\n ]);\n return json({\n contacts: { total: contactsCount, recent: recentContacts },\n forms: { total: formsCount, submissions: formSubmissionsCount, recentSubmissions },\n users: usersCount,\n blogs: blogsCount,\n });\n } catch (err) {\n return json({ error: 'Failed to fetch dashboard stats' }, { status: 500 });\n }\n };\n}\n\nexport interface AnalyticsHandlerConfig extends CmsHandlersBase {\n getAnalyticsData?: (days: number) => Promise<unknown>;\n getPropertyId?: () => ({ currentViewId?: string; [k: string]: unknown });\n getPermissions?: () => ({ serviceAccountEmail?: string; currentViewId?: string; [k: string]: unknown });\n}\n\nexport function createAnalyticsHandlers(config: AnalyticsHandlerConfig) {\n const { json, getAnalyticsData, getPropertyId, getPermissions } = config;\n return {\n async GET(req: Request): Promise<Response> {\n if (!getAnalyticsData) return json({ error: 'Analytics not configured' }, { status: 404 });\n try {\n const url = new URL(req.url);\n const days = parseInt(url.searchParams.get('days') || '30', 10);\n const data = await getAnalyticsData(days);\n return json(data);\n } catch (err: unknown) {\n const msg = err instanceof Error ? err.message : '';\n if (msg.includes('authentication credential')) return json({ error: 'Google Analytics authentication failed.' }, { status: 401 });\n if (msg.includes('sufficient permissions')) return json({ error: 'Service account does not have access.' }, { status: 403 });\n return json({ error: 'Failed to fetch analytics data' }, { status: 500 });\n }\n },\n propertyId: async (): Promise<Response> => {\n const payload = getPropertyId ? getPropertyId() : { currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID };\n return json({ message: 'Property ID Information', ...payload });\n },\n permissions: async (): Promise<Response> => {\n const payload = getPermissions ? getPermissions() : {\n serviceAccountEmail: process.env.GOOGLE_ANALYTICS_CLIENT_EMAIL,\n currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID,\n };\n return json({ message: 'Permission Troubleshooting Guide', ...payload });\n },\n };\n}\n\nimport type { StorageService } from '../plugins/storage';\n\nexport type { StorageService };\n\nexport interface UploadHandlerConfig extends CmsHandlersBase {\n /** Storage plugin instance or getter (e.g. () => getCms().then(c => c.getPlugin('storage'))). If not set, uses local fallback. */\n storage?: StorageService | (() => StorageService | undefined) | (() => Promise<StorageService | undefined>);\n /** Fallback when storage not set: dir relative to cwd (e.g. \"public/uploads\") */\n localUploadDir?: string;\n allowedTypes?: string[];\n maxSizeBytes?: number;\n}\n\nexport function createUploadHandler(config: UploadHandlerConfig) {\n const { json, requireAuth, requireEntityPermission, storage, localUploadDir = 'public/uploads', allowedTypes, maxSizeBytes = 10 * 1024 * 1024 } = config;\n const allowed = allowedTypes ?? ['image/jpeg', 'image/png', 'image/gif', 'image/webp', 'application/pdf', 'text/plain'];\n return async function POST(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'upload', 'create');\n if (pe) return pe;\n }\n try {\n const formData = await req.formData();\n const file = formData.get('file') as File | null;\n if (!file) return json({ error: 'No file uploaded' }, { status: 400 });\n if (!allowed.includes(file.type)) return json({ error: 'File type not allowed' }, { status: 400 });\n if (file.size > maxSizeBytes) return json({ error: 'File size exceeds limit' }, { status: 400 });\n const buffer = Buffer.from(await file.arrayBuffer());\n const fileName = `${Date.now()}-${file.name}`;\n const contentType = file.type || 'application/octet-stream';\n const raw = typeof storage === 'function' ? storage() : storage;\n const storageService = raw instanceof Promise ? await raw : raw;\n if (storageService) {\n const fileUrl = await storageService.upload(buffer, `uploads/${fileName}`, contentType);\n return json({ filePath: fileUrl });\n }\n const fs = await import('fs/promises');\n const path = await import('path');\n const dir = path.join(process.cwd(), localUploadDir);\n await fs.mkdir(dir, { recursive: true });\n const filePath = path.join(dir, fileName);\n await fs.writeFile(filePath, buffer);\n return json({ filePath: `/${localUploadDir.replace(/^\\/+/, '').replace(/\\\\/g, '/')}/${fileName}` });\n } catch (err) {\n return json({ error: 'File upload failed' }, { status: 500 });\n }\n };\n}\n\nexport interface BlogBySlugConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\nexport function createBlogBySlugHandler(config: BlogBySlugConfig) {\n const { dataSource, entityMap, json } = config;\n return async function GET(_req: Request, slug: string): Promise<Response> {\n try {\n const blogRepo = dataSource.getRepository(entityMap.blogs);\n const blog = await blogRepo.findOne({\n where: { slug, published: true },\n relations: ['author', 'category', 'tags', 'seo'],\n });\n if (!blog) return json({ error: 'Blog not found' }, { status: 404 });\n return json(blog);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormBySlugConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\nexport function createFormBySlugHandler(config: FormBySlugConfig) {\n const { dataSource, entityMap, json } = config;\n return async function GET(_req: Request, slug: string): Promise<Response> {\n try {\n const formRepo = dataSource.getRepository(entityMap.forms);\n const form = await formRepo.findOne({\n where: { slug, published: true, deleted: false },\n relations: ['fields'],\n order: { fields: { order: 'ASC' } },\n });\n if (!form) return json({ error: 'Form not found' }, { status: 404 });\n const out = form as { fields?: unknown[] };\n if (Array.isArray(out.fields)) out.fields = out.fields.filter((f) => !(f as { deleted?: boolean }).deleted);\n return json(form);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormSaveHandlersConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n}\n\n/** Normalize a field from payload for DB: only include columns that exist on form_fields. */\nfunction normalizeFieldRow(f: Record<string, unknown>, formId: number): Record<string, unknown> {\n const order = typeof f.order === 'number' ? f.order : Number(f.order) || 0;\n const groupId = typeof f.groupId === 'number' ? f.groupId : Number(f.groupId) || 1;\n const columnWidth = typeof f.columnWidth === 'number' ? f.columnWidth : Number(f.columnWidth) || 12;\n return {\n formId,\n label: f.label != null ? String(f.label) : '',\n type: f.type != null ? String(f.type) : 'text',\n placeholder: f.placeholder != null ? String(f.placeholder) : null,\n options: f.options != null ? (typeof f.options === 'string' ? f.options : JSON.stringify(f.options)) : null,\n required: Boolean(f.required),\n validation: f.validation != null ? (typeof f.validation === 'string' ? f.validation : JSON.stringify(f.validation)) : null,\n order,\n groupId,\n columnWidth,\n };\n}\n\nexport function createFormSaveHandlers(config: FormSaveHandlersConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n const formRepo = () => dataSource.getRepository(entityMap.forms);\n const fieldRepo = () => dataSource.getRepository(entityMap.form_fields);\n\n return {\n async GET(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'read');\n if (pe) return pe;\n }\n try {\n const formId = Number(id);\n if (!Number.isInteger(formId) || formId <= 0) return json({ error: 'Invalid form id' }, { status: 400 });\n const form = await formRepo().findOne({\n where: { id: formId },\n relations: ['fields'],\n order: { fields: { order: 'ASC' } },\n });\n if (!form) return json({ message: 'Not found' }, { status: 404 });\n const out = form as { fields?: { deleted?: boolean }[] };\n if (Array.isArray(out.fields)) out.fields = out.fields.filter((f) => !f.deleted);\n return json(form);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async POST(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'create');\n if (pe) return pe;\n }\n try {\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid request payload' }, { status: 400 });\n const fields = Array.isArray(body.fields) ? body.fields : [];\n const { fields: _f, ...formRow } = body;\n const form = await formRepo().save(formRepo().create(formRow as object));\n for (let i = 0; i < fields.length; i++) {\n const row = normalizeFieldRow(fields[i] as Record<string, unknown>, form.id);\n (row as Record<string, unknown>).order = i + 1;\n await fieldRepo().save(fieldRepo().create(row as object));\n }\n const saved = await formRepo().findOne({ where: { id: form.id }, relations: ['fields'], order: { fields: { order: 'ASC' } } });\n return json(saved ?? form, { status: 201 });\n } catch (e) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async PUT(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'forms', 'update');\n if (pe) return pe;\n }\n try {\n const formId = Number(id);\n if (!Number.isInteger(formId) || formId <= 0) return json({ error: 'Invalid form id' }, { status: 400 });\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid request payload' }, { status: 400 });\n const existing = await formRepo().findOne({ where: { id: formId } });\n if (!existing) return json({ message: 'Not found' }, { status: 404 });\n const fields = Array.isArray(body.fields) ? body.fields : [];\n const formRow: Record<string, unknown> = {};\n for (const key of ['name', 'description', 'campaign', 'slug', 'published']) {\n if (body[key] !== undefined) formRow[key] = body[key];\n }\n if (Object.keys(formRow).length > 0) await formRepo().update(formId, formRow as object);\n await fieldRepo().delete({ formId });\n for (let i = 0; i < fields.length; i++) {\n const row = normalizeFieldRow(fields[i] as Record<string, unknown>, formId);\n (row as Record<string, unknown>).order = i + 1;\n await fieldRepo().save(fieldRepo().create(row as object));\n }\n const saved = await formRepo().findOne({ where: { id: formId }, relations: ['fields'], order: { fields: { order: 'ASC' } } });\n return saved ? json(saved) : json({ message: 'Not found' }, { status: 404 });\n } catch (e) {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface FormSubmissionHandlerConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n /** When set, form submission notification email is queued (CRM recipient). */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n getRecipientForChannel?: (channel: 'crm' | 'sales' | 'fulfilment') => Promise<string | null>;\n}\n\nasync function isErpIntegrationEnabled(dataSource: DataSource, entityMap: EntityMap): Promise<boolean> {\n const repo = dataSource.getRepository(entityMap.configs);\n const rows = await repo.find({ where: { settings: 'erp', deleted: false } as object });\n for (const row of rows) {\n const r = row as { key: string; value: string };\n if (r.key === 'enabled') return r.value !== 'false';\n }\n return true;\n}\n\n/** Form IDs that use `form.submitted` (opportunity). `null` = config key absent (treat as no IDs → all submissions use lead). */\nasync function getErpOpportunityFormIds(dataSource: DataSource, entityMap: EntityMap): Promise<number[] | null> {\n const repo = dataSource.getRepository(entityMap.configs);\n const row = await repo.findOne({\n where: { settings: 'erp', key: 'opportunityFormIds', deleted: false } as object,\n });\n if (!row) return null;\n const raw = ((row as { value: string }).value ?? '').trim();\n if (!raw) return [];\n try {\n const parsed = JSON.parse(raw) as unknown;\n if (!Array.isArray(parsed)) return [];\n const ids = parsed\n .map((x) => (typeof x === 'number' ? x : Number(x)))\n .filter((n): n is number => Number.isInteger(n) && n > 0);\n return [...new Set(ids)];\n } catch {\n return [];\n }\n}\n\nexport interface FormSubmissionGetByIdConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n}\n\nexport function createFormSubmissionGetByIdHandler(config: FormSubmissionGetByIdConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n return async function GET(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'form_submissions', 'read');\n if (pe) return pe;\n }\n try {\n const submissionId = Number(id);\n if (!Number.isInteger(submissionId) || submissionId <= 0) return json({ error: 'Invalid id' }, { status: 400 });\n const repo = dataSource.getRepository(entityMap.form_submissions);\n const submission = await repo.findOne({\n where: { id: submissionId },\n relations: ['form', 'contact'],\n });\n if (!submission) return json({ message: 'Not found' }, { status: 404 });\n const form = submission as { form?: { id: number; fields?: unknown[] } };\n if (form.form?.id) {\n const formRepo = dataSource.getRepository(entityMap.forms);\n const formWithFields = await formRepo.findOne({\n where: { id: form.form.id },\n relations: ['fields'],\n order: { fields: { order: 'ASC' as const } },\n });\n if (formWithFields) (submission as { form?: unknown }).form = formWithFields;\n }\n return json(submission);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface FormSubmissionListConfig extends FormSubmissionGetByIdConfig {}\n\nexport function createFormSubmissionListHandler(config: FormSubmissionListConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n return async function GET(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'form_submissions', 'read');\n if (pe) return pe;\n }\n try {\n const repo = dataSource.getRepository(entityMap.form_submissions);\n const { searchParams } = new URL(req.url);\n const page = Number(searchParams.get('page')) || 1;\n const limit = Math.min(100, Number(searchParams.get('limit')) || 10);\n const skip = (page - 1) * limit;\n const sortField = searchParams.get('sortField') || 'createdAt';\n const sortOrder = searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const [data, total] = await repo.findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n relations: ['form', 'contact'],\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nfunction formatSubmissionFieldValue(raw: unknown): string {\n if (raw == null || raw === '') return '—';\n if (typeof raw === 'object') return JSON.stringify(raw);\n return String(raw);\n}\n\nfunction pickContactFromSubmission(\n fields: { id: number; type: string; label: string }[],\n data: Record<string, unknown>\n): { name: string; email: string; phone: string | null } | null {\n let email: string | null = null;\n let name: string | null = null;\n let phone: string | null = null;\n for (const f of fields) {\n const val = data[String(f.id)];\n if (val == null || val === '') continue;\n const str = String(val).trim();\n if (f.type === 'email' || (f.label && f.label.toLowerCase().includes('email'))) {\n if (str && !email) email = str;\n } else if (f.type === 'phone' || (f.label && f.label.toLowerCase().includes('phone'))) {\n if (str && !phone) phone = str;\n } else if (f.label && f.label.toLowerCase().includes('name') && (f.type === 'text' || !f.type)) {\n if (str && !name) name = str;\n }\n }\n if (!email) return null;\n return { name: name || email, email, phone: phone || null };\n}\n\nexport function createFormSubmissionHandler(config: FormSubmissionHandlerConfig) {\n const { dataSource, entityMap, json, getCms } = config;\n return async function POST(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as Record<string, unknown> | null;\n if (!body || typeof body !== 'object') {\n return json({ error: 'Invalid request payload' }, { status: 400 });\n }\n const captchaErr = await assertCaptchaOk(getCms, body, req, json);\n if (captchaErr) return captchaErr;\n const formId = typeof body.formId === 'number' ? body.formId : Number(body.formId);\n if (!Number.isInteger(formId) || formId <= 0) {\n return json({ error: 'formId is required and must be a positive integer' }, { status: 400 });\n }\n const data = body.data;\n if (!data || typeof data !== 'object' || Array.isArray(data)) {\n return json({ error: 'data is required and must be an object' }, { status: 400 });\n }\n const formRepo = dataSource.getRepository(entityMap.forms);\n const form = await formRepo.findOne({\n where: { id: formId, published: true, deleted: false },\n relations: ['fields'],\n });\n if (!form) {\n return json({ error: 'Form not found' }, { status: 404 });\n }\n const fields = (form as { fields?: { id: number; type: string; label: string; deleted?: boolean }[] }).fields ?? [];\n const activeFields = fields.filter((f) => !f.deleted);\n\n let contactId: number | null =\n body.contactId != null && body.contactId !== ''\n ? (typeof body.contactId === 'number' ? body.contactId : Number(body.contactId))\n : null;\n\n if (!contactId) {\n const contactData = pickContactFromSubmission(activeFields, data as Record<string, unknown>);\n if (contactData) {\n const contactRepo = dataSource.getRepository(entityMap.contacts);\n let contact = await contactRepo.findOne({ where: { email: contactData.email } });\n if (!contact) {\n contact = await contactRepo.save(\n contactRepo.create({\n name: contactData.name,\n email: contactData.email,\n phone: contactData.phone,\n })\n );\n }\n contactId = contact.id;\n }\n }\n\n const ipAddress = (req.headers.get('x-forwarded-for') ?? req.headers.get('x-real-ip') ?? null) as string | null;\n const userAgent = req.headers.get('user-agent') ?? null;\n const submissionRepo = dataSource.getRepository(entityMap.form_submissions);\n const created = await submissionRepo.save(\n submissionRepo.create({\n formId,\n contactId: Number.isInteger(contactId) ? contactId : null,\n data: data as Record<string, unknown>,\n ipAddress: ipAddress?.slice(0, 255) ?? null,\n userAgent: userAgent?.slice(0, 500) ?? null,\n })\n );\n\n const formWithName = form as { name?: string };\n const formName = formWithName.name ?? 'Form';\n let contactName = 'Unknown';\n let contactEmail = '';\n if (Number.isInteger(contactId)) {\n const contactRepo = dataSource.getRepository(entityMap.contacts);\n const contact = await contactRepo.findOne({ where: { id: contactId }, select: ['name', 'email'] });\n if (contact) {\n contactName = (contact as { name: string }).name ?? contactName;\n contactEmail = (contact as { email: string }).email ?? contactEmail;\n }\n } else {\n const contactData = pickContactFromSubmission(activeFields, data as Record<string, unknown>);\n if (contactData) {\n contactName = contactData.name;\n contactEmail = contactData.email;\n }\n }\n\n if (config.getCms) {\n try {\n const cms = await config.getCms();\n if (config.getCompanyDetails && config.getRecipientForChannel) {\n const to = await config.getRecipientForChannel('crm');\n if (to) {\n const companyDetails = await config.getCompanyDetails();\n const formFieldRows = activeFields.map((f) => ({\n label: (f.label && String(f.label).trim()) || `Field ${f.id}`,\n value: formatSubmissionFieldValue(data[String(f.id)]),\n }));\n await queueEmail(cms, {\n to,\n templateName: 'formSubmission',\n ctx: {\n formName,\n contactName,\n contactEmail,\n formData: data,\n formFieldRows,\n companyDetails: companyDetails ?? {},\n },\n });\n }\n }\n if (await isErpIntegrationEnabled(dataSource, entityMap)) {\n const erp = cms.getPlugin('erp') as { submission: ERPSubmissionService } | undefined;\n if (erp) {\n const contact = erp.submission.extractContactData(data as Record<string, unknown>, activeFields);\n if (contact?.email?.trim()) {\n const opportunityFormIds = await getErpOpportunityFormIds(dataSource, entityMap);\n const asOpportunity =\n opportunityFormIds != null &&\n opportunityFormIds.length > 0 &&\n opportunityFormIds.includes(formId);\n await queueErp(\n cms,\n asOpportunity ? { kind: 'formOpportunity', contact } : { kind: 'lead', contact }\n );\n }\n }\n }\n } catch {\n // do not fail the submission if notification / ERP queue fails\n }\n }\n\n return json(created, { status: 201 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n };\n}\n\nexport interface UsersApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n baseUrl: string;\n /** When set with email queue/plugin, invite emails are sent on user create and regenerate-invite. */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n}\n\nexport function createUsersApiHandlers(config: UsersApiConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission, baseUrl, getCms, getCompanyDetails } = config;\n\n async function trySendInviteEmail(toEmail: string, inviteLink: string, inviteeName: string): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n await queueEmail(cms, {\n to: toEmail,\n templateName: 'invite',\n ctx: {\n inviteLink,\n email: toEmail,\n inviteeName: inviteeName.trim(),\n companyDetails: companyDetails ?? {},\n },\n });\n } catch {\n // do not fail user APIs if email fails\n }\n }\n const userRepo = () => dataSource.getRepository(entityMap.users);\n return {\n async list(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'read');\n if (pe) return pe;\n }\n try {\n const url = new URL(req.url);\n const page = Math.max(1, parseInt(url.searchParams.get('page') || '1', 10));\n const limit = Math.min(100, parseInt(url.searchParams.get('limit') || '10', 10));\n const skip = (page - 1) * limit;\n const sortField = url.searchParams.get('sortField') || 'createdAt';\n const sortOrder = url.searchParams.get('sortOrder') === 'desc' ? 'DESC' : 'ASC';\n const search = url.searchParams.get('search');\n const where = search ? [{ name: ILike(`%${search}%`) }, { email: ILike(`%${search}%`) }] : {};\n const [data, total] = await userRepo().findAndCount({\n skip,\n take: limit,\n order: { [sortField]: sortOrder },\n where,\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n return json({ total, page, limit, totalPages: Math.ceil(total / limit), data });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async create(req: Request): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'create');\n if (pe) return pe;\n }\n try {\n const body = await req.json() as Record<string, unknown>;\n if (!body?.name || !body?.email) return json({ error: 'Name and email are required' }, { status: 400 });\n const existing = await userRepo().findOne({ where: { email: body.email as string } });\n if (existing) return json({ error: 'User with this email already exists' }, { status: 400 });\n const groupRepo = dataSource.getRepository(entityMap.user_groups);\n const customerG = await groupRepo.findOne({ where: { name: 'Customer', deleted: false } });\n const gid = (body.groupId as number) ?? null;\n const isCustomer = !!(customerG && gid === customerG.id);\n const adminAccess = isCustomer ? false : body.adminAccess === false ? false : true;\n const newUser = await userRepo().save(\n userRepo().create({\n name: body.name,\n email: body.email,\n password: null,\n blocked: true,\n groupId: gid,\n adminAccess,\n })\n );\n if (entityMap.contacts) {\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, newUser.id, newUser.email as string);\n }\n const emailToken = Buffer.from(newUser.email).toString('base64');\n const inviteLink = `${baseUrl}/admin/invite?token=${emailToken}`;\n await trySendInviteEmail(\n newUser.email as string,\n inviteLink,\n (newUser.name as string) ?? ''\n );\n return json({ message: 'User created successfully (blocked until password is set)', user: newUser, inviteLink }, { status: 201 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async getById(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'read');\n if (pe) return pe;\n }\n try {\n const user = await userRepo().findOne({\n where: { id: parseInt(id, 10) },\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n return json(user);\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async update(req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'users', 'update');\n if (pe) return pe;\n }\n try {\n const body = await req.json() as Record<string, unknown>;\n const { password: _p, ...safe } = body;\n await userRepo().update(parseInt(id, 10), safe as object);\n const updated = await userRepo().findOne({\n where: { id: parseInt(id, 10) },\n relations: ['group'],\n select: ['id', 'name', 'email', 'blocked', 'createdAt', 'updatedAt', 'groupId'],\n });\n return updated ? json(updated) : json({ error: 'Not found' }, { status: 404 });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async delete(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'delete');\n if (pe) return pe;\n }\n try {\n const r = await userRepo().delete(parseInt(id, 10));\n if (r.affected === 0) return json({ error: 'User not found' }, { status: 404 });\n return json({ message: 'User deleted successfully' });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n async regenerateInvite(_req: Request, id: string): Promise<Response> {\n const authErr = await requireAuth(new Request(_req.url));\n if (authErr) return authErr;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(_req, 'users', 'update');\n if (pe) return pe;\n }\n try {\n const user = await userRepo().findOne({ where: { id: parseInt(id, 10) }, select: ['email', 'name'] });\n if (!user) return json({ error: 'User not found' }, { status: 404 });\n const emailToken = Buffer.from(user.email).toString('base64');\n const inviteLink = `${baseUrl}/admin/invite?token=${emailToken}`;\n await trySendInviteEmail(user.email as string, inviteLink, (user.name as string) ?? '');\n return json({ message: 'New invite link generated successfully', inviteLink });\n } catch {\n return json({ error: 'Server Error' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface UserAvatarConfig extends CmsHandlersBase {\n getSession: () => Promise<{ user?: { email?: string | null } } | null>;\n /** Save avatar (buffer, fileName) => publicUrl. Default: local public/uploads/avatars */\n saveAvatar?: (buffer: Buffer, fileName: string) => Promise<string>;\n}\n\nexport function createUserAvatarHandler(config: UserAvatarConfig) {\n const { json, getSession, saveAvatar } = config;\n return async function POST(req: Request): Promise<Response> {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n try {\n const formData = await req.formData();\n const file = formData.get('avatar') as File | null;\n if (!file) return json({ error: 'No file uploaded' }, { status: 400 });\n if (!file.type.startsWith('image/')) return json({ error: 'File must be an image' }, { status: 400 });\n if (file.size > 5 * 1024 * 1024) return json({ error: 'File size must be less than 5MB' }, { status: 400 });\n const buffer = Buffer.from(await file.arrayBuffer());\n const ext = file.name.split('.').pop() || 'jpg';\n const fileName = `avatar_${session.user.email}_${Date.now()}.${ext}`;\n const avatarUrl = saveAvatar\n ? await saveAvatar(buffer, fileName)\n : await (async () => {\n const fs = await import('fs/promises');\n const path = await import('path');\n const dir = path.join(process.cwd(), 'public', 'uploads', 'avatars');\n await fs.mkdir(dir, { recursive: true });\n await fs.writeFile(path.join(dir, fileName), buffer);\n return `/uploads/avatars/${fileName}`;\n })();\n return json({ message: 'Avatar uploaded successfully', avatarUrl });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface UserProfileConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n getSession: () => Promise<{ user?: { email?: string | null; name?: string | null } } | null>;\n}\n\nexport function createUserProfileHandler(config: UserProfileConfig) {\n const { dataSource, entityMap, json, getSession } = config;\n return async function PUT(req: Request): Promise<Response> {\n const session = await getSession();\n if (!session?.user?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n try {\n const body = await req.json() as { name?: string };\n if (!body?.name) return json({ error: 'Name is required' }, { status: 400 });\n const userRepo = dataSource.getRepository(entityMap.users);\n await userRepo.update({ email: session.user.email }, { name: body.name, updatedAt: new Date() });\n const updated = await userRepo.findOne({ where: { email: session.user.email }, select: ['id', 'name', 'email'] });\n if (!updated) return json({ error: 'Not found' }, { status: 404 });\n return json({ message: 'Profile updated successfully', user: { id: updated.id, name: updated.name, email: updated.email } });\n } catch {\n return json({ error: 'Internal server error' }, { status: 500 });\n }\n };\n}\n\nexport interface SettingsApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n encryptionKey?: string;\n /** Groups in this list are readable without auth (GET returns all keys for the group). */\n publicGetGroups?: string[];\n}\n\nfunction simpleEncrypt(text: string, key: string): string {\n const buf = Buffer.from(text, 'utf8');\n const keyBuf = Buffer.from(key.padEnd(32, '0').slice(0, 32), 'utf8');\n const out = Buffer.alloc(buf.length);\n for (let i = 0; i < buf.length; i++) out[i] = buf[i] ^ keyBuf[i % keyBuf.length];\n return out.toString('base64');\n}\n\nfunction simpleDecrypt(encoded: string, key: string): string {\n const buf = Buffer.from(encoded, 'base64');\n const keyBuf = Buffer.from(key.padEnd(32, '0').slice(0, 32), 'utf8');\n const out = Buffer.alloc(buf.length);\n for (let i = 0; i < buf.length; i++) out[i] = buf[i] ^ keyBuf[i % keyBuf.length];\n return out.toString('utf8');\n}\n\n/**\n * Structural types so apps with their own `typeorm` install (e.g. npm link) typecheck without duplicate-package errors.\n */\nexport type GetPublicSettingsGroupDataSource = {\n getRepository(entity: unknown): {\n find(options: object): Promise<unknown[]>;\n };\n};\n\nexport interface GetPublicSettingsGroupConfig {\n dataSource: GetPublicSettingsGroupDataSource;\n entityMap: Record<string, unknown>;\n encryptionKey?: string;\n}\n\n/** Same rows as unauthenticated GET /api/settings/:group when the group is in publicGetGroups. */\nexport async function getPublicSettingsGroup(\n config: GetPublicSettingsGroupConfig,\n group: string\n): Promise<Record<string, string>> {\n const { dataSource, entityMap, encryptionKey } = config;\n const repo = dataSource.getRepository(entityMap.configs);\n const rows = await repo.find({ where: { settings: group, deleted: false } });\n const result: Record<string, string> = {};\n for (const row of rows) {\n const r = row as { key: string; value: string; encrypted?: boolean };\n let val = r.value;\n if (r.encrypted && encryptionKey) {\n try {\n val = simpleDecrypt(val, encryptionKey);\n } catch {\n /* keep raw if decrypt fails */\n }\n }\n result[r.key] = val;\n }\n return result;\n}\n\nexport function createSettingsApiHandlers(config: SettingsApiConfig) {\n const { dataSource, entityMap, json, requireAuth, encryptionKey, publicGetGroups } = config;\n const configRepo = () => dataSource.getRepository(entityMap.configs);\n\n return {\n async GET(req: Request, group: string): Promise<Response> {\n const isPublicGroup = publicGetGroups?.includes(group);\n const authErr = isPublicGroup ? null : await requireAuth(req);\n const isAuthed = !authErr;\n\n try {\n if (isPublicGroup) {\n const result = await getPublicSettingsGroup(\n { dataSource, entityMap, encryptionKey },\n group\n );\n return json(result);\n }\n\n const where: Record<string, unknown> = { settings: group, deleted: false };\n if (!isAuthed && !isPublicGroup) where.type = 'public';\n\n const rows = await configRepo().find({ where });\n const result: Record<string, unknown> = {};\n for (const row of rows) {\n const r = row as { key: string; value: string; encrypted?: boolean; type?: string };\n let val = r.value;\n if (r.encrypted && encryptionKey) {\n try { val = simpleDecrypt(val, encryptionKey); } catch { /* return raw if decrypt fails */ }\n }\n result[r.key] = val;\n }\n return json(result);\n } catch {\n return json({ error: 'Failed to fetch settings' }, { status: 500 });\n }\n },\n\n async PUT(req: Request, group: string): Promise<Response> {\n const authErr = await requireAuth(req);\n if (authErr) return authErr;\n\n try {\n const body = await req.json() as Record<string, { value: string; type?: 'public' | 'private'; encrypted?: boolean }>;\n if (!body || typeof body !== 'object') return json({ error: 'Invalid payload' }, { status: 400 });\n\n const repo = configRepo();\n for (const [key, entry] of Object.entries(body)) {\n const val = typeof entry === 'string' ? entry : entry.value;\n const type = (typeof entry === 'object' && entry.type) || 'private';\n const encrypted = !!(typeof entry === 'object' && entry.encrypted);\n\n let storedValue = val;\n if (encrypted && encryptionKey) {\n storedValue = simpleEncrypt(val, encryptionKey);\n }\n\n const existing = await repo.findOne({ where: { settings: group, key } });\n if (existing) {\n await repo.update(existing.id, { value: storedValue, type, encrypted, updatedAt: new Date() } as object);\n } else {\n await repo.save(repo.create({ settings: group, key, value: storedValue, type, encrypted } as object));\n }\n }\n return json({ message: 'Settings saved' });\n } catch {\n return json({ error: 'Failed to save settings' }, { status: 500 });\n }\n },\n };\n}\n\nexport interface ChatApiConfig extends CmsHandlersBase {\n dataSource: DataSource;\n entityMap: EntityMap;\n getCms: () => Promise<{ getPlugin: (name: string) => unknown }>;\n}\n\nconst KB_CHUNK_LIMIT = 10;\nconst KB_CONTEXT_MAX_CHARS = 4000;\n\nfunction getQueryTerms(message: string): string[] {\n return message\n .replace(/[^\\w\\s]/g, ' ')\n .split(/\\s+/)\n .filter((w) => w.length > 2)\n .slice(0, 6);\n}\n\nexport function createChatHandlers(config: ChatApiConfig) {\n const { dataSource, entityMap, json, getCms } = config;\n const contactRepo = () => dataSource.getRepository(entityMap.contacts);\n const convRepo = () => dataSource.getRepository(entityMap.chat_conversations);\n const msgRepo = () => dataSource.getRepository(entityMap.chat_messages);\n const chunkRepo = () => dataSource.getRepository(entityMap.knowledge_base_chunks);\n\n return {\n async identify(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as { name?: string; email?: string; phone?: string };\n const name = body?.name?.trim();\n const email = body?.email?.trim();\n if (!name || !email) return json({ error: 'name and email required' }, { status: 400 });\n const repo = contactRepo();\n let contact = await repo.findOne({ where: { email, deleted: false } });\n if (!contact) {\n const created = repo.create({ name, email, phone: body.phone?.trim() || null } as object);\n contact = await repo.save(created);\n }\n const convRepoInst = convRepo();\n const conv = await convRepoInst.save(convRepoInst.create({ contactId: (contact as { id: number }).id } as object));\n return json({\n contactId: (contact as { id: number }).id,\n conversationId: (conv as { id: number }).id,\n });\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Failed to identify';\n return json({ error: 'Failed to identify', detail: message }, { status: 500 });\n }\n },\n async getMessages(req: Request, conversationId: string): Promise<Response> {\n try {\n const conv = await convRepo().findOne({\n where: { id: parseInt(conversationId, 10) },\n relations: ['messages'],\n });\n if (!conv) return json({ error: 'Conversation not found' }, { status: 404 });\n const messages = ((conv as { messages?: Array<{ role: string; content: string; createdAt: string }> }).messages ?? [])\n .sort((a, b) => new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime())\n .map((m) => ({ role: m.role, content: m.content }));\n return json({ messages });\n } catch {\n return json({ error: 'Failed to fetch messages' }, { status: 500 });\n }\n },\n async postMessage(req: Request): Promise<Response> {\n try {\n const body = (await req.json()) as { conversationId?: number; message?: string };\n const conversationId = body?.conversationId;\n const message = body?.message?.trim();\n if (!conversationId || !message) return json({ error: 'conversationId and message required' }, { status: 400 });\n const conv = await convRepo().findOne({\n where: { id: conversationId },\n relations: ['messages'],\n });\n if (!conv) return json({ error: 'Conversation not found' }, { status: 404 });\n const msgRepoInst = msgRepo();\n await msgRepoInst.save(msgRepoInst.create({ conversationId, role: 'user', content: message } as object));\n const cms = await getCms();\n const llm = cms.getPlugin('llm') as {\n chat: (messages: Array<{ role: string; content: string }>, opts?: object) => Promise<{ content: string }>;\n embed?: (text: string) => Promise<number[]>;\n } | undefined;\n if (!llm?.chat) return json({ error: 'LLM not configured' }, { status: 503 });\n let contextParts: string[] = [];\n const queryEmbedding = llm.embed ? await llm.embed(message) : null;\n if (queryEmbedding && queryEmbedding.length > 0) {\n const vectorStr = '[' + queryEmbedding.join(',') + ']';\n try {\n const rows = (await dataSource.query(\n `SELECT id, content FROM knowledge_base_chunks WHERE embedding IS NOT NULL ORDER BY embedding <=> $1::vector LIMIT $2`,\n [vectorStr, KB_CHUNK_LIMIT]\n )) as Array<{ id: number; content: string }>;\n let totalLen = 0;\n for (const r of rows) {\n const text = (r.content ?? '').trim();\n if (!text || totalLen + text.length > KB_CONTEXT_MAX_CHARS) continue;\n contextParts.push(text);\n totalLen += text.length;\n }\n } catch {\n // pgvector not available or query failed; fall back to keyword\n }\n }\n if (contextParts.length === 0) {\n const terms = getQueryTerms(message);\n if (terms.length > 0) {\n const conditions = terms.map((t) => ({ content: ILike(`%${t}%`) }));\n const chunks = await chunkRepo().find({\n where: conditions,\n take: KB_CHUNK_LIMIT,\n order: { id: 'ASC' },\n });\n const seen = new Set<string>();\n let totalLen = 0;\n for (const c of chunks as Array<{ content: string }>) {\n const text = c.content.trim();\n if (seen.has(text) || totalLen + text.length > KB_CONTEXT_MAX_CHARS) continue;\n seen.add(text);\n contextParts.push(text);\n totalLen += text.length;\n }\n }\n }\n const history = ((conv as { messages?: Array<{ role: string; content: string; createdAt?: string }> }).messages ?? [])\n .sort((a, b) => new Date(a.createdAt ?? 0).getTime() - new Date(b.createdAt ?? 0).getTime())\n .map((m) => ({ role: m.role as 'user' | 'assistant' | 'system', content: m.content }));\n const systemContent = contextParts.length > 0\n ? `Use the following context about the company and its products to answer. If the answer is not in the context, say so.\\n\\nContext:\\n${contextParts.join('\\n\\n')}`\n : 'You are a helpful assistant for the company. If you do not have specific information, say so.';\n const messages = [\n { role: 'system' as const, content: systemContent },\n ...history,\n { role: 'user' as const, content: message },\n ];\n const { content } = await llm.chat(messages);\n await msgRepoInst.save(msgRepoInst.create({ conversationId, role: 'assistant', content } as object));\n return json({ content });\n } catch (err) {\n const msg = err instanceof Error ? err.message : '';\n return json({ error: msg || 'Failed to send message' }, { status: 500 });\n }\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport { SMS_MESSAGE_TEMPLATE_DEFAULTS, getSmsTemplateDefault } from '../message-templates/sms-defaults';\nimport type { RequireEntityPermissionFn } from './cms-handlers';\n\nexport interface MessageTemplateAdminHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n requireAuth: (req: Request) => Promise<Response | null>;\n requireEntityPermission?: RequireEntityPermissionFn;\n}\n\nexport function createSmsMessageTemplateHandlers(config: MessageTemplateAdminHandlersConfig) {\n const { dataSource, entityMap, json, requireAuth, requireEntityPermission } = config;\n\n const repo = () => dataSource.getRepository(entityMap.message_templates);\n\n async function requireSettingsRead(req: Request): Promise<Response | null> {\n const a = await requireAuth(req);\n if (a) return a;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'settings', 'read');\n if (pe) return pe;\n }\n return null;\n }\n\n async function requireSettingsUpdate(req: Request): Promise<Response | null> {\n const a = await requireAuth(req);\n if (a) return a;\n if (requireEntityPermission) {\n const pe = await requireEntityPermission(req, 'settings', 'update');\n if (pe) return pe;\n }\n return null;\n }\n\n return {\n async GET(req: Request): Promise<Response> {\n const err = await requireSettingsRead(req);\n if (err) return err;\n try {\n const rows = await repo().find({ where: { channel: 'sms', deleted: false } as object });\n const byKey = new Map(rows.map((r) => [r.templateKey, r]));\n const items = SMS_MESSAGE_TEMPLATE_DEFAULTS.map((def) => {\n const row = byKey.get(def.templateKey);\n return {\n templateKey: def.templateKey,\n name: def.name,\n defaultBody: def.body,\n body: row?.body?.trim() ? row.body : def.body,\n externalTemplateRef: row?.externalTemplateRef?.trim() ?? '',\n otpVarKey:\n row?.providerMeta && typeof row.providerMeta.otpVarKey === 'string'\n ? String(row.providerMeta.otpVarKey)\n : def.providerMeta?.otpVarKey ?? 'var1',\n enabled: row ? row.enabled : false,\n dbId: row?.id ?? null,\n };\n });\n return json({ items });\n } catch {\n return json({ error: 'Failed to load templates' }, { status: 500 });\n }\n },\n\n async PUT(req: Request): Promise<Response> {\n const err = await requireSettingsUpdate(req);\n if (err) return err;\n try {\n const raw = (await req.json().catch(() => null)) as {\n items?: Array<{\n templateKey?: string;\n body?: string;\n externalTemplateRef?: string;\n otpVarKey?: string;\n enabled?: boolean;\n }>;\n } | null;\n if (!raw?.items || !Array.isArray(raw.items)) {\n return json({ error: 'Invalid payload' }, { status: 400 });\n }\n\n for (const item of raw.items) {\n const templateKey = typeof item.templateKey === 'string' ? item.templateKey.trim() : '';\n if (!getSmsTemplateDefault(templateKey)) continue;\n\n const body = typeof item.body === 'string' ? item.body : '';\n const externalTemplateRef =\n typeof item.externalTemplateRef === 'string' ? item.externalTemplateRef.trim() : '';\n const otpVarKey =\n typeof item.otpVarKey === 'string' && item.otpVarKey.trim()\n ? item.otpVarKey.trim()\n : 'var1';\n const enabled = item.enabled !== false;\n\n const existing = await repo().findOne({\n where: { channel: 'sms', templateKey, deleted: false } as object,\n });\n const def = getSmsTemplateDefault(templateKey)!;\n const providerMeta = { otpVarKey };\n\n if (existing) {\n await repo().update(existing.id, {\n name: def.name,\n body,\n externalTemplateRef: externalTemplateRef || null,\n providerMeta,\n enabled,\n updatedAt: new Date(),\n } as object);\n } else {\n await repo().save(\n repo().create({\n channel: 'sms',\n templateKey,\n name: def.name,\n subject: null,\n body,\n externalTemplateRef: externalTemplateRef || null,\n providerMeta,\n enabled,\n deleted: false,\n } as object)\n );\n }\n }\n return json({ ok: true });\n } catch {\n return json({ error: 'Failed to save templates' }, { status: 500 });\n }\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityMap } from './crud';\nimport { getPermissionableEntityKeys, isSuperAdminGroupName } from '../auth/permission-entities';\nimport type { SessionUser } from '../auth/helpers';\nimport { canManageRoles } from '../auth/helpers';\n\nexport type GetSessionUser = () => Promise<SessionUser | null>;\n\nexport interface AdminRolesHandlersConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number }) => Response;\n getSessionUser: GetSessionUser;\n}\n\nexport function createAdminRolesHandlers(config: AdminRolesHandlersConfig) {\n const { dataSource, entityMap, json, getSessionUser } = config;\n const baseEntities = getPermissionableEntityKeys(entityMap as Record<string, unknown>);\n const allowEntities = new Set([...baseEntities, 'users']);\n const groupRepo = () => dataSource.getRepository(entityMap.user_groups);\n const permRepo = () => dataSource.getRepository(entityMap.permissions);\n const userRepo = () => dataSource.getRepository(entityMap.users);\n\n async function gate(): Promise<Response | null> {\n const u = await getSessionUser();\n if (!u?.email) return json({ error: 'Unauthorized' }, { status: 401 });\n if (!canManageRoles(u)) return json({ error: 'Forbidden' }, { status: 403 });\n return null;\n }\n\n return {\n async list(): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const groups = await groupRepo().find({\n where: { deleted: false },\n order: { id: 'ASC' },\n relations: ['permissions'],\n });\n const entities = [...allowEntities].sort();\n return json({\n entities,\n groups: groups.map((g) => ({\n id: g.id,\n name: g.name,\n permissions: (g.permissions ?? [])\n .filter((p) => !p.deleted)\n .map((p) => ({\n entity: p.entity,\n canCreate: p.canCreate,\n canRead: p.canRead,\n canUpdate: p.canUpdate,\n canDelete: p.canDelete,\n })),\n })),\n });\n },\n\n async createGroup(req: Request): Promise<Response> {\n const err = await gate();\n if (err) return err;\n try {\n const body = (await req.json()) as { name?: string };\n const name = body?.name?.trim();\n if (!name) return json({ error: 'Name is required' }, { status: 400 });\n const repo = groupRepo();\n const existing = await repo.findOne({ where: { name } });\n if (existing) return json({ error: 'Group name already exists' }, { status: 400 });\n const g = await repo.save(repo.create({ name }));\n return json({ id: g.id, name: g.name, permissions: [] }, { status: 201 });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n\n async patchGroup(req: Request, idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const id = parseInt(idStr, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n try {\n const body = (await req.json()) as { name?: string };\n const name = body?.name?.trim();\n if (!name) return json({ error: 'Name is required' }, { status: 400 });\n const repo = groupRepo();\n const g = await repo.findOne({ where: { id, deleted: false } });\n if (!g) return json({ error: 'Not found' }, { status: 404 });\n if (isSuperAdminGroupName(g.name) && !isSuperAdminGroupName(name)) {\n return json({ error: 'Cannot rename the administrator group' }, { status: 400 });\n }\n const dup = await repo.findOne({ where: { name } });\n if (dup && dup.id !== id) return json({ error: 'Name already in use' }, { status: 400 });\n g.name = name;\n await repo.save(g);\n return json({ id: g.id, name: g.name });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n\n async deleteGroup(idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const id = parseInt(idStr, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const repo = groupRepo();\n const g = await repo.findOne({ where: { id, deleted: false } });\n if (!g) return json({ error: 'Not found' }, { status: 404 });\n if (isSuperAdminGroupName(g.name)) return json({ error: 'Cannot delete the administrator group' }, { status: 400 });\n const userCount = await userRepo().count({ where: { groupId: id } });\n if (userCount > 0) return json({ error: 'Reassign users before deleting this group' }, { status: 409 });\n await permRepo().delete({ groupId: id });\n await repo.update(id, { deleted: true, deletedAt: new Date() } as object);\n return json({ ok: true });\n },\n\n async putPermissions(req: Request, idStr: string): Promise<Response> {\n const err = await gate();\n if (err) return err;\n const groupId = parseInt(idStr, 10);\n if (!Number.isFinite(groupId)) return json({ error: 'Invalid id' }, { status: 400 });\n const groupRepository = groupRepo();\n const g = await groupRepository.findOne({ where: { id: groupId, deleted: false } });\n if (!g) return json({ error: 'Group not found' }, { status: 404 });\n try {\n const body = (await req.json()) as {\n permissions?: Array<{\n entity: string;\n canCreate?: boolean;\n canRead?: boolean;\n canUpdate?: boolean;\n canDelete?: boolean;\n }>;\n };\n const rows = body?.permissions;\n if (!Array.isArray(rows)) return json({ error: 'permissions array required' }, { status: 400 });\n for (const r of rows) {\n if (!r?.entity || !allowEntities.has(r.entity)) {\n return json({ error: `Invalid entity: ${r?.entity ?? ''}` }, { status: 400 });\n }\n }\n await dataSource.transaction(async (em) => {\n await em.getRepository(entityMap.permissions).delete({ groupId });\n for (const r of rows) {\n await em.getRepository(entityMap.permissions).save(\n em.getRepository(entityMap.permissions).create({\n groupId,\n entity: r.entity,\n canCreate: !!r.canCreate,\n canRead: !!r.canRead,\n canUpdate: !!r.canUpdate,\n canDelete: !!r.canDelete,\n })\n );\n }\n });\n const updated = await groupRepository.findOne({\n where: { id: groupId },\n relations: ['permissions'],\n });\n return json({\n id: groupId,\n permissions: (updated?.permissions ?? []).map((p) => ({\n entity: p.entity,\n canCreate: p.canCreate,\n canRead: p.canRead,\n canUpdate: p.canUpdate,\n canDelete: p.canDelete,\n })),\n });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n };\n}\n","/**\n * Single CMS API handler: dashboard, analytics, upload, blog/form by slug, users API, user auth, CRUD. Mount once (e.g. app/api/[[...path]]/route.ts).\n */\nimport type { DataSource } from 'typeorm';\nimport { createCrudHandler, createCrudByIdHandler } from './crud';\nimport type { CrudHandlerOptions, EntityMap } from './crud';\nimport { createUserAuthApiRouter } from './auth-handlers';\nimport type { UserAuthApiConfig } from './auth-handlers';\nimport {\n createDashboardStatsHandler,\n createAnalyticsHandlers,\n createUploadHandler,\n createBlogBySlugHandler,\n createFormBySlugHandler,\n createFormSaveHandlers,\n createFormSubmissionHandler,\n createFormSubmissionGetByIdHandler,\n createFormSubmissionListHandler,\n createUsersApiHandlers,\n createUserAvatarHandler,\n createUserProfileHandler,\n createSettingsApiHandlers,\n createChatHandlers,\n} from './cms-handlers';\nimport { createSmsMessageTemplateHandlers } from './message-template-admin-handlers';\nimport type {\n DashboardStatsConfig,\n AnalyticsHandlerConfig,\n UploadHandlerConfig,\n BlogBySlugConfig,\n FormBySlugConfig,\n FormSaveHandlersConfig,\n FormSubmissionHandlerConfig,\n FormSubmissionGetByIdConfig,\n UsersApiConfig,\n UserAvatarConfig,\n UserProfileConfig,\n SettingsApiConfig,\n ChatApiConfig,\n} from './cms-handlers';\nimport type { EntityCrudAction } from '../auth/permission-entities';\nimport type { SessionUser } from '../auth/helpers';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { createAdminRolesHandlers } from './admin-roles-handlers';\n\n/** CMS instance with getPlugin; when provided, analytics and userAuth.sendEmail can be resolved from plugins when not passed. */\nexport type CmsGetter = () => Promise<{ getPlugin: (name: string) => unknown }>;\n\nexport interface CmsApiHandlerConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n requireAuth: (req: Request) => Promise<Response | null>;\n json: (body: unknown, init?: { status?: number }) => Response;\n pathToModel?: (segment: string) => string;\n crudResources?: string[];\n /** When set, analytics and userAuth.sendEmail can be derived from getPlugin('analytics') and getPlugin('email') when not provided. */\n getCms?: CmsGetter;\n /** Optional: used when deriving userAuth.sendEmail to pass company details into email templates (e.g. from settings). */\n getCompanyDetails?: () => Promise<CompanyDetails>;\n /** Optional: used for form submission and other channel-based email recipients (from settings group \"email\"). */\n getRecipientForChannel?: (channel: 'crm' | 'sales' | 'fulfilment') => Promise<string | null>;\n userAuth?: UserAuthApiConfig;\n /** GET /api/dashboard/stats */\n dashboard?: DashboardStatsConfig;\n /** GET /api/analytics. If omitted and getCms is set, uses getCms().getPlugin('analytics'). */\n analytics?: AnalyticsHandlerConfig;\n /** POST /api/upload (S3 or local) */\n upload?: UploadHandlerConfig;\n /** GET /api/blogs/slug/:slug (public) */\n blogBySlug?: BlogBySlugConfig;\n /** GET /api/forms/slug/:slug (public) */\n formBySlug?: FormBySlugConfig;\n /** POST/PUT /api/forms (save form + fields; when set, overrides CRUD for forms) */\n formSave?: FormSaveHandlersConfig;\n /** POST /api/form-submissions (public, no auth) */\n formSubmission?: FormSubmissionHandlerConfig;\n /** GET /api/form-submissions/:id (auth) with form + contact relations */\n formSubmissionGetById?: FormSubmissionGetByIdConfig;\n /** GET/POST /api/users, GET/PUT/DELETE /api/users/:id, POST /api/users/:id/regenerate-invite */\n usersApi?: UsersApiConfig;\n /** POST /api/users/avatar */\n userAvatar?: UserAvatarConfig;\n /** PUT /api/users/profile */\n userProfile?: UserProfileConfig;\n /** GET/PUT /api/settings/:group */\n settings?: SettingsApiConfig;\n /** POST /api/chat/identify, GET /api/chat/conversations/:id/messages, POST /api/chat/messages */\n chat?: ChatApiConfig;\n /** When set, CRUD and admin routes enforce entity-level permissions from session */\n requireEntityPermission?: (req: Request, entity: string, action: EntityCrudAction) => Promise<Response | null>;\n /** Required for GET/POST/PATCH/DELETE /api/admin/roles */\n getSessionUser?: () => Promise<SessionUser | null>;\n}\n\nconst DEFAULT_EXCLUDE = new Set([\n 'users',\n 'password_reset_tokens',\n 'user_groups',\n 'permissions',\n 'comments',\n 'form_fields',\n 'configs',\n 'carts',\n 'cart_items',\n 'wishlists',\n 'wishlist_items',\n 'message_templates',\n]);\n\nexport function createCmsApiHandler(config: CmsApiHandlerConfig) {\n const {\n dataSource,\n entityMap,\n pathToModel = (s) => s,\n crudResources = Object.keys(entityMap).filter((k) => !DEFAULT_EXCLUDE.has(k)),\n getCms,\n userAuth: userAuthConfig,\n dashboard,\n analytics: analyticsConfig,\n upload,\n blogBySlug,\n formBySlug,\n formSave: formSaveConfig,\n formSubmission: formSubmissionConfig,\n formSubmissionGetById: formSubmissionGetByIdConfig,\n usersApi,\n userAvatar,\n userProfile,\n settings: settingsConfig,\n chat: chatConfig,\n requireEntityPermission: reqEntityPerm,\n getSessionUser,\n } = config;\n\n const analytics: AnalyticsHandlerConfig | undefined =\n analyticsConfig ??\n (getCms\n ? {\n json: config.json,\n requireAuth: async () => null,\n getAnalyticsData: async (days) => {\n const cms = await getCms();\n const a = cms.getPlugin('analytics') as { getAnalyticsData?: (days: number) => Promise<unknown> } | undefined;\n if (!a?.getAnalyticsData) throw new Error('Analytics not configured');\n return a.getAnalyticsData(days);\n },\n getPropertyId: () => ({ currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID }),\n getPermissions: () => ({\n serviceAccountEmail: process.env.GOOGLE_ANALYTICS_CLIENT_EMAIL,\n currentViewId: process.env.GOOGLE_ANALYTICS_VIEW_ID,\n }),\n }\n : undefined);\n\n const userAuth: UserAuthApiConfig | undefined =\n userAuthConfig && getCms && userAuthConfig.sendEmail === undefined\n ? {\n ...userAuthConfig,\n sendEmail: async (opts) => {\n const cms = await getCms();\n const queue = cms.getPlugin('queue') as { add: (name: string, data: object) => Promise<void> } | undefined;\n const companyDetails = config.getCompanyDetails ? await config.getCompanyDetails() : {};\n const resetLink =\n (typeof opts.resetLink === 'string' && opts.resetLink.trim()) ||\n (typeof opts.text === 'string' && opts.text.trim()) ||\n (typeof opts.html === 'string' ? (opts.html.match(/href\\s*=\\s*[\"']([^\"']+)[\"']/)?.[1] ?? '') : '');\n const ctx = { resetLink, companyDetails };\n if (queue) {\n const { queueEmail } = await import('../plugins/email/email-queue');\n await queueEmail(cms as { getPlugin: (name: string) => unknown }, { to: opts.to, templateName: 'passwordReset', ctx });\n return;\n }\n const email = cms.getPlugin('email') as { send: (data: { subject: string; html: string; text?: string; to?: string }) => Promise<boolean>; renderTemplate: (name: string, ctx: unknown) => { subject: string; html: string; text?: string } } | undefined;\n if (!email?.send) return;\n const rendered = email.renderTemplate('passwordReset', ctx);\n await email.send({ subject: rendered.subject, html: rendered.html, text: rendered.text, to: opts.to });\n },\n }\n : userAuthConfig;\n\n const crudOpts: CrudHandlerOptions = {\n requireAuth: config.requireAuth,\n json: config.json,\n requireEntityPermission: reqEntityPerm,\n getCms,\n };\n const crud = createCrudHandler(dataSource, entityMap, crudOpts);\n const crudById = createCrudByIdHandler(dataSource, entityMap, crudOpts);\n\n const mergePerm = <T extends object>(c: T | undefined): T | undefined =>\n !c ? undefined : reqEntityPerm ? ({ ...c, requireEntityPermission: reqEntityPerm } as T) : c;\n\n const adminRoles =\n getSessionUser &&\n createAdminRolesHandlers({\n dataSource,\n entityMap,\n json: config.json,\n getSessionUser,\n });\n const userAuthRouter = userAuth ? createUserAuthApiRouter(userAuth) : null;\n\n const dashboardGet = dashboard ? createDashboardStatsHandler(mergePerm(dashboard) ?? dashboard) : null;\n const analyticsHandlers = analytics ? createAnalyticsHandlers(analytics) : null;\n const uploadPost = upload ? createUploadHandler(mergePerm(upload) ?? upload) : null;\n const blogBySlugGet = blogBySlug ? createBlogBySlugHandler(blogBySlug) : null;\n const formBySlugGet = formBySlug ? createFormBySlugHandler(formBySlug) : null;\n const formSaveHandlers = formSaveConfig ? createFormSaveHandlers(mergePerm(formSaveConfig) ?? formSaveConfig) : null;\n const formSubmissionPost = formSubmissionConfig ? createFormSubmissionHandler(formSubmissionConfig) : null;\n const formSubmissionGetById = formSubmissionGetByIdConfig\n ? createFormSubmissionGetByIdHandler(mergePerm(formSubmissionGetByIdConfig) ?? formSubmissionGetByIdConfig)\n : null;\n const formSubmissionList = formSubmissionGetByIdConfig\n ? createFormSubmissionListHandler(mergePerm(formSubmissionGetByIdConfig) ?? formSubmissionGetByIdConfig)\n : null;\n const usersApiMerged =\n usersApi && getCms\n ? {\n ...usersApi,\n getCms: usersApi.getCms ?? getCms,\n getCompanyDetails: usersApi.getCompanyDetails ?? config.getCompanyDetails,\n }\n : usersApi;\n const usersHandlers = usersApiMerged ? createUsersApiHandlers(mergePerm(usersApiMerged) ?? usersApiMerged) : null;\n const avatarPost = userAvatar ? createUserAvatarHandler(userAvatar) : null;\n const profilePut = userProfile ? createUserProfileHandler(userProfile) : null;\n const settingsHandlers = settingsConfig ? createSettingsApiHandlers(settingsConfig) : null;\n const smsMessageTemplateHandlers = createSmsMessageTemplateHandlers({\n dataSource,\n entityMap,\n json: config.json,\n requireAuth: config.requireAuth,\n requireEntityPermission: reqEntityPerm,\n });\n const chatHandlers = chatConfig ? createChatHandlers(chatConfig) : null;\n\n function resolveResource(segment: string): string {\n const model = pathToModel(segment);\n return crudResources.includes(model) ? model : segment;\n }\n\n return {\n async handle(method: string, path: string[], req: Request): Promise<Response> {\n const perm = reqEntityPerm;\n async function analyticsGate(): Promise<Response | null> {\n const a = await config.requireAuth(req);\n if (a) return a;\n if (perm) return perm(req, 'analytics', 'read');\n return null;\n }\n\n // Admin roles: [\"admin\", \"roles\"], [\"admin\", \"roles\", id], [\"admin\", \"roles\", id, \"permissions\"]\n if (path[0] === 'admin' && path[1] === 'roles') {\n if (!adminRoles) return config.json({ error: 'Not found' }, { status: 404 });\n if (path.length === 2 && method === 'GET') return adminRoles.list();\n if (path.length === 2 && method === 'POST') return adminRoles.createGroup(req);\n if (path.length === 3 && method === 'PATCH') return adminRoles.patchGroup(req, path[2]!);\n if (path.length === 3 && method === 'DELETE') return adminRoles.deleteGroup(path[2]!);\n if (path.length === 4 && path[3] === 'permissions' && method === 'PUT') return adminRoles.putPermissions(req, path[2]!);\n return config.json({ error: 'Not found' }, { status: 404 });\n }\n\n // Dashboard: [\"dashboard\", \"stats\"]\n if (path[0] === 'dashboard' && path[1] === 'stats' && path.length === 2 && method === 'GET' && dashboardGet) {\n return dashboardGet(req);\n }\n // Analytics: [\"analytics\"], [\"analytics\", \"property-id\"], [\"analytics\", \"permissions\"]\n if (path[0] === 'analytics' && analyticsHandlers) {\n if (path.length === 1 && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.GET(req);\n }\n if (path.length === 2 && path[1] === 'property-id' && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.propertyId();\n }\n if (path.length === 2 && path[1] === 'permissions' && method === 'GET') {\n const g = await analyticsGate();\n if (g) return g;\n return analyticsHandlers.permissions();\n }\n }\n // Upload: [\"upload\"]\n if (path[0] === 'upload' && path.length === 1 && method === 'POST' && uploadPost) return uploadPost(req);\n // Blog by slug: [\"blogs\", \"slug\", slug] (public)\n if (path[0] === 'blogs' && path[1] === 'slug' && path.length === 3 && method === 'GET' && blogBySlugGet) {\n return blogBySlugGet(req, path[2]);\n }\n // Form by slug: [\"forms\", \"slug\", slug] (public)\n if (path[0] === 'forms' && path[1] === 'slug' && path.length === 3 && method === 'GET' && formBySlugGet) {\n return formBySlugGet(req, path[2]);\n }\n // Form submission: [\"form-submissions\"] GET (auth, list with relations) or POST (public); [\"form-submissions\", id] GET (auth, with relations)\n if (path[0] === 'form-submissions') {\n if (path.length === 1) {\n if (method === 'GET' && formSubmissionList) return formSubmissionList(req);\n if (method === 'POST' && formSubmissionPost) return formSubmissionPost(req);\n }\n if (path.length === 2 && method === 'GET' && formSubmissionGetById) return formSubmissionGetById(req, path[1]);\n }\n // Form save: [\"forms\"] POST; [\"forms\", id] GET (with fields) or PUT/PATCH (saves form + fields)\n if (path[0] === 'forms' && formSaveHandlers) {\n if (path.length === 1 && method === 'POST') return formSaveHandlers.POST(req);\n if (path.length === 2) {\n if (method === 'GET') return formSaveHandlers.GET(req, path[1]);\n if (method === 'PUT' || method === 'PATCH') return formSaveHandlers.PUT(req, path[1]);\n }\n }\n // Users API\n if (path[0] === 'users' && usersHandlers) {\n if (path.length === 1) {\n if (method === 'GET') return usersHandlers.list(req);\n if (method === 'POST') return usersHandlers.create(req);\n }\n if (path.length === 2) {\n if (path[1] === 'avatar' && method === 'POST' && avatarPost) return avatarPost(req);\n if (path[1] === 'profile' && method === 'PUT' && profilePut) return profilePut(req);\n // users/:id\n const id = path[1];\n if (method === 'GET') return usersHandlers.getById(req, id);\n if (method === 'PUT' || method === 'PATCH') return usersHandlers.update(req, id);\n if (method === 'DELETE') return usersHandlers.delete(req, id);\n }\n if (path.length === 3 && path[2] === 'regenerate-invite' && method === 'POST') {\n return usersHandlers.regenerateInvite(req, path[1]);\n }\n }\n // User auth: [\"users\", \"forgot-password\"] etc.\n if (path[0] === 'users' && path.length === 2 && userAuthRouter && method === 'POST') {\n return userAuthRouter.POST(req, path[1]);\n }\n // Settings: [\"settings\", group]\n if (path[0] === 'settings' && path.length === 2 && settingsHandlers) {\n const group = path[1]!;\n const isPublic = settingsConfig?.publicGetGroups?.includes(group);\n if (method === 'GET') {\n if (!isPublic && perm) {\n const a = await config.requireAuth(req);\n if (a) return a;\n const pe = await perm(req, 'settings', 'read');\n if (pe) return pe;\n }\n return settingsHandlers.GET(req, group);\n }\n if (method === 'PUT') {\n if (perm) {\n const pe = await perm(req, 'settings', 'update');\n if (pe) return pe;\n }\n return settingsHandlers.PUT(req, group);\n }\n }\n // SMS message templates (plugins page only): [\"message-templates\", \"sms\"]\n if (path[0] === 'message-templates' && path[1] === 'sms' && path.length === 2) {\n if (method === 'GET') return smsMessageTemplateHandlers.GET(req);\n if (method === 'PUT') return smsMessageTemplateHandlers.PUT(req);\n }\n // Chat: [\"chat\", \"identify\"] POST; [\"chat\", \"conversations\", id, \"messages\"] GET; [\"chat\", \"messages\"] POST\n if (path[0] === 'chat' && chatHandlers) {\n if (path.length === 2 && path[1] === 'identify' && method === 'POST') return chatHandlers.identify(req);\n if (path.length === 4 && path[1] === 'conversations' && path[3] === 'messages' && method === 'GET') return chatHandlers.getMessages(req, path[2]);\n if (path.length === 2 && path[1] === 'messages' && method === 'POST') return chatHandlers.postMessage(req);\n }\n\n // Invoice PDF: [\"orders\", id, \"invoice\"] (admin auth + orders:read)\n if (path[0] === 'orders' && path.length === 3 && path[2] === 'invoice' && method === 'GET' && getCms) {\n const a = await config.requireAuth(req);\n if (a) return a;\n if (perm) {\n const pe = await perm(req, 'orders', 'read');\n if (pe) return pe;\n }\n const cms = await getCms();\n const { streamOrderInvoicePdf } = await import('../plugins/erp/erp-order-invoice');\n const oid = Number(path[1]);\n if (!Number.isFinite(oid)) return config.json({ error: 'Invalid id' }, { status: 400 });\n return streamOrderInvoicePdf(cms, dataSource, entityMap, oid, {});\n }\n // ERP repost: [\"orders\", id, \"repost-erp\"] (admin auth + orders:update)\n if (path[0] === 'orders' && path.length === 3 && path[2] === 'repost-erp' && getCms) {\n const a = await config.requireAuth(req);\n if (a) return a;\n if (perm) {\n const pe = await perm(req, 'orders', method === 'GET' ? 'read' : 'update');\n if (pe) return pe;\n }\n const oid = Number(path[1]);\n if (!Number.isFinite(oid)) return config.json({ error: 'Invalid id' }, { status: 400 });\n const cms = await getCms();\n const { isErpIntegrationEnabled } = await import('../plugins/erp/erp-config-enabled');\n const enabled = await isErpIntegrationEnabled(cms, dataSource, entityMap);\n if (method === 'GET') {\n return config.json({ enabled });\n }\n if (method === 'POST') {\n if (!enabled) return config.json({ error: 'ERP integration is disabled' }, { status: 409 });\n const { queueErpPaidOrderForOrderId } = await import('../plugins/erp/paid-order-erp');\n await queueErpPaidOrderForOrderId(cms, dataSource, entityMap, oid);\n return config.json({ ok: true });\n }\n return config.json({ error: 'Method not allowed' }, { status: 405 });\n }\n\n // CRUD: [\"blogs\"] or [\"blogs\", \"123\"]\n if (path.length === 0) return config.json({ error: 'Not found' }, { status: 404 });\n const resource = resolveResource(path[0]);\n if (!crudResources.includes(resource)) return config.json({ error: 'Invalid resource' }, { status: 400 });\n\n // Bulk operations: [\"products\", \"metadata\"], [\"products\", \"bulk\"], [\"products\", \"export\"]\n if (path.length === 2) {\n if (path[1] === 'metadata' && method === 'GET') {\n return crud.GET_METADATA(req, resource);\n }\n if (path[1] === 'bulk' && method === 'POST') {\n return crud.BULK_POST(req, resource);\n }\n if (path[1] === 'export' && method === 'GET') {\n return crud.GET_EXPORT(req, resource);\n }\n }\n\n if (path.length === 1) {\n if (method === 'GET') return crud.GET(req, resource);\n if (method === 'POST') return crud.POST(req, resource);\n return config.json({ error: 'Method not allowed' }, { status: 405 });\n }\n if (path.length === 2) {\n const id = path[1];\n if (method === 'GET') return crudById.GET(req, resource, id);\n if (method === 'PUT' || method === 'PATCH') return crudById.PUT(req, resource, id);\n if (method === 'DELETE') return crudById.DELETE(req, resource, id);\n return config.json({ error: 'Method not allowed' }, { status: 405 });\n }\n return config.json({ error: 'Not found' }, { status: 404 });\n },\n };\n}\n","import type { DataSource } from 'typeorm';\nimport type { EntityTarget, ObjectLiteral } from 'typeorm';\nimport { In, IsNull } from 'typeorm';\nimport type { SessionUser } from '../auth/helpers';\nimport type { EntityMap } from './crud';\nimport { linkUnclaimedContactToUser } from '../lib/link-contact-to-user';\nimport { isValidSignupEmail } from '../lib/is-valid-signup-email';\nimport { queueEmail } from '../plugins/email/email-queue';\nimport { queueErpCreateContactIfEnabled } from '../plugins/erp/erp-contact-sync';\nimport { buildCanonicalOrderNumber, temporaryOrderNumberPlaceholder } from '../lib/order-number';\nimport { tryRefreshOrderFromErpForStorefront } from '../plugins/erp/erp-order-sync';\nimport { streamOrderInvoicePdf } from '../plugins/erp/erp-order-invoice';\nimport type { CompanyDetails } from '../plugins/email/templates/types';\nimport { assertCaptchaOk } from '../plugins/captcha/assert';\nimport { queueSms } from '../plugins/sms/sms-queue';\nimport {\n createOtpChallenge,\n verifyAndConsumeOtpChallenge,\n generateNumericOtp,\n normalizePhoneE164,\n type OtpPurpose,\n} from '../lib/otp-challenge';\nconst GUEST_COOKIE = 'guest_id';\nconst ONE_YEAR = 60 * 60 * 24 * 365;\n\nexport interface StorefrontOtpFlags {\n login?: boolean;\n verifyEmail?: boolean;\n verifyPhone?: boolean;\n}\n\nfunction parseCookies(header: string | null): Record<string, string> {\n const out: Record<string, string> = {};\n if (!header) return out;\n for (const part of header.split(';')) {\n const i = part.indexOf('=');\n if (i === -1) continue;\n const k = part.slice(0, i).trim();\n const v = part.slice(i + 1).trim();\n out[k] = decodeURIComponent(v);\n }\n return out;\n}\n\nfunction guestCookieHeader(name: string, token: string): string {\n return `${name}=${encodeURIComponent(token)}; Path=/; HttpOnly; SameSite=Lax; Max-Age=${ONE_YEAR}`;\n}\n\nexport interface StorefrontApiConfig {\n dataSource: DataSource;\n entityMap: EntityMap;\n json: (body: unknown, init?: { status?: number; headers?: HeadersInit }) => Response;\n getSessionUser: () => Promise<SessionUser | null>;\n guestCookieName?: string;\n /** Required for POST storefront/register */\n hashPassword?: (plain: string) => Promise<string>;\n /** When set, new registrations are blocked until email is verified and a signup email is sent. */\n getCms?: () => Promise<{ getPlugin: (name: string) => unknown }>;\n getCompanyDetails?: () => Promise<CompanyDetails>;\n /** Origin for verify links (e.g. process.env.NEXTAUTH_URL). Required with getCms for verification URLs. */\n publicSiteUrl?: string;\n /** When a flag is true, the corresponding OTP storefront route is enabled. */\n otpFlags?: StorefrontOtpFlags;\n /** Defaults to OTP_PEPPER or NEXTAUTH_SECRET. */\n otpPepper?: string;\n defaultPhoneCountryCode?: string;\n /** When false, login OTP send accepts email only (no phone). Default true. */\n otpAllowPhoneLogin?: boolean;\n}\n\nconst SIGNUP_VERIFY_EXPIRY_HOURS = 72;\n\nexport function createStorefrontApiHandler(config: StorefrontApiConfig) {\n const { dataSource, entityMap, json, getSessionUser, getCms, getCompanyDetails, publicSiteUrl } = config;\n const cookieName = config.guestCookieName ?? GUEST_COOKIE;\n const otpFlags = config.otpFlags;\n const otpPepper = config.otpPepper;\n const defaultPhoneCc = config.defaultPhoneCountryCode;\n const otpAllowPhoneLogin = config.otpAllowPhoneLogin !== false;\n\n function otpOff(key: keyof StorefrontOtpFlags) {\n return !otpFlags || otpFlags[key] !== true;\n }\n\n const cartRepo = () => dataSource.getRepository(entityMap.carts);\n const cartItemRepo = () => dataSource.getRepository(entityMap.cart_items);\n const productRepo = () => dataSource.getRepository(entityMap.products);\n const contactRepo = () => dataSource.getRepository(entityMap.contacts);\n const addressRepo = () => dataSource.getRepository(entityMap.addresses);\n const orderRepo = () => dataSource.getRepository(entityMap.orders);\n const orderItemRepo = () => dataSource.getRepository(entityMap.order_items);\n const wishlistRepo = () => dataSource.getRepository(entityMap.wishlists);\n const wishlistItemRepo = () => dataSource.getRepository(entityMap.wishlist_items);\n const userRepo = () => dataSource.getRepository(entityMap.users);\n const tokenRepo = () => dataSource.getRepository(entityMap.password_reset_tokens);\n const collectionRepo = () => dataSource.getRepository(entityMap.collections);\n const groupRepo = () => dataSource.getRepository(entityMap.user_groups);\n const configRepo = () => dataSource.getRepository(entityMap.configs);\n\n const CART_CHECKOUT_RELATIONS = ['items', 'items.product', 'items.product.taxes', 'items.product.taxes.tax'] as const;\n\n function roundMoney(n: number): number {\n return Math.round(n * 100) / 100;\n }\n\n type CheckoutDraftLine = {\n productId: number;\n quantity: number;\n unitPrice: number;\n tax: number;\n total: number;\n hsn: string | null;\n uom: string | null;\n productType: string | null;\n taxRate: number | null;\n taxCode: string | null;\n };\n\n async function getStoreDefaultTaxRate(): Promise<number | null> {\n const rows = await configRepo().find({ where: { settings: 'store', deleted: false } as ObjectLiteral });\n for (const row of rows) {\n const r = row as { key: string; value: string };\n if (r.key === 'defaultTaxRate') {\n const n = parseFloat(String(r.value ?? '').trim());\n return Number.isFinite(n) && n >= 0 ? n : null;\n }\n }\n return null;\n }\n\n function computeTaxForProductLine(\n p: ObjectLiteral,\n lineSubtotal: number,\n defaultRate: number | null\n ): { tax: number; taxRate: number | null; taxCode: string | null } {\n const pts = (p.taxes as ObjectLiteral[] | undefined) ?? [];\n const activePts = pts.filter((pt) => {\n const t = pt.tax as ObjectLiteral | undefined;\n return t != null && t.active !== false;\n });\n if (activePts.length) {\n let sumRate = 0;\n const slugs: string[] = [];\n for (const pt of activePts) {\n const t = pt.tax as ObjectLiteral;\n const r = Number(pt.rate != null && pt.rate !== '' ? pt.rate : (t.rate ?? 0));\n if (Number.isFinite(r)) sumRate += r;\n const slug = String(t.slug ?? '').trim();\n if (slug) slugs.push(slug);\n }\n const tax = roundMoney((lineSubtotal * sumRate) / 100);\n return {\n tax,\n taxRate: sumRate > 0 ? roundMoney(sumRate) : null,\n taxCode: slugs.length ? [...new Set(slugs)].sort().join(',') : null,\n };\n }\n if (defaultRate != null && defaultRate > 0) {\n return {\n tax: roundMoney((lineSubtotal * defaultRate) / 100),\n taxRate: roundMoney(defaultRate),\n taxCode: null,\n };\n }\n return { tax: 0, taxRate: null, taxCode: null };\n }\n\n function parseInlineAddress(raw: unknown): Record<string, string> | null {\n if (!raw || typeof raw !== 'object' || Array.isArray(raw)) return null;\n const o = raw as Record<string, unknown>;\n const line1 = String(o.line1 ?? '').trim();\n if (!line1) return null;\n return {\n line1,\n line2: o.line2 != null ? String(o.line2) : '',\n city: o.city != null ? String(o.city) : '',\n state: o.state != null ? String(o.state) : '',\n postalCode: o.postalCode != null ? String(o.postalCode) : '',\n country: o.country != null ? String(o.country) : '',\n };\n }\n\n function intFromBody(v: unknown): number | undefined {\n if (typeof v === 'number' && Number.isInteger(v)) return v;\n if (typeof v === 'string' && /^\\d+$/.test(v)) return parseInt(v, 10);\n return undefined;\n }\n\n async function resolveCheckoutAddress(\n contactId: number,\n idVal: unknown,\n inlineVal: unknown\n ): Promise<{ id: number | null; error?: string }> {\n const aid = intFromBody(idVal);\n if (aid != null) {\n const existing = await addressRepo().findOne({\n where: { id: aid, contactId } as ObjectLiteral,\n });\n if (!existing) return { id: null, error: 'Address not found' };\n return { id: aid };\n }\n const addr = parseInlineAddress(inlineVal);\n if (addr) {\n const saved = await addressRepo().save(\n addressRepo().create({\n contactId,\n line1: addr.line1,\n line2: addr.line2?.trim() ? addr.line2 : null,\n city: addr.city?.trim() ? addr.city : null,\n state: addr.state?.trim() ? addr.state : null,\n postalCode: addr.postalCode?.trim() ? addr.postalCode : null,\n country: addr.country?.trim() ? addr.country : null,\n } as ObjectLiteral)\n );\n return { id: (saved as { id: number }).id };\n }\n return { id: null };\n }\n\n async function prepareCheckoutFromCart(\n b: Record<string, unknown>,\n cart: ObjectLiteral,\n contactId: number\n ): Promise<\n | {\n ok: true;\n lines: CheckoutDraftLine[];\n subtotal: number;\n orderTax: number;\n orderTotal: number;\n billingAddressId: number;\n shippingAddressId: number | null;\n }\n | { ok: false; status: number; message: string }\n > {\n const defaultRate = await getStoreDefaultTaxRate();\n const lines: CheckoutDraftLine[] = [];\n let subtotal = 0;\n let orderTax = 0;\n let needsShipping = false;\n\n for (const it of (cart.items as ObjectLiteral[]) || []) {\n const p = it.product as ObjectLiteral | undefined;\n if (!p || p.deleted || p.status !== 'available') continue;\n const unit = Number(p.price);\n const qty = (it.quantity as number) || 1;\n const lineSubtotal = unit * qty;\n const pType = p.type === 'service' ? 'service' : 'product';\n if (pType === 'product') needsShipping = true;\n\n const { tax, taxRate, taxCode } = computeTaxForProductLine(p, lineSubtotal, defaultRate);\n const lineTotal = roundMoney(lineSubtotal + tax);\n subtotal = roundMoney(subtotal + lineSubtotal);\n orderTax = roundMoney(orderTax + tax);\n\n lines.push({\n productId: p.id as number,\n quantity: qty,\n unitPrice: unit,\n tax,\n total: lineTotal,\n hsn: (p.hsn as string | null) ?? null,\n uom: (p.uom as string | null) ?? null,\n productType: pType,\n taxRate,\n taxCode,\n });\n }\n\n if (!lines.length) return { ok: false, status: 400, message: 'No available items in cart' };\n\n const bill = await resolveCheckoutAddress(contactId, b.billingAddressId, b.billingAddress);\n if (bill.error) return { ok: false, status: 400, message: bill.error };\n if (bill.id == null) return { ok: false, status: 400, message: 'Billing address required' };\n\n const ship = await resolveCheckoutAddress(contactId, b.shippingAddressId, b.shippingAddress);\n if (ship.error) return { ok: false, status: 400, message: ship.error };\n\n let shippingAddressId: number | null = ship.id;\n if (needsShipping && shippingAddressId == null) shippingAddressId = bill.id;\n if (needsShipping && shippingAddressId == null) {\n return { ok: false, status: 400, message: 'Shipping address required' };\n }\n\n const orderTotal = roundMoney(subtotal + orderTax);\n\n return {\n ok: true,\n lines,\n subtotal,\n orderTax,\n orderTotal,\n billingAddressId: bill.id,\n shippingAddressId,\n };\n }\n\n async function syncContactToErp(contact: ObjectLiteral): Promise<void> {\n if (!getCms) return;\n try {\n const cms = await getCms();\n await queueErpCreateContactIfEnabled(cms, dataSource, entityMap, {\n name: String((contact as { name: string }).name ?? ''),\n email: String((contact as { email: string }).email ?? '').trim(),\n phone: (contact as { phone: string | null }).phone,\n type: (contact as { type: string | null }).type,\n company: (contact as { company: string | null }).company,\n notes: (contact as { notes: string | null }).notes,\n });\n } catch {\n /* ignore */\n }\n }\n\n async function ensureContactForUser(userId: number): Promise<{ id: number } | null> {\n let c = await contactRepo().findOne({ where: { userId, deleted: false } as ObjectLiteral });\n if (c) return c as { id: number };\n const u = await userRepo().findOne({ where: { id: userId } });\n if (!u) return null;\n const unclaimed = await contactRepo().findOne({\n where: { email: u.email, userId: IsNull(), deleted: false } as ObjectLiteral,\n });\n if (unclaimed) {\n await contactRepo().update((unclaimed as { id: number }).id, { userId });\n return { id: (unclaimed as { id: number }).id };\n }\n const created = await contactRepo().save(\n contactRepo().create({\n name: u.name,\n email: u.email,\n phone: null,\n userId,\n deleted: false,\n } as ObjectLiteral)\n );\n await syncContactToErp(created as ObjectLiteral);\n return { id: (created as { id: number }).id };\n }\n\n async function getOrCreateCart(req: Request): Promise<{ cart: ObjectLiteral; setCookie: string | null; err: Response | null }> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return { cart: {} as ObjectLiteral, setCookie: null, err: json({ error: 'User not found' }, { status: 400 }) };\n let cart = await cartRepo().findOne({\n where: { contactId: contact.id } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ contactId: contact.id, guestToken: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: null, err: null };\n }\n const cookies = parseCookies(req.headers.get('cookie'));\n let token = cookies[cookieName] || '';\n if (!token) {\n token = crypto.randomUUID();\n let cart = await cartRepo().findOne({\n where: { guestToken: token } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ guestToken: token, contactId: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: guestCookieHeader(cookieName, token), err: null };\n }\n let cart = await cartRepo().findOne({\n where: { guestToken: token } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!cart) {\n cart = await cartRepo().save(\n cartRepo().create({ guestToken: token, contactId: null, currency: 'INR' } as ObjectLiteral)\n );\n cart = await cartRepo().findOne({\n where: { id: (cart as { id: number }).id },\n relations: ['items', 'items.product'],\n });\n }\n return { cart: cart!, setCookie: null, err: null };\n }\n\n function primaryProductImageUrl(metadata: unknown): string | null {\n const meta = metadata as { images?: Array<{ url?: string; isDefault?: boolean }> } | null;\n const images = meta?.images;\n if (!Array.isArray(images) || !images.length) return null;\n const sorted = images.filter((i) => i?.url) as Array<{ url: string; isDefault?: boolean }>;\n if (!sorted.length) return null;\n const di = sorted.findIndex((i) => i.isDefault);\n if (di > 0) {\n const [d] = sorted.splice(di, 1);\n sorted.unshift(d!);\n }\n return sorted[0]!.url;\n }\n\n function serializeCart(cart: ObjectLiteral) {\n const items = (cart.items as ObjectLiteral[]) || [];\n return {\n id: cart.id,\n currency: cart.currency,\n items: items.map((it) => {\n const p = it.product as ObjectLiteral | undefined;\n return {\n id: it.id,\n productId: it.productId,\n quantity: it.quantity,\n metadata: it.metadata,\n product: p\n ? {\n id: p.id,\n name: p.name,\n slug: p.slug,\n price: p.price,\n sku: p.sku,\n type: p.type === 'service' ? 'service' : 'product',\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n }),\n };\n }\n\n function serializeSeo(seo: unknown): ObjectLiteral | undefined {\n if (!seo || typeof seo !== 'object') return undefined;\n const s = seo as ObjectLiteral;\n return {\n title: s.title ?? null,\n description: s.description ?? null,\n keywords: s.keywords ?? null,\n ogTitle: s.ogTitle ?? null,\n ogDescription: s.ogDescription ?? null,\n ogImage: s.ogImage ?? null,\n slug: s.slug ?? null,\n };\n }\n\n function serializeProduct(p: ObjectLiteral) {\n const seo = serializeSeo(p.seo);\n return {\n id: p.id,\n name: p.name,\n slug: p.slug,\n sku: p.sku,\n hsn: p.hsn,\n uom: p.uom ?? null,\n type: p.type === 'service' ? 'service' : 'product',\n price: p.price,\n compareAtPrice: p.compareAtPrice,\n status: p.status,\n collectionId: p.collectionId,\n metadata: p.metadata,\n ...(seo ? { seo } : {}),\n };\n }\n\n return {\n async handle(method: string, path: string[], req: Request): Promise<Response> {\n try {\n // --- products GET list ---\n if (path[0] === 'products' && path.length === 1 && method === 'GET') {\n const url = new URL(req.url || '', 'http://localhost');\n const collectionSlug = url.searchParams.get('collection')?.trim();\n const collectionId = url.searchParams.get('collectionId');\n const limit = Math.min(100, Math.max(1, parseInt(url.searchParams.get('limit') || '20', 10)));\n const offset = Math.max(0, parseInt(url.searchParams.get('offset') || '0', 10));\n const where: ObjectLiteral = { status: 'available', deleted: false };\n let collectionFilter: { name: string | null; slug: string | null } | null = null;\n if (collectionSlug) {\n let col: ObjectLiteral | null = null;\n if (/^\\d+$/.test(collectionSlug)) {\n col = (await collectionRepo().findOne({\n where: {\n id: parseInt(collectionSlug, 10),\n active: true,\n deleted: false,\n } as ObjectLiteral,\n })) as ObjectLiteral | null;\n } else {\n col = (await collectionRepo()\n .createQueryBuilder('c')\n .where('LOWER(c.slug) = LOWER(:slug)', { slug: collectionSlug })\n .andWhere('c.active = :a', { a: true })\n .andWhere('c.deleted = :d', { d: false })\n .getOne()) as ObjectLiteral | null;\n }\n if (!col) {\n return json({ products: [], total: 0, collection: null });\n }\n where.collectionId = col.id;\n collectionFilter = { name: col.name as string | null, slug: col.slug as string | null };\n } else if (collectionId) {\n const cid = parseInt(collectionId, 10);\n if (Number.isFinite(cid)) where.collectionId = cid;\n }\n const [items, total] = await productRepo().findAndCount({\n where,\n order: { id: 'ASC' },\n take: limit,\n skip: offset,\n });\n return json({\n products: items.map(serializeProduct),\n total,\n ...(collectionFilter && { collection: collectionFilter }),\n });\n }\n\n // --- products GET by id or slug ---\n if (path[0] === 'products' && path.length === 2 && method === 'GET') {\n const idOrSlug = path[1]!;\n const byId = /^\\d+$/.test(idOrSlug);\n const product = await productRepo().findOne({\n where: byId\n ? { id: parseInt(idOrSlug, 10), status: 'available', deleted: false }\n : { slug: idOrSlug, status: 'available', deleted: false },\n relations: ['attributes', 'attributes.attribute', 'seo'],\n } as ObjectLiteral);\n if (!product) return json({ error: 'Not found' }, { status: 404 });\n const p = product as ObjectLiteral;\n const attrRows = (p.attributes as ObjectLiteral[] | undefined) ?? [];\n const attributeTags = attrRows.map((pa) => ({\n name: ((pa.attribute as ObjectLiteral)?.name as string) ?? '',\n value: String(pa.value ?? ''),\n })).filter((t) => t.name || t.value);\n return json({ ...serializeProduct(p), attributes: attributeTags });\n }\n\n // --- collections GET list ---\n if (path[0] === 'collections' && path.length === 1 && method === 'GET') {\n const items = await collectionRepo().find({\n where: { active: true, deleted: false } as ObjectLiteral,\n order: { sortOrder: 'ASC', id: 'ASC' },\n });\n const ids = items.map((c) => (c as ObjectLiteral).id as number);\n const countByCollection: Record<number, number> = {};\n if (ids.length > 0) {\n const rows = await productRepo()\n .createQueryBuilder('p')\n .select('p.collectionId', 'collectionId')\n .addSelect('COUNT(p.id)', 'cnt')\n .where('p.collectionId IN (:...ids)', { ids })\n .andWhere('p.status = :status', { status: 'available' })\n .andWhere('p.deleted = :del', { del: false })\n .groupBy('p.collectionId')\n .getRawMany();\n for (const r of rows) {\n const cid = (r as ObjectLiteral).collectionId;\n if (cid != null) countByCollection[Number(cid)] = parseInt(String((r as ObjectLiteral).cnt), 10);\n }\n }\n return json({\n collections: items.map((c) => {\n const col = c as ObjectLiteral;\n const id = col.id as number;\n return {\n id,\n name: col.name,\n slug: col.slug,\n description: col.description,\n image: col.image,\n productCount: countByCollection[id] ?? 0,\n };\n }),\n });\n }\n\n // --- collections GET by id or slug ---\n if (path[0] === 'collections' && path.length === 2 && method === 'GET') {\n const idOrSlug = path[1]!;\n const byId = /^\\d+$/.test(idOrSlug);\n const collection = await collectionRepo().findOne({\n where: byId\n ? { id: parseInt(idOrSlug, 10), active: true, deleted: false }\n : { slug: idOrSlug, active: true, deleted: false },\n relations: ['seo'],\n } as ObjectLiteral);\n if (!collection) return json({ error: 'Not found' }, { status: 404 });\n const col = collection as ObjectLiteral;\n const products = await productRepo().find({\n where: { collectionId: col.id, status: 'available', deleted: false } as ObjectLiteral,\n order: { id: 'ASC' },\n });\n const colSeo = serializeSeo(col.seo);\n return json({\n id: col.id,\n name: col.name,\n slug: col.slug,\n description: col.description,\n image: col.image,\n ...(colSeo ? { seo: colSeo } : {}),\n products: products.map((p) => serializeProduct(p as ObjectLiteral)),\n });\n }\n\n // --- profile GET (auth) ---\n if (path[0] === 'profile' && path.length === 1 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const user = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n if (!user) return json({ error: 'Not found' }, { status: 404 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n return json({\n user: { id: (user as ObjectLiteral).id, name: (user as ObjectLiteral).name, email: (user as ObjectLiteral).email },\n contact: contact\n ? {\n id: (contact as ObjectLiteral).id,\n name: (contact as ObjectLiteral).name,\n email: (contact as ObjectLiteral).email,\n phone: (contact as ObjectLiteral).phone,\n }\n : null,\n });\n }\n\n // --- profile PUT (auth) ---\n if (path[0] === 'profile' && path.length === 1 && method === 'PUT') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const b = (await req.json().catch(() => ({}))) as { name?: string; phone?: string };\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (contact) {\n const updates: ObjectLiteral = {};\n if (typeof b.name === 'string' && b.name.trim()) updates.name = b.name.trim();\n if (b.phone !== undefined) updates.phone = b.phone === null || b.phone === '' ? null : String(b.phone);\n if (Object.keys(updates).length) await contactRepo().update((contact as { id: number }).id, updates);\n }\n const user = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n if (user && typeof b.name === 'string' && b.name.trim()) {\n await userRepo().update(uid, { name: b.name.trim() });\n }\n const updatedContact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (updatedContact) await syncContactToErp(updatedContact as ObjectLiteral);\n const updatedUser = await userRepo().findOne({ where: { id: uid }, select: ['id', 'name', 'email'] });\n return json({\n user: updatedUser ? { id: (updatedUser as ObjectLiteral).id, name: (updatedUser as ObjectLiteral).name, email: (updatedUser as ObjectLiteral).email } : null,\n contact: updatedContact\n ? {\n id: (updatedContact as ObjectLiteral).id,\n name: (updatedContact as ObjectLiteral).name,\n email: (updatedContact as ObjectLiteral).email,\n phone: (updatedContact as ObjectLiteral).phone,\n }\n : null,\n });\n }\n\n // --- addresses (auth: contact's addresses) ---\n async function getContactForAddresses(): Promise<{ contactId: number } | Response> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Contact not found' }, { status: 404 });\n return { contactId: (contact as { id: number }).id };\n }\n function serializeAddress(a: ObjectLiteral) {\n return {\n id: a.id,\n contactId: a.contactId,\n tag: a.tag,\n line1: a.line1,\n line2: a.line2,\n city: a.city,\n state: a.state,\n postalCode: a.postalCode,\n country: a.country,\n };\n }\n if (path[0] === 'addresses' && path.length === 1 && method === 'GET') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const list = await addressRepo().find({\n where: { contactId: contactOrErr.contactId },\n order: { id: 'ASC' },\n });\n return json({ addresses: list.map((a) => serializeAddress(a as ObjectLiteral)) });\n }\n if (path[0] === 'addresses' && path.length === 1 && method === 'POST') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const created = await addressRepo().save(\n addressRepo().create({\n contactId: contactOrErr.contactId,\n tag: typeof b.tag === 'string' ? b.tag.trim() || null : null,\n line1: typeof b.line1 === 'string' ? b.line1.trim() || null : null,\n line2: typeof b.line2 === 'string' ? b.line2.trim() || null : null,\n city: typeof b.city === 'string' ? b.city.trim() || null : null,\n state: typeof b.state === 'string' ? b.state.trim() || null : null,\n postalCode: typeof b.postalCode === 'string' ? b.postalCode.trim() || null : null,\n country: typeof b.country === 'string' ? b.country.trim() || null : null,\n } as ObjectLiteral)\n );\n return json(serializeAddress(created as ObjectLiteral));\n }\n if (path[0] === 'addresses' && path.length === 2 && (method === 'PATCH' || method === 'PUT')) {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const id = parseInt(path[1]!, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const existing = await addressRepo().findOne({ where: { id, contactId: contactOrErr.contactId } as ObjectLiteral });\n if (!existing) return json({ error: 'Not found' }, { status: 404 });\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const updates: ObjectLiteral = {};\n if (b.tag !== undefined) updates.tag = typeof b.tag === 'string' ? b.tag.trim() || null : null;\n if (b.line1 !== undefined) updates.line1 = typeof b.line1 === 'string' ? b.line1.trim() || null : null;\n if (b.line2 !== undefined) updates.line2 = typeof b.line2 === 'string' ? b.line2.trim() || null : null;\n if (b.city !== undefined) updates.city = typeof b.city === 'string' ? b.city.trim() || null : null;\n if (b.state !== undefined) updates.state = typeof b.state === 'string' ? b.state.trim() || null : null;\n if (b.postalCode !== undefined) updates.postalCode = typeof b.postalCode === 'string' ? b.postalCode.trim() || null : null;\n if (b.country !== undefined) updates.country = typeof b.country === 'string' ? b.country.trim() || null : null;\n if (Object.keys(updates).length) await addressRepo().update(id, updates);\n const updated = await addressRepo().findOne({ where: { id } as ObjectLiteral });\n return json(serializeAddress(updated as ObjectLiteral));\n }\n if (path[0] === 'addresses' && path.length === 2 && method === 'DELETE') {\n const contactOrErr = await getContactForAddresses();\n if (contactOrErr instanceof Response) return contactOrErr;\n const id = parseInt(path[1]!, 10);\n if (!Number.isFinite(id)) return json({ error: 'Invalid id' }, { status: 400 });\n const existing = await addressRepo().findOne({ where: { id, contactId: contactOrErr.contactId } as ObjectLiteral });\n if (!existing) return json({ error: 'Not found' }, { status: 404 });\n await addressRepo().delete(id);\n return json({ deleted: true });\n }\n\n // --- verify-email POST (public) ---\n if (path[0] === 'verify-email' && path.length === 1 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as { token?: string };\n const token = typeof b.token === 'string' ? b.token.trim() : '';\n if (!token) return json({ error: 'token is required' }, { status: 400 });\n const record = await tokenRepo().findOne({ where: { token } as ObjectLiteral });\n if (!record || (record as { expiresAt: Date }).expiresAt < new Date()) {\n return json({ error: 'Invalid or expired link. Please sign up again or contact support.' }, { status: 400 });\n }\n const email = (record as { email: string }).email;\n const user = await userRepo().findOne({ where: { email }, select: ['id', 'blocked'] });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n await userRepo().update((user as { id: number }).id, {\n blocked: false,\n emailVerifiedAt: new Date(),\n updatedAt: new Date(),\n } as ObjectLiteral);\n await tokenRepo().delete({ email } as ObjectLiteral);\n return json({ success: true, message: 'Email verified. You can sign in.' });\n }\n\n // --- auth/otp/send POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'send' && path.length === 3 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const purposeRaw = typeof b.purpose === 'string' ? b.purpose.trim() : '';\n const purpose = (purposeRaw === 'login' || purposeRaw === 'verify_email' || purposeRaw === 'verify_phone'\n ? purposeRaw\n : '') as OtpPurpose | '';\n if (!purpose) return json({ error: 'purpose must be login, verify_email, or verify_phone' }, { status: 400 });\n if (purpose === 'login' && otpOff('login')) return json({ error: 'otp_disabled' }, { status: 403 });\n if (purpose === 'verify_email' && otpOff('verifyEmail')) return json({ error: 'otp_disabled' }, { status: 403 });\n if (purpose === 'verify_phone' && otpOff('verifyPhone')) return json({ error: 'otp_disabled' }, { status: 403 });\n\n const capOtp = await assertCaptchaOk(getCms, b, req, json);\n if (capOtp) return capOtp;\n\n const emailIn = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const phoneIn = typeof b.phone === 'string' ? b.phone.trim() : '';\n let identifier: string;\n let channel: 'email' | 'sms';\n\n if (purpose === 'login') {\n if (emailIn) {\n identifier = emailIn;\n channel = 'email';\n } else if (phoneIn) {\n if (!otpAllowPhoneLogin) {\n return json({ error: 'Phone sign-in is not enabled' }, { status: 403 });\n }\n const p = normalizePhoneE164(phoneIn, defaultPhoneCc);\n if (!p) return json({ error: 'Invalid phone' }, { status: 400 });\n identifier = p;\n channel = 'sms';\n } else {\n return json({ error: 'email or phone required' }, { status: 400 });\n }\n const user =\n channel === 'email'\n ? await userRepo().findOne({ where: { email: identifier } as ObjectLiteral })\n : await userRepo().findOne({ where: { phone: identifier } as ObjectLiteral });\n if (!user || (user as { deleted?: boolean }).deleted || (user as { blocked?: boolean }).blocked) {\n return json({ ok: true });\n }\n } else if (purpose === 'verify_email') {\n if (!emailIn || !isValidSignupEmail(emailIn)) return json({ error: 'Valid email required' }, { status: 400 });\n identifier = emailIn;\n channel = 'email';\n const user = await userRepo().findOne({ where: { email: identifier } as ObjectLiteral });\n if (!user || (user as { deleted?: boolean }).deleted) return json({ ok: true });\n } else {\n const su = await getSessionUser();\n const uid = su?.id ? parseInt(String(su.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const p = normalizePhoneE164(phoneIn, defaultPhoneCc);\n if (!p) return json({ error: 'Valid phone required' }, { status: 400 });\n identifier = p;\n channel = 'sms';\n const taken = await userRepo().findOne({\n where: { phone: identifier } as ObjectLiteral,\n select: ['id'],\n });\n if (taken && (taken as { id: number }).id !== uid) {\n return json({ error: 'Phone already in use' }, { status: 400 });\n }\n }\n\n const code = generateNumericOtp(6);\n const created = await createOtpChallenge(dataSource, entityMap, {\n purpose,\n channel,\n identifier,\n code,\n pepper: otpPepper,\n });\n if (!created.ok) return json({ error: created.error }, { status: created.status });\n\n if (!getCms) return json({ error: 'OTP delivery not configured' }, { status: 503 });\n\n try {\n const cms = await getCms();\n if (channel === 'email') {\n if (!cms.getPlugin('email')) return json({ error: 'Email not configured' }, { status: 503 });\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n await queueEmail(cms, {\n to: identifier,\n templateName: 'otp',\n ctx: { code, companyDetails: companyDetails ?? {} },\n });\n } else {\n if (!cms.getPlugin('sms')) return json({ error: 'SMS not configured' }, { status: 503 });\n const templateKey = purpose === 'verify_phone' ? 'auth.otp_verify_phone' : 'auth.otp_login';\n await queueSms(cms, { to: identifier, templateKey, variables: { code } });\n }\n } catch {\n return json({ error: 'Failed to send code' }, { status: 500 });\n }\n return json({ ok: true });\n }\n\n // --- auth/otp/verify-email POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'verify-email' && path.length === 3 && method === 'POST') {\n if (otpOff('verifyEmail')) return json({ error: 'otp_disabled' }, { status: 403 });\n const b = (await req.json().catch(() => ({}))) as { email?: string; code?: string };\n const email = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const code = typeof b.code === 'string' ? b.code.trim() : '';\n if (!email || !code) return json({ error: 'email and code required' }, { status: 400 });\n const v = await verifyAndConsumeOtpChallenge(dataSource, entityMap, {\n purpose: 'verify_email',\n identifier: email,\n code,\n pepper: otpPepper,\n });\n if (!v.ok) return json({ error: v.error }, { status: v.status });\n const user = await userRepo().findOne({ where: { email } as ObjectLiteral });\n if (!user) return json({ error: 'User not found' }, { status: 400 });\n await userRepo().update((user as { id: number }).id, {\n blocked: false,\n emailVerifiedAt: new Date(),\n updatedAt: new Date(),\n } as ObjectLiteral);\n await tokenRepo().delete({ email } as ObjectLiteral);\n return json({ success: true, message: 'Email verified. You can sign in.' });\n }\n\n // --- auth/otp/verify-phone POST ---\n if (path[0] === 'auth' && path[1] === 'otp' && path[2] === 'verify-phone' && path.length === 3 && method === 'POST') {\n if (otpOff('verifyPhone')) return json({ error: 'otp_disabled' }, { status: 403 });\n const su = await getSessionUser();\n const uid = su?.id ? parseInt(String(su.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const b = (await req.json().catch(() => ({}))) as { phone?: string; code?: string };\n const phoneRaw = typeof b.phone === 'string' ? b.phone.trim() : '';\n const code = typeof b.code === 'string' ? b.code.trim() : '';\n const phone = normalizePhoneE164(phoneRaw, defaultPhoneCc);\n if (!phone || !code) return json({ error: 'phone and code required' }, { status: 400 });\n const v = await verifyAndConsumeOtpChallenge(dataSource, entityMap, {\n purpose: 'verify_phone',\n identifier: phone,\n code,\n pepper: otpPepper,\n });\n if (!v.ok) return json({ error: v.error }, { status: v.status });\n const taken = await userRepo().findOne({ where: { phone } as ObjectLiteral, select: ['id'] });\n if (taken && (taken as { id: number }).id !== uid) {\n return json({ error: 'Phone already in use' }, { status: 400 });\n }\n await userRepo().update(uid, { phone, phoneVerifiedAt: new Date(), updatedAt: new Date() } as ObjectLiteral);\n const contact = await ensureContactForUser(uid);\n if (contact) {\n await contactRepo().update(contact.id, { phone } as ObjectLiteral);\n }\n return json({ success: true });\n }\n\n // --- register POST ---\n if (path[0] === 'register' && path.length === 1 && method === 'POST') {\n if (!config.hashPassword) return json({ error: 'Registration not configured' }, { status: 501 });\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capReg = await assertCaptchaOk(getCms, b, req, json);\n if (capReg) return capReg;\n const name = typeof b.name === 'string' ? b.name.trim() : '';\n const email = typeof b.email === 'string' ? b.email.trim().toLowerCase() : '';\n const password = typeof b.password === 'string' ? b.password : '';\n if (!name || !email || !password) return json({ error: 'name, email and password are required' }, { status: 400 });\n if (!isValidSignupEmail(email)) return json({ error: 'Invalid email address' }, { status: 400 });\n const existing = await userRepo().findOne({ where: { email } });\n if (existing) return json({ error: 'User with this email already exists' }, { status: 400 });\n const customerG = await groupRepo().findOne({ where: { name: 'Customer', deleted: false } as ObjectLiteral });\n const groupId = customerG ? (customerG as { id: number }).id : null;\n const hashed = await config.hashPassword(password);\n const requireEmailVerification = Boolean(getCms);\n const newUser = await userRepo().save(\n userRepo().create({\n name,\n email,\n password: hashed,\n blocked: requireEmailVerification,\n groupId,\n adminAccess: false,\n } as ObjectLiteral)\n );\n const userId = (newUser as { id: number }).id;\n await linkUnclaimedContactToUser(dataSource, entityMap.contacts, userId, email);\n\n let emailVerificationSent = false;\n if (requireEmailVerification && getCms) {\n try {\n const crypto = await import('crypto');\n const rawToken = crypto.randomBytes(32).toString('hex');\n const expiresAt = new Date(Date.now() + SIGNUP_VERIFY_EXPIRY_HOURS * 60 * 60 * 1000);\n await tokenRepo().save(\n tokenRepo().create({ email, token: rawToken, expiresAt } as ObjectLiteral)\n );\n const cms = await getCms();\n const companyDetails = getCompanyDetails ? await getCompanyDetails() : {};\n const base = (publicSiteUrl || '').replace(/\\/$/, '').trim() || 'http://localhost:3000';\n const verifyEmailUrl = `${base}/verify-email?token=${encodeURIComponent(rawToken)}`;\n await queueEmail(cms, {\n to: email,\n templateName: 'signup',\n ctx: { name, verifyEmailUrl, companyDetails: companyDetails ?? {} },\n });\n emailVerificationSent = true;\n } catch {\n await userRepo().update(userId, { blocked: false, updatedAt: new Date() } as ObjectLiteral);\n }\n }\n\n return json({\n success: true,\n userId,\n emailVerificationSent,\n });\n }\n\n // --- cart GET ---\n if (path[0] === 'cart' && path.length === 1 && method === 'GET') {\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const body = serializeCart(cart);\n if (setCookie) return json(body, { headers: { 'Set-Cookie': setCookie } });\n return json(body);\n }\n\n // --- cart/items POST ---\n if (path[0] === 'cart' && path[1] === 'items' && path.length === 2 && method === 'POST') {\n const body = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capCart = await assertCaptchaOk(getCms, body, req, json);\n if (capCart) return capCart;\n const productId = Number(body.productId);\n const quantity = Math.max(1, Number(body.quantity) || 1);\n if (!Number.isFinite(productId)) return json({ error: 'productId required' }, { status: 400 });\n const product = await productRepo().findOne({ where: { id: productId, deleted: false } as ObjectLiteral });\n if (!product) return json({ error: 'Product not found' }, { status: 404 });\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const cartId = (cart as { id: number }).id;\n const existing = await cartItemRepo().findOne({ where: { cartId, productId } as ObjectLiteral });\n if (existing) {\n await cartItemRepo().update((existing as { id: number }).id, {\n quantity: (existing as { quantity: number }).quantity + quantity,\n });\n } else {\n await cartItemRepo().save(\n cartItemRepo().create({ cartId, productId, quantity } as ObjectLiteral)\n );\n }\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n\n // --- cart/items/:id PATCH DELETE ---\n if (path[0] === 'cart' && path[1] === 'items' && path.length === 3) {\n const itemId = parseInt(path[2]!, 10);\n if (!Number.isFinite(itemId)) return json({ error: 'Invalid item id' }, { status: 400 });\n const { cart, setCookie, err } = await getOrCreateCart(req);\n if (err) return err;\n const cartId = (cart as { id: number }).id;\n const item = await cartItemRepo().findOne({ where: { id: itemId, cartId } as ObjectLiteral });\n if (!item) return json({ error: 'Not found' }, { status: 404 });\n if (method === 'DELETE') {\n await cartItemRepo().delete(itemId);\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n if (method === 'PATCH') {\n const b = (await req.json().catch(() => ({}))) as { quantity?: number };\n const q = Math.max(0, Number(b.quantity) || 0);\n if (q === 0) await cartItemRepo().delete(itemId);\n else await cartItemRepo().update(itemId, { quantity: q });\n await cartRepo().update(cartId, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: cartId },\n relations: ['items', 'items.product'],\n });\n const out = serializeCart(fresh!);\n if (setCookie) return json(out, { headers: { 'Set-Cookie': setCookie } });\n return json(out);\n }\n }\n\n // --- cart/merge POST (auth) ---\n if (path[0] === 'cart' && path[1] === 'merge' && method === 'POST') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact not found' }, { status: 400 });\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ merged: false, message: 'No guest cart' });\n const guestCart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: ['items'],\n });\n if (!guestCart || !((guestCart as { items?: unknown[] }).items || []).length) {\n let uc = await cartRepo().findOne({\n where: { contactId: contact.id } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!uc) uc = { items: [] } as ObjectLiteral;\n return json(\n { merged: false, cart: serializeCart(uc) },\n { headers: { 'Set-Cookie': `${cookieName}=; Path=/; Max-Age=0` } }\n );\n }\n let userCart = await cartRepo().findOne({ where: { contactId: contact.id } as ObjectLiteral });\n if (!userCart) {\n userCart = await cartRepo().save(\n cartRepo().create({ contactId: contact.id, guestToken: null, currency: (guestCart as { currency: string }).currency } as ObjectLiteral)\n );\n }\n const uidCart = (userCart as { id: number }).id;\n const gItems = ((guestCart as { items: ObjectLiteral[] }).items) || [];\n for (const gi of gItems) {\n const existing = await cartItemRepo().findOne({\n where: { cartId: uidCart, productId: (gi as { productId: number }).productId } as ObjectLiteral,\n });\n if (existing) {\n await cartItemRepo().update((existing as { id: number }).id, {\n quantity: (existing as { quantity: number }).quantity + (gi as { quantity: number }).quantity,\n });\n } else {\n await cartItemRepo().save(\n cartItemRepo().create({\n cartId: uidCart,\n productId: (gi as { productId: number }).productId,\n quantity: (gi as { quantity: number }).quantity,\n metadata: (gi as { metadata: unknown }).metadata,\n } as ObjectLiteral)\n );\n }\n }\n await cartRepo().delete((guestCart as { id: number }).id);\n await cartRepo().update(uidCart, { updatedAt: new Date() });\n const fresh = await cartRepo().findOne({\n where: { id: uidCart },\n relations: ['items', 'items.product'],\n });\n // Merge guest wishlist into contact wishlist (same guestId cookie)\n const guestWishlist = await wishlistRepo().findOne({\n where: { guestId: guestToken } as ObjectLiteral,\n relations: ['items'],\n });\n if (guestWishlist && ((guestWishlist as { items?: unknown[] }).items || []).length > 0) {\n const userWishlist = await getDefaultWishlist(contact.id);\n const gItems = ((guestWishlist as { items: ObjectLiteral[] }).items) || [];\n for (const gi of gItems) {\n const pid = (gi as { productId: number }).productId;\n const ex = await wishlistItemRepo().findOne({ where: { wishlistId: userWishlist.id, productId: pid } as ObjectLiteral });\n if (!ex) await wishlistItemRepo().save(wishlistItemRepo().create({ wishlistId: userWishlist.id, productId: pid } as ObjectLiteral));\n }\n await wishlistRepo().delete((guestWishlist as { id: number }).id);\n }\n return json({ merged: true, cart: serializeCart(fresh!) }, { headers: { 'Set-Cookie': `${cookieName}=; Path=/; Max-Age=0` } });\n }\n\n // --- wishlist (guest or auth: same guestId cookie as cart) ---\n async function getDefaultWishlist(contactId: number) {\n let w = await wishlistRepo().findOne({ where: { contactId, name: 'default' } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ contactId, guestId: null, name: 'default' } as ObjectLiteral));\n }\n return w as { id: number };\n }\n\n async function getOrCreateWishlist(req: Request): Promise<{ wishlist: ObjectLiteral; setCookie: string | null; err: Response | null }> {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return { wishlist: {} as ObjectLiteral, setCookie: null, err: json({ error: 'User not found' }, { status: 400 }) };\n const w = await getDefaultWishlist(contact.id);\n const wishlist = await wishlistRepo().findOne({ where: { id: w.id } as ObjectLiteral });\n return { wishlist: wishlist!, setCookie: null, err: null };\n }\n const cookies = parseCookies(req.headers.get('cookie'));\n let token = cookies[cookieName] || '';\n if (!token) {\n token = crypto.randomUUID();\n let w = await wishlistRepo().findOne({ where: { guestId: token } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ guestId: token, contactId: null, name: 'default' } as ObjectLiteral));\n }\n return { wishlist: w!, setCookie: guestCookieHeader(cookieName, token), err: null };\n }\n let w = await wishlistRepo().findOne({ where: { guestId: token } as ObjectLiteral });\n if (!w) {\n w = await wishlistRepo().save(wishlistRepo().create({ guestId: token, contactId: null, name: 'default' } as ObjectLiteral));\n }\n return { wishlist: w!, setCookie: null, err: null };\n }\n\n if (path[0] === 'wishlist' && path.length === 1 && method === 'GET') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const items = await wishlistItemRepo().find({\n where: { wishlistId: (wishlist as { id: number }).id } as ObjectLiteral,\n relations: ['product'],\n });\n const body = {\n wishlistId: (wishlist as { id: number }).id,\n items: items.map((it) => {\n const p = (it as ObjectLiteral).product as ObjectLiteral | undefined;\n return {\n id: (it as ObjectLiteral).id,\n productId: (it as ObjectLiteral).productId,\n product: p\n ? {\n id: p.id,\n name: p.name,\n slug: p.slug,\n price: p.price,\n sku: p.sku,\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n }),\n };\n if (setCookie) return json(body, { headers: { 'Set-Cookie': setCookie } });\n return json(body);\n }\n\n if (path[0] === 'wishlist' && path[1] === 'items' && path.length === 2 && method === 'POST') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capWl = await assertCaptchaOk(getCms, b, req, json);\n if (capWl) return capWl;\n const productId = Number(b.productId);\n if (!Number.isFinite(productId)) return json({ error: 'productId required' }, { status: 400 });\n const wid = (wishlist as { id: number }).id;\n const ex = await wishlistItemRepo().findOne({ where: { wishlistId: wid, productId } as ObjectLiteral });\n if (!ex) await wishlistItemRepo().save(wishlistItemRepo().create({ wishlistId: wid, productId } as ObjectLiteral));\n if (setCookie) return json({ ok: true }, { headers: { 'Set-Cookie': setCookie } });\n return json({ ok: true });\n }\n\n if (path[0] === 'wishlist' && path[1] === 'items' && path.length === 3 && method === 'DELETE') {\n const { wishlist, setCookie, err } = await getOrCreateWishlist(req);\n if (err) return err;\n const productId = parseInt(path[2]!, 10);\n await wishlistItemRepo().delete({ wishlistId: (wishlist as { id: number }).id, productId } as ObjectLiteral);\n if (setCookie) return json({ ok: true }, { headers: { 'Set-Cookie': setCookie } });\n return json({ ok: true });\n }\n\n // --- checkout/order POST (create order only, for Razorpay; do not clear cart) ---\n if (path[0] === 'checkout' && path[1] === 'order' && path.length === 2 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capOrd = await assertCaptchaOk(getCms, b, req, json);\n if (capOrd) return capOrd;\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n let contactId: number;\n let cart: ObjectLiteral | null;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact required' }, { status: 400 });\n contactId = contact.id;\n cart = await cartRepo().findOne({\n where: { contactId } as ObjectLiteral,\n relations: [...CART_CHECKOUT_RELATIONS],\n });\n } else {\n const email = String(b.email ?? '').trim();\n const name = String(b.name ?? '').trim();\n if (!email || !name) return json({ error: 'name and email required for guest checkout' }, { status: 400 });\n let contact = await contactRepo().findOne({ where: { email, deleted: false } as ObjectLiteral });\n if (contact && (contact as { userId: number | null }).userId != null) {\n return json({ error: 'Please sign in to complete checkout' }, { status: 400 });\n }\n if (!contact) {\n contact = await contactRepo().save(\n contactRepo().create({\n name,\n email,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : null,\n userId: null,\n deleted: false,\n } as ObjectLiteral)\n );\n } else if (name)\n await contactRepo().update((contact as { id: number }).id, {\n name,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : (contact as { phone: string | null }).phone,\n });\n contactId = (contact as { id: number }).id;\n const guestForErp = await contactRepo().findOne({ where: { id: contactId } as ObjectLiteral });\n if (guestForErp) await syncContactToErp(guestForErp);\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ error: 'Cart not found' }, { status: 400 });\n cart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: [...CART_CHECKOUT_RELATIONS],\n });\n }\n if (!cart || !((cart.items as ObjectLiteral[]) || []).length) {\n return json({ error: 'Cart is empty' }, { status: 400 });\n }\n const prepOrd = await prepareCheckoutFromCart(b, cart, contactId);\n if (!prepOrd.ok) return json({ error: prepOrd.message }, { status: prepOrd.status });\n const cartId = (cart as { id: number }).id;\n const ord = await orderRepo().save(\n orderRepo().create({\n orderNumber: temporaryOrderNumberPlaceholder(),\n orderKind: 'sale',\n parentOrderId: null,\n contactId,\n billingAddressId: prepOrd.billingAddressId,\n shippingAddressId: prepOrd.shippingAddressId,\n status: 'pending',\n subtotal: prepOrd.subtotal,\n tax: prepOrd.orderTax,\n discount: 0,\n total: prepOrd.orderTotal,\n currency: (cart.currency as string) || 'INR',\n metadata: { cartId },\n } as ObjectLiteral)\n );\n const oid = (ord as { id: number }).id;\n await orderRepo().update(oid, {\n orderNumber: buildCanonicalOrderNumber('sale', oid, (ord as { createdAt?: Date }).createdAt ?? new Date()),\n } as ObjectLiteral);\n for (const line of prepOrd.lines) {\n await orderItemRepo().save(\n orderItemRepo().create({\n orderId: oid,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n hsn: line.hsn,\n uom: line.uom,\n productType: line.productType,\n taxRate: line.taxRate,\n taxCode: line.taxCode,\n } as ObjectLiteral)\n );\n }\n return json({\n orderId: oid,\n orderNumber: (ord as { orderNumber: string }).orderNumber,\n subtotal: prepOrd.subtotal,\n tax: prepOrd.orderTax,\n total: prepOrd.orderTotal,\n currency: (cart.currency as string) || 'INR',\n });\n }\n\n // --- checkout POST ---\n if (path[0] === 'checkout' && path.length === 1 && method === 'POST') {\n const b = (await req.json().catch(() => ({}))) as Record<string, unknown>;\n const capChk = await assertCaptchaOk(getCms, b, req, json);\n if (capChk) return capChk;\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n let contactId: number;\n let cart: ObjectLiteral | null;\n if (Number.isFinite(uid)) {\n const contact = await ensureContactForUser(uid);\n if (!contact) return json({ error: 'Contact required' }, { status: 400 });\n contactId = contact.id;\n cart = await cartRepo().findOne({\n where: { contactId } as ObjectLiteral,\n relations: [...CART_CHECKOUT_RELATIONS],\n });\n } else {\n const email = String(b.email ?? '').trim();\n const name = String(b.name ?? '').trim();\n if (!email || !name) return json({ error: 'name and email required for guest checkout' }, { status: 400 });\n let contact = await contactRepo().findOne({ where: { email, deleted: false } as ObjectLiteral });\n if (contact && (contact as { userId: number | null }).userId != null) {\n return json({ error: 'Please sign in to complete checkout' }, { status: 400 });\n }\n if (!contact) {\n contact = await contactRepo().save(\n contactRepo().create({\n name,\n email,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : null,\n userId: null,\n deleted: false,\n } as ObjectLiteral)\n );\n } else if (name)\n await contactRepo().update((contact as { id: number }).id, {\n name,\n phone: b.phone != null && b.phone !== '' ? String(b.phone) : (contact as { phone: string | null }).phone,\n });\n contactId = (contact as { id: number }).id;\n const guestForErp2 = await contactRepo().findOne({ where: { id: contactId } as ObjectLiteral });\n if (guestForErp2) await syncContactToErp(guestForErp2);\n const cookies = parseCookies(req.headers.get('cookie'));\n const guestToken = cookies[cookieName];\n if (!guestToken) return json({ error: 'Cart not found' }, { status: 400 });\n cart = await cartRepo().findOne({\n where: { guestToken } as ObjectLiteral,\n relations: [...CART_CHECKOUT_RELATIONS],\n });\n }\n if (!cart || !((cart.items as ObjectLiteral[]) || []).length) {\n return json({ error: 'Cart is empty' }, { status: 400 });\n }\n const prepChk = await prepareCheckoutFromCart(b, cart, contactId);\n if (!prepChk.ok) return json({ error: prepChk.message }, { status: prepChk.status });\n const ord = await orderRepo().save(\n orderRepo().create({\n orderNumber: temporaryOrderNumberPlaceholder(),\n orderKind: 'sale',\n parentOrderId: null,\n contactId,\n billingAddressId: prepChk.billingAddressId,\n shippingAddressId: prepChk.shippingAddressId,\n status: 'pending',\n subtotal: prepChk.subtotal,\n tax: prepChk.orderTax,\n discount: 0,\n total: prepChk.orderTotal,\n currency: (cart.currency as string) || 'INR',\n } as ObjectLiteral)\n );\n const oid = (ord as { id: number }).id;\n await orderRepo().update(oid, {\n orderNumber: buildCanonicalOrderNumber('sale', oid, (ord as { createdAt?: Date }).createdAt ?? new Date()),\n } as ObjectLiteral);\n for (const line of prepChk.lines) {\n await orderItemRepo().save(\n orderItemRepo().create({\n orderId: oid,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n hsn: line.hsn,\n uom: line.uom,\n productType: line.productType,\n taxRate: line.taxRate,\n taxCode: line.taxCode,\n } as ObjectLiteral)\n );\n }\n await cartItemRepo().delete({ cartId: (cart as { id: number }).id } as ObjectLiteral);\n await cartRepo().delete((cart as { id: number }).id);\n return json({\n orderId: oid,\n orderNumber: (ord as { orderNumber: string }).orderNumber,\n subtotal: prepChk.subtotal,\n tax: prepChk.orderTax,\n total: prepChk.orderTotal,\n });\n }\n\n // --- orders GET ---\n if (path[0] === 'orders' && path.length === 1 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ orders: [] });\n const orders = await orderRepo().find({\n where: { contactId: (contact as { id: number }).id, deleted: false, orderKind: 'sale' } as ObjectLiteral,\n order: { createdAt: 'DESC' },\n take: 50,\n });\n const orderIds = orders.map((o) => (o as ObjectLiteral).id as number);\n const previewByOrder: Record<number, string[]> = {};\n if (orderIds.length) {\n const oItems = await orderItemRepo().find({\n where: { orderId: In(orderIds) },\n relations: ['product'],\n order: { id: 'ASC' },\n });\n for (const oi of oItems) {\n const oid = (oi as ObjectLiteral).orderId as number;\n if (!previewByOrder[oid]) previewByOrder[oid] = [];\n if (previewByOrder[oid].length >= 4) continue;\n const url = primaryProductImageUrl(((oi as ObjectLiteral).product as ObjectLiteral)?.metadata);\n if (url && !previewByOrder[oid].includes(url)) previewByOrder[oid].push(url);\n }\n }\n return json({\n orders: orders.map((o) => {\n const ol = o as ObjectLiteral;\n return {\n id: ol.id,\n orderNumber: ol.orderNumber,\n status: ol.status,\n total: ol.total,\n currency: ol.currency,\n createdAt: ol.createdAt,\n previewImages: previewByOrder[ol.id as number] ?? [],\n };\n }),\n });\n }\n\n if (path[0] === 'orders' && path.length === 3 && path[2] === 'invoice' && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n if (!getCms) return json({ error: 'Not found' }, { status: 404 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Not found' }, { status: 404 });\n const orderId = parseInt(path[1]!, 10);\n if (!Number.isFinite(orderId)) return json({ error: 'Invalid id' }, { status: 400 });\n const cms = await getCms();\n return streamOrderInvoicePdf(cms, dataSource, entityMap, orderId, {\n ownerContactId: (contact as { id: number }).id,\n });\n }\n\n if (path[0] === 'orders' && path.length === 2 && method === 'GET') {\n const u = await getSessionUser();\n const uid = u?.id ? parseInt(String(u.id), 10) : NaN;\n if (!Number.isFinite(uid)) return json({ error: 'Unauthorized' }, { status: 401 });\n const contact = await contactRepo().findOne({ where: { userId: uid, deleted: false } as ObjectLiteral });\n if (!contact) return json({ error: 'Not found' }, { status: 404 });\n const orderId = parseInt(path[1]!, 10);\n let order = await orderRepo().findOne({\n where: { id: orderId, contactId: (contact as { id: number }).id, deleted: false } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n if (!order) return json({ error: 'Not found' }, { status: 404 });\n if (getCms) {\n const cms = await getCms();\n await tryRefreshOrderFromErpForStorefront(cms, dataSource, entityMap, order as ObjectLiteral);\n order = await orderRepo().findOne({\n where: { id: orderId, contactId: (contact as { id: number }).id, deleted: false } as ObjectLiteral,\n relations: ['items', 'items.product'],\n });\n }\n if (!order) return json({ error: 'Not found' }, { status: 404 });\n const o = order as ObjectLiteral;\n const lines = ((o.items as ObjectLiteral[]) || []).map((line) => {\n const p = line.product as ObjectLiteral | undefined;\n return {\n id: line.id,\n productId: line.productId,\n quantity: line.quantity,\n unitPrice: line.unitPrice,\n tax: line.tax,\n total: line.total,\n product: p\n ? {\n name: p.name,\n slug: p.slug,\n sku: p.sku,\n image: primaryProductImageUrl(p.metadata),\n }\n : null,\n };\n });\n const kind = (o.orderKind as string) || 'sale';\n let relatedOrders: ObjectLiteral[] = [];\n if (kind === 'sale') {\n relatedOrders = await orderRepo().find({\n where: { parentOrderId: orderId, deleted: false } as ObjectLiteral,\n order: { id: 'ASC' },\n });\n }\n const meta = o.metadata as Record<string, unknown> | null | undefined;\n const fulfillmentPreview =\n meta && typeof meta.fulfillment === 'object' && meta.fulfillment && 'status' in (meta.fulfillment as object)\n ? String((meta.fulfillment as { status?: string }).status ?? '')\n : '';\n return json({\n order: {\n id: o.id,\n orderNumber: o.orderNumber,\n orderKind: kind,\n parentOrderId: o.parentOrderId ?? null,\n status: o.status,\n subtotal: o.subtotal,\n tax: o.tax,\n discount: o.discount,\n total: o.total,\n currency: o.currency,\n createdAt: o.createdAt,\n metadata: o.metadata ?? null,\n items: lines,\n },\n relatedOrders: relatedOrders.map((r) => ({\n id: r.id,\n orderNumber: r.orderNumber,\n orderKind: r.orderKind ?? 'return',\n status: r.status,\n createdAt: r.createdAt,\n fulfillmentStatus:\n r.metadata &&\n typeof r.metadata === 'object' &&\n (r.metadata as { fulfillment?: { status?: string } }).fulfillment?.status,\n })),\n fulfillmentPreview: fulfillmentPreview || undefined,\n });\n }\n\n return json({ error: 'Not found' }, { status: 404 });\n } catch {\n return json({ error: 'Server error' }, { status: 500 });\n }\n },\n };\n}\n","/** Max lengths per RFC 5321 / 5322 (practical subset). */\nconst MAX_EMAIL = 254;\nconst MAX_LOCAL = 64;\n\n/**\n * Rejects obviously invalid signup addresses (format, length, missing TLD).\n * Used by storefront register so junk strings cannot create accounts.\n */\nexport function isValidSignupEmail(email: string): boolean {\n if (!email || email.length > MAX_EMAIL) return false;\n const at = email.indexOf('@');\n if (at <= 0 || at !== email.lastIndexOf('@')) return false;\n const local = email.slice(0, at);\n const domain = email.slice(at + 1);\n if (!local || local.length > MAX_LOCAL || !domain || domain.length > 253) return false;\n if (local.startsWith('.') || local.endsWith('.') || local.includes('..')) return false;\n if (domain.startsWith('.') || domain.endsWith('.') || domain.includes('..')) return false;\n if (!/^[a-z0-9._%+-]+$/i.test(local)) return false;\n if (!/^[a-z0-9](?:[a-z0-9-]*[a-z0-9])?(?:\\.[a-z0-9](?:[a-z0-9-]*[a-z0-9])?)+$/i.test(domain)) return false;\n const tld = domain.split('.').pop()!;\n return tld.length >= 2;\n}\n","export interface AdminNavItem {\n href: string;\n label: string;\n icon?: string;\n}\n\nexport const DEFAULT_ADMIN_NAV: AdminNavItem[] = [\n { href: '/admin/dashboard', label: 'Dashboard' },\n { href: '/admin/contacts', label: 'Contacts' },\n { href: '/admin/blogs', label: 'Blogs' },\n { href: '/admin/users', label: 'Users' },\n { href: '/admin/roles', label: 'Roles' },\n { href: '/admin/forms', label: 'Forms' },\n { href: '/admin/submissions', label: 'Submissions' },\n { href: '/admin/pages', label: 'Pages' },\n];\n"],"mappings":"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","names":["pickInvoiceId","Stripe","Razorpay","crypto","pick","escapeHtml","subject","escapeHtml","bodyHtml","text","html","render","escapeHtml","render","escapeHtml","escapeHtml","render","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","escapeHtml","render","render","nodemailer","resolveSmsTemplateForSend","applyTemplateVars","StripePaymentService","RazorpayPaymentService","Redis","fastq","import_crypto","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","import_typeorm","isPublicMethod","_CredentialsProvider","import_typeorm","repo","sortField","total","data","typeFilter","updatePayload","updated","crypto","import_typeorm","isErpIntegrationEnabled","result","queueEmail","streamOrderInvoicePdf","isErpIntegrationEnabled","queueErpPaidOrderForOrderId","import_typeorm","roundMoney","cart","serializeAddress","crypto","gItems","req","w"]}