@wtree/payload-ecommerce-coupon 3.78.9 → 3.78.10

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,"file":"index.mjs","names":["normalizeCode","relationId","relationId","readField","writeField","normalizeCode","ensureRequestData","findByNormalizedCode","relationId","readField","asNumber","relationId","validateCouponCode","relationId","readField","normalizePath","formatCurrency","formatCurrency","formatCurrency"],"sources":["../src/collections/createCouponsCollection.ts","../src/utilities/userRoles.ts","../src/collections/createReferralCodesCollection.ts","../src/collections/createReferralProgramsCollection.ts","../src/utilities/pricing.ts","../src/utilities/calculateValues.ts","../src/utilities/roundTo2.ts","../src/endpoints/applyCoupon.ts","../src/endpoints/partnerStats.ts","../src/endpoints/validateCoupon.ts","../src/hooks/recalculateCart.ts","../src/utilities/recordCouponUsageForOrder.ts","../src/utilities/sanitizePluginConfig.ts","../src/plugin.ts","../src/client/hooks.ts","../node_modules/react/cjs/react.production.js","../node_modules/react/index.js","../node_modules/react/cjs/react-jsx-runtime.production.js","../node_modules/react/jsx-runtime.js","../src/components/PartnerDashboard/EarningsSummary.tsx","../src/components/PartnerDashboard/ReferralPerformance.tsx","../src/components/PartnerDashboard/RecentReferrals.tsx","../src/components/PartnerDashboard/ReferralCodes.tsx","../src/components/PartnerDashboard/ProgramOverview.tsx","../src/components/PartnerDashboard/CommissionBreakdown.tsx","../src/components/PartnerDashboard/index.tsx","../src/utilities/getCartTotalWithDiscounts.ts"],"sourcesContent":["import type { CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\nexport const createCouponsCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, defaultCurrency, adminGroups, integration } = pluginConfig\n const usersSlug = integration.collections.usersSlug\n\n return {\n slug: collections.couponsSlug,\n admin: {\n useAsTitle: 'code',\n defaultColumns: ['code', 'type', 'value', 'activeFrom', 'activeUntil'],\n group: adminGroups.couponsGroup,\n },\n access: {\n read: access.canUseCoupons || (() => true),\n create: access.isAdmin || (() => false),\n update: access.isAdmin || (() => false),\n delete: access.isAdmin || (() => false),\n },\n fields: [\n {\n name: 'code',\n type: 'text',\n required: true,\n unique: true,\n admin: {\n description: 'The coupon code that customers will enter',\n },\n },\n {\n name: 'normalizedCode',\n type: 'text',\n unique: true,\n index: true,\n admin: {\n hidden: true,\n description: 'Uppercased, trimmed code used for fast case-insensitive lookups',\n },\n },\n {\n name: 'description',\n type: 'text',\n admin: {\n description: 'Optional description for admin reference',\n },\n },\n {\n name: 'type',\n type: 'select',\n required: true,\n options: [\n { label: 'Percentage', value: 'percentage' },\n { label: 'Fixed Amount', value: 'fixed' },\n ],\n defaultValue: 'percentage',\n admin: {\n description: 'Whether this is a percentage or fixed amount discount',\n },\n },\n {\n name: 'value',\n type: 'number',\n required: true,\n admin: {\n description: `If percentage, 10 = 10%. If fixed, interpreted in ${defaultCurrency} (smallest currency units)`,\n step: 0.01,\n },\n },\n {\n name: 'maxDiscountAmount',\n type: 'number',\n admin: {\n description: `Maximum discount amount in ${defaultCurrency} (smallest currency unit). Leave empty for no cap.`,\n },\n },\n {\n name: 'usageLimit',\n type: 'number',\n admin: {\n description:\n 'Total times this coupon can be used across all customers. Empty = unlimited.',\n },\n },\n {\n name: 'perCustomerLimit',\n type: 'number',\n admin: {\n description: 'Times a single customer can use this coupon. Empty = unlimited.',\n },\n },\n {\n name: 'activeFrom',\n type: 'date',\n admin: {\n description:\n 'Coupon becomes active from this date. Leave empty for immediate activation.',\n },\n },\n {\n name: 'activeUntil',\n type: 'date',\n admin: {\n description: 'Coupon expires after this date. Leave empty for no expiration.',\n },\n },\n {\n name: 'minOrderValue',\n type: 'number',\n admin: {\n description: `Minimum order value required in ${defaultCurrency} (smallest currency units)`,\n },\n },\n {\n name: 'maxOrderValue',\n type: 'number',\n admin: {\n description: `Maximum order value allowed in ${defaultCurrency} (smallest currency units)`,\n },\n },\n {\n name: 'usageCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'How many times this coupon has been used',\n readOnly: true,\n },\n },\n {\n name: 'createdBy',\n type: 'relationship',\n relationTo: usersSlug,\n admin: {\n readOnly: true,\n position: 'sidebar',\n },\n },\n ],\n hooks: {\n beforeValidate: [\n ({ data }) => {\n if (data && typeof data.code === 'string') {\n data.code = data.code.trim()\n data.normalizedCode = data.code.toUpperCase()\n }\n return data\n },\n ],\n beforeChange: [\n ({ operation, req, data }) => {\n if (data && typeof data.code === 'string') {\n data.code = data.code.trim()\n data.normalizedCode = data.code.toUpperCase()\n }\n\n if (operation === 'create' && req.user && !data.createdBy) {\n data.createdBy = (req.user as { id?: string | number }).id\n }\n\n return data\n },\n ],\n },\n timestamps: true,\n }\n}\n","import type { SanitizedCouponPluginOptions } from '../types'\nimport type { Where } from 'payload'\n\ntype RoleConfig = SanitizedCouponPluginOptions['roleConfig']\n\nfunction readByPath(input: unknown, path: string): unknown {\n if (!input || typeof input !== 'object' || !path) return undefined\n return path.split('.').reduce<unknown>((acc, key) => {\n if (!acc || typeof acc !== 'object') return undefined\n return (acc as Record<string, unknown>)[key]\n }, input)\n}\n\nfunction toRoleArray(value: unknown): string[] {\n if (typeof value === 'string' && value.trim().length > 0) return [value]\n if (Array.isArray(value)) {\n return value\n .filter((item): item is string => typeof item === 'string')\n .map((item) => item.trim())\n .filter(Boolean)\n }\n return []\n}\n\nconst normalizeRoleValue = (value: string): string => value.trim().toLowerCase()\n\nexport const resolveUserRoles = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): string[] => {\n if (!user) return []\n\n if (typeof roleConfig.customRoleResolver === 'function') {\n const custom = roleConfig.customRoleResolver(user)\n return Array.isArray(custom) ? custom.map(normalizeRoleValue).filter(Boolean) : []\n }\n\n const roles = roleConfig.roleFieldPaths.flatMap((path) => toRoleArray(readByPath(user, path)))\n return [...new Set(roles.map(normalizeRoleValue).filter(Boolean))]\n}\n\nexport const userHasAnyRole = ({\n user,\n roleConfig,\n targetRoles,\n}: {\n user: unknown\n roleConfig: RoleConfig\n targetRoles: string[]\n}): boolean => {\n const userRoles = new Set(resolveUserRoles({ user, roleConfig }))\n for (const target of targetRoles) {\n if (userRoles.has(normalizeRoleValue(target))) return true\n }\n return false\n}\n\nexport const isPartnerUser = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): boolean =>\n userHasAnyRole({\n user,\n roleConfig,\n targetRoles: roleConfig.partnerRoleValues,\n })\n\nexport const isAdminUser = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): boolean =>\n userHasAnyRole({\n user,\n roleConfig,\n targetRoles: roleConfig.adminRoleValues,\n })\n\nexport const buildPartnerUserFilterWhere = ({\n roleConfig,\n}: {\n roleConfig: RoleConfig\n}): Where | true => {\n if (!roleConfig.roleFieldPaths.length || !roleConfig.partnerRoleValues.length) return true\n\n const conditions = roleConfig.roleFieldPaths.map((fieldPath) => ({\n [fieldPath]: { in: roleConfig.partnerRoleValues },\n }))\n\n if (conditions.length === 1) return conditions[0] as Where\n return { or: conditions } as Where\n}\n","import type { CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport { buildPartnerUserFilterWhere, isAdminUser, isPartnerUser } from '../utilities/userRoles'\n\nconst normalizeCode = (value: unknown): string =>\n typeof value === 'string' ? value.trim().toUpperCase() : ''\n\nexport const createReferralCodesCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, adminGroups, defaultCurrency, roleConfig, policies, integration } =\n pluginConfig\n\n const usersSlug = integration.collections.usersSlug\n\n return {\n slug: collections.referralCodesSlug,\n admin: {\n useAsTitle: 'code',\n defaultColumns: ['code', 'normalizedCode', 'partner', 'program', 'usageCount', 'isActive'],\n group: adminGroups.referralsGroup,\n },\n access: {\n read: async ({ req }) => {\n const user = req?.user as { id?: string | number } | undefined\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n if (isAdmin) return true\n\n const hasPolicyAccess = await Promise.resolve(\n policies.canApplyReferral({ req, user, payload: req?.payload }),\n )\n\n if (!hasPolicyAccess) return false\n\n const isPartner =\n isPartnerUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isPartner?.({ req } as any)))\n\n if (isPartner) {\n return {\n partner: {\n equals: user.id,\n },\n }\n }\n\n return access.canUseReferrals ? access.canUseReferrals({ req } as any) : false\n },\n\n create: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const policyAllowed = await Promise.resolve(\n policies.canApplyReferral({ req, user, payload: req?.payload }),\n )\n if (!policyAllowed) return false\n\n if (\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n ) {\n return true\n }\n\n if (\n isPartnerUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isPartner?.({ req } as any)))\n ) {\n return true\n }\n\n return access.isAdmin ? access.isAdmin({ req } as any) : false\n },\n\n update: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n if (isAdmin) return true\n\n return false\n },\n\n delete: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n return isAdmin\n },\n },\n\n fields: [\n {\n name: 'code',\n type: 'text',\n required: true,\n unique: true,\n admin: {\n description: 'The referral code that customers will enter',\n },\n },\n {\n name: 'normalizedCode',\n type: 'text',\n required: true,\n unique: true,\n index: true,\n admin: {\n readOnly: true,\n description: 'Uppercased normalized code for fast case-insensitive lookup',\n },\n },\n {\n name: 'program',\n type: 'relationship',\n relationTo: collections.referralProgramsSlug,\n required: true,\n admin: {\n description: 'The referral program this code belongs to',\n },\n },\n {\n name: 'partner',\n type: 'relationship',\n relationTo: usersSlug,\n required: true,\n filterOptions: async ({ req, user }) => {\n const currentUser = (user || req?.user) as unknown\n\n if (\n isAdminUser({ user: currentUser, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n ) {\n return true\n }\n\n return buildPartnerUserFilterWhere({ roleConfig })\n },\n admin: {\n description: `The partner who owns this referral code (relation: ${usersSlug})`,\n },\n },\n {\n name: 'isActive',\n type: 'checkbox',\n defaultValue: true,\n admin: {\n description: 'Whether this referral code is currently active',\n },\n },\n {\n name: 'usageCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'How many times this referral code has been used',\n readOnly: true,\n },\n },\n {\n name: 'usageLimit',\n type: 'number',\n admin: {\n description: 'Maximum times this code can be used. Empty = unlimited.',\n },\n },\n {\n name: 'expiresAt',\n type: 'date',\n admin: {\n description: 'When this referral code expires',\n },\n },\n {\n name: 'successfulReferralsCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'Total count of successful referrals using this code',\n readOnly: true,\n },\n },\n {\n name: 'totalEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Total earnings generated by this code in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'pendingEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Pending earnings awaiting payout in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'paidEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Total earnings paid out in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'metadata',\n type: 'json',\n admin: {\n description: 'Additional metadata for the referral code',\n position: 'sidebar',\n },\n },\n ],\n\n hooks: {\n beforeValidate: [\n ({ data }) => {\n if (!data) return data\n data.normalizedCode = normalizeCode(data.code)\n return data\n },\n ],\n beforeChange: [\n ({ operation, req, data }) => {\n if (!data) return data\n\n if (operation === 'create' && !data.code && data.partner) {\n const timestamp = Date.now().toString(36)\n const random = Math.random().toString(36).slice(2, 8)\n data.code = `REF-${timestamp}-${random}`.toUpperCase()\n }\n\n data.normalizedCode = normalizeCode(data.code)\n\n if (operation === 'create' && req.user) {\n const user = req.user as { id?: string | number }\n if (isPartnerUser({ user, roleConfig }) && user.id != null) {\n data.partner = user.id\n }\n }\n\n return data\n },\n ],\n },\n\n timestamps: true,\n }\n}\n","import { APIError, type CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\ntype CommissionType = 'fixed' | 'percentage'\n\ntype RuleData = {\n appliesTo?: 'all' | 'products' | 'segments' | 'categories'\n products?: unknown[]\n categories?: unknown[]\n tags?: unknown[]\n totalCommission?: { type?: CommissionType; value?: number; maxAmount?: number }\n partnerSplit?: number\n customerSplit?: number\n partnerPercent?: number\n customerPercent?: number\n partnerAmount?: number\n customerAmount?: number\n}\n\nfunction toNumber(value: unknown): number | null {\n return typeof value === 'number' && Number.isFinite(value) ? value : null\n}\n\n/**\n * Scaling policy:\n * - Admin inputs normal currency values (e.g. 100 means $100).\n * - Internally, maxPartnerCommissionPerOrder, maxCustomerDiscountPerOrder, and\n * minOrderAmount are stored in x100 (integer cents) form to avoid floating-point\n * drift on monetary cap fields.\n * - beforeChange → multiply by 100 before persisting.\n * - afterRead → divide by 100 before returning to admin / calculation code.\n * - partnerAmount / customerAmount per-rule fixed amounts are NOT scaled here;\n * they represent per-item fixed currency amounts and are stored as-is.\n */\n\nexport const createReferralProgramsCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, adminGroups, referralConfig, integration } = pluginConfig\n const allowedTotalCommissionTypes = referralConfig.allowedTotalCommissionTypes\n const relationSlugs = integration.collections\n\n // ---------------------------------------------------------------------------\n // beforeChange — validate, scale monetary cap fields ×100, normalise rules\n // ---------------------------------------------------------------------------\n const beforeChange: NonNullable<CollectionConfig['hooks']>['beforeChange'] = [\n ({ data }: { data: Record<string, unknown> }) => {\n if (\n !data.commissionRules ||\n !Array.isArray(data.commissionRules) ||\n data.commissionRules.length === 0\n ) {\n throw new APIError('At least one commission rule is required', 400)\n }\n\n // --- top-level monetary caps: validate then scale ×100 for storage ---\n\n const rawMaxPartner = toNumber(data.maxPartnerCommissionPerOrder)\n if (rawMaxPartner != null && rawMaxPartner < 0) {\n throw new APIError(\n 'Maximum commission per order for partner must be a non-negative number',\n 400,\n )\n }\n\n const rawMaxCustomer = toNumber(data.maxCustomerDiscountPerOrder)\n if (rawMaxCustomer != null && rawMaxCustomer < 0) {\n throw new APIError(\n 'Maximum discount for customer per order must be a non-negative number',\n 400,\n )\n }\n\n const rawMinOrder = toNumber(data.minOrderAmount)\n if (rawMinOrder != null && rawMinOrder < 0) {\n throw new APIError('Minimum Order Amount must be a non-negative number', 400)\n }\n\n // Store as x100 (cents). null when not provided.\n data.maxPartnerCommissionPerOrder =\n rawMaxPartner != null ? Math.round(rawMaxPartner * 100) : null\n data.maxCustomerDiscountPerOrder =\n rawMaxCustomer != null ? Math.round(rawMaxCustomer * 100) : null\n data.minOrderAmount = rawMinOrder != null ? Math.round(rawMinOrder * 100) : null\n\n // --- commission rules ---\n data.commissionRules = data.commissionRules.map(\n (rule: Record<string, unknown>, index: number) => {\n const r = rule as RuleData\n\n if (!r.totalCommission) {\n throw new APIError(`Commission rule ${index + 1}: Total Commission is required`, 400)\n }\n\n if (\n !r.totalCommission.type ||\n !allowedTotalCommissionTypes.includes(r.totalCommission.type)\n ) {\n throw new APIError(\n `Commission rule ${index + 1}: Total Commission type must be one of ${allowedTotalCommissionTypes.join(', ')}`,\n 400,\n )\n }\n\n const type = r.totalCommission.type\n\n const appliesTo = r.appliesTo ?? 'all'\n if (appliesTo === 'products' && (!r.products || r.products.length === 0)) {\n throw new APIError(\n `Commission rule ${index + 1}: At least one product is required`,\n 400,\n )\n }\n\n if (\n (appliesTo === 'segments' || appliesTo === 'categories') &&\n (!r.categories || r.categories.length === 0) &&\n (!r.tags || r.tags.length === 0)\n ) {\n throw new APIError(\n `Commission rule ${index + 1}: At least one category or tag is required`,\n 400,\n )\n }\n\n let partnerSplit: number\n let customerSplit: number\n let partnerPercent: number | null = null\n let customerPercent: number | null = null\n let partnerAmount: number | null = null\n let customerAmount: number | null = null\n let splitWarning: string | null = null\n\n if (type === 'percentage') {\n const partnerPctInput = toNumber(r.partnerPercent) ?? toNumber(r.partnerSplit)\n const customerPctInput = toNumber(r.customerPercent) ?? toNumber(r.customerSplit)\n\n if (partnerPctInput == null || partnerPctInput < 0 || partnerPctInput > 100) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner Split must be between 0 and 100`,\n 400,\n )\n }\n\n if (customerPctInput != null && (customerPctInput < 0 || customerPctInput > 100)) {\n throw new APIError(\n `Commission rule ${index + 1}: Customer percentage must be between 0 and 100`,\n 400,\n )\n }\n\n const customerPctComputed =\n customerPctInput != null ? customerPctInput : 100 - partnerPctInput\n const percentTotal = partnerPctInput + customerPctComputed\n\n if (percentTotal > 100) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner percentage + Customer percentage cannot exceed 100`,\n 400,\n )\n }\n\n if (percentTotal > 50) {\n splitWarning = `High total split configured: ${percentTotal}% (partner + customer).`\n }\n\n partnerPercent = partnerPctInput\n customerPercent = customerPctComputed\n partnerSplit = partnerPctInput\n customerSplit = customerPctComputed\n } else {\n // Fixed commission type.\n // partnerAmount / customerAmount are per-item fixed currency amounts — stored as-is\n // (no x100 scaling: they feed directly into per-unit arithmetic in calculateValues).\n const partnerAmountInput = toNumber(r.partnerAmount)\n const customerAmountInput = toNumber(r.customerAmount)\n const legacyPartnerSplitInput = toNumber(r.partnerSplit)\n const legacyCustomerSplitInput = toNumber(r.customerSplit)\n\n const hasNewFixedInputs = partnerAmountInput != null || customerAmountInput != null\n const hasLegacyFixedInputs =\n legacyPartnerSplitInput != null || legacyCustomerSplitInput != null\n\n if (hasNewFixedInputs) {\n if (partnerAmountInput == null || partnerAmountInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner fixed amount must be a non-negative number`,\n 400,\n )\n }\n\n if (customerAmountInput == null || customerAmountInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: Customer fixed amount must be a non-negative number`,\n 400,\n )\n }\n\n partnerAmount = partnerAmountInput\n customerAmount = customerAmountInput\n partnerSplit = partnerAmountInput\n customerSplit = customerAmountInput\n } else if (hasLegacyFixedInputs) {\n if (legacyPartnerSplitInput == null || legacyPartnerSplitInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be non-negative numbers`,\n 400,\n )\n }\n\n const resolvedLegacyCustomerSplit =\n legacyCustomerSplitInput ?? 100 - legacyPartnerSplitInput\n\n if (resolvedLegacyCustomerSplit == null || resolvedLegacyCustomerSplit < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be non-negative numbers`,\n 400,\n )\n }\n\n partnerSplit = legacyPartnerSplitInput\n customerSplit = resolvedLegacyCustomerSplit\n partnerAmount = null\n customerAmount = null\n } else {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be provided`,\n 400,\n )\n }\n }\n\n return {\n ...rule,\n appliesTo: appliesTo === 'categories' ? 'segments' : appliesTo,\n totalCommission: {\n type,\n ...(typeof r.totalCommission.value === 'number'\n ? { value: r.totalCommission.value }\n : {}),\n ...(typeof r.totalCommission.maxAmount === 'number'\n ? { maxAmount: r.totalCommission.maxAmount }\n : {}),\n },\n partnerPercent,\n customerPercent,\n partnerAmount,\n customerAmount,\n partnerSplit,\n customerSplit,\n splitWarning,\n }\n },\n )\n\n return data\n },\n ]\n\n // ---------------------------------------------------------------------------\n // afterRead — divide x100 monetary cap fields back to normal currency values\n // so that admin UI and calculation utilities always see normal currency units.\n // ---------------------------------------------------------------------------\n const afterRead: NonNullable<CollectionConfig['hooks']>['afterRead'] = [\n ({ doc }: { doc: Record<string, unknown> }) => {\n if (!doc) return doc\n\n const unscale = (value: unknown): number | null => {\n const n = toNumber(value)\n if (n == null) return null\n // Divide stored x100 value back to normal currency, rounded to 2 dp.\n return Math.round((n / 100) * 100) / 100\n }\n\n doc.maxPartnerCommissionPerOrder = unscale(doc.maxPartnerCommissionPerOrder)\n doc.maxCustomerDiscountPerOrder = unscale(doc.maxCustomerDiscountPerOrder)\n doc.minOrderAmount = unscale(doc.minOrderAmount)\n\n return doc\n },\n ]\n\n return {\n slug: collections.referralProgramsSlug,\n admin: {\n useAsTitle: 'name',\n defaultColumns: ['id', 'name', 'commissionRules', 'isActive'],\n group: adminGroups.referralsGroup,\n },\n access: {\n read: access.canUseReferrals || (() => true),\n create: access.isAdmin || (() => false),\n update: access.isAdmin || (() => false),\n delete: access.isAdmin || (() => false),\n },\n hooks: {\n beforeChange,\n afterRead,\n },\n fields: [\n {\n name: 'name',\n type: 'text',\n required: true,\n admin: {\n description: 'Name of the referral program',\n },\n },\n {\n name: 'isActive',\n type: 'checkbox',\n defaultValue: true,\n admin: {\n description: 'Whether this referral program is currently active',\n },\n },\n {\n name: 'maxPartnerCommissionPerOrder',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Maximum commission per order for partner (enter normal currency value, e.g. 50 for $50). Leave empty for no cap.',\n },\n },\n {\n name: 'maxCustomerDiscountPerOrder',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Maximum customer discount per order (enter normal currency value, e.g. 25 for $25). Leave empty for no cap.',\n },\n },\n {\n name: 'minOrderAmount',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Minimum cart subtotal required for this program (enter normal currency value, e.g. 100 for $100). Leave empty for no minimum.',\n },\n },\n {\n name: 'commissionRules',\n type: 'array',\n required: true,\n minRows: 1,\n admin: {\n description: 'Rules for referral commission and customer discount distribution.',\n },\n fields: [\n {\n name: 'appliesTo',\n type: 'select',\n required: true,\n options: [\n { label: 'All Products', value: 'all' },\n { label: 'Specific Products', value: 'products' },\n { label: 'Categories and Tags', value: 'segments' },\n ],\n defaultValue: 'all',\n },\n {\n name: 'products',\n type: 'relationship',\n relationTo: relationSlugs.productsSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'products',\n description: 'Products this rule applies to',\n },\n },\n {\n name: 'categories',\n type: 'relationship',\n relationTo: relationSlugs.categoriesSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'segments',\n description: 'Any matching category can activate this rule',\n },\n },\n {\n name: 'tags',\n type: 'relationship',\n relationTo: relationSlugs.tagsSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'segments',\n description: 'Any matching tag can activate this rule',\n },\n },\n {\n name: 'totalCommission',\n type: 'group',\n admin: {\n description: 'Total commission pool configuration',\n },\n fields: [\n {\n name: 'type',\n type: 'select',\n required: true,\n options: allowedTotalCommissionTypes.map((value) => ({\n label: value === 'fixed' ? 'Fixed Amount' : 'Percentage of Order',\n value,\n })),\n defaultValue: allowedTotalCommissionTypes.includes('fixed')\n ? 'fixed'\n : 'percentage',\n },\n ],\n },\n {\n name: 'partnerPercent',\n type: 'number',\n min: 0,\n max: 100,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'percentage',\n description: 'Partner share in percent (0–100)',\n },\n },\n {\n name: 'customerPercent',\n type: 'number',\n min: 0,\n max: 100,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'percentage',\n description:\n 'Customer share percentage (0–100). Partner + Customer cannot exceed 100.',\n },\n },\n {\n name: 'partnerAmount',\n type: 'number',\n min: 0,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'fixed',\n description:\n 'Fixed partner commission amount per item (normal currency value, e.g. 12.50 for $12.50).',\n },\n },\n {\n name: 'customerAmount',\n type: 'number',\n min: 0,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'fixed',\n description:\n 'Fixed customer discount amount per item (normal currency value, e.g. 5 for $5).',\n },\n },\n {\n name: 'partnerSplit',\n type: 'number',\n min: 0,\n admin: {\n hidden: true,\n description:\n 'Canonical storage field. Percentage mode: percent value. Fixed mode: per-item amount.',\n },\n },\n {\n name: 'customerSplit',\n type: 'number',\n min: 0,\n admin: {\n hidden: true,\n description:\n 'Canonical storage field. Percentage mode: percent value. Fixed mode: per-item amount.',\n },\n },\n ],\n },\n ],\n timestamps: true,\n }\n}\n","export const DEFAULT_PRICE_CURRENCY = 'AED'\n\ntype PriceEntity = {\n price?: number | null\n} & Record<string, unknown>\n\nfunction normalizeCurrencyCode(currencyCode?: string): string {\n if (!currencyCode) return DEFAULT_PRICE_CURRENCY\n return currencyCode.toUpperCase()\n}\n\nfunction readNumberField(entity: unknown, key: string): number | undefined {\n if (!entity || typeof entity !== 'object') return undefined\n const value = (entity as Record<string, unknown>)[key]\n return typeof value === 'number' ? value : undefined\n}\n\nexport function getPriceFieldKey(currencyCode: string): string {\n return `priceIn${normalizeCurrencyCode(currencyCode)}`\n}\n\nexport function readMoneyField(\n entity: PriceEntity | null | undefined,\n currencyCode: string,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n): number | undefined {\n if (!entity) return undefined\n\n const primaryField = getPriceFieldKey(currencyCode)\n const primary = readNumberField(entity, primaryField)\n if (typeof primary === 'number') return primary\n\n const fallbackField = getPriceFieldKey(defaultCurrencyCode)\n if (fallbackField !== primaryField) {\n const fallback = readNumberField(entity, fallbackField)\n if (typeof fallback === 'number') return fallback\n }\n\n return typeof entity.price === 'number' ? entity.price : undefined\n}\n\nexport function resolveMoneyField(\n entity: PriceEntity | null | undefined,\n currencyCode: string,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n): number {\n return readMoneyField(entity, currencyCode, defaultCurrencyCode) ?? 0\n}\n\nexport function getCartItemUnitPrice({\n item,\n product,\n variant,\n currencyCode,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n}: {\n item?: {\n price?: number | null\n unitPrice?: number | null\n } | null\n product?: PriceEntity | null\n variant?: PriceEntity | null\n currencyCode: string\n defaultCurrencyCode?: string\n}): number {\n if (typeof item?.price === 'number') return item.price\n if (typeof item?.unitPrice === 'number') return item.unitPrice\n if (variant) return resolveMoneyField(variant, currencyCode, defaultCurrencyCode)\n return resolveMoneyField(product, currencyCode, defaultCurrencyCode)\n}\n","import { getCartItemUnitPrice } from './pricing'\n\n// ---------------------------------------------------------------------------\n// Cent-safe arithmetic helpers\n//\n// Policy:\n// • All monetary values arrive from the DB / admin in NORMAL CURRENCY units\n// (e.g. 10.50 means $10.50).\n// • Before any arithmetic we scale UP to integer CENTS (×100) so every\n// intermediate result is a safe integer with no floating-point drift.\n// • Results are scaled back DOWN (÷100) before being returned to callers,\n// who then store them in DB as normal currency again.\n// • Percentage values (0-100) stay as-is; they act as divisors inside the\n// formula and do not need independent scaling.\n// ---------------------------------------------------------------------------\n\n/** Convert a normal-currency amount to integer cents. */\nfunction toCents(amount: number): number {\n return Math.round(amount * 100)\n}\n\n/** Convert integer cents back to a normal-currency amount (2 dp max). */\nfunction fromCents(cents: number): number {\n return Math.round(cents) / 100\n}\n\n// ---------------------------------------------------------------------------\n// Public: coupon discount\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate the discount amount for a coupon.\n *\n * @param coupon - Coupon document from DB (values in normal currency).\n * @param cartTotal - Cart subtotal in normal currency.\n * @returns Discount amount in normal currency (2 dp).\n */\nexport function calculateCouponDiscount({\n coupon,\n cartTotal,\n}: {\n coupon: any\n cartTotal: number\n}): number {\n const cartCents = toCents(cartTotal)\n let discountCents = 0\n\n if (coupon.type === 'percentage') {\n // percentage value is 0-100, no scaling needed\n discountCents = Math.floor((cartCents * coupon.value) / 100)\n if (coupon.maxDiscountAmount != null) {\n const maxCents = toCents(coupon.maxDiscountAmount)\n if (discountCents > maxCents) discountCents = maxCents\n }\n } else if (coupon.type === 'fixed') {\n discountCents = toCents(coupon.value)\n if (discountCents > cartCents) discountCents = cartCents\n }\n\n return fromCents(discountCents)\n}\n\n// ---------------------------------------------------------------------------\n// Internal helpers\n// ---------------------------------------------------------------------------\n\nfunction relationId(value: any): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nconst allowedCommissionTypesSet = (\n allowed: Array<'fixed' | 'percentage'> | undefined,\n): Set<string> =>\n new Set((allowed && allowed.length ? allowed : ['fixed', 'percentage']).map((v) => v))\n\nfunction normalizeIds(values: any[] | null | undefined): Array<string | number> {\n if (!Array.isArray(values)) return []\n return values.map(relationId).filter((v): v is string | number => v != null)\n}\n\nfunction getRuleSplits(rule: any): { partnerSplit: number; customerSplit: number } | null {\n const partnerRaw =\n typeof rule.partnerSplit === 'number'\n ? rule.partnerSplit\n : typeof rule.referrerSplit === 'number'\n ? rule.referrerSplit\n : null\n if (partnerRaw == null) return null\n\n const customerRaw =\n typeof rule.customerSplit === 'number'\n ? rule.customerSplit\n : typeof rule.refereeSplit === 'number'\n ? rule.refereeSplit\n : 100 - partnerRaw\n\n return { partnerSplit: partnerRaw, customerSplit: customerRaw }\n}\n\n// ---------------------------------------------------------------------------\n// Core per-item reward calculation (all values in CENTS)\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate partner and customer reward for a single line item.\n *\n * ALL inputs are expected in CENTS.\n * Returns rewards in CENTS, or null if the rule is inapplicable.\n */\nfunction calculateItemRewardByRule({\n rule,\n itemTotalCents,\n quantity,\n allowedTotalCommissionTypes,\n}: {\n rule: any\n itemTotalCents: number\n quantity: number\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { partner: number; customer: number } | null {\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n // ── Shared / v2 model ────────────────────────────────────────────────────\n if (rule.totalCommission) {\n if (!allowedTypes.has(rule.totalCommission.type)) return null\n\n // maxAmount is stored in normal currency → convert to cents\n const resolvedMaxAmountCents =\n typeof rule.totalCommission.maxAmount === 'number' &&\n Number.isFinite(rule.totalCommission.maxAmount)\n ? toCents(rule.totalCommission.maxAmount)\n : null\n\n // ── Fixed direct mode: partnerSplit / customerSplit are per-unit currency amounts\n if (rule.totalCommission.type === 'fixed' && rule.totalCommission.value == null) {\n const partnerAmtPerUnitCents =\n typeof rule.partnerSplit === 'number' ? toCents(rule.partnerSplit) : null\n const customerAmtPerUnitCents =\n typeof rule.customerSplit === 'number' ? toCents(rule.customerSplit) : null\n if (partnerAmtPerUnitCents == null || customerAmtPerUnitCents == null) return null\n\n let partner = partnerAmtPerUnitCents * quantity\n let customer = customerAmtPerUnitCents * quantity\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n const totalPot = partner + customer\n if (totalPot > maxPotForLine && totalPot > 0) {\n const ratio = maxPotForLine / totalPot\n partner = Math.floor(partner * ratio)\n customer = Math.floor(customer * ratio)\n }\n }\n\n return { partner, customer }\n }\n\n // ── Percentage mode ───────────────────────────────────────────────────\n if (rule.totalCommission.type === 'percentage') {\n const commissionValue =\n typeof rule.totalCommission.value === 'number' &&\n Number.isFinite(rule.totalCommission.value)\n ? rule.totalCommission.value\n : null\n\n if (commissionValue == null) {\n // Direct percent splits (no totalCommission.value)\n const partnerPercentInput =\n typeof rule.partnerPercent === 'number'\n ? rule.partnerPercent\n : typeof rule.partnerSplit === 'number'\n ? rule.partnerSplit\n : null\n if (partnerPercentInput == null || partnerPercentInput < 0 || partnerPercentInput > 100) {\n return null\n }\n\n const customerPercentInput =\n typeof rule.customerPercent === 'number'\n ? rule.customerPercent\n : typeof rule.customerSplit === 'number'\n ? rule.customerSplit\n : 100 - partnerPercentInput\n\n if (\n customerPercentInput == null ||\n customerPercentInput < 0 ||\n customerPercentInput > 100\n ) {\n return null\n }\n\n let partner = Math.floor((itemTotalCents * partnerPercentInput) / 100)\n let customer = Math.floor((itemTotalCents * customerPercentInput) / 100)\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n const totalForLine = partner + customer\n if (totalForLine > maxPotForLine && totalForLine > 0) {\n const ratio = maxPotForLine / totalForLine\n partner = Math.floor(partner * ratio)\n customer = Math.floor(customer * ratio)\n }\n }\n\n return { partner, customer }\n }\n\n // totalCommission.value drives the total pool (percentage of item total)\n let totalPotCents = Math.floor((itemTotalCents * commissionValue) / 100)\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n if (totalPotCents > maxPotForLine) totalPotCents = maxPotForLine\n }\n\n const splits = getRuleSplits(rule)\n if (!splits) return null\n\n return {\n partner: Math.floor((totalPotCents * splits.partnerSplit) / 100),\n customer: Math.floor((totalPotCents * splits.customerSplit) / 100),\n }\n }\n\n // ── Fixed pool mode: totalCommission.value is a per-unit currency amount\n {\n const splits = getRuleSplits(rule)\n if (!splits) return null\n\n // totalCommission.value is a per-unit currency amount → convert to cents\n let totalPotCents = toCents(rule.totalCommission.value) * quantity\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n if (totalPotCents > maxPotForLine) totalPotCents = maxPotForLine\n }\n\n return {\n partner: Math.floor((totalPotCents * splits.partnerSplit) / 100),\n customer: Math.floor((totalPotCents * splits.customerSplit) / 100),\n }\n }\n }\n\n // ── Legacy direct-reward model (migration compatibility) ─────────────────\n if (rule.referrerReward && rule.refereeReward) {\n let partner = 0\n if (rule.referrerReward.type === 'percentage') {\n partner = Math.floor((itemTotalCents * rule.referrerReward.value) / 100)\n } else {\n partner = toCents(rule.referrerReward.value) * quantity\n }\n if (rule.referrerReward.maxReward != null) {\n const maxCents = toCents(rule.referrerReward.maxReward)\n if (partner > maxCents) partner = maxCents\n }\n\n let customer = 0\n if (rule.refereeReward.type === 'percentage') {\n customer = Math.floor((itemTotalCents * rule.refereeReward.value) / 100)\n } else {\n customer = toCents(rule.refereeReward.value) * quantity\n }\n if (rule.refereeReward.maxReward != null) {\n const maxCents = toCents(rule.refereeReward.maxReward)\n if (customer > maxCents) customer = maxCents\n }\n\n return { partner, customer }\n }\n\n return null\n}\n\n// ---------------------------------------------------------------------------\n// Rule selection helpers\n// ---------------------------------------------------------------------------\n\nfunction getItemCategoryIds(item: any): Array<string | number> {\n const productCategories = Array.isArray(item?.product?.categories)\n ? normalizeIds(item.product.categories)\n : []\n const singleCategory = relationId(item?.category ?? item?.product?.category)\n return [...productCategories, ...(singleCategory != null ? [singleCategory] : [])]\n}\n\nfunction getItemTagIds(item: any): Array<string | number> {\n return Array.isArray(item?.product?.tags) ? normalizeIds(item.product.tags) : []\n}\n\nfunction selectBestRuleForItem({\n rules,\n item,\n itemTotalCents,\n quantity,\n cartTotalCents,\n minOrderAmountCents,\n allowedTotalCommissionTypes,\n}: {\n rules: any[]\n item: any\n itemTotalCents: number\n quantity: number\n cartTotalCents: number\n minOrderAmountCents?: number | null\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { rule: any; reward: { partner: number; customer: number } } | null {\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n const eligibleRules = rules.filter((rule: any) => {\n const hasSharedType = rule?.totalCommission?.type\n ? allowedTypes.has(rule.totalCommission.type)\n : true\n if (!hasSharedType) return false\n\n // minOrderAmount on the rule itself is stored in normal currency → cents\n const resolvedMinCents =\n minOrderAmountCents != null && Number.isFinite(minOrderAmountCents)\n ? minOrderAmountCents\n : typeof rule?.minOrderAmount === 'number' && Number.isFinite(rule.minOrderAmount)\n ? toCents(rule.minOrderAmount)\n : null\n\n if (resolvedMinCents != null) {\n return cartTotalCents >= resolvedMinCents\n }\n return true\n })\n\n const productId = relationId(item.product)\n const itemCategoryIds = new Set(getItemCategoryIds(item))\n const itemTagIds = new Set(getItemTagIds(item))\n\n const productCandidates = eligibleRules.filter(\n (r: any) =>\n r.appliesTo === 'products' &&\n normalizeIds(r.products).some((id) => productId != null && id === productId),\n )\n\n const segmentCategoryCandidates = eligibleRules.filter((r: any) => {\n const isSegment = r.appliesTo === 'segments' || r.appliesTo === 'categories'\n if (!isSegment) return false\n return normalizeIds(r.categories).some((id) => itemCategoryIds.has(id))\n })\n\n const segmentTagCandidates = eligibleRules.filter((r: any) => {\n if (r.appliesTo !== 'segments') return false\n return normalizeIds(r.tags).some((id) => itemTagIds.has(id))\n })\n\n const allCandidates = eligibleRules.filter((r: any) => r.appliesTo === 'all')\n\n const levels = [productCandidates, segmentCategoryCandidates, segmentTagCandidates, allCandidates]\n const candidates = levels.find((level) => level.length > 0) ?? []\n if (!candidates.length) return null\n\n let best: { rule: any; reward: { partner: number; customer: number } } | null = null\n\n for (const rule of candidates) {\n const reward = calculateItemRewardByRule({\n rule,\n itemTotalCents,\n quantity,\n allowedTotalCommissionTypes,\n })\n if (!reward) continue\n\n if (!best) {\n best = { rule, reward }\n continue\n }\n\n if (reward.customer > best.reward.customer) {\n best = { rule, reward }\n continue\n }\n\n if (reward.customer === best.reward.customer && reward.partner > best.reward.partner) {\n best = { rule, reward }\n }\n }\n\n return best\n}\n\n// ---------------------------------------------------------------------------\n// Public: minimum order amount\n// ---------------------------------------------------------------------------\n\n/**\n * Returns the effective minimum order amount for a program in NORMAL CURRENCY.\n * Returns null if there is no minimum.\n */\nexport function getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes,\n}: {\n program: any\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): number | null {\n if (typeof program?.minOrderAmount === 'number' && Number.isFinite(program.minOrderAmount)) {\n return program.minOrderAmount\n }\n\n const rules = Array.isArray(program?.commissionRules) ? program.commissionRules : []\n if (!rules.length) return null\n\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n const minValues = rules\n .filter((rule: any) => {\n if (rule?.totalCommission?.type) return allowedTypes.has(rule.totalCommission.type)\n return true\n })\n .map((rule: any) => rule?.minOrderAmount)\n .filter(\n (value: unknown): value is number => typeof value === 'number' && Number.isFinite(value),\n )\n\n if (!minValues.length) return null\n return Math.min(...minValues)\n}\n\n// ---------------------------------------------------------------------------\n// Public: commission + discount calculation\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate total partner commission and customer discount for a cart.\n *\n * All monetary inputs are in NORMAL CURRENCY.\n * Returns results in NORMAL CURRENCY (2 dp).\n */\nexport function calculateCommissionAndDiscount({\n cartItems,\n program,\n currencyCode = 'AED',\n cartTotal = 0,\n allowedTotalCommissionTypes,\n}: {\n cartItems: any[]\n program: any\n currencyCode?: string\n cartTotal?: number\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { partnerCommission: number; customerDiscount: number } {\n const rules = Array.isArray(program?.commissionRules) ? program.commissionRules : []\n\n if (!rules.length) {\n return { partnerCommission: 0, customerDiscount: 0 }\n }\n\n // Scale cart total to cents for eligibility checks\n const cartTotalCents = toCents(cartTotal)\n\n // Scale program-level minOrderAmount to cents (if present) for rule filtering\n const programMinOrderAmountCents =\n typeof program?.minOrderAmount === 'number' && Number.isFinite(program.minOrderAmount)\n ? toCents(program.minOrderAmount)\n : null\n\n let totalPartnerCents = 0\n let totalCustomerCents = 0\n\n for (const item of cartItems) {\n const product = typeof item.product === 'object' ? item.product : {}\n const variant = typeof item.variant === 'object' ? item.variant : {}\n\n // Unit price from DB is in normal currency → convert to cents\n const itemPriceCurrency = getCartItemUnitPrice({\n item,\n product,\n variant,\n currencyCode,\n })\n\n const quantity = item.quantity ?? 1\n const itemTotalCents = toCents(itemPriceCurrency) * quantity\n\n const bestMatch = selectBestRuleForItem({\n rules,\n item: { ...item, product },\n itemTotalCents,\n quantity,\n cartTotalCents,\n minOrderAmountCents: programMinOrderAmountCents,\n allowedTotalCommissionTypes,\n })\n\n if (!bestMatch) continue\n\n totalPartnerCents += bestMatch.reward.partner\n totalCustomerCents += bestMatch.reward.customer\n }\n\n // Per-order caps are stored in normal currency → convert to cents for comparison\n const maxPartnerCents =\n typeof program?.maxPartnerCommissionPerOrder === 'number' &&\n Number.isFinite(program.maxPartnerCommissionPerOrder)\n ? toCents(program.maxPartnerCommissionPerOrder)\n : null\n\n const maxCustomerCents =\n typeof program?.maxCustomerDiscountPerOrder === 'number' &&\n Number.isFinite(program.maxCustomerDiscountPerOrder)\n ? toCents(program.maxCustomerDiscountPerOrder)\n : null\n\n if (maxPartnerCents != null) {\n totalPartnerCents = Math.min(totalPartnerCents, maxPartnerCents)\n }\n if (maxCustomerCents != null) {\n totalCustomerCents = Math.min(totalCustomerCents, maxCustomerCents)\n }\n\n // Convert back to normal currency for return\n return {\n partnerCommission: fromCents(totalPartnerCents),\n customerDiscount: fromCents(totalCustomerCents),\n }\n}\n","/**\n * Rounds a number to 2 decimal places (standard for monetary values).\n */\nexport function roundTo2(value: number): number {\n return Math.round(value * 100) / 100\n}\n","import { addDataAndFileToRequest, type Endpoint, type PayloadHandler } from 'payload'\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n calculateCouponDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction writeField(doc: Record<string, unknown>, field: string, value: unknown): void {\n doc[field] = value\n}\n\nfunction normalizeCode(value: unknown): string {\n return typeof value === 'string' ? value.trim().toUpperCase() : ''\n}\n\nasync function ensureRequestData(req: any): Promise<Record<string, unknown>> {\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n await addDataAndFileToRequest(req)\n } catch {\n // Fallback for non-standard test/mocked requests where payload parser cannot run.\n }\n\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n const parsed = await req?.json?.()\n if (parsed && typeof parsed === 'object') {\n req.data = parsed\n return parsed as Record<string, unknown>\n }\n } catch {\n // Ignore malformed/empty body; validation below will return proper 400 errors.\n }\n\n req.data = {}\n return req.data\n}\n\nasync function findByNormalizedCode({\n payload,\n collection,\n normalizedCode,\n}: {\n payload: any\n collection: string\n normalizedCode: string\n}): Promise<any | null> {\n const exactQuery = await payload.find({\n collection,\n where: {\n normalizedCode: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n if (exactQuery?.docs?.[0]) return exactQuery.docs[0]\n\n const lowerQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toLowerCase() },\n },\n limit: 1,\n })\n\n if (lowerQuery?.docs?.[0]) return lowerQuery.docs[0]\n\n const upperQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toUpperCase() },\n },\n limit: 1,\n })\n\n if (upperQuery?.docs?.[0]) return upperQuery.docs[0]\n\n const exactCodeQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n return exactCodeQuery?.docs?.[0] ?? null\n}\n\nexport const applyCouponHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload } = req\n const fields = pluginConfig.integration.fields\n const collections = pluginConfig.integration.collections\n const data = await ensureRequestData(req)\n\n const rawCode = data?.code\n const cartIDRaw = data?.cartID\n const cartID = typeof cartIDRaw === 'string' || typeof cartIDRaw === 'number' ? cartIDRaw : null\n const customerEmail = typeof data?.customerEmail === 'string' ? data.customerEmail : undefined\n\n const normalizedCode = normalizeCode(rawCode)\n\n if (!normalizedCode || !cartID) {\n return Response.json(\n {\n success: false,\n error: `${pluginConfig.enableReferrals ? 'Referral code' : 'Coupon code'} and cart ID are required`,\n },\n { status: 400 },\n )\n }\n\n const allowCoupon = await Promise.resolve(\n pluginConfig.policies.canApplyCoupon({ req, user: req?.user, payload }),\n )\n\n const allowReferral = await Promise.resolve(\n pluginConfig.policies.canApplyReferral({ req, user: req?.user, payload }),\n )\n\n if (!allowCoupon && !(pluginConfig.enableReferrals && allowReferral)) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n try {\n const cart = await payload.findByID({\n collection: collections.cartsSlug,\n id: cartID,\n depth: 2,\n })\n\n if (!cart) {\n return Response.json({ success: false, error: 'Cart not found' }, { status: 404 })\n }\n\n const cartAppliedCoupon = relationId(\n readField(cart, fields.cartAppliedCouponField) as RelationValue,\n )\n const cartAppliedReferral = relationId(\n readField(cart, fields.cartAppliedReferralCodeField) as RelationValue,\n )\n\n if (\n pluginConfig.referralConfig.singleCodePerCart &&\n (cartAppliedCoupon || cartAppliedReferral)\n ) {\n return Response.json(\n {\n success: false,\n error:\n 'A code has already been applied to this cart. Only one code can be used per order.',\n },\n { status: 400 },\n )\n }\n\n if (pluginConfig.enableReferrals && allowReferral) {\n const referralResult = await handleReferralCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n pluginConfig,\n })\n\n if (\n !referralResult.ok &&\n referralResult.status === 404 &&\n pluginConfig.referralConfig.allowBothSystems &&\n allowCoupon\n ) {\n return await handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n })\n }\n\n return referralResult\n }\n\n if (!allowCoupon) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n })\n } catch (error) {\n console.error('Code application error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n }\n\nasync function handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n}: {\n payload: any\n cart: any\n cartID: string | number\n normalizedCode: string\n customerEmail?: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const resolvers = pluginConfig.integration.resolvers\n const coupon = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.couponsSlug,\n normalizedCode,\n })\n\n if (!coupon) {\n return Response.json({ success: false, error: 'Invalid coupon code' }, { status: 404 })\n }\n\n const now = new Date()\n const activeFrom = coupon.activeFrom ? new Date(coupon.activeFrom) : null\n const activeUntil = coupon.activeUntil ? new Date(coupon.activeUntil) : null\n\n if (activeFrom && now < activeFrom) {\n return Response.json({ success: false, error: 'Coupon is not yet active' }, { status: 400 })\n }\n\n if (activeUntil && now > activeUntil) {\n return Response.json({ success: false, error: 'Coupon has expired' }, { status: 400 })\n }\n\n if (coupon.usageLimit && coupon.usageCount >= coupon.usageLimit) {\n return Response.json({ success: false, error: 'Coupon usage limit exceeded' }, { status: 400 })\n }\n\n if (coupon.perCustomerLimit != null && coupon.perCustomerLimit > 0) {\n const email = typeof customerEmail === 'string' ? customerEmail.trim() : ''\n if (!email) {\n return Response.json(\n {\n success: false,\n error: 'Customer email is required for this coupon.',\n },\n { status: 400 },\n )\n }\n\n const ordersQuery = await payload.find({\n collection: pluginConfig.orderIntegration.ordersSlug,\n where: {\n and: [\n { [fields.orderAppliedCouponField]: { equals: coupon.id } },\n {\n [pluginConfig.orderIntegration.orderCustomerEmailField]: {\n equals: email,\n },\n },\n {\n [pluginConfig.orderIntegration.orderPaymentStatusField]: {\n equals: pluginConfig.orderIntegration.orderPaidStatusValue,\n },\n },\n ],\n },\n limit: 0,\n })\n\n if (ordersQuery.totalDocs >= coupon.perCustomerLimit) {\n return Response.json(\n {\n success: false,\n error: 'You have reached the maximum uses for this coupon.',\n },\n { status: 400 },\n )\n }\n }\n\n const existingCouponId = relationId(\n readField(cart, fields.cartAppliedCouponField) as RelationValue,\n )\n if (existingCouponId === coupon.id) {\n return Response.json(\n { success: false, error: 'Coupon already applied to this cart' },\n { status: 400 },\n )\n }\n\n // Use cartSubtotal (pre-discount baseline) for all order-value eligibility checks\n // and discount calculations. cartTotal may already reflect a previously applied\n // discount and would produce an inconsistent baseline for min/max enforcement.\n const cartSubtotal = Number(resolvers.getCartSubtotal(cart)) || 0\n const cartTotal = Number(resolvers.getCartTotal(cart)) || cartSubtotal || 0\n\n if (coupon.minOrderValue && cartSubtotal < coupon.minOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${coupon.minOrderValue} ${pluginConfig.defaultCurrency} required`,\n },\n { status: 400 },\n )\n }\n\n if (coupon.maxOrderValue && cartSubtotal > coupon.maxOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Maximum order value of ${coupon.maxOrderValue} ${pluginConfig.defaultCurrency} exceeded`,\n },\n { status: 400 },\n )\n }\n\n const discountAmount = calculateCouponDiscount({ coupon, cartTotal: cartSubtotal })\n const nextTotal = roundTo2(Math.max(0, cartTotal - discountAmount))\n\n const data: Record<string, unknown> = {}\n writeField(data, fields.cartAppliedCouponField, coupon.id)\n writeField(data, fields.cartDiscountAmountField, discountAmount)\n writeField(data, fields.cartTotalField, nextTotal)\n\n await payload.update({\n collection: pluginConfig.integration.collections.cartsSlug,\n id: cartID,\n data,\n })\n\n return Response.json({\n success: true,\n message: 'Coupon applied successfully',\n coupon: {\n code: coupon.code,\n type: coupon.type,\n value: coupon.value,\n },\n discount: discountAmount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nasync function handleReferralCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n pluginConfig,\n}: {\n payload: any\n cart: any\n cartID: string | number\n normalizedCode: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const resolvers = pluginConfig.integration.resolvers\n\n const referralCode = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.referralCodesSlug,\n normalizedCode,\n })\n\n if (!referralCode) {\n return Response.json({ success: false, error: 'Invalid referral code' }, { status: 404 })\n }\n\n if (!referralCode.isActive) {\n return Response.json({ success: false, error: 'Referral code is not active' }, { status: 400 })\n }\n\n if (referralCode.expiresAt && new Date() > new Date(referralCode.expiresAt)) {\n return Response.json({ success: false, error: 'Referral code has expired' }, { status: 400 })\n }\n\n if (referralCode.usageLimit && referralCode.usageCount >= referralCode.usageLimit) {\n return Response.json(\n { success: false, error: 'Referral code usage limit exceeded' },\n { status: 400 },\n )\n }\n\n const programId =\n typeof referralCode.program === 'string' || typeof referralCode.program === 'number'\n ? referralCode.program\n : referralCode.program?.id\n\n const program = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n\n if (!program || !program.isActive) {\n return Response.json(\n { success: false, error: 'Referral program is not active' },\n { status: 400 },\n )\n }\n\n const existingReferralId = relationId(\n readField(cart, fields.cartAppliedReferralCodeField) as RelationValue,\n )\n\n if (existingReferralId === referralCode.id) {\n return Response.json(\n { success: false, error: 'Referral code already applied to this cart' },\n { status: 400 },\n )\n }\n\n const cartItems = resolvers.getCartItems(cart)\n // Use cartSubtotal as the pre-discount baseline for min-order enforcement and\n // commission/discount calculations, matching the recalculateCart hook policy.\n const cartSubtotal = Number(resolvers.getCartSubtotal(cart)) || 0\n const cartTotal = Number(resolvers.getCartTotal(cart)) || cartSubtotal || 0\n\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && cartSubtotal < minOrderAmount) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderAmount} ${pluginConfig.defaultCurrency} required for this referral program`,\n },\n { status: 400 },\n )\n }\n\n const { partnerCommission, customerDiscount } = calculateCommissionAndDiscount({\n cartItems,\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal: cartSubtotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const roundedPartnerCommission = roundTo2(partnerCommission)\n const roundedCustomerDiscount = roundTo2(customerDiscount)\n const nextTotal = roundTo2(Math.max(0, cartTotal - roundedCustomerDiscount))\n\n const data: Record<string, unknown> = {}\n writeField(data, fields.cartAppliedReferralCodeField, referralCode.id)\n writeField(data, fields.cartPartnerCommissionField, roundedPartnerCommission)\n writeField(data, fields.cartCustomerDiscountField, roundedCustomerDiscount)\n writeField(data, fields.cartTotalField, nextTotal)\n\n await payload.update({\n collection: pluginConfig.integration.collections.cartsSlug,\n id: cartID,\n data,\n })\n\n return Response.json({\n success: true,\n message: 'Referral code applied successfully',\n referralCode: {\n code: referralCode.code,\n },\n partnerCommission: roundedPartnerCommission,\n customerDiscount: roundedCustomerDiscount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nexport const applyCouponEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.applyCoupon,\n method: 'post',\n handler: applyCouponHandler({ pluginConfig }),\n})\n","import type { Endpoint, PayloadHandler } from 'payload'\n\nimport type { PartnerDashboardData, PartnerStats, SanitizedCouponPluginOptions } from '../types'\nimport { isAdminUser, isPartnerUser } from '../utilities/userRoles'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction asNumber(value: unknown): number {\n return typeof value === 'number' && Number.isFinite(value) ? value : 0\n}\n\nfunction asString(value: unknown): string {\n return typeof value === 'string' ? value : ''\n}\n\nfunction toStatsStatus(value: unknown): 'pending' | 'paid' | 'cancelled' {\n if (value === 'paid') return 'paid'\n if (value === 'cancelled') return 'cancelled'\n return 'pending'\n}\n\nexport const partnerStatsHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload, user } = req\n const fields = pluginConfig.integration.fields\n const collections = pluginConfig.integration.collections\n\n if (!user) {\n return Response.json({ success: false, error: 'Authentication required' }, { status: 401 })\n }\n\n const typedUser = user as { id?: string | number }\n const userID = pluginConfig.integration.resolvers.getUserID({ req, user })\n\n if (userID == null) {\n return Response.json(\n { success: false, error: 'Unable to resolve user identity' },\n { status: 403 },\n )\n }\n\n const isPartner =\n isPartnerUser({ user: typedUser, roleConfig: pluginConfig.roleConfig }) ||\n (await Promise.resolve(pluginConfig.access.isPartner?.({ req } as any)))\n\n const isAdmin =\n isAdminUser({ user: typedUser, roleConfig: pluginConfig.roleConfig }) ||\n (await Promise.resolve(pluginConfig.access.isAdmin?.({ req } as any)))\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canViewPartnerStats({\n req,\n user,\n payload,\n requestedPartnerID: userID,\n }),\n )\n\n if (!policyAllowed && !isAdmin && !isPartner) {\n return Response.json({ success: false, error: 'Partner access required' }, { status: 403 })\n }\n\n try {\n const referralCodesQuery = await payload.find({\n collection: pluginConfig.collections.referralCodesSlug,\n where: {\n partner: { equals: userID },\n },\n limit: 100,\n })\n\n const referralCodes = Array.isArray(referralCodesQuery?.docs) ? referralCodesQuery.docs : []\n\n let totalEarnings = 0\n let pendingEarnings = 0\n let paidEarnings = 0\n let totalReferrals = 0\n let successfulReferrals = 0\n\n const referralCodeData = referralCodes.map((code: any) => {\n totalEarnings += asNumber(code?.totalEarnings)\n pendingEarnings += asNumber(code?.pendingEarnings)\n paidEarnings += asNumber(code?.paidEarnings)\n totalReferrals += asNumber(code?.usageCount)\n successfulReferrals += asNumber(code?.successfulReferralsCount)\n\n return {\n id: String(code?.id ?? ''),\n code: asString(code?.code),\n usageCount: asNumber(code?.usageCount),\n totalEarnings: asNumber(code?.totalEarnings),\n isActive: Boolean(code?.isActive),\n }\n })\n\n const conversionRate = totalReferrals > 0 ? (successfulReferrals / totalReferrals) * 100 : 0\n const recentReferrals: PartnerStats['recentReferrals'] = []\n\n try {\n const referralCodeIDs = referralCodes\n .map((c: any) => relationId(c?.id as RelationValue))\n .filter((id): id is string | number => id != null)\n\n if (referralCodeIDs.length > 0) {\n const ordersQuery = await payload.find({\n collection: collections.ordersSlug,\n where: {\n [fields.orderAppliedReferralCodeField]: {\n in: referralCodeIDs,\n },\n },\n limit: 10,\n sort: `-${fields.orderCreatedAtField}`,\n })\n\n for (const order of (ordersQuery?.docs || []) as any[]) {\n const orderReferralRelation = readField(order, fields.orderAppliedReferralCodeField)\n const orderReferralID = relationId(orderReferralRelation as RelationValue)\n\n const matchedCode = referralCodes.find(\n (c: any) => relationId(c?.id as RelationValue) === orderReferralID,\n )\n\n const paymentStatus = readField(order, fields.orderPaymentStatusField)\n const createdAt = readField(order, fields.orderCreatedAtField)\n\n recentReferrals.push({\n id: String(order?.id ?? ''),\n code: asString(matchedCode?.code),\n orderValue: asNumber(readField(order, fields.cartTotalField) ?? order?.total),\n commission: asNumber(readField(order, fields.orderPartnerCommissionField)),\n date: asString(createdAt),\n status: toStatsStatus(paymentStatus),\n })\n }\n }\n } catch {\n // Host app may not expose expected order structure.\n }\n\n const monthlyEarnings: PartnerStats['monthlyEarnings'] = []\n const now = new Date()\n\n for (let i = 5; i >= 0; i--) {\n const monthDate = new Date(now.getFullYear(), now.getMonth() - i, 1)\n const monthName = monthDate.toLocaleString('default', { month: 'short', year: 'numeric' })\n\n monthlyEarnings.push({\n month: monthName,\n earnings: 0,\n referrals: 0,\n })\n }\n\n let program: PartnerDashboardData['program'] = null\n\n if (referralCodes.length > 0) {\n const firstCode = referralCodes[0] as any\n const programID = relationId(firstCode?.program as RelationValue)\n\n if (programID != null) {\n try {\n const programData = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programID,\n })\n\n if (programData) {\n const typedProgram = programData as any\n const firstRule = typedProgram?.commissionRules?.[0]\n\n const partnerSplit =\n asNumber(firstRule?.partnerSplit) ||\n asNumber(firstRule?.referrerSplit) ||\n asNumber(firstRule?.split?.partnerPercentage)\n\n const customerSplit =\n asNumber(firstRule?.customerSplit) ||\n asNumber(firstRule?.refereeSplit) ||\n asNumber(firstRule?.split?.customerPercentage) ||\n Math.max(0, 100 - partnerSplit)\n\n program = {\n name: asString(typedProgram?.name),\n commissionRate: partnerSplit,\n customerDiscount: customerSplit,\n }\n }\n } catch {\n // Program lookup failed or removed.\n }\n }\n }\n\n const stats: PartnerStats = {\n totalEarnings,\n pendingEarnings,\n paidEarnings,\n totalReferrals,\n successfulReferrals,\n conversionRate: Math.round(conversionRate * 100) / 100,\n recentReferrals,\n monthlyEarnings,\n }\n\n const dashboardData: PartnerDashboardData = {\n stats,\n referralCodes: referralCodeData,\n program,\n }\n\n return Response.json({\n success: true,\n data: dashboardData,\n currency: pluginConfig.defaultCurrency,\n })\n } catch (error) {\n console.error('Partner stats error:', error)\n return Response.json(\n { success: false, error: 'Failed to fetch partner stats' },\n { status: 500 },\n )\n }\n }\n\nexport const partnerStatsEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.partnerStats,\n method: 'get',\n handler: partnerStatsHandler({ pluginConfig }),\n})\n","import { addDataAndFileToRequest, type Endpoint, type PayloadHandler } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction normalizeCode(value: unknown): string {\n return typeof value === 'string' ? value.trim().toUpperCase() : ''\n}\n\nasync function ensureRequestData(req: any): Promise<Record<string, unknown>> {\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n await addDataAndFileToRequest(req)\n } catch {\n // Fallback for non-standard test/mocked requests where payload parser cannot run.\n }\n\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n const parsed = await req?.json?.()\n if (parsed && typeof parsed === 'object') {\n req.data = parsed\n return parsed as Record<string, unknown>\n }\n } catch {\n // Ignore malformed/empty body; validation below will return proper 400 errors.\n }\n\n req.data = {}\n return req.data\n}\n\nasync function findByNormalizedCode({\n payload,\n collection,\n normalizedCode,\n}: {\n payload: any\n collection: string\n normalizedCode: string\n}): Promise<any | null> {\n const normalizedQuery = await payload.find({\n collection,\n where: {\n normalizedCode: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n if (normalizedQuery?.docs?.length) return normalizedQuery.docs[0]\n\n const lowerQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toLowerCase() },\n },\n limit: 1,\n })\n\n if (lowerQuery?.docs?.length) return lowerQuery.docs[0]\n\n const upperQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toUpperCase() },\n },\n limit: 1,\n })\n\n if (upperQuery?.docs?.length) return upperQuery.docs[0]\n\n const exactQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n return exactQuery?.docs?.[0] ?? null\n}\n\nexport const validateCouponHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload } = req\n const data = await ensureRequestData(req)\n\n const rawCode = data?.code\n const cartValue = typeof data?.cartValue === 'number' ? data.cartValue : undefined\n const cartIDRaw = data?.cartID\n const cartID =\n typeof cartIDRaw === 'string' || typeof cartIDRaw === 'number' ? cartIDRaw : undefined\n const customerEmail = typeof data?.customerEmail === 'string' ? data.customerEmail : undefined\n\n const normalizedCode = normalizeCode(rawCode)\n\n if (!normalizedCode) {\n return Response.json(\n {\n success: false,\n error: 'Code is required',\n },\n { status: 400 },\n )\n }\n\n try {\n if (pluginConfig.enableReferrals) {\n const canApplyReferral = await Promise.resolve(\n pluginConfig.policies.canApplyReferral({\n req,\n user: req?.user,\n payload,\n }),\n )\n\n if (!canApplyReferral) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await validateReferralCode({\n payload,\n normalizedCode,\n cartID,\n pluginConfig,\n })\n }\n\n const canApplyCoupon = await Promise.resolve(\n pluginConfig.policies.canApplyCoupon({ req, user: req?.user, payload }),\n )\n\n if (!canApplyCoupon) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await validateCouponCode({\n payload,\n normalizedCode,\n cartValue,\n customerEmail,\n pluginConfig,\n })\n } catch (error) {\n console.error('Code validation error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n }\n\nasync function validateCouponCode({\n payload,\n normalizedCode,\n cartValue,\n customerEmail,\n pluginConfig,\n}: {\n payload: any\n normalizedCode: string\n cartValue?: number\n customerEmail?: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const couponData = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.couponsSlug,\n normalizedCode,\n })\n\n if (!couponData) {\n return Response.json({ success: false, error: 'Invalid coupon code' }, { status: 404 })\n }\n\n const now = new Date()\n const activeFrom = couponData.activeFrom ? new Date(couponData.activeFrom) : null\n const activeUntil = couponData.activeUntil ? new Date(couponData.activeUntil) : null\n\n if (activeFrom && now < activeFrom) {\n return Response.json({ success: false, error: 'Coupon is not yet active' }, { status: 400 })\n }\n\n if (activeUntil && now > activeUntil) {\n return Response.json({ success: false, error: 'Coupon has expired' }, { status: 400 })\n }\n\n if (couponData.usageLimit && couponData.usageCount >= couponData.usageLimit) {\n return Response.json({ success: false, error: 'Coupon usage limit exceeded' }, { status: 400 })\n }\n\n if (\n couponData.perCustomerLimit != null &&\n couponData.perCustomerLimit > 0 &&\n typeof customerEmail === 'string' &&\n customerEmail.trim().length > 0\n ) {\n const email = customerEmail.trim()\n const ordersQuery = await payload.find({\n collection: pluginConfig.orderIntegration.ordersSlug,\n where: {\n and: [\n { [fields.orderAppliedCouponField]: { equals: couponData.id } },\n {\n [pluginConfig.orderIntegration.orderCustomerEmailField]: {\n equals: email,\n },\n },\n {\n [pluginConfig.orderIntegration.orderPaymentStatusField]: {\n equals: pluginConfig.orderIntegration.orderPaidStatusValue,\n },\n },\n ],\n },\n limit: 0,\n })\n\n if (ordersQuery.totalDocs >= couponData.perCustomerLimit) {\n return Response.json(\n {\n success: false,\n error: 'You have reached the maximum uses for this coupon.',\n },\n { status: 400 },\n )\n }\n }\n\n // cartValue is the caller-supplied cart subtotal (pre-discount baseline).\n // Using the subtotal ensures min/max checks and discount calculations are\n // always based on the original item total, not a post-discount total.\n if (cartValue !== undefined) {\n const minOrderValue = couponData.minOrderValue\n const maxOrderValue = couponData.maxOrderValue\n\n if (minOrderValue && cartValue < minOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderValue} ${pluginConfig.defaultCurrency} required`,\n },\n { status: 400 },\n )\n }\n\n if (maxOrderValue && cartValue > maxOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Maximum order value of ${maxOrderValue} ${pluginConfig.defaultCurrency} exceeded`,\n },\n { status: 400 },\n )\n }\n }\n\n let discount = 0\n if (cartValue !== undefined) {\n if (couponData.type === 'percentage') {\n discount = roundTo2((cartValue * couponData.value) / 100)\n if (couponData.maxDiscountAmount != null && discount > couponData.maxDiscountAmount) {\n discount = roundTo2(couponData.maxDiscountAmount)\n }\n } else if (couponData.type === 'fixed') {\n discount = roundTo2(couponData.value)\n if (discount > cartValue) discount = roundTo2(cartValue)\n }\n }\n\n return Response.json({\n success: true,\n coupon: {\n code: couponData.code,\n type: couponData.type,\n value: couponData.value,\n description: couponData.description,\n },\n discount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nasync function validateReferralCode({\n payload,\n normalizedCode,\n cartID,\n pluginConfig,\n}: {\n payload: any\n normalizedCode: string\n cartID?: string | number\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const collections = pluginConfig.integration.collections\n const resolvers = pluginConfig.integration.resolvers\n\n const referralData = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.referralCodesSlug,\n normalizedCode,\n })\n\n if (!referralData) {\n return Response.json({ success: false, error: 'Referral code not found' }, { status: 404 })\n }\n\n if (!referralData.isActive) {\n return Response.json({ success: false, error: 'Referral code is not active' }, { status: 400 })\n }\n\n if (referralData.expiresAt && new Date() > new Date(referralData.expiresAt)) {\n return Response.json({ success: false, error: 'Referral code has expired' }, { status: 400 })\n }\n\n if (referralData.usageLimit && referralData.usageCount >= referralData.usageLimit) {\n return Response.json(\n { success: false, error: 'Referral code usage limit exceeded' },\n { status: 400 },\n )\n }\n\n const programId = relationId(referralData.program as RelationValue)\n if (programId == null) {\n return Response.json({ success: false, error: 'Referral program not found' }, { status: 404 })\n }\n\n const program = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n\n if (!program || !program.isActive) {\n return Response.json(\n { success: false, error: 'Referral program is not active' },\n { status: 400 },\n )\n }\n\n const cart = cartID\n ? await payload.findByID({\n collection: collections.cartsSlug,\n id: cartID,\n depth: 2,\n })\n : null\n\n // Use cartSubtotal (pre-discount baseline) for min-order enforcement and\n // commission calculations, consistent with applyCoupon and recalculateCart.\n const cartSubtotal = cart\n ? Number(resolvers.getCartSubtotal(cart)) || Number(resolvers.getCartTotal(cart)) || 0\n : 0\n const cartTotal = cartSubtotal\n\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && cartTotal < minOrderAmount) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderAmount} ${pluginConfig.defaultCurrency} required for this referral program`,\n },\n { status: 400 },\n )\n }\n\n const { partnerCommission, customerDiscount } = calculateCommissionAndDiscount({\n cartItems: cart ? resolvers.getCartItems(cart) : [],\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const cappedCustomerDiscount =\n cartTotal > 0 ? Math.min(customerDiscount, cartTotal) : customerDiscount\n const roundedPartnerCommission = roundTo2(partnerCommission)\n const roundedCustomerDiscount = roundTo2(cappedCustomerDiscount)\n\n return Response.json({\n success: true,\n referralCode: {\n code: referralData.code,\n description: `Get ${roundedCustomerDiscount.toFixed(2)} discount with this referral code`,\n },\n partnerCommission: roundedPartnerCommission,\n customerDiscount: roundedCustomerDiscount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nexport const validateCouponEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.validateCoupon,\n method: 'post',\n handler: validateCouponHandler({ pluginConfig }),\n})\n","import type { CollectionBeforeChangeHook } from 'payload'\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n calculateCouponDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction writeField(doc: Record<string, unknown>, field: string, value: unknown): void {\n doc[field] = value\n}\n\nfunction clearCouponFields(\n target: Record<string, unknown>,\n fields: SanitizedCouponPluginOptions['integration']['fields'],\n): void {\n writeField(target, fields.cartAppliedCouponField, null)\n writeField(target, fields.cartDiscountAmountField, 0)\n}\n\nfunction clearReferralFields(\n target: Record<string, unknown>,\n fields: SanitizedCouponPluginOptions['integration']['fields'],\n): void {\n writeField(target, fields.cartAppliedReferralCodeField, null)\n writeField(target, fields.cartPartnerCommissionField, 0)\n writeField(target, fields.cartCustomerDiscountField, 0)\n}\n\nexport const recalculateCartHook =\n (pluginConfig: SanitizedCouponPluginOptions): CollectionBeforeChangeHook =>\n async ({ data, req, originalDoc }) => {\n if (!req.payload) return data\n\n const integration = pluginConfig.integration || ({} as any)\n const collections = integration.collections || {\n cartsSlug: 'carts',\n ordersSlug: 'orders',\n productsSlug: 'products',\n usersSlug: 'users',\n categoriesSlug: 'categories',\n tagsSlug: 'tags',\n }\n const fields = integration.fields || {\n cartItemsField: 'items',\n cartSubtotalField: 'subtotal',\n cartTotalField: 'total',\n cartAppliedCouponField: 'appliedCoupon',\n cartAppliedReferralCodeField: 'appliedReferralCode',\n cartDiscountAmountField: 'discountAmount',\n cartCustomerDiscountField: 'customerDiscount',\n cartPartnerCommissionField: 'partnerCommission',\n orderAppliedCouponField: 'appliedCoupon',\n orderAppliedReferralCodeField: 'appliedReferralCode',\n orderDiscountAmountField: 'discountAmount',\n orderCustomerDiscountField: 'customerDiscount',\n orderPartnerCommissionField: 'partnerCommission',\n orderCustomerEmailField: 'customerEmail',\n orderPaymentStatusField: 'paymentStatus',\n orderCreatedAtField: 'createdAt',\n productPriceField: 'price',\n productCurrencyCodeField: 'currencyCode',\n }\n const resolvers = integration.resolvers || {\n getUserID: ({ user }: { user?: unknown }) => {\n if (!user || typeof user !== 'object') return null\n const id = (user as Record<string, unknown>).id\n if (typeof id === 'string' || typeof id === 'number') return id\n return null\n },\n getCartItems: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return []\n const value = (cart as Record<string, unknown>)[fields.cartItemsField]\n return Array.isArray(value) ? value : []\n },\n getCartSubtotal: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[fields.cartSubtotalField]\n return typeof value === 'number' ? value : 0\n },\n getCartTotal: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[fields.cartTotalField]\n return typeof value === 'number' ? value : 0\n },\n isOrderPaid: (_order: unknown) => false,\n getProductUnitPrice: ({ item, product, variant, currencyCode }: any) => {\n if (item && typeof item === 'object') {\n const itemPrice = (item as Record<string, unknown>).price\n if (typeof itemPrice === 'number') return itemPrice\n const unitPrice = (item as Record<string, unknown>).unitPrice\n if (typeof unitPrice === 'number') return unitPrice\n }\n\n const readPrice = (entity: unknown, code?: string) => {\n if (!entity || typeof entity !== 'object') return undefined\n const map = entity as Record<string, unknown>\n if (code && typeof code === 'string') {\n const key = `priceIn${code.toUpperCase()}`\n const value = map[key]\n if (typeof value === 'number') return value\n }\n const base = map.price\n return typeof base === 'number' ? base : undefined\n }\n\n return readPrice(variant, currencyCode) ?? readPrice(product, currencyCode) ?? 0\n },\n }\n\n const mutableData = (data || {}) as Record<string, unknown>\n const original = (originalDoc || {}) as Record<string, unknown>\n\n const effectiveItems =\n readField<any[]>(mutableData, fields.cartItemsField) ??\n readField<any[]>(original, fields.cartItemsField) ??\n []\n\n const effectiveAppliedReferral =\n readField<RelationValue>(mutableData, fields.cartAppliedReferralCodeField) !== undefined\n ? readField<RelationValue>(mutableData, fields.cartAppliedReferralCodeField)\n : readField<RelationValue>(original, fields.cartAppliedReferralCodeField)\n\n const effectiveAppliedCoupon =\n readField<RelationValue>(mutableData, fields.cartAppliedCouponField) !== undefined\n ? readField<RelationValue>(mutableData, fields.cartAppliedCouponField)\n : readField<RelationValue>(original, fields.cartAppliedCouponField)\n\n if (!Array.isArray(effectiveItems) || effectiveItems.length === 0) {\n clearReferralFields(mutableData, fields)\n clearCouponFields(mutableData, fields)\n writeField(mutableData, fields.cartTotalField, 0)\n return mutableData\n }\n\n const getRelationID = (value: unknown): string | number | null =>\n relationId(value as RelationValue)\n\n const productIds = effectiveItems\n .map((item: any) => getRelationID(item?.product))\n .filter((id: string | number | null): id is string | number => id != null)\n\n let productsMap = new Map<string, any>()\n if (productIds.length > 0) {\n const productsQuery = await req.payload.find({\n collection: collections.productsSlug,\n where: {\n id: { in: productIds },\n },\n limit: productIds.length,\n })\n productsMap = new Map((productsQuery?.docs || []).map((p: any) => [String(p.id), p]))\n }\n\n let calculatedSubtotal = 0\n const enrichedItems = effectiveItems.map((item: any) => {\n const pid = getRelationID(item?.product)\n const product = pid != null ? productsMap.get(String(pid)) || {} : {}\n const variant = typeof item?.variant === 'object' ? item.variant : undefined\n\n const unitPrice = Number(\n resolvers.getProductUnitPrice({\n item,\n product,\n variant,\n currencyCode: pluginConfig.defaultCurrency,\n }),\n )\n\n const quantity =\n typeof item?.quantity === 'number' && Number.isFinite(item.quantity) ? item.quantity : 1\n const safeUnitPrice = Number.isFinite(unitPrice) ? unitPrice : 0\n calculatedSubtotal += safeUnitPrice * quantity\n\n return {\n ...item,\n product,\n price: safeUnitPrice,\n quantity,\n }\n })\n\n writeField(mutableData, fields.cartSubtotalField, roundTo2(calculatedSubtotal))\n\n let customerDiscount = 0\n let couponDiscount = 0\n\n const appliedReferralID = relationId(effectiveAppliedReferral)\n if (pluginConfig.enableReferrals && appliedReferralID != null) {\n const referralQuery = await req.payload.find({\n collection: pluginConfig.collections.referralCodesSlug,\n where: { id: { equals: appliedReferralID } },\n limit: 1,\n depth: 1,\n })\n\n const referralCode = referralQuery?.docs?.[0]\n if (!referralCode || referralCode.isActive === false) {\n clearReferralFields(mutableData, fields)\n } else {\n const programId = relationId(referralCode.program as RelationValue)\n const program =\n typeof referralCode.program === 'object'\n ? referralCode.program\n : programId != null\n ? await req.payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n : null\n\n if (!program || program.isActive === false) {\n clearReferralFields(mutableData, fields)\n } else {\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && calculatedSubtotal < minOrderAmount) {\n clearReferralFields(mutableData, fields)\n } else {\n const result = calculateCommissionAndDiscount({\n cartItems: enrichedItems,\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal: calculatedSubtotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const roundedPartnerCommission = roundTo2(result.partnerCommission)\n const roundedCustomerDiscount = roundTo2(Math.max(0, result.customerDiscount))\n\n writeField(mutableData, fields.cartPartnerCommissionField, roundedPartnerCommission)\n writeField(mutableData, fields.cartCustomerDiscountField, roundedCustomerDiscount)\n customerDiscount = roundedCustomerDiscount\n }\n }\n }\n } else {\n if (readField(mutableData, fields.cartAppliedReferralCodeField) === null) {\n clearReferralFields(mutableData, fields)\n }\n }\n\n const appliedCouponID = relationId(effectiveAppliedCoupon)\n const canUseCouponWithReferral =\n !pluginConfig.enableReferrals ||\n pluginConfig.referralConfig.allowBothSystems ||\n relationId(readField(mutableData, fields.cartAppliedReferralCodeField) as RelationValue) ==\n null\n\n if (appliedCouponID != null && canUseCouponWithReferral) {\n const couponQuery = await req.payload.find({\n collection: pluginConfig.collections.couponsSlug,\n where: { id: { equals: appliedCouponID } },\n limit: 1,\n })\n\n const coupon = couponQuery?.docs?.[0]\n if (!coupon) {\n clearCouponFields(mutableData, fields)\n } else {\n const now = new Date()\n const activeFrom = coupon.activeFrom ? new Date(coupon.activeFrom) : null\n const activeUntil = coupon.activeUntil ? new Date(coupon.activeUntil) : null\n const isValidDate =\n (!activeFrom || now >= activeFrom) && (!activeUntil || now <= activeUntil)\n const underUsage =\n !coupon.usageLimit || Number(coupon.usageCount || 0) < Number(coupon.usageLimit || 0)\n\n if (!isValidDate || !underUsage) {\n clearCouponFields(mutableData, fields)\n } else {\n couponDiscount = roundTo2(\n calculateCouponDiscount({ coupon, cartTotal: calculatedSubtotal }),\n )\n writeField(mutableData, fields.cartDiscountAmountField, couponDiscount)\n }\n }\n } else {\n if (readField(mutableData, fields.cartAppliedCouponField) === null) {\n clearCouponFields(mutableData, fields)\n writeField(mutableData, fields.cartCustomerDiscountField, 0)\n writeField(mutableData, fields.cartPartnerCommissionField, 0)\n writeField(\n mutableData,\n fields.cartTotalField,\n roundTo2(Number(resolvers.getCartSubtotal(mutableData)) || calculatedSubtotal),\n )\n return mutableData\n }\n }\n\n const nextTotal = roundTo2(Math.max(0, calculatedSubtotal - customerDiscount - couponDiscount))\n writeField(mutableData, fields.cartTotalField, nextTotal)\n\n return mutableData\n }\n","import type { Payload } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nexport type OrderWithCouponFields = {\n id?: string | number\n [key: string]: unknown\n}\n\nexport type RecordCouponUsageResult = {\n recordedCoupon: boolean\n recordedReferral: boolean\n alreadyRecorded: boolean\n}\n\nconst USAGE_MARKER_FIELD = 'couponUsageRecordedAt'\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction asNumber(value: unknown): number {\n return typeof value === 'number' && Number.isFinite(value) ? value : 0\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nasync function markOrderUsageRecorded({\n payload,\n pluginConfig,\n orderID,\n}: {\n payload: Payload\n pluginConfig: SanitizedCouponPluginOptions\n orderID: string | number\n}): Promise<boolean> {\n const ordersSlug = pluginConfig.integration.collections.ordersSlug\n\n const latestOrder = await payload.findByID({\n collection: ordersSlug as any,\n id: orderID,\n depth: 0,\n })\n\n if (!latestOrder) return false\n\n const alreadyRecorded = Boolean(readField(latestOrder, USAGE_MARKER_FIELD))\n if (alreadyRecorded) return false\n\n await payload.update({\n collection: ordersSlug as any,\n id: orderID,\n data: {\n [USAGE_MARKER_FIELD]: new Date().toISOString(),\n },\n })\n\n return true\n}\n\n/**\n * Record coupon and referral usage when an order is placed successfully.\n * This function is idempotent and integration-field aware.\n *\n * Behavior:\n * - Uses configured order field names from `pluginConfig.integration.fields`\n * - Uses configured paid-order resolver (`integration.resolvers.isOrderPaid`)\n * - Marks the order with `couponUsageRecordedAt` before mutating counters to avoid duplicate counting\n * - If marker already exists, returns `alreadyRecorded: true` and performs no increments\n */\nexport async function recordCouponUsageForOrder(\n payload: Payload,\n order: OrderWithCouponFields,\n pluginConfig: SanitizedCouponPluginOptions,\n): Promise<RecordCouponUsageResult> {\n const result: RecordCouponUsageResult = {\n recordedCoupon: false,\n recordedReferral: false,\n alreadyRecorded: false,\n }\n\n const orderID = order.id\n if (orderID == null) return result\n\n const isPaid = await Promise.resolve(pluginConfig.integration.resolvers.isOrderPaid(order))\n if (!isPaid) return result\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canRecordOrderUsage({\n req: {},\n user: undefined,\n payload,\n order,\n }),\n )\n\n if (!policyAllowed) return result\n\n const acquiredMarker = await markOrderUsageRecorded({\n payload,\n pluginConfig,\n orderID,\n })\n\n if (!acquiredMarker) {\n result.alreadyRecorded = true\n return result\n }\n\n const fields = pluginConfig.integration.fields\n\n const couponField = fields.orderAppliedCouponField\n const referralField = fields.orderAppliedReferralCodeField\n const partnerCommissionField = fields.orderPartnerCommissionField\n\n const couponId = relationId(readField(order, couponField) as RelationValue)\n const referralCodeId = relationId(readField(order, referralField) as RelationValue)\n const commission = asNumber(readField(order, partnerCommissionField))\n\n if (couponId) {\n const coupon = await payload.findByID({\n collection: pluginConfig.collections.couponsSlug as any,\n id: couponId,\n depth: 0,\n })\n\n if (coupon) {\n const currentUsage = asNumber((coupon as Record<string, unknown>).usageCount)\n await payload.update({\n collection: pluginConfig.collections.couponsSlug as any,\n id: couponId,\n data: {\n usageCount: currentUsage + 1,\n },\n })\n result.recordedCoupon = true\n }\n }\n\n if (referralCodeId) {\n const referralCode = await payload.findByID({\n collection: pluginConfig.collections.referralCodesSlug as any,\n id: referralCodeId,\n depth: 0,\n })\n\n if (referralCode) {\n const rc = referralCode as Record<string, unknown>\n const currentTotal = asNumber(rc.totalEarnings)\n const currentPending = asNumber(rc.pendingEarnings)\n const currentUsage = asNumber(rc.usageCount)\n const currentSuccessful = asNumber(rc.successfulReferralsCount)\n\n await payload.update({\n collection: pluginConfig.collections.referralCodesSlug as any,\n id: referralCodeId,\n data: {\n usageCount: currentUsage + 1,\n successfulReferralsCount: currentSuccessful + 1,\n totalEarnings: currentTotal + commission,\n pendingEarnings: currentPending + commission,\n },\n })\n\n result.recordedReferral = true\n }\n }\n\n return result\n}\n","import type { CouponPluginOptions, SanitizedCouponPluginOptions } from '../types'\nimport { getCartItemUnitPrice } from './pricing'\nimport { isAdminUser, isPartnerUser } from './userRoles'\n\nconst toCleanStringArray = (value: unknown): string[] =>\n Array.isArray(value)\n ? value\n .filter((item): item is string => typeof item === 'string')\n .map((item) => item.trim())\n .filter(Boolean)\n : []\n\nconst toBoolean = (value: unknown, fallback: boolean): boolean => {\n if (typeof value === 'boolean') return value\n if (typeof value === 'number') {\n if (value === 1) return true\n if (value === 0) return false\n }\n if (typeof value === 'string') {\n const normalized = value.trim().toLowerCase()\n if (\n normalized === 'true' ||\n normalized === '1' ||\n normalized === 'yes' ||\n normalized === 'on'\n ) {\n return true\n }\n if (\n normalized === 'false' ||\n normalized === '0' ||\n normalized === 'no' ||\n normalized === 'off'\n ) {\n return false\n }\n }\n return fallback\n}\n\nexport const sanitizePluginConfig = ({\n pluginConfig,\n}: {\n pluginConfig: CouponPluginOptions\n}): SanitizedCouponPluginOptions => {\n const roleConfig = {\n roleFieldPaths:\n toCleanStringArray(pluginConfig?.roleConfig?.roleFieldPaths).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.roleFieldPaths)\n : ['role', 'roles'],\n adminRoleValues:\n toCleanStringArray(pluginConfig?.roleConfig?.adminRoleValues).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.adminRoleValues)\n : ['admin'],\n partnerRoleValues:\n toCleanStringArray(pluginConfig?.roleConfig?.partnerRoleValues).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.partnerRoleValues)\n : ['partner'],\n customRoleResolver:\n typeof pluginConfig?.roleConfig?.customRoleResolver === 'function'\n ? pluginConfig.roleConfig.customRoleResolver\n : undefined,\n }\n\n const normalizedAllowedTotalCommissionTypes = Array.isArray(\n pluginConfig?.referralConfig?.allowedTotalCommissionTypes,\n )\n ? [\n ...new Set(\n pluginConfig.referralConfig.allowedTotalCommissionTypes.filter(\n (value): value is 'fixed' | 'percentage' => value === 'fixed' || value === 'percentage',\n ),\n ),\n ]\n : []\n\n const integrationCollections = {\n cartsSlug:\n typeof pluginConfig?.integration?.collections?.cartsSlug === 'string' &&\n pluginConfig.integration.collections.cartsSlug.trim().length > 0\n ? pluginConfig.integration.collections.cartsSlug.trim()\n : 'carts',\n ordersSlug:\n typeof pluginConfig?.integration?.collections?.ordersSlug === 'string' &&\n pluginConfig.integration.collections.ordersSlug.trim().length > 0\n ? pluginConfig.integration.collections.ordersSlug.trim()\n : 'orders',\n productsSlug:\n typeof pluginConfig?.integration?.collections?.productsSlug === 'string' &&\n pluginConfig.integration.collections.productsSlug.trim().length > 0\n ? pluginConfig.integration.collections.productsSlug.trim()\n : 'products',\n usersSlug:\n typeof pluginConfig?.integration?.collections?.usersSlug === 'string' &&\n pluginConfig.integration.collections.usersSlug.trim().length > 0\n ? pluginConfig.integration.collections.usersSlug.trim()\n : 'users',\n categoriesSlug:\n typeof pluginConfig?.integration?.collections?.categoriesSlug === 'string' &&\n pluginConfig.integration.collections.categoriesSlug.trim().length > 0\n ? pluginConfig.integration.collections.categoriesSlug.trim()\n : 'categories',\n tagsSlug:\n typeof pluginConfig?.integration?.collections?.tagsSlug === 'string' &&\n pluginConfig.integration.collections.tagsSlug.trim().length > 0\n ? pluginConfig.integration.collections.tagsSlug.trim()\n : 'tags',\n }\n\n const integrationFields = {\n cartItemsField: pluginConfig?.integration?.fields?.cartItemsField?.trim() || 'items',\n cartSubtotalField: pluginConfig?.integration?.fields?.cartSubtotalField?.trim() || 'subtotal',\n cartTotalField: pluginConfig?.integration?.fields?.cartTotalField?.trim() || 'total',\n cartAppliedCouponField:\n pluginConfig?.integration?.fields?.cartAppliedCouponField?.trim() || 'appliedCoupon',\n cartAppliedReferralCodeField:\n pluginConfig?.integration?.fields?.cartAppliedReferralCodeField?.trim() ||\n 'appliedReferralCode',\n cartDiscountAmountField:\n pluginConfig?.integration?.fields?.cartDiscountAmountField?.trim() || 'discountAmount',\n cartCustomerDiscountField:\n pluginConfig?.integration?.fields?.cartCustomerDiscountField?.trim() || 'customerDiscount',\n cartPartnerCommissionField:\n pluginConfig?.integration?.fields?.cartPartnerCommissionField?.trim() || 'partnerCommission',\n orderAppliedCouponField:\n pluginConfig?.integration?.fields?.orderAppliedCouponField?.trim() || 'appliedCoupon',\n orderAppliedReferralCodeField:\n pluginConfig?.integration?.fields?.orderAppliedReferralCodeField?.trim() ||\n 'appliedReferralCode',\n orderDiscountAmountField:\n pluginConfig?.integration?.fields?.orderDiscountAmountField?.trim() || 'discountAmount',\n orderCustomerDiscountField:\n pluginConfig?.integration?.fields?.orderCustomerDiscountField?.trim() || 'customerDiscount',\n orderPartnerCommissionField:\n pluginConfig?.integration?.fields?.orderPartnerCommissionField?.trim() || 'partnerCommission',\n orderCustomerEmailField:\n pluginConfig?.integration?.fields?.orderCustomerEmailField?.trim() ||\n pluginConfig?.orderIntegration?.orderCustomerEmailField?.trim() ||\n 'customerEmail',\n orderPaymentStatusField:\n pluginConfig?.integration?.fields?.orderPaymentStatusField?.trim() ||\n pluginConfig?.orderIntegration?.orderPaymentStatusField?.trim() ||\n 'paymentStatus',\n orderCreatedAtField:\n pluginConfig?.integration?.fields?.orderCreatedAtField?.trim() || 'createdAt',\n productPriceField: pluginConfig?.integration?.fields?.productPriceField?.trim() || 'price',\n productCurrencyCodeField:\n pluginConfig?.integration?.fields?.productCurrencyCodeField?.trim() || 'currencyCode',\n }\n\n const integrationResolvers = {\n getUserID:\n pluginConfig?.integration?.resolvers?.getUserID ||\n (({ user }: { user?: unknown }) => {\n if (!user || typeof user !== 'object') return null\n const id = (user as Record<string, unknown>).id\n if (typeof id === 'string' || typeof id === 'number') return id\n return null\n }),\n getCartItems:\n pluginConfig?.integration?.resolvers?.getCartItems ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return []\n const value = (cart as Record<string, unknown>)[integrationFields.cartItemsField]\n return Array.isArray(value) ? value : []\n }),\n getCartSubtotal:\n pluginConfig?.integration?.resolvers?.getCartSubtotal ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[integrationFields.cartSubtotalField]\n return typeof value === 'number' ? value : 0\n }),\n getCartTotal:\n pluginConfig?.integration?.resolvers?.getCartTotal ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[integrationFields.cartTotalField]\n return typeof value === 'number' ? value : 0\n }),\n isOrderPaid:\n pluginConfig?.integration?.resolvers?.isOrderPaid ||\n ((order: unknown) => {\n if (!order || typeof order !== 'object') return false\n const status = (order as Record<string, unknown>)[integrationFields.orderPaymentStatusField]\n return status === (pluginConfig?.orderIntegration?.orderPaidStatusValue ?? 'paid')\n }),\n getProductUnitPrice:\n pluginConfig?.integration?.resolvers?.getProductUnitPrice ||\n ((args: { item: unknown; product: unknown; variant?: unknown; currencyCode?: string }) =>\n getCartItemUnitPrice({\n item: (args.item as any) ?? null,\n product: (args.product as any) ?? null,\n variant: (args.variant as any) ?? null,\n currencyCode: args.currencyCode || 'USD',\n })),\n }\n\n return {\n enabled: toBoolean(pluginConfig?.enabled, true),\n enableReferrals: toBoolean(pluginConfig?.enableReferrals, false),\n allowStackWithOtherCoupons: toBoolean(pluginConfig?.allowStackWithOtherCoupons, false),\n defaultCurrency:\n typeof pluginConfig?.defaultCurrency === 'string' &&\n pluginConfig.defaultCurrency.length > 0 &&\n pluginConfig.defaultCurrency.length <= 3\n ? pluginConfig.defaultCurrency\n : 'USD',\n collections: {\n couponsSlug:\n typeof pluginConfig?.collections?.couponsSlug === 'string' &&\n pluginConfig.collections.couponsSlug.trim().length > 0 &&\n pluginConfig.collections.couponsSlug.length <= 100\n ? pluginConfig.collections.couponsSlug\n : 'coupons',\n referralProgramsSlug:\n typeof pluginConfig?.collections?.referralProgramsSlug === 'string' &&\n pluginConfig.collections.referralProgramsSlug.trim().length > 0 &&\n pluginConfig.collections.referralProgramsSlug.length <= 100\n ? pluginConfig.collections.referralProgramsSlug\n : 'referral-programs',\n referralCodesSlug:\n typeof pluginConfig?.collections?.referralCodesSlug === 'string' &&\n pluginConfig.collections.referralCodesSlug.trim().length > 0 &&\n pluginConfig.collections.referralCodesSlug.length <= 100\n ? pluginConfig.collections.referralCodesSlug\n : 'referral-codes',\n referralPartnersSlug:\n typeof pluginConfig?.collections?.referralPartnersSlug === 'string' &&\n pluginConfig.collections.referralPartnersSlug.trim().length > 0 &&\n pluginConfig.collections.referralPartnersSlug.length <= 100\n ? pluginConfig.collections.referralPartnersSlug\n : 'referral-partners',\n },\n endpoints: {\n applyCoupon:\n typeof pluginConfig?.endpoints?.applyCoupon === 'string' &&\n pluginConfig.endpoints.applyCoupon.trim().length > 0\n ? pluginConfig.endpoints.applyCoupon.trim()\n : '/coupons/apply',\n validateCoupon:\n typeof pluginConfig?.endpoints?.validateCoupon === 'string' &&\n pluginConfig.endpoints.validateCoupon.trim().length > 0\n ? pluginConfig.endpoints.validateCoupon.trim()\n : '/coupons/validate',\n partnerStats:\n typeof pluginConfig?.endpoints?.partnerStats === 'string' &&\n pluginConfig.endpoints.partnerStats.trim().length > 0\n ? pluginConfig.endpoints.partnerStats.trim()\n : '/referrals/partner-stats',\n recordOrderUsage:\n typeof pluginConfig?.endpoints?.recordOrderUsage === 'string' &&\n pluginConfig.endpoints.recordOrderUsage.trim().length > 0\n ? pluginConfig.endpoints.recordOrderUsage.trim()\n : '/coupons/record-order-usage',\n },\n autoIntegrate: pluginConfig?.autoIntegrate !== false,\n access: {\n canUseCoupons:\n typeof pluginConfig?.access?.canUseCoupons === 'function'\n ? pluginConfig.access.canUseCoupons\n : () => true,\n canUseReferrals:\n typeof pluginConfig?.access?.canUseReferrals === 'function'\n ? pluginConfig.access.canUseReferrals\n : () => false,\n isAdmin:\n typeof pluginConfig?.access?.isAdmin === 'function'\n ? pluginConfig.access.isAdmin\n : ({ req }) => isAdminUser({ user: req?.user, roleConfig }),\n isPartner:\n typeof pluginConfig?.access?.isPartner === 'function'\n ? pluginConfig.access.isPartner\n : ({ req }) => isPartnerUser({ user: req?.user, roleConfig }),\n },\n policies: {\n canApplyCoupon:\n typeof pluginConfig?.policies?.canApplyCoupon === 'function'\n ? pluginConfig.policies.canApplyCoupon\n : () => true,\n canApplyReferral:\n typeof pluginConfig?.policies?.canApplyReferral === 'function'\n ? pluginConfig.policies.canApplyReferral\n : () => true,\n canViewPartnerStats:\n typeof pluginConfig?.policies?.canViewPartnerStats === 'function'\n ? pluginConfig.policies.canViewPartnerStats\n : ({ req }) =>\n isPartnerUser({ user: (req as any)?.user, roleConfig }) ||\n isAdminUser({ user: (req as any)?.user, roleConfig }),\n canRecordOrderUsage:\n typeof pluginConfig?.policies?.canRecordOrderUsage === 'function'\n ? pluginConfig.policies.canRecordOrderUsage\n : () => true,\n },\n integration: {\n collections: integrationCollections,\n fields: integrationFields,\n resolvers: integrationResolvers,\n },\n referralConfig: {\n allowBothSystems: pluginConfig?.referralConfig?.allowBothSystems ?? false,\n singleCodePerCart: pluginConfig?.referralConfig?.singleCodePerCart ?? true,\n defaultPartnerSplit: pluginConfig?.referralConfig?.defaultPartnerSplit ?? 70,\n defaultCustomerSplit: pluginConfig?.referralConfig?.defaultCustomerSplit ?? 30,\n allowedTotalCommissionTypes:\n normalizedAllowedTotalCommissionTypes.length > 0\n ? normalizedAllowedTotalCommissionTypes\n : ['fixed', 'percentage'],\n },\n adminGroups: {\n couponsGroup: pluginConfig?.adminGroups?.couponsGroup ?? 'Coupons',\n referralsGroup: pluginConfig?.adminGroups?.referralsGroup ?? 'Referrals',\n },\n partnerDashboard: {\n enabled: pluginConfig?.partnerDashboard?.enabled ?? true,\n showEarningsSummary: pluginConfig?.partnerDashboard?.showEarningsSummary ?? true,\n showReferralPerformance: pluginConfig?.partnerDashboard?.showReferralPerformance ?? true,\n showRecentReferrals: pluginConfig?.partnerDashboard?.showRecentReferrals ?? true,\n showCommissionBreakdown: pluginConfig?.partnerDashboard?.showCommissionBreakdown ?? true,\n },\n orderIntegration: {\n ordersSlug:\n typeof pluginConfig?.orderIntegration?.ordersSlug === 'string' &&\n pluginConfig.orderIntegration.ordersSlug.trim().length > 0\n ? pluginConfig.orderIntegration.ordersSlug\n : integrationCollections.ordersSlug,\n orderCustomerEmailField: integrationFields.orderCustomerEmailField,\n orderPaymentStatusField: integrationFields.orderPaymentStatusField,\n orderPaidStatusValue:\n typeof pluginConfig?.orderIntegration?.orderPaidStatusValue === 'string'\n ? pluginConfig.orderIntegration.orderPaidStatusValue\n : 'paid',\n },\n roleConfig,\n }\n}\n","import type { Config } from 'payload'\n\nimport { createCouponsCollection } from './collections/createCouponsCollection'\nimport { createReferralCodesCollection } from './collections/createReferralCodesCollection'\nimport { createReferralProgramsCollection } from './collections/createReferralProgramsCollection'\nimport { applyCouponEndpoint } from './endpoints/applyCoupon'\nimport { partnerStatsEndpoint } from './endpoints/partnerStats'\nimport { validateCouponEndpoint } from './endpoints/validateCoupon'\nimport { recalculateCartHook } from './hooks/recalculateCart'\nimport { CouponPluginOptions } from './types'\nimport { recordCouponUsageForOrder } from './utilities/recordCouponUsageForOrder'\nimport { sanitizePluginConfig } from './utilities/sanitizePluginConfig'\n\nconst RECALCULATE_HOOK_KEY = '__payloadEcommerceCouponRecalculateHook__'\n\ntype GenericCollection = {\n slug: string\n fields?: Array<Record<string, unknown>>\n endpoints?: Array<Record<string, unknown>>\n hooks?: {\n beforeChange?: Array<unknown>\n [key: string]: unknown\n }\n [key: string]: unknown\n}\n\nconst asArray = <T>(value: T[] | null | undefined): T[] => (Array.isArray(value) ? value : [])\n\nconst hasNamedField = (collection: GenericCollection, fieldName: string): boolean =>\n asArray(collection.fields).some((f) => f?.name === fieldName)\n\nconst normalizePath = (path: string): string => {\n if (!path) return '/'\n const withLeadingSlash = path.startsWith('/') ? path : `/${path}`\n return withLeadingSlash.replace(/\\/+$/, '') || '/'\n}\n\nconst toCollectionEndpointPath = ({\n endpointPath,\n collectionSlug,\n}: {\n endpointPath: string\n collectionSlug: string\n}): string | null => {\n const normalizedEndpointPath = normalizePath(endpointPath)\n const normalizedCollectionPath = normalizePath(`/${collectionSlug}`)\n\n if (!normalizedEndpointPath.startsWith(`${normalizedCollectionPath}/`)) return null\n const relative = normalizedEndpointPath.slice(normalizedCollectionPath.length)\n return relative.startsWith('/') ? relative : `/${relative}`\n}\n\nconst ensureCouponCollectionEndpoints = ({\n collection,\n pluginConfig,\n}: {\n collection: GenericCollection\n pluginConfig: ReturnType<typeof sanitizePluginConfig>\n}): GenericCollection => {\n const couponsSlug = pluginConfig.collections.couponsSlug\n const applyPath = toCollectionEndpointPath({\n endpointPath: pluginConfig.endpoints.applyCoupon,\n collectionSlug: couponsSlug,\n })\n const validatePath = toCollectionEndpointPath({\n endpointPath: pluginConfig.endpoints.validateCoupon,\n collectionSlug: couponsSlug,\n })\n\n if (!applyPath && !validatePath) return collection\n\n const endpoints = asArray(collection.endpoints)\n const endpointKeys = new Set(\n endpoints.map((e: any) => `${(e?.method || 'get').toLowerCase()}:${e?.path || ''}`),\n )\n\n if (validatePath) {\n const validateKey = `post:${validatePath}`\n if (!endpointKeys.has(validateKey)) {\n endpointKeys.add(validateKey)\n endpoints.push({\n path: validatePath,\n method: 'post',\n handler: validateCouponEndpoint({ pluginConfig }).handler,\n })\n }\n }\n\n if (applyPath) {\n const applyKey = `post:${applyPath}`\n if (!endpointKeys.has(applyKey)) {\n endpointKeys.add(applyKey)\n endpoints.push({\n path: applyPath,\n method: 'post',\n handler: applyCouponEndpoint({ pluginConfig }).handler,\n })\n }\n }\n\n collection.endpoints = endpoints\n return collection\n}\n\nconst addFieldsToCollection = (\n config: Config,\n targetSlug: string,\n newFields: Array<Record<string, unknown>>,\n): void => {\n const collections = asArray(config.collections as GenericCollection[])\n const idx = collections.findIndex((c) => c.slug === targetSlug)\n if (idx === -1) return\n\n const collection = collections[idx]\n collection.fields = asArray(collection.fields)\n\n for (const field of newFields) {\n const name = typeof field.name === 'string' ? field.name : ''\n if (!name) continue\n if (!hasNamedField(collection, name)) {\n collection.fields.push(field)\n }\n }\n\n collections[idx] = collection\n config.collections = collections as any\n}\n\ntype RecalculateHookFn = (...args: any[]) => any\n\nconst markHook = <T extends RecalculateHookFn>(fn: T): T => {\n ;(fn as any)[RECALCULATE_HOOK_KEY] = true\n return fn\n}\n\nconst hasMarkedHook = (hook: unknown): boolean =>\n Boolean(hook && typeof hook === 'function' && (hook as any)[RECALCULATE_HOOK_KEY])\n\nconst createRecordOrderUsageEndpoint = ({\n pluginConfig,\n}: {\n pluginConfig: ReturnType<typeof sanitizePluginConfig>\n}) => ({\n path: pluginConfig.endpoints.recordOrderUsage,\n method: 'post' as const,\n handler: async (req: any) => {\n try {\n const payload = req?.payload\n const orderId = req?.data?.orderId ?? req?.json?.orderId\n\n if (!payload) {\n return Response.json(\n { success: false, error: 'Payload instance is required' },\n { status: 500 },\n )\n }\n\n if (!orderId) {\n return Response.json({ success: false, error: 'orderId is required' }, { status: 400 })\n }\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canRecordOrderUsage({\n req,\n user: req?.user,\n payload,\n order: { id: orderId },\n }),\n )\n\n if (!policyAllowed) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n const order = await payload.findByID({\n collection: pluginConfig.integration.collections.ordersSlug as any,\n id: orderId,\n })\n\n if (!order) {\n return Response.json({ success: false, error: 'Order not found' }, { status: 404 })\n }\n\n const result = await recordCouponUsageForOrder(payload, order, pluginConfig)\n\n return Response.json({\n success: true,\n result,\n })\n } catch (error) {\n console.error('record-order-usage endpoint error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n },\n})\n\nexport const payloadEcommerceCouponPlugin =\n (pluginOptions: CouponPluginOptions = {}) =>\n async (incomingConfig: Config): Promise<Config> => {\n const pluginConfig = sanitizePluginConfig({ pluginConfig: pluginOptions })\n\n if (!pluginConfig.enabled) return incomingConfig || {}\n\n if (!incomingConfig) {\n incomingConfig = { collections: [], endpoints: [] } as any\n }\n\n incomingConfig.collections = asArray(incomingConfig.collections as any) as Config['collections']\n incomingConfig.endpoints = asArray(incomingConfig.endpoints as any) as Config['endpoints']\n\n const collectionsToAdd: GenericCollection[] = []\n\n if (pluginConfig.enableReferrals) {\n let referralProgramsCollection = createReferralProgramsCollection(\n pluginConfig,\n ) as GenericCollection\n let referralCodesCollection = createReferralCodesCollection(pluginConfig) as GenericCollection\n\n if (pluginOptions.collections?.referralProgramsCollectionOverride) {\n referralProgramsCollection =\n await pluginOptions.collections.referralProgramsCollectionOverride({\n defaultCollection: referralProgramsCollection,\n })\n }\n\n if (pluginOptions.collections?.referralCodesCollectionOverride) {\n referralCodesCollection = await pluginOptions.collections.referralCodesCollectionOverride({\n defaultCollection: referralCodesCollection,\n })\n }\n\n collectionsToAdd.push(referralProgramsCollection, referralCodesCollection)\n\n if (pluginConfig.referralConfig.allowBothSystems) {\n let couponsCollection = createCouponsCollection(pluginConfig) as GenericCollection\n if (pluginOptions.collections?.couponsCollectionOverride) {\n couponsCollection = await pluginOptions.collections.couponsCollectionOverride({\n defaultCollection: couponsCollection,\n })\n }\n couponsCollection = ensureCouponCollectionEndpoints({\n collection: couponsCollection,\n pluginConfig,\n })\n collectionsToAdd.push(couponsCollection)\n }\n } else {\n let couponsCollection = createCouponsCollection(pluginConfig) as GenericCollection\n if (pluginOptions.collections?.couponsCollectionOverride) {\n couponsCollection = await pluginOptions.collections.couponsCollectionOverride({\n defaultCollection: couponsCollection,\n })\n }\n couponsCollection = ensureCouponCollectionEndpoints({\n collection: couponsCollection,\n pluginConfig,\n })\n collectionsToAdd.push(couponsCollection)\n }\n\n const existingSlugs = new Set(\n asArray(incomingConfig.collections as GenericCollection[]).map((c) => c.slug),\n )\n const toAppend = collectionsToAdd.filter((c) => !existingSlugs.has(c.slug))\n incomingConfig.collections = [\n ...asArray(incomingConfig.collections as GenericCollection[]),\n ...toAppend,\n ] as any\n\n const endpointPaths = new Set(\n asArray(incomingConfig.endpoints as any[]).map(\n (e: any) => `${e?.method || 'get'}:${e?.path || ''}`,\n ),\n )\n\n const maybePushEndpoint = (endpoint: any) => {\n const key = `${endpoint?.method || 'get'}:${endpoint?.path || ''}`\n if (!endpointPaths.has(key)) {\n endpointPaths.add(key)\n ;(incomingConfig.endpoints as any[]).push(endpoint)\n }\n }\n\n maybePushEndpoint(validateCouponEndpoint({ pluginConfig }))\n maybePushEndpoint(applyCouponEndpoint({ pluginConfig }))\n\n if (pluginOptions.endpoints?.recordOrderUsage) {\n maybePushEndpoint(createRecordOrderUsageEndpoint({ pluginConfig }))\n }\n\n if (pluginConfig.enableReferrals) {\n maybePushEndpoint(partnerStatsEndpoint({ pluginConfig }))\n }\n\n if (pluginConfig.autoIntegrate) {\n const allSlugs = new Set(\n asArray(incomingConfig.collections as GenericCollection[]).map((c) => c.slug),\n )\n\n const cartsSlug = pluginConfig.integration.collections.cartsSlug\n const ordersSlug = pluginConfig.integration.collections.ordersSlug\n\n const {\n cartAppliedReferralCodeField,\n cartPartnerCommissionField,\n cartCustomerDiscountField,\n cartAppliedCouponField,\n cartDiscountAmountField,\n orderAppliedReferralCodeField,\n orderPartnerCommissionField,\n orderCustomerDiscountField,\n orderAppliedCouponField,\n orderDiscountAmountField,\n } = pluginConfig.integration.fields\n\n if (\n pluginConfig.enableReferrals &&\n allSlugs.has(pluginConfig.collections.referralCodesSlug)\n ) {\n const cartReferralFields: Array<Record<string, unknown>> = [\n {\n name: cartAppliedReferralCodeField,\n type: 'relationship',\n relationTo: pluginConfig.collections.referralCodesSlug,\n admin: { description: 'Referral code applied to this cart' },\n },\n {\n name: cartPartnerCommissionField,\n type: 'number',\n admin: { description: 'Partner commission amount for this cart' },\n },\n {\n name: cartCustomerDiscountField,\n type: 'number',\n admin: { description: 'Customer discount amount for this cart' },\n },\n ]\n\n if (\n pluginConfig.referralConfig.allowBothSystems &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n cartReferralFields.push(\n {\n name: cartAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: { description: 'Coupon applied to this cart' },\n },\n {\n name: cartDiscountAmountField,\n type: 'number',\n admin: { description: 'Discount amount from coupon' },\n },\n )\n }\n\n addFieldsToCollection(incomingConfig, cartsSlug, cartReferralFields)\n\n const orderReferralFields: Array<Record<string, unknown>> = [\n {\n name: orderAppliedReferralCodeField,\n type: 'relationship',\n relationTo: pluginConfig.collections.referralCodesSlug,\n admin: {\n description: 'Referral code applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderPartnerCommissionField,\n type: 'number',\n admin: {\n description: 'Partner commission amount for this order',\n readOnly: true,\n },\n },\n {\n name: orderCustomerDiscountField,\n type: 'number',\n admin: {\n description: 'Customer discount amount for this order',\n readOnly: true,\n },\n },\n ]\n\n if (\n pluginConfig.referralConfig.allowBothSystems &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n orderReferralFields.push(\n {\n name: orderAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: {\n description: 'Coupon applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderDiscountAmountField,\n type: 'number',\n admin: {\n description: 'Discount amount from coupon',\n readOnly: true,\n },\n },\n )\n }\n\n addFieldsToCollection(incomingConfig, ordersSlug, orderReferralFields)\n } else if (\n !pluginConfig.enableReferrals &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n const cartCouponFields: Array<Record<string, unknown>> = [\n {\n name: cartAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: { description: 'Coupon applied to this cart' },\n },\n {\n name: cartDiscountAmountField,\n type: 'number',\n admin: { description: 'Discount amount from coupon' },\n },\n ]\n\n const orderCouponFields: Array<Record<string, unknown>> = [\n {\n name: orderAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: {\n description: 'Coupon applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderDiscountAmountField,\n type: 'number',\n admin: {\n description: 'Discount amount from coupon',\n readOnly: true,\n },\n },\n ]\n\n addFieldsToCollection(incomingConfig, cartsSlug, cartCouponFields)\n addFieldsToCollection(incomingConfig, ordersSlug, orderCouponFields)\n }\n }\n\n const cartsSlug = pluginConfig.integration.collections.cartsSlug\n const cartIndex = asArray(incomingConfig.collections as GenericCollection[]).findIndex(\n (c) => c.slug === cartsSlug,\n )\n\n if (cartIndex > -1) {\n const collection = (incomingConfig.collections as GenericCollection[])[cartIndex]\n const beforeChangeHooks = asArray(collection.hooks?.beforeChange)\n\n const alreadyAdded = beforeChangeHooks.some((h) => hasMarkedHook(h))\n if (!alreadyAdded) {\n const hook = markHook(recalculateCartHook(pluginConfig))\n collection.hooks = {\n ...(collection.hooks || {}),\n beforeChange: [...beforeChangeHooks, hook],\n }\n }\n\n ;(incomingConfig.collections as GenericCollection[])[cartIndex] = collection\n }\n\n return incomingConfig\n }\n","import type { ApplyCouponHook, ApplyCouponResponse, PartnerDashboardData } from '../types'\n\ntype EndpointInput =\n | string\n | {\n applyCoupon?: string\n validateCoupon?: string\n partnerStats?: string\n baseURL?: string\n }\n\nconst DEFAULT_ENDPOINTS = {\n applyCoupon: '/api/coupons/apply',\n validateCoupon: '/api/coupons/validate',\n partnerStats: '/api/referrals/partner-stats',\n} as const\n\nfunction normalizePath(path: string): string {\n if (!path) return ''\n return path.startsWith('/') ? path : `/${path}`\n}\n\nfunction withBaseURL(path: string, baseURL?: string): string {\n if (!baseURL) return path\n const trimmedBase = baseURL.endsWith('/') ? baseURL.slice(0, -1) : baseURL\n const normalizedPath = normalizePath(path)\n return `${trimmedBase}${normalizedPath}`\n}\n\nfunction resolveEndpoints(input?: EndpointInput) {\n if (typeof input === 'string') {\n return {\n ...DEFAULT_ENDPOINTS,\n partnerStats: input,\n }\n }\n\n const baseURL = input?.baseURL\n return {\n applyCoupon: withBaseURL(input?.applyCoupon || DEFAULT_ENDPOINTS.applyCoupon, baseURL),\n validateCoupon: withBaseURL(input?.validateCoupon || DEFAULT_ENDPOINTS.validateCoupon, baseURL),\n partnerStats: withBaseURL(input?.partnerStats || DEFAULT_ENDPOINTS.partnerStats, baseURL),\n }\n}\n\n/**\n * Apply a coupon/referral code to a cart\n * @param options - Code, cart ID, and optional customerEmail\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function useCouponCode(\n options: ApplyCouponHook,\n endpointConfig?: EndpointInput,\n): Promise<ApplyCouponResponse> {\n const { code, cartID, customerEmail } = options\n\n if (!code) {\n return {\n success: false,\n message: 'Coupon code is required',\n error: 'Code is missing',\n }\n }\n\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.applyCoupon, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ code, cartID, customerEmail }),\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n message: (data.error as string) || 'Failed to apply coupon',\n error: data.error as string,\n }\n }\n\n const couponData = data.coupon as Record<string, unknown> | undefined\n const referralData = data.referralCode as Record<string, unknown> | undefined\n\n return {\n success: Boolean(data.success),\n message: (data.message as string) || 'Code applied',\n discount: (data.discount as number) || (data.customerDiscount as number),\n partnerCommission: data.partnerCommission as number,\n customerDiscount: data.customerDiscount as number,\n\n coupon: couponData\n ? {\n code: (couponData.code as string) || '',\n type: (couponData.type as 'percentage' | 'fixed') || 'percentage',\n value: (couponData.value as number) || 0,\n }\n : undefined,\n referralCode: referralData\n ? {\n code: (referralData.code as string) || '',\n }\n : undefined,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, message, error: message }\n }\n}\n\n/**\n * Validate a coupon/referral code without applying it\n * @param code - Code to validate\n * @param cartValue - Optional cart value\n * @param cartID - Optional cart ID\n * @param customerEmail - Optional customer email (for per-customer limits)\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function validateCouponCode(\n code: string,\n cartValue?: number,\n cartID?: string,\n customerEmail?: string,\n endpointConfig?: EndpointInput,\n): Promise<ApplyCouponResponse> {\n if (!code) {\n return {\n success: false,\n message: 'Code required',\n error: 'Code missing',\n }\n }\n\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.validateCoupon, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ code, cartValue, cartID, customerEmail }),\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n message: (data.error as string) || 'Invalid coupon',\n error: data.error as string,\n }\n }\n\n const couponData = data.coupon as Record<string, unknown> | undefined\n const referralData = data.referralCode as Record<string, unknown> | undefined\n\n return {\n success: Boolean(data.success),\n message: (data.message as string) || 'Code is valid',\n coupon: couponData\n ? {\n code: (couponData.code as string) || '',\n type: (couponData.type as 'percentage' | 'fixed') || 'percentage',\n value: (couponData.value as number) || 0,\n }\n : undefined,\n referralCode: referralData\n ? {\n code: (referralData.code as string) || '',\n }\n : undefined,\n discount: data.discount as number,\n partnerCommission: data.partnerCommission as number,\n customerDiscount: data.customerDiscount as number,\n currency: data.currency as string,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, message, error: message }\n }\n}\n\nexport type PartnerStatsResponse = {\n success: boolean\n data?: PartnerDashboardData\n currency?: string\n error?: string\n}\n\n/**\n * Fetch partner dashboard statistics\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function usePartnerStats(\n endpointConfig?: EndpointInput,\n): Promise<PartnerStatsResponse> {\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.partnerStats, {\n method: 'GET',\n headers: { 'Content-Type': 'application/json' },\n credentials: 'include',\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n error: (data.error as string) || 'Failed to fetch partner stats',\n }\n }\n\n return {\n success: Boolean(data.success),\n data: data.data as PartnerDashboardData,\n currency: data.currency as string,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, error: message }\n }\n}\n","/**\n * @license React\n * react.production.js\n *\n * Copyright (c) Meta Platforms, Inc. and affiliates.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n */\n\n\"use strict\";\nvar REACT_ELEMENT_TYPE = Symbol.for(\"react.transitional.element\"),\n REACT_PORTAL_TYPE = Symbol.for(\"react.portal\"),\n REACT_FRAGMENT_TYPE = Symbol.for(\"react.fragment\"),\n REACT_STRICT_MODE_TYPE = Symbol.for(\"react.strict_mode\"),\n REACT_PROFILER_TYPE = Symbol.for(\"react.profiler\"),\n REACT_CONSUMER_TYPE = Symbol.for(\"react.consumer\"),\n REACT_CONTEXT_TYPE = Symbol.for(\"react.context\"),\n REACT_FORWARD_REF_TYPE = Symbol.for(\"react.forward_ref\"),\n REACT_SUSPENSE_TYPE = Symbol.for(\"react.suspense\"),\n REACT_MEMO_TYPE = Symbol.for(\"react.memo\"),\n REACT_LAZY_TYPE = Symbol.for(\"react.lazy\"),\n REACT_ACTIVITY_TYPE = Symbol.for(\"react.activity\"),\n MAYBE_ITERATOR_SYMBOL = Symbol.iterator;\nfunction getIteratorFn(maybeIterable) {\n if (null === maybeIterable || \"object\" !== typeof maybeIterable) return null;\n maybeIterable =\n (MAYBE_ITERATOR_SYMBOL && maybeIterable[MAYBE_ITERATOR_SYMBOL]) ||\n maybeIterable[\"@@iterator\"];\n return \"function\" === typeof maybeIterable ? maybeIterable : null;\n}\nvar ReactNoopUpdateQueue = {\n isMounted: function () {\n return !1;\n },\n enqueueForceUpdate: function () {},\n enqueueReplaceState: function () {},\n enqueueSetState: function () {}\n },\n assign = Object.assign,\n emptyObject = {};\nfunction Component(props, context, updater) {\n this.props = props;\n this.context = context;\n this.refs = emptyObject;\n this.updater = updater || ReactNoopUpdateQueue;\n}\nComponent.prototype.isReactComponent = {};\nComponent.prototype.setState = function (partialState, callback) {\n if (\n \"object\" !== typeof partialState &&\n \"function\" !== typeof partialState &&\n null != partialState\n )\n throw Error(\n \"takes an object of state variables to update or a function which returns an object of state variables.\"\n );\n this.updater.enqueueSetState(this, partialState, callback, \"setState\");\n};\nComponent.prototype.forceUpdate = function (callback) {\n this.updater.enqueueForceUpdate(this, callback, \"forceUpdate\");\n};\nfunction ComponentDummy() {}\nComponentDummy.prototype = Component.prototype;\nfunction PureComponent(props, context, updater) {\n this.props = props;\n this.context = context;\n this.refs = emptyObject;\n this.updater = updater || ReactNoopUpdateQueue;\n}\nvar pureComponentPrototype = (PureComponent.prototype = new ComponentDummy());\npureComponentPrototype.constructor = PureComponent;\nassign(pureComponentPrototype, Component.prototype);\npureComponentPrototype.isPureReactComponent = !0;\nvar isArrayImpl = Array.isArray;\nfunction noop() {}\nvar ReactSharedInternals = { H: null, A: null, T: null, S: null },\n hasOwnProperty = Object.prototype.hasOwnProperty;\nfunction ReactElement(type, key, props) {\n var refProp = props.ref;\n return {\n $$typeof: REACT_ELEMENT_TYPE,\n type: type,\n key: key,\n ref: void 0 !== refProp ? refProp : null,\n props: props\n };\n}\nfunction cloneAndReplaceKey(oldElement, newKey) {\n return ReactElement(oldElement.type, newKey, oldElement.props);\n}\nfunction isValidElement(object) {\n return (\n \"object\" === typeof object &&\n null !== object &&\n object.$$typeof === REACT_ELEMENT_TYPE\n );\n}\nfunction escape(key) {\n var escaperLookup = { \"=\": \"=0\", \":\": \"=2\" };\n return (\n \"$\" +\n key.replace(/[=:]/g, function (match) {\n return escaperLookup[match];\n })\n );\n}\nvar userProvidedKeyEscapeRegex = /\\/+/g;\nfunction getElementKey(element, index) {\n return \"object\" === typeof element && null !== element && null != element.key\n ? escape(\"\" + element.key)\n : index.toString(36);\n}\nfunction resolveThenable(thenable) {\n switch (thenable.status) {\n case \"fulfilled\":\n return thenable.value;\n case \"rejected\":\n throw thenable.reason;\n default:\n switch (\n (\"string\" === typeof thenable.status\n ? thenable.then(noop, noop)\n : ((thenable.status = \"pending\"),\n thenable.then(\n function (fulfilledValue) {\n \"pending\" === thenable.status &&\n ((thenable.status = \"fulfilled\"),\n (thenable.value = fulfilledValue));\n },\n function (error) {\n \"pending\" === thenable.status &&\n ((thenable.status = \"rejected\"), (thenable.reason = error));\n }\n )),\n thenable.status)\n ) {\n case \"fulfilled\":\n return thenable.value;\n case \"rejected\":\n throw thenable.reason;\n }\n }\n throw thenable;\n}\nfunction mapIntoArray(children, array, escapedPrefix, nameSoFar, callback) {\n var type = typeof children;\n if (\"undefined\" === type || \"boolean\" === type) children = null;\n var invokeCallback = !1;\n if (null === children) invokeCallback = !0;\n else\n switch (type) {\n case \"bigint\":\n case \"string\":\n case \"number\":\n invokeCallback = !0;\n break;\n case \"object\":\n switch (children.$$typeof) {\n case REACT_ELEMENT_TYPE:\n case REACT_PORTAL_TYPE:\n invokeCallback = !0;\n break;\n case REACT_LAZY_TYPE:\n return (\n (invokeCallback = children._init),\n mapIntoArray(\n invokeCallback(children._payload),\n array,\n escapedPrefix,\n nameSoFar,\n callback\n )\n );\n }\n }\n if (invokeCallback)\n return (\n (callback = callback(children)),\n (invokeCallback =\n \"\" === nameSoFar ? \".\" + getElementKey(children, 0) : nameSoFar),\n isArrayImpl(callback)\n ? ((escapedPrefix = \"\"),\n null != invokeCallback &&\n (escapedPrefix =\n invokeCallback.replace(userProvidedKeyEscapeRegex, \"$&/\") + \"/\"),\n mapIntoArray(callback, array, escapedPrefix, \"\", function (c) {\n return c;\n }))\n : null != callback &&\n (isValidElement(callback) &&\n (callback = cloneAndReplaceKey(\n callback,\n escapedPrefix +\n (null == callback.key ||\n (children && children.key === callback.key)\n ? \"\"\n : (\"\" + callback.key).replace(\n userProvidedKeyEscapeRegex,\n \"$&/\"\n ) + \"/\") +\n invokeCallback\n )),\n array.push(callback)),\n 1\n );\n invokeCallback = 0;\n var nextNamePrefix = \"\" === nameSoFar ? \".\" : nameSoFar + \":\";\n if (isArrayImpl(children))\n for (var i = 0; i < children.length; i++)\n (nameSoFar = children[i]),\n (type = nextNamePrefix + getElementKey(nameSoFar, i)),\n (invokeCallback += mapIntoArray(\n nameSoFar,\n array,\n escapedPrefix,\n type,\n callback\n ));\n else if (((i = getIteratorFn(children)), \"function\" === typeof i))\n for (\n children = i.call(children), i = 0;\n !(nameSoFar = children.next()).done;\n\n )\n (nameSoFar = nameSoFar.value),\n (type = nextNamePrefix + getElementKey(nameSoFar, i++)),\n (invokeCallback += mapIntoArray(\n nameSoFar,\n array,\n escapedPrefix,\n type,\n callback\n ));\n else if (\"object\" === type) {\n if (\"function\" === typeof children.then)\n return mapIntoArray(\n resolveThenable(children),\n array,\n escapedPrefix,\n nameSoFar,\n callback\n );\n array = String(children);\n throw Error(\n \"Objects are not valid as a React child (found: \" +\n (\"[object Object]\" === array\n ? \"object with keys {\" + Object.keys(children).join(\", \") + \"}\"\n : array) +\n \"). If you meant to render a collection of children, use an array instead.\"\n );\n }\n return invokeCallback;\n}\nfunction mapChildren(children, func, context) {\n if (null == children) return children;\n var result = [],\n count = 0;\n mapIntoArray(children, result, \"\", \"\", function (child) {\n return func.call(context, child, count++);\n });\n return result;\n}\nfunction lazyInitializer(payload) {\n if (-1 === payload._status) {\n var ctor = payload._result;\n ctor = ctor();\n ctor.then(\n function (moduleObject) {\n if (0 === payload._status || -1 === payload._status)\n (payload._status = 1), (payload._result = moduleObject);\n },\n function (error) {\n if (0 === payload._status || -1 === payload._status)\n (payload._status = 2), (payload._result = error);\n }\n );\n -1 === payload._status && ((payload._status = 0), (payload._result = ctor));\n }\n if (1 === payload._status) return payload._result.default;\n throw payload._result;\n}\nvar reportGlobalError =\n \"function\" === typeof reportError\n ? reportError\n : function (error) {\n if (\n \"object\" === typeof window &&\n \"function\" === typeof window.ErrorEvent\n ) {\n var event = new window.ErrorEvent(\"error\", {\n bubbles: !0,\n cancelable: !0,\n message:\n \"object\" === typeof error &&\n null !== error &&\n \"string\" === typeof error.message\n ? String(error.message)\n : String(error),\n error: error\n });\n if (!window.dispatchEvent(event)) return;\n } else if (\n \"object\" === typeof process &&\n \"function\" === typeof process.emit\n ) {\n process.emit(\"uncaughtException\", error);\n return;\n }\n console.error(error);\n },\n Children = {\n map: mapChildren,\n forEach: function (children, forEachFunc, forEachContext) {\n mapChildren(\n children,\n function () {\n forEachFunc.apply(this, arguments);\n },\n forEachContext\n );\n },\n count: function (children) {\n var n = 0;\n mapChildren(children, function () {\n n++;\n });\n return n;\n },\n toArray: function (children) {\n return (\n mapChildren(children, function (child) {\n return child;\n }) || []\n );\n },\n only: function (children) {\n if (!isValidElement(children))\n throw Error(\n \"React.Children.only expected to receive a single React element child.\"\n );\n return children;\n }\n };\nexports.Activity = REACT_ACTIVITY_TYPE;\nexports.Children = Children;\nexports.Component = Component;\nexports.Fragment = REACT_FRAGMENT_TYPE;\nexports.Profiler = REACT_PROFILER_TYPE;\nexports.PureComponent = PureComponent;\nexports.StrictMode = REACT_STRICT_MODE_TYPE;\nexports.Suspense = REACT_SUSPENSE_TYPE;\nexports.__CLIENT_INTERNALS_DO_NOT_USE_OR_WARN_USERS_THEY_CANNOT_UPGRADE =\n ReactSharedInternals;\nexports.__COMPILER_RUNTIME = {\n __proto__: null,\n c: function (size) {\n return ReactSharedInternals.H.useMemoCache(size);\n }\n};\nexports.cache = function (fn) {\n return function () {\n return fn.apply(null, arguments);\n };\n};\nexports.cacheSignal = function () {\n return null;\n};\nexports.cloneElement = function (element, config, children) {\n if (null === element || void 0 === element)\n throw Error(\n \"The argument must be a React element, but you passed \" + element + \".\"\n );\n var props = assign({}, element.props),\n key = element.key;\n if (null != config)\n for (propName in (void 0 !== config.key && (key = \"\" + config.key), config))\n !hasOwnProperty.call(config, propName) ||\n \"key\" === propName ||\n \"__self\" === propName ||\n \"__source\" === propName ||\n (\"ref\" === propName && void 0 === config.ref) ||\n (props[propName] = config[propName]);\n var propName = arguments.length - 2;\n if (1 === propName) props.children = children;\n else if (1 < propName) {\n for (var childArray = Array(propName), i = 0; i < propName; i++)\n childArray[i] = arguments[i + 2];\n props.children = childArray;\n }\n return ReactElement(element.type, key, props);\n};\nexports.createContext = function (defaultValue) {\n defaultValue = {\n $$typeof: REACT_CONTEXT_TYPE,\n _currentValue: defaultValue,\n _currentValue2: defaultValue,\n _threadCount: 0,\n Provider: null,\n Consumer: null\n };\n defaultValue.Provider = defaultValue;\n defaultValue.Consumer = {\n $$typeof: REACT_CONSUMER_TYPE,\n _context: defaultValue\n };\n return defaultValue;\n};\nexports.createElement = function (type, config, children) {\n var propName,\n props = {},\n key = null;\n if (null != config)\n for (propName in (void 0 !== config.key && (key = \"\" + config.key), config))\n hasOwnProperty.call(config, propName) &&\n \"key\" !== propName &&\n \"__self\" !== propName &&\n \"__source\" !== propName &&\n (props[propName] = config[propName]);\n var childrenLength = arguments.length - 2;\n if (1 === childrenLength) props.children = children;\n else if (1 < childrenLength) {\n for (var childArray = Array(childrenLength), i = 0; i < childrenLength; i++)\n childArray[i] = arguments[i + 2];\n props.children = childArray;\n }\n if (type && type.defaultProps)\n for (propName in ((childrenLength = type.defaultProps), childrenLength))\n void 0 === props[propName] &&\n (props[propName] = childrenLength[propName]);\n return ReactElement(type, key, props);\n};\nexports.createRef = function () {\n return { current: null };\n};\nexports.forwardRef = function (render) {\n return { $$typeof: REACT_FORWARD_REF_TYPE, render: render };\n};\nexports.isValidElement = isValidElement;\nexports.lazy = function (ctor) {\n return {\n $$typeof: REACT_LAZY_TYPE,\n _payload: { _status: -1, _result: ctor },\n _init: lazyInitializer\n };\n};\nexports.memo = function (type, compare) {\n return {\n $$typeof: REACT_MEMO_TYPE,\n type: type,\n compare: void 0 === compare ? null : compare\n };\n};\nexports.startTransition = function (scope) {\n var prevTransition = ReactSharedInternals.T,\n currentTransition = {};\n ReactSharedInternals.T = currentTransition;\n try {\n var returnValue = scope(),\n onStartTransitionFinish = ReactSharedInternals.S;\n null !== onStartTransitionFinish &&\n onStartTransitionFinish(currentTransition, returnValue);\n \"object\" === typeof returnValue &&\n null !== returnValue &&\n \"function\" === typeof returnValue.then &&\n returnValue.then(noop, reportGlobalError);\n } catch (error) {\n reportGlobalError(error);\n } finally {\n null !== prevTransition &&\n null !== currentTransition.types &&\n (prevTransition.types = currentTransition.types),\n (ReactSharedInternals.T = prevTransition);\n }\n};\nexports.unstable_useCacheRefresh = function () {\n return ReactSharedInternals.H.useCacheRefresh();\n};\nexports.use = function (usable) {\n return ReactSharedInternals.H.use(usable);\n};\nexports.useActionState = function (action, initialState, permalink) {\n return ReactSharedInternals.H.useActionState(action, initialState, permalink);\n};\nexports.useCallback = function (callback, deps) {\n return ReactSharedInternals.H.useCallback(callback, deps);\n};\nexports.useContext = function (Context) {\n return ReactSharedInternals.H.useContext(Context);\n};\nexports.useDebugValue = function () {};\nexports.useDeferredValue = function (value, initialValue) {\n return ReactSharedInternals.H.useDeferredValue(value, initialValue);\n};\nexports.useEffect = function (create, deps) {\n return ReactSharedInternals.H.useEffect(create, deps);\n};\nexports.useEffectEvent = function (callback) {\n return ReactSharedInternals.H.useEffectEvent(callback);\n};\nexports.useId = function () {\n return ReactSharedInternals.H.useId();\n};\nexports.useImperativeHandle = function (ref, create, deps) {\n return ReactSharedInternals.H.useImperativeHandle(ref, create, deps);\n};\nexports.useInsertionEffect = function (create, deps) {\n return ReactSharedInternals.H.useInsertionEffect(create, deps);\n};\nexports.useLayoutEffect = function (create, deps) {\n return ReactSharedInternals.H.useLayoutEffect(create, deps);\n};\nexports.useMemo = function (create, deps) {\n return ReactSharedInternals.H.useMemo(create, deps);\n};\nexports.useOptimistic = function (passthrough, reducer) {\n return ReactSharedInternals.H.useOptimistic(passthrough, reducer);\n};\nexports.useReducer = function (reducer, initialArg, init) {\n return ReactSharedInternals.H.useReducer(reducer, initialArg, init);\n};\nexports.useRef = function (initialValue) {\n return ReactSharedInternals.H.useRef(initialValue);\n};\nexports.useState = function (initialState) {\n return ReactSharedInternals.H.useState(initialState);\n};\nexports.useSyncExternalStore = function (\n subscribe,\n getSnapshot,\n getServerSnapshot\n) {\n return ReactSharedInternals.H.useSyncExternalStore(\n subscribe,\n getSnapshot,\n getServerSnapshot\n );\n};\nexports.useTransition = function () {\n return ReactSharedInternals.H.useTransition();\n};\nexports.version = \"19.2.4\";\n","'use strict';\n\nif (process.env.NODE_ENV === 'production') {\n module.exports = require('./cjs/react.production.js');\n} else {\n module.exports = require('./cjs/react.development.js');\n}\n","/**\n * @license React\n * react-jsx-runtime.production.js\n *\n * Copyright (c) Meta Platforms, Inc. and affiliates.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n */\n\n\"use strict\";\nvar REACT_ELEMENT_TYPE = Symbol.for(\"react.transitional.element\"),\n REACT_FRAGMENT_TYPE = Symbol.for(\"react.fragment\");\nfunction jsxProd(type, config, maybeKey) {\n var key = null;\n void 0 !== maybeKey && (key = \"\" + maybeKey);\n void 0 !== config.key && (key = \"\" + config.key);\n if (\"key\" in config) {\n maybeKey = {};\n for (var propName in config)\n \"key\" !== propName && (maybeKey[propName] = config[propName]);\n } else maybeKey = config;\n config = maybeKey.ref;\n return {\n $$typeof: REACT_ELEMENT_TYPE,\n type: type,\n key: key,\n ref: void 0 !== config ? config : null,\n props: maybeKey\n };\n}\nexports.Fragment = REACT_FRAGMENT_TYPE;\nexports.jsx = jsxProd;\nexports.jsxs = jsxProd;\n","'use strict';\n\nif (process.env.NODE_ENV === 'production') {\n module.exports = require('./cjs/react-jsx-runtime.production.js');\n} else {\n module.exports = require('./cjs/react-jsx-runtime.development.js');\n}\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type EarningsSummaryProps = {\n stats: PartnerStats\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nexport const EarningsSummary: React.FC<EarningsSummaryProps> = ({ stats, currency }) => {\n return (\n <div className=\"partner-widget partner-widget--earnings\">\n <h3 className=\"partner-widget__title\">Earnings Summary</h3>\n <div className=\"partner-widget__content\">\n <div className=\"earnings-grid\">\n <div className=\"earnings-card earnings-card--total\">\n <span className=\"earnings-card__label\">Total Earnings</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.totalEarnings, currency)}\n </span>\n </div>\n <div className=\"earnings-card earnings-card--pending\">\n <span className=\"earnings-card__label\">Pending</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.pendingEarnings, currency)}\n </span>\n </div>\n <div className=\"earnings-card earnings-card--paid\">\n <span className=\"earnings-card__label\">Paid Out</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.paidEarnings, currency)}\n </span>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default EarningsSummary\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type ReferralPerformanceProps = {\n stats: PartnerStats\n}\n\nexport const ReferralPerformance: React.FC<ReferralPerformanceProps> = ({ stats }) => {\n return (\n <div className=\"partner-widget partner-widget--performance\">\n <h3 className=\"partner-widget__title\">Referral Performance</h3>\n <div className=\"partner-widget__content\">\n <div className=\"performance-grid\">\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.totalReferrals}</span>\n <span className=\"performance-stat__label\">Total Referrals</span>\n </div>\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.successfulReferrals}</span>\n <span className=\"performance-stat__label\">Successful</span>\n </div>\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.conversionRate.toFixed(1)}%</span>\n <span className=\"performance-stat__label\">Conversion Rate</span>\n </div>\n </div>\n\n {stats.monthlyEarnings.length > 0 && (\n <div className=\"performance-chart\">\n <h4 className=\"performance-chart__title\">Monthly Trend</h4>\n <div className=\"performance-chart__bars\">\n {stats.monthlyEarnings.map((month, index) => {\n const maxEarnings = Math.max(...stats.monthlyEarnings.map((m) => m.earnings), 1)\n const heightPercent = (month.earnings / maxEarnings) * 100\n\n return (\n <div key={index} className=\"performance-chart__bar-container\">\n <div\n className=\"performance-chart__bar\"\n style={{ height: `${Math.max(heightPercent, 5)}%` }}\n title={`${month.month}: ${month.referrals} referrals`}\n />\n <span className=\"performance-chart__label\">{month.month.split(' ')[0]}</span>\n </div>\n )\n })}\n </div>\n </div>\n )}\n </div>\n </div>\n )\n}\n\nexport default ReferralPerformance\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type RecentReferralsProps = {\n referrals: PartnerStats['recentReferrals']\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nconst formatDate = (dateString: string): string => {\n return new Date(dateString).toLocaleDateString('en-US', {\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })\n}\n\nexport const RecentReferrals: React.FC<RecentReferralsProps> = ({ referrals, currency }) => {\n return (\n <div className=\"partner-widget partner-widget--recent\">\n <h3 className=\"partner-widget__title\">Recent Referrals</h3>\n <div className=\"partner-widget__content\">\n <div className=\"referrals-table\">\n <div className=\"referrals-table__header\">\n <span>Code</span>\n <span>Order Value</span>\n <span>Commission</span>\n <span>Date</span>\n <span>Status</span>\n </div>\n <div className=\"referrals-table__body\">\n {referrals.map((referral) => (\n <div key={referral.id} className=\"referrals-table__row\">\n <span className=\"referrals-table__code\">{referral.code}</span>\n <span className=\"referrals-table__value\">\n {formatCurrency(referral.orderValue, currency)}\n </span>\n <span className=\"referrals-table__commission\">\n {formatCurrency(referral.commission, currency)}\n </span>\n <span className=\"referrals-table__date\">{formatDate(referral.date)}</span>\n <span\n className={`referrals-table__status referrals-table__status--${referral.status}`}\n >\n {referral.status}\n </span>\n </div>\n ))}\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default RecentReferrals\n","'use client'\n\nimport React, { useState } from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nexport type ReferralCodesProps = {\n codes: PartnerDashboardData['referralCodes']\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nexport const ReferralCodes: React.FC<ReferralCodesProps> = ({ codes, currency }) => {\n const [copiedCode, setCopiedCode] = useState<string | null>(null)\n\n const copyToClipboard = async (code: string) => {\n try {\n await navigator.clipboard.writeText(code)\n setCopiedCode(code)\n setTimeout(() => setCopiedCode(null), 2000)\n } catch {\n // Fallback for older browsers\n const textArea = document.createElement('textarea')\n textArea.value = code\n document.body.appendChild(textArea)\n textArea.select()\n document.execCommand('copy')\n document.body.removeChild(textArea)\n setCopiedCode(code)\n setTimeout(() => setCopiedCode(null), 2000)\n }\n }\n\n return (\n <div className=\"partner-widget partner-widget--codes\">\n <h3 className=\"partner-widget__title\">Your Referral Codes</h3>\n <div className=\"partner-widget__content\">\n <div className=\"codes-list\">\n {codes.map((codeData) => (\n <div\n key={codeData.id}\n className={`code-card ${!codeData.isActive ? 'code-card--inactive' : ''}`}\n >\n <div className=\"code-card__header\">\n <span className=\"code-card__code\">{codeData.code}</span>\n <button\n type=\"button\"\n className=\"code-card__copy\"\n onClick={() => copyToClipboard(codeData.code)}\n title=\"Copy code\"\n >\n {copiedCode === codeData.code ? '✓ Copied' : 'Copy'}\n </button>\n </div>\n <div className=\"code-card__stats\">\n <div className=\"code-card__stat\">\n <span className=\"code-card__stat-value\">{codeData.usageCount}</span>\n <span className=\"code-card__stat-label\">Uses</span>\n </div>\n <div className=\"code-card__stat\">\n <span className=\"code-card__stat-value\">\n {formatCurrency(codeData.totalEarnings, currency)}\n </span>\n <span className=\"code-card__stat-label\">Earnings</span>\n </div>\n </div>\n {!codeData.isActive && (\n <span className=\"code-card__badge code-card__badge--inactive\">Inactive</span>\n )}\n </div>\n ))}\n </div>\n </div>\n </div>\n )\n}\n\nexport default ReferralCodes\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nexport type ProgramOverviewProps = {\n program: PartnerDashboardData['program']\n}\n\nexport const ProgramOverview: React.FC<ProgramOverviewProps> = ({ program }) => {\n if (!program) {\n return (\n <div className=\"partner-widget partner-widget--program-overview\">\n <h3 className=\"partner-widget__title\">Program Overview</h3>\n <div className=\"partner-widget__content\">\n <div className=\"program-overview program-overview--empty\">\n <p>No referral program details are available yet.</p>\n </div>\n </div>\n </div>\n )\n }\n\n return (\n <div className=\"partner-widget partner-widget--program-overview\">\n <h3 className=\"partner-widget__title\">Program Overview</h3>\n <div className=\"partner-widget__content\">\n <div className=\"program-overview\">\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Program Name</span>\n <span className=\"program-overview__value\">{program.name}</span>\n </div>\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Commission Rate</span>\n <span className=\"program-overview__value\">{program.commissionRate}%</span>\n </div>\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Customer Discount</span>\n <span className=\"program-overview__value\">{program.customerDiscount}%</span>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default ProgramOverview\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type CommissionBreakdownProps = {\n stats: PartnerStats\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nconst formatPercent = (value: number): string => {\n if (!Number.isFinite(value)) return '0%'\n return `${value.toFixed(1)}%`\n}\n\nexport const CommissionBreakdown: React.FC<CommissionBreakdownProps> = ({ stats, currency }) => {\n const totalEarnings = Math.max(stats.totalEarnings || 0, 0)\n const paidEarnings = Math.max(stats.paidEarnings || 0, 0)\n const pendingEarnings = Math.max(stats.pendingEarnings || 0, 0)\n\n const paidPercent = totalEarnings > 0 ? (paidEarnings / totalEarnings) * 100 : 0\n const pendingPercent = totalEarnings > 0 ? (pendingEarnings / totalEarnings) * 100 : 0\n const unpaidOrOther = Math.max(totalEarnings - (paidEarnings + pendingEarnings), 0)\n const otherPercent = totalEarnings > 0 ? (unpaidOrOther / totalEarnings) * 100 : 0\n\n return (\n <div className=\"partner-widget partner-widget--commission-breakdown\">\n <h3 className=\"partner-widget__title\">Commission Breakdown</h3>\n\n <div className=\"partner-widget__content\">\n <div className=\"commission-breakdown\">\n <div className=\"commission-breakdown__summary\">\n <span className=\"commission-breakdown__total-label\">Total Commission</span>\n <span className=\"commission-breakdown__total-value\">\n {formatCurrency(totalEarnings, currency)}\n </span>\n </div>\n\n <div\n className=\"commission-breakdown__bars\"\n role=\"img\"\n aria-label=\"Commission distribution\"\n >\n <div className=\"commission-breakdown__track\">\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--paid\"\n style={{ width: `${Math.max(Math.min(paidPercent, 100), 0)}%` }}\n title={`Paid: ${formatCurrency(paidEarnings, currency)} (${formatPercent(paidPercent)})`}\n />\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--pending\"\n style={{ width: `${Math.max(Math.min(pendingPercent, 100), 0)}%` }}\n title={`Pending: ${formatCurrency(pendingEarnings, currency)} (${formatPercent(pendingPercent)})`}\n />\n {otherPercent > 0 && (\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--other\"\n style={{ width: `${Math.max(Math.min(otherPercent, 100), 0)}%` }}\n title={`Other: ${formatCurrency(unpaidOrOther, currency)} (${formatPercent(otherPercent)})`}\n />\n )}\n </div>\n </div>\n\n <div className=\"commission-breakdown__legend\">\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--paid\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Paid</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(paidEarnings, currency)} · {formatPercent(paidPercent)}\n </span>\n </div>\n </div>\n\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--pending\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Pending</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(pendingEarnings, currency)} · {formatPercent(pendingPercent)}\n </span>\n </div>\n </div>\n\n {otherPercent > 0 && (\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--other\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Other</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(unpaidOrOther, currency)} · {formatPercent(otherPercent)}\n </span>\n </div>\n </div>\n )}\n </div>\n\n <div className=\"commission-breakdown__footnote\">\n Based on current referral payouts and pending settlements.\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default CommissionBreakdown\n","'use client'\n\nimport React, { useEffect, useState } from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nimport { EarningsSummary } from './EarningsSummary'\nimport { ReferralPerformance } from './ReferralPerformance'\nimport { RecentReferrals } from './RecentReferrals'\nimport { ReferralCodes } from './ReferralCodes'\nimport { ProgramOverview } from './ProgramOverview'\nimport { CommissionBreakdown } from './CommissionBreakdown'\n\nexport type PartnerDashboardProps = {\n showEarningsSummary?: boolean\n showReferralPerformance?: boolean\n showRecentReferrals?: boolean\n showReferralCodes?: boolean\n showProgramOverview?: boolean\n showCommissionBreakdown?: boolean\n apiEndpoint?: string\n}\n\nexport const PartnerDashboard: React.FC<PartnerDashboardProps> = ({\n showEarningsSummary = true,\n showReferralPerformance = true,\n showRecentReferrals = true,\n showReferralCodes = true,\n showProgramOverview = true,\n showCommissionBreakdown = true,\n apiEndpoint = '/api/referrals/partner-stats',\n}) => {\n const [data, setData] = useState<PartnerDashboardData | null>(null)\n const [loading, setLoading] = useState(true)\n const [error, setError] = useState<string | null>(null)\n const [currency, setCurrency] = useState('USD')\n\n useEffect(() => {\n const fetchData = async () => {\n try {\n setLoading(true)\n const response = await fetch(apiEndpoint, {\n credentials: 'include',\n })\n\n if (!response.ok) {\n if (response.status === 401) {\n setError('Please log in to view your partner dashboard')\n return\n }\n if (response.status === 403) {\n setError('Partner access required')\n return\n }\n throw new Error('Failed to fetch partner data')\n }\n\n const result = (await response.json()) as {\n success: boolean\n data?: PartnerDashboardData\n currency?: string\n error?: string\n }\n\n if (result.success) {\n setData(result.data ?? null)\n setCurrency(result.currency ?? 'USD')\n } else {\n setError(result.error ?? 'Failed to load dashboard data')\n }\n } catch (err) {\n setError(err instanceof Error ? err.message : 'An error occurred')\n } finally {\n setLoading(false)\n }\n }\n\n fetchData()\n }, [apiEndpoint])\n\n if (loading) {\n return (\n <div className=\"partner-dashboard partner-dashboard--loading\">\n <div className=\"partner-dashboard__loader\">\n <div className=\"partner-dashboard__spinner\" />\n <p>Loading partner dashboard...</p>\n </div>\n </div>\n )\n }\n\n if (error) {\n return (\n <div className=\"partner-dashboard partner-dashboard--error\">\n <div className=\"partner-dashboard__error\">\n <h3>Unable to load dashboard</h3>\n <p>{error}</p>\n </div>\n </div>\n )\n }\n\n if (!data) {\n return (\n <div className=\"partner-dashboard partner-dashboard--empty\">\n <div className=\"partner-dashboard__empty\">\n <h3>No data available</h3>\n <p>Your partner dashboard will appear here once you have referral activity.</p>\n </div>\n </div>\n )\n }\n\n return (\n <div className=\"partner-dashboard\">\n <div className=\"partner-dashboard__header\">\n <h2>Partner Dashboard</h2>\n {data.program && (\n <div className=\"partner-dashboard__program\">\n <span className=\"partner-dashboard__program-name\">{data.program.name}</span>\n <span className=\"partner-dashboard__program-rate\">\n {data.program.commissionRate}% commission\n </span>\n </div>\n )}\n </div>\n\n <div className=\"partner-dashboard__grid\">\n {showEarningsSummary && <EarningsSummary stats={data.stats} currency={currency} />}\n\n {showReferralPerformance && <ReferralPerformance stats={data.stats} />}\n\n {showRecentReferrals && data.stats.recentReferrals.length > 0 && (\n <RecentReferrals referrals={data.stats.recentReferrals} currency={currency} />\n )}\n\n {showReferralCodes && data.referralCodes.length > 0 && (\n <ReferralCodes codes={data.referralCodes} currency={currency} />\n )}\n\n {showProgramOverview && <ProgramOverview program={data.program} />}\n\n {showCommissionBreakdown && <CommissionBreakdown stats={data.stats} currency={currency} />}\n </div>\n </div>\n )\n}\n\nexport default PartnerDashboard\n","import { roundTo2 } from './roundTo2'\n\nexport type CartLike = {\n subtotal?: number\n total?: number\n discountAmount?: number\n customerDiscount?: number\n}\n\n/**\n * Computes the cart total after applying plugin discounts.\n * Use this in your host app's cart beforeChange (or wherever you compute total)\n * so the amount always reflects coupon/referral discounts and is not overwritten incorrectly.\n *\n * Formula: subtotal - discountAmount - customerDiscount (each defaulting to 0).\n */\nexport function getCartTotalWithDiscounts(cart: CartLike): number {\n const subtotal = cart.subtotal ?? cart.total ?? 0\n const discountAmount = cart.discountAmount ?? 0\n const customerDiscount = cart.customerDiscount ?? 0\n return roundTo2(Math.max(0, subtotal - discountAmount - customerDiscount))\n}\n"],"x_google_ignoreList":[15,16,17,18],"mappings":"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"}
1
+ {"version":3,"file":"index.mjs","names":["normalizeCode","relationId","relationId","readField","writeField","normalizeCode","ensureRequestData","findByNormalizedCode","relationId","readField","asNumber","relationId","validateCouponCode","relationId","readField","normalizePath","formatCurrency","formatCurrency","formatCurrency"],"sources":["../src/collections/createCouponsCollection.ts","../src/utilities/userRoles.ts","../src/collections/createReferralCodesCollection.ts","../src/collections/createReferralProgramsCollection.ts","../src/utilities/pricing.ts","../src/utilities/calculateValues.ts","../src/utilities/roundTo2.ts","../src/endpoints/applyCoupon.ts","../src/endpoints/partnerStats.ts","../src/endpoints/validateCoupon.ts","../src/hooks/recalculateCart.ts","../src/utilities/recordCouponUsageForOrder.ts","../src/utilities/sanitizePluginConfig.ts","../src/plugin.ts","../src/client/hooks.ts","../node_modules/react/cjs/react.production.js","../node_modules/react/index.js","../node_modules/react/cjs/react-jsx-runtime.production.js","../node_modules/react/jsx-runtime.js","../src/components/PartnerDashboard/EarningsSummary.tsx","../src/components/PartnerDashboard/ReferralPerformance.tsx","../src/components/PartnerDashboard/RecentReferrals.tsx","../src/components/PartnerDashboard/ReferralCodes.tsx","../src/components/PartnerDashboard/ProgramOverview.tsx","../src/components/PartnerDashboard/CommissionBreakdown.tsx","../src/components/PartnerDashboard/index.tsx","../src/utilities/getCartTotalWithDiscounts.ts"],"sourcesContent":["import type { CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\nexport const createCouponsCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, defaultCurrency, adminGroups, integration } = pluginConfig\n const usersSlug = integration.collections.usersSlug\n\n return {\n slug: collections.couponsSlug,\n admin: {\n useAsTitle: 'code',\n defaultColumns: ['code', 'type', 'value', 'activeFrom', 'activeUntil'],\n group: adminGroups.couponsGroup,\n },\n access: {\n read: access.canUseCoupons || (() => true),\n create: access.isAdmin || (() => false),\n update: access.isAdmin || (() => false),\n delete: access.isAdmin || (() => false),\n },\n fields: [\n {\n name: 'code',\n type: 'text',\n required: true,\n unique: true,\n admin: {\n description: 'The coupon code that customers will enter',\n },\n },\n {\n name: 'normalizedCode',\n type: 'text',\n unique: true,\n index: true,\n admin: {\n hidden: true,\n description: 'Uppercased, trimmed code used for fast case-insensitive lookups',\n },\n },\n {\n name: 'description',\n type: 'text',\n admin: {\n description: 'Optional description for admin reference',\n },\n },\n {\n name: 'type',\n type: 'select',\n required: true,\n options: [\n { label: 'Percentage', value: 'percentage' },\n { label: 'Fixed Amount', value: 'fixed' },\n ],\n defaultValue: 'percentage',\n admin: {\n description: 'Whether this is a percentage or fixed amount discount',\n },\n },\n {\n name: 'value',\n type: 'number',\n required: true,\n admin: {\n description: `If percentage, 10 = 10%. If fixed, interpreted in ${defaultCurrency} (smallest currency units)`,\n step: 0.01,\n },\n },\n {\n name: 'maxDiscountAmount',\n type: 'number',\n admin: {\n description: `Maximum discount amount in ${defaultCurrency} (smallest currency unit). Leave empty for no cap.`,\n },\n },\n {\n name: 'usageLimit',\n type: 'number',\n admin: {\n description:\n 'Total times this coupon can be used across all customers. Empty = unlimited.',\n },\n },\n {\n name: 'perCustomerLimit',\n type: 'number',\n admin: {\n description: 'Times a single customer can use this coupon. Empty = unlimited.',\n },\n },\n {\n name: 'activeFrom',\n type: 'date',\n admin: {\n description:\n 'Coupon becomes active from this date. Leave empty for immediate activation.',\n },\n },\n {\n name: 'activeUntil',\n type: 'date',\n admin: {\n description: 'Coupon expires after this date. Leave empty for no expiration.',\n },\n },\n {\n name: 'minOrderValue',\n type: 'number',\n admin: {\n description: `Minimum order value required in ${defaultCurrency} (smallest currency units)`,\n },\n },\n {\n name: 'maxOrderValue',\n type: 'number',\n admin: {\n description: `Maximum order value allowed in ${defaultCurrency} (smallest currency units)`,\n },\n },\n {\n name: 'usageCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'How many times this coupon has been used',\n readOnly: true,\n },\n },\n {\n name: 'createdBy',\n type: 'relationship',\n relationTo: usersSlug,\n admin: {\n readOnly: true,\n position: 'sidebar',\n },\n },\n ],\n hooks: {\n beforeValidate: [\n ({ data }) => {\n if (data && typeof data.code === 'string') {\n data.code = data.code.trim()\n data.normalizedCode = data.code.toUpperCase()\n }\n return data\n },\n ],\n beforeChange: [\n ({ operation, req, data }) => {\n if (data && typeof data.code === 'string') {\n data.code = data.code.trim()\n data.normalizedCode = data.code.toUpperCase()\n }\n\n if (operation === 'create' && req.user && !data.createdBy) {\n data.createdBy = (req.user as { id?: string | number }).id\n }\n\n return data\n },\n ],\n },\n timestamps: true,\n }\n}\n","import type { SanitizedCouponPluginOptions } from '../types'\nimport type { Where } from 'payload'\n\ntype RoleConfig = SanitizedCouponPluginOptions['roleConfig']\n\nfunction readByPath(input: unknown, path: string): unknown {\n if (!input || typeof input !== 'object' || !path) return undefined\n return path.split('.').reduce<unknown>((acc, key) => {\n if (!acc || typeof acc !== 'object') return undefined\n return (acc as Record<string, unknown>)[key]\n }, input)\n}\n\nfunction toRoleArray(value: unknown): string[] {\n if (typeof value === 'string' && value.trim().length > 0) return [value]\n if (Array.isArray(value)) {\n return value\n .filter((item): item is string => typeof item === 'string')\n .map((item) => item.trim())\n .filter(Boolean)\n }\n return []\n}\n\nconst normalizeRoleValue = (value: string): string => value.trim().toLowerCase()\n\nexport const resolveUserRoles = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): string[] => {\n if (!user) return []\n\n if (typeof roleConfig.customRoleResolver === 'function') {\n const custom = roleConfig.customRoleResolver(user)\n return Array.isArray(custom) ? custom.map(normalizeRoleValue).filter(Boolean) : []\n }\n\n const roles = roleConfig.roleFieldPaths.flatMap((path) => toRoleArray(readByPath(user, path)))\n return [...new Set(roles.map(normalizeRoleValue).filter(Boolean))]\n}\n\nexport const userHasAnyRole = ({\n user,\n roleConfig,\n targetRoles,\n}: {\n user: unknown\n roleConfig: RoleConfig\n targetRoles: string[]\n}): boolean => {\n const userRoles = new Set(resolveUserRoles({ user, roleConfig }))\n for (const target of targetRoles) {\n if (userRoles.has(normalizeRoleValue(target))) return true\n }\n return false\n}\n\nexport const isPartnerUser = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): boolean =>\n userHasAnyRole({\n user,\n roleConfig,\n targetRoles: roleConfig.partnerRoleValues,\n })\n\nexport const isAdminUser = ({\n user,\n roleConfig,\n}: {\n user: unknown\n roleConfig: RoleConfig\n}): boolean =>\n userHasAnyRole({\n user,\n roleConfig,\n targetRoles: roleConfig.adminRoleValues,\n })\n\nexport const buildPartnerUserFilterWhere = ({\n roleConfig,\n}: {\n roleConfig: RoleConfig\n}): Where | true => {\n if (!roleConfig.roleFieldPaths.length || !roleConfig.partnerRoleValues.length) return true\n\n const conditions = roleConfig.roleFieldPaths.map((fieldPath) => ({\n [fieldPath]: { in: roleConfig.partnerRoleValues },\n }))\n\n if (conditions.length === 1) return conditions[0] as Where\n return { or: conditions } as Where\n}\n","import type { CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport { buildPartnerUserFilterWhere, isAdminUser, isPartnerUser } from '../utilities/userRoles'\n\nconst normalizeCode = (value: unknown): string =>\n typeof value === 'string' ? value.trim().toUpperCase() : ''\n\nexport const createReferralCodesCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, adminGroups, defaultCurrency, roleConfig, policies, integration } =\n pluginConfig\n\n const usersSlug = integration.collections.usersSlug\n\n return {\n slug: collections.referralCodesSlug,\n admin: {\n useAsTitle: 'code',\n defaultColumns: ['code', 'normalizedCode', 'partner', 'program', 'usageCount', 'isActive'],\n group: adminGroups.referralsGroup,\n },\n access: {\n read: async ({ req }) => {\n const user = req?.user as { id?: string | number } | undefined\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n if (isAdmin) return true\n\n const hasPolicyAccess = await Promise.resolve(\n policies.canApplyReferral({ req, user, payload: req?.payload }),\n )\n\n if (!hasPolicyAccess) return false\n\n const isPartner =\n isPartnerUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isPartner?.({ req } as any)))\n\n if (isPartner) {\n return {\n partner: {\n equals: user.id,\n },\n }\n }\n\n return access.canUseReferrals ? access.canUseReferrals({ req } as any) : false\n },\n\n create: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const policyAllowed = await Promise.resolve(\n policies.canApplyReferral({ req, user, payload: req?.payload }),\n )\n if (!policyAllowed) return false\n\n if (\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n ) {\n return true\n }\n\n if (\n isPartnerUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isPartner?.({ req } as any)))\n ) {\n return true\n }\n\n return access.isAdmin ? access.isAdmin({ req } as any) : false\n },\n\n update: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n if (isAdmin) return true\n\n return false\n },\n\n delete: async ({ req }) => {\n const user = req?.user\n if (!user) return false\n\n const isAdmin =\n isAdminUser({ user, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n return isAdmin\n },\n },\n\n fields: [\n {\n name: 'code',\n type: 'text',\n required: true,\n unique: true,\n admin: {\n description: 'The referral code that customers will enter',\n },\n },\n {\n name: 'normalizedCode',\n type: 'text',\n required: true,\n unique: true,\n index: true,\n admin: {\n readOnly: true,\n description: 'Uppercased normalized code for fast case-insensitive lookup',\n },\n },\n {\n name: 'program',\n type: 'relationship',\n relationTo: collections.referralProgramsSlug,\n required: true,\n admin: {\n description: 'The referral program this code belongs to',\n },\n },\n {\n name: 'partner',\n type: 'relationship',\n relationTo: usersSlug,\n required: true,\n filterOptions: async ({ req, user }) => {\n const currentUser = (user || req?.user) as unknown\n\n if (\n isAdminUser({ user: currentUser, roleConfig }) ||\n (await Promise.resolve(access.isAdmin?.({ req } as any)))\n ) {\n return true\n }\n\n return buildPartnerUserFilterWhere({ roleConfig })\n },\n admin: {\n description: `The partner who owns this referral code (relation: ${usersSlug})`,\n },\n },\n {\n name: 'isActive',\n type: 'checkbox',\n defaultValue: true,\n admin: {\n description: 'Whether this referral code is currently active',\n },\n },\n {\n name: 'usageCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'How many times this referral code has been used',\n readOnly: true,\n },\n },\n {\n name: 'usageLimit',\n type: 'number',\n admin: {\n description: 'Maximum times this code can be used. Empty = unlimited.',\n },\n },\n {\n name: 'expiresAt',\n type: 'date',\n admin: {\n description: 'When this referral code expires',\n },\n },\n {\n name: 'successfulReferralsCount',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: 'Total count of successful referrals using this code',\n readOnly: true,\n },\n },\n {\n name: 'totalEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Total earnings generated by this code in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'pendingEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Pending earnings awaiting payout in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'paidEarnings',\n type: 'number',\n defaultValue: 0,\n admin: {\n description: `Total earnings paid out in ${defaultCurrency}`,\n readOnly: true,\n },\n },\n {\n name: 'metadata',\n type: 'json',\n admin: {\n description: 'Additional metadata for the referral code',\n position: 'sidebar',\n },\n },\n ],\n\n hooks: {\n beforeValidate: [\n ({ data }) => {\n if (!data) return data\n data.normalizedCode = normalizeCode(data.code)\n return data\n },\n ],\n beforeChange: [\n ({ operation, req, data }) => {\n if (!data) return data\n\n if (operation === 'create' && !data.code && data.partner) {\n const timestamp = Date.now().toString(36)\n const random = Math.random().toString(36).slice(2, 8)\n data.code = `REF-${timestamp}-${random}`.toUpperCase()\n }\n\n data.normalizedCode = normalizeCode(data.code)\n\n if (operation === 'create' && req.user) {\n const user = req.user as { id?: string | number }\n if (isPartnerUser({ user, roleConfig }) && user.id != null) {\n data.partner = user.id\n }\n }\n\n return data\n },\n ],\n },\n\n timestamps: true,\n }\n}\n","import { APIError, type CollectionConfig } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\ntype CommissionType = 'fixed' | 'percentage'\n\ntype RuleData = {\n appliesTo?: 'all' | 'products' | 'segments' | 'categories'\n products?: unknown[]\n categories?: unknown[]\n tags?: unknown[]\n totalCommission?: { type?: CommissionType; value?: number; maxAmount?: number }\n partnerSplit?: number\n customerSplit?: number\n partnerPercent?: number\n customerPercent?: number\n partnerAmount?: number\n customerAmount?: number\n}\n\nfunction toNumber(value: unknown): number | null {\n return typeof value === 'number' && Number.isFinite(value) ? value : null\n}\n\n/**\n * Scaling policy:\n * - Admin inputs normal currency values (e.g. 100 means $100).\n * - Internally, maxPartnerCommissionPerOrder, maxCustomerDiscountPerOrder, and\n * minOrderAmount are stored in x100 (integer cents) form to avoid floating-point\n * drift on monetary cap fields.\n * - beforeChange → multiply by 100 before persisting.\n * - afterRead → divide by 100 before returning to admin / calculation code.\n * - partnerAmount / customerAmount per-rule fixed amounts are NOT scaled here;\n * they represent per-item fixed currency amounts and are stored as-is.\n */\n\nexport const createReferralProgramsCollection = (\n pluginConfig: SanitizedCouponPluginOptions,\n): CollectionConfig => {\n const { collections, access, adminGroups, referralConfig, integration } = pluginConfig\n const allowedTotalCommissionTypes = referralConfig.allowedTotalCommissionTypes\n const relationSlugs = integration.collections\n\n // ---------------------------------------------------------------------------\n // beforeChange — validate, scale monetary cap fields ×100, normalise rules\n // ---------------------------------------------------------------------------\n const beforeChange: NonNullable<CollectionConfig['hooks']>['beforeChange'] = [\n ({ data }: { data: Record<string, unknown> }) => {\n if (\n !data.commissionRules ||\n !Array.isArray(data.commissionRules) ||\n data.commissionRules.length === 0\n ) {\n throw new APIError('At least one commission rule is required', 400)\n }\n\n // --- top-level monetary caps: validate and store as normal currency ---\n // The calculation layer (calculateValues.ts) converts to integer cents\n // internally via toCents() before doing any arithmetic, so these fields\n // must be stored as normal currency units (e.g. 100 = $100.00).\n\n const rawMaxPartner = toNumber(data.maxPartnerCommissionPerOrder)\n if (rawMaxPartner != null && rawMaxPartner < 0) {\n throw new APIError(\n 'Maximum commission per order for partner must be a non-negative number',\n 400,\n )\n }\n\n const rawMaxCustomer = toNumber(data.maxCustomerDiscountPerOrder)\n if (rawMaxCustomer != null && rawMaxCustomer < 0) {\n throw new APIError(\n 'Maximum discount for customer per order must be a non-negative number',\n 400,\n )\n }\n\n const rawMinOrder = toNumber(data.minOrderAmount)\n if (rawMinOrder != null && rawMinOrder < 0) {\n throw new APIError('Minimum Order Amount must be a non-negative number', 400)\n }\n\n // Store as normal currency. null when not provided.\n data.maxPartnerCommissionPerOrder = rawMaxPartner ?? null\n data.maxCustomerDiscountPerOrder = rawMaxCustomer ?? null\n data.minOrderAmount = rawMinOrder ?? null\n\n // --- commission rules ---\n data.commissionRules = data.commissionRules.map(\n (rule: Record<string, unknown>, index: number) => {\n const r = rule as RuleData\n\n if (!r.totalCommission) {\n throw new APIError(`Commission rule ${index + 1}: Total Commission is required`, 400)\n }\n\n if (\n !r.totalCommission.type ||\n !allowedTotalCommissionTypes.includes(r.totalCommission.type)\n ) {\n throw new APIError(\n `Commission rule ${index + 1}: Total Commission type must be one of ${allowedTotalCommissionTypes.join(', ')}`,\n 400,\n )\n }\n\n const type = r.totalCommission.type\n\n const appliesTo = r.appliesTo ?? 'all'\n if (appliesTo === 'products' && (!r.products || r.products.length === 0)) {\n throw new APIError(\n `Commission rule ${index + 1}: At least one product is required`,\n 400,\n )\n }\n\n if (\n (appliesTo === 'segments' || appliesTo === 'categories') &&\n (!r.categories || r.categories.length === 0) &&\n (!r.tags || r.tags.length === 0)\n ) {\n throw new APIError(\n `Commission rule ${index + 1}: At least one category or tag is required`,\n 400,\n )\n }\n\n let partnerSplit: number\n let customerSplit: number\n let partnerPercent: number | null = null\n let customerPercent: number | null = null\n let partnerAmount: number | null = null\n let customerAmount: number | null = null\n let splitWarning: string | null = null\n\n if (type === 'percentage') {\n const partnerPctInput = toNumber(r.partnerPercent) ?? toNumber(r.partnerSplit)\n const customerPctInput = toNumber(r.customerPercent) ?? toNumber(r.customerSplit)\n\n if (partnerPctInput == null || partnerPctInput < 0 || partnerPctInput > 100) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner Split must be between 0 and 100`,\n 400,\n )\n }\n\n if (customerPctInput != null && (customerPctInput < 0 || customerPctInput > 100)) {\n throw new APIError(\n `Commission rule ${index + 1}: Customer percentage must be between 0 and 100`,\n 400,\n )\n }\n\n const customerPctComputed =\n customerPctInput != null ? customerPctInput : 100 - partnerPctInput\n const percentTotal = partnerPctInput + customerPctComputed\n\n if (percentTotal > 100) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner percentage + Customer percentage cannot exceed 100`,\n 400,\n )\n }\n\n if (percentTotal > 50) {\n splitWarning = `High total split configured: ${percentTotal}% (partner + customer).`\n }\n\n partnerPercent = partnerPctInput\n customerPercent = customerPctComputed\n partnerSplit = partnerPctInput\n customerSplit = customerPctComputed\n } else {\n // Fixed commission type.\n // partnerAmount / customerAmount are per-item fixed currency amounts — stored as-is\n // (no x100 scaling: they feed directly into per-unit arithmetic in calculateValues).\n const partnerAmountInput = toNumber(r.partnerAmount)\n const customerAmountInput = toNumber(r.customerAmount)\n const legacyPartnerSplitInput = toNumber(r.partnerSplit)\n const legacyCustomerSplitInput = toNumber(r.customerSplit)\n\n const hasNewFixedInputs = partnerAmountInput != null || customerAmountInput != null\n const hasLegacyFixedInputs =\n legacyPartnerSplitInput != null || legacyCustomerSplitInput != null\n\n if (hasNewFixedInputs) {\n if (partnerAmountInput == null || partnerAmountInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: Partner fixed amount must be a non-negative number`,\n 400,\n )\n }\n\n if (customerAmountInput == null || customerAmountInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: Customer fixed amount must be a non-negative number`,\n 400,\n )\n }\n\n partnerAmount = partnerAmountInput\n customerAmount = customerAmountInput\n partnerSplit = partnerAmountInput\n customerSplit = customerAmountInput\n } else if (hasLegacyFixedInputs) {\n if (legacyPartnerSplitInput == null || legacyPartnerSplitInput < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be non-negative numbers`,\n 400,\n )\n }\n\n const resolvedLegacyCustomerSplit =\n legacyCustomerSplitInput ?? 100 - legacyPartnerSplitInput\n\n if (resolvedLegacyCustomerSplit == null || resolvedLegacyCustomerSplit < 0) {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be non-negative numbers`,\n 400,\n )\n }\n\n partnerSplit = legacyPartnerSplitInput\n customerSplit = resolvedLegacyCustomerSplit\n partnerAmount = null\n customerAmount = null\n } else {\n throw new APIError(\n `Commission rule ${index + 1}: For fixed commissions, both partner and customer values must be provided`,\n 400,\n )\n }\n }\n\n return {\n ...rule,\n appliesTo: appliesTo === 'categories' ? 'segments' : appliesTo,\n totalCommission: {\n type,\n ...(typeof r.totalCommission.value === 'number'\n ? { value: r.totalCommission.value }\n : {}),\n ...(typeof r.totalCommission.maxAmount === 'number'\n ? { maxAmount: r.totalCommission.maxAmount }\n : {}),\n },\n partnerPercent,\n customerPercent,\n partnerAmount,\n customerAmount,\n partnerSplit,\n customerSplit,\n splitWarning,\n }\n },\n )\n\n return data\n },\n ]\n\n return {\n slug: collections.referralProgramsSlug,\n admin: {\n useAsTitle: 'name',\n defaultColumns: ['id', 'name', 'commissionRules', 'isActive'],\n group: adminGroups.referralsGroup,\n },\n access: {\n read: access.canUseReferrals || (() => true),\n create: access.isAdmin || (() => false),\n update: access.isAdmin || (() => false),\n delete: access.isAdmin || (() => false),\n },\n hooks: {\n beforeChange,\n },\n fields: [\n {\n name: 'name',\n type: 'text',\n required: true,\n admin: {\n description: 'Name of the referral program',\n },\n },\n {\n name: 'isActive',\n type: 'checkbox',\n defaultValue: true,\n admin: {\n description: 'Whether this referral program is currently active',\n },\n },\n {\n name: 'maxPartnerCommissionPerOrder',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Maximum commission per order for partner (enter normal currency value, e.g. 50 for $50). Leave empty for no cap.',\n },\n },\n {\n name: 'maxCustomerDiscountPerOrder',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Maximum customer discount per order (enter normal currency value, e.g. 25 for $25). Leave empty for no cap.',\n },\n },\n {\n name: 'minOrderAmount',\n type: 'number',\n min: 0,\n admin: {\n description:\n 'Minimum cart subtotal required for this program (enter normal currency value, e.g. 100 for $100). Leave empty for no minimum.',\n },\n },\n {\n name: 'commissionRules',\n type: 'array',\n required: true,\n minRows: 1,\n admin: {\n description: 'Rules for referral commission and customer discount distribution.',\n },\n fields: [\n {\n name: 'appliesTo',\n type: 'select',\n required: true,\n options: [\n { label: 'All Products', value: 'all' },\n { label: 'Specific Products', value: 'products' },\n { label: 'Categories and Tags', value: 'segments' },\n ],\n defaultValue: 'all',\n },\n {\n name: 'products',\n type: 'relationship',\n relationTo: relationSlugs.productsSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'products',\n description: 'Products this rule applies to',\n },\n },\n {\n name: 'categories',\n type: 'relationship',\n relationTo: relationSlugs.categoriesSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'segments',\n description: 'Any matching category can activate this rule',\n },\n },\n {\n name: 'tags',\n type: 'relationship',\n relationTo: relationSlugs.tagsSlug,\n hasMany: true,\n admin: {\n condition: (_: unknown, siblingData: { appliesTo?: string }) =>\n siblingData?.appliesTo === 'segments',\n description: 'Any matching tag can activate this rule',\n },\n },\n {\n name: 'totalCommission',\n type: 'group',\n admin: {\n description: 'Total commission pool configuration',\n },\n fields: [\n {\n name: 'type',\n type: 'select',\n required: true,\n options: allowedTotalCommissionTypes.map((value) => ({\n label: value === 'fixed' ? 'Fixed Amount' : 'Percentage of Order',\n value,\n })),\n defaultValue: allowedTotalCommissionTypes.includes('fixed')\n ? 'fixed'\n : 'percentage',\n },\n ],\n },\n {\n name: 'partnerPercent',\n type: 'number',\n min: 0,\n max: 100,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'percentage',\n description: 'Partner share in percent (0–100)',\n },\n },\n {\n name: 'customerPercent',\n type: 'number',\n min: 0,\n max: 100,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'percentage',\n description:\n 'Customer share percentage (0–100). Partner + Customer cannot exceed 100.',\n },\n },\n {\n name: 'partnerAmount',\n type: 'number',\n min: 0,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'fixed',\n description:\n 'Fixed partner commission amount per item (normal currency value, e.g. 12.50 for $12.50).',\n },\n },\n {\n name: 'customerAmount',\n type: 'number',\n min: 0,\n admin: {\n condition: (_: unknown, siblingData: { totalCommission?: { type?: string } }) =>\n siblingData?.totalCommission?.type === 'fixed',\n description:\n 'Fixed customer discount amount per item (normal currency value, e.g. 5 for $5).',\n },\n },\n {\n name: 'partnerSplit',\n type: 'number',\n min: 0,\n admin: {\n hidden: true,\n description:\n 'Canonical storage field. Percentage mode: percent value. Fixed mode: per-item amount.',\n },\n },\n {\n name: 'customerSplit',\n type: 'number',\n min: 0,\n admin: {\n hidden: true,\n description:\n 'Canonical storage field. Percentage mode: percent value. Fixed mode: per-item amount.',\n },\n },\n ],\n },\n ],\n timestamps: true,\n }\n}\n","export const DEFAULT_PRICE_CURRENCY = 'AED'\n\ntype PriceEntity = {\n price?: number | null\n} & Record<string, unknown>\n\nfunction normalizeCurrencyCode(currencyCode?: string): string {\n if (!currencyCode) return DEFAULT_PRICE_CURRENCY\n return currencyCode.toUpperCase()\n}\n\nfunction readNumberField(entity: unknown, key: string): number | undefined {\n if (!entity || typeof entity !== 'object') return undefined\n const value = (entity as Record<string, unknown>)[key]\n return typeof value === 'number' ? value : undefined\n}\n\nexport function getPriceFieldKey(currencyCode: string): string {\n return `priceIn${normalizeCurrencyCode(currencyCode)}`\n}\n\nexport function readMoneyField(\n entity: PriceEntity | null | undefined,\n currencyCode: string,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n): number | undefined {\n if (!entity) return undefined\n\n const primaryField = getPriceFieldKey(currencyCode)\n const primary = readNumberField(entity, primaryField)\n if (typeof primary === 'number') return primary\n\n const fallbackField = getPriceFieldKey(defaultCurrencyCode)\n if (fallbackField !== primaryField) {\n const fallback = readNumberField(entity, fallbackField)\n if (typeof fallback === 'number') return fallback\n }\n\n return typeof entity.price === 'number' ? entity.price : undefined\n}\n\nexport function resolveMoneyField(\n entity: PriceEntity | null | undefined,\n currencyCode: string,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n): number {\n return readMoneyField(entity, currencyCode, defaultCurrencyCode) ?? 0\n}\n\nexport function getCartItemUnitPrice({\n item,\n product,\n variant,\n currencyCode,\n defaultCurrencyCode = DEFAULT_PRICE_CURRENCY,\n}: {\n item?: {\n price?: number | null\n unitPrice?: number | null\n } | null\n product?: PriceEntity | null\n variant?: PriceEntity | null\n currencyCode: string\n defaultCurrencyCode?: string\n}): number {\n if (typeof item?.price === 'number') return item.price\n if (typeof item?.unitPrice === 'number') return item.unitPrice\n if (variant) return resolveMoneyField(variant, currencyCode, defaultCurrencyCode)\n return resolveMoneyField(product, currencyCode, defaultCurrencyCode)\n}\n","import { getCartItemUnitPrice } from './pricing'\n\n// ---------------------------------------------------------------------------\n// Cent-safe arithmetic helpers\n//\n// Policy:\n// • All monetary values arrive from the DB / admin in NORMAL CURRENCY units\n// (e.g. 10.50 means $10.50).\n// • Before any arithmetic we scale UP to integer CENTS (×100) so every\n// intermediate result is a safe integer with no floating-point drift.\n// • Results are scaled back DOWN (÷100) before being returned to callers,\n// who then store them in DB as normal currency again.\n// • Percentage values (0-100) stay as-is; they act as divisors inside the\n// formula and do not need independent scaling.\n// ---------------------------------------------------------------------------\n\n/** Convert a normal-currency amount to integer cents. */\nfunction toCents(amount: number): number {\n return Math.round(amount * 100)\n}\n\n/** Convert integer cents back to a normal-currency amount (2 dp max). */\nfunction fromCents(cents: number): number {\n return Math.round(cents) / 100\n}\n\n// ---------------------------------------------------------------------------\n// Public: coupon discount\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate the discount amount for a coupon.\n *\n * @param coupon - Coupon document from DB (values in normal currency).\n * @param cartTotal - Cart subtotal in normal currency.\n * @returns Discount amount in normal currency (2 dp).\n */\nexport function calculateCouponDiscount({\n coupon,\n cartTotal,\n}: {\n coupon: any\n cartTotal: number\n}): number {\n const cartCents = toCents(cartTotal)\n let discountCents = 0\n\n if (coupon.type === 'percentage') {\n // percentage value is 0-100, no scaling needed\n discountCents = Math.floor((cartCents * coupon.value) / 100)\n if (coupon.maxDiscountAmount != null) {\n const maxCents = toCents(coupon.maxDiscountAmount)\n if (discountCents > maxCents) discountCents = maxCents\n }\n } else if (coupon.type === 'fixed') {\n discountCents = toCents(coupon.value)\n if (discountCents > cartCents) discountCents = cartCents\n }\n\n return fromCents(discountCents)\n}\n\n// ---------------------------------------------------------------------------\n// Internal helpers\n// ---------------------------------------------------------------------------\n\nfunction relationId(value: any): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nconst allowedCommissionTypesSet = (\n allowed: Array<'fixed' | 'percentage'> | undefined,\n): Set<string> =>\n new Set((allowed && allowed.length ? allowed : ['fixed', 'percentage']).map((v) => v))\n\nfunction normalizeIds(values: any[] | null | undefined): Array<string | number> {\n if (!Array.isArray(values)) return []\n return values.map(relationId).filter((v): v is string | number => v != null)\n}\n\nfunction getRuleSplits(rule: any): { partnerSplit: number; customerSplit: number } | null {\n const partnerRaw =\n typeof rule.partnerSplit === 'number'\n ? rule.partnerSplit\n : typeof rule.referrerSplit === 'number'\n ? rule.referrerSplit\n : null\n if (partnerRaw == null) return null\n\n const customerRaw =\n typeof rule.customerSplit === 'number'\n ? rule.customerSplit\n : typeof rule.refereeSplit === 'number'\n ? rule.refereeSplit\n : 100 - partnerRaw\n\n return { partnerSplit: partnerRaw, customerSplit: customerRaw }\n}\n\n// ---------------------------------------------------------------------------\n// Core per-item reward calculation (all values in CENTS)\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate partner and customer reward for a single line item.\n *\n * ALL inputs are expected in CENTS.\n * Returns rewards in CENTS, or null if the rule is inapplicable.\n */\nfunction calculateItemRewardByRule({\n rule,\n itemTotalCents,\n quantity,\n allowedTotalCommissionTypes,\n}: {\n rule: any\n itemTotalCents: number\n quantity: number\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { partner: number; customer: number } | null {\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n // ── Shared / v2 model ────────────────────────────────────────────────────\n if (rule.totalCommission) {\n if (!allowedTypes.has(rule.totalCommission.type)) return null\n\n // maxAmount is stored in normal currency → convert to cents\n const resolvedMaxAmountCents =\n typeof rule.totalCommission.maxAmount === 'number' &&\n Number.isFinite(rule.totalCommission.maxAmount)\n ? toCents(rule.totalCommission.maxAmount)\n : null\n\n // ── Fixed direct mode: partnerSplit / customerSplit are per-unit currency amounts\n if (rule.totalCommission.type === 'fixed' && rule.totalCommission.value == null) {\n const partnerAmtPerUnitCents =\n typeof rule.partnerSplit === 'number' ? toCents(rule.partnerSplit) : null\n const customerAmtPerUnitCents =\n typeof rule.customerSplit === 'number' ? toCents(rule.customerSplit) : null\n if (partnerAmtPerUnitCents == null || customerAmtPerUnitCents == null) return null\n\n let partner = partnerAmtPerUnitCents * quantity\n let customer = customerAmtPerUnitCents * quantity\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n const totalPot = partner + customer\n if (totalPot > maxPotForLine && totalPot > 0) {\n const ratio = maxPotForLine / totalPot\n partner = Math.floor(partner * ratio)\n customer = Math.floor(customer * ratio)\n }\n }\n\n return { partner, customer }\n }\n\n // ── Percentage mode ───────────────────────────────────────────────────\n if (rule.totalCommission.type === 'percentage') {\n const commissionValue =\n typeof rule.totalCommission.value === 'number' &&\n Number.isFinite(rule.totalCommission.value)\n ? rule.totalCommission.value\n : null\n\n if (commissionValue == null) {\n // Direct percent splits (no totalCommission.value)\n const partnerPercentInput =\n typeof rule.partnerPercent === 'number'\n ? rule.partnerPercent\n : typeof rule.partnerSplit === 'number'\n ? rule.partnerSplit\n : null\n if (partnerPercentInput == null || partnerPercentInput < 0 || partnerPercentInput > 100) {\n return null\n }\n\n const customerPercentInput =\n typeof rule.customerPercent === 'number'\n ? rule.customerPercent\n : typeof rule.customerSplit === 'number'\n ? rule.customerSplit\n : 100 - partnerPercentInput\n\n if (\n customerPercentInput == null ||\n customerPercentInput < 0 ||\n customerPercentInput > 100\n ) {\n return null\n }\n\n let partner = Math.floor((itemTotalCents * partnerPercentInput) / 100)\n let customer = Math.floor((itemTotalCents * customerPercentInput) / 100)\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n const totalForLine = partner + customer\n if (totalForLine > maxPotForLine && totalForLine > 0) {\n const ratio = maxPotForLine / totalForLine\n partner = Math.floor(partner * ratio)\n customer = Math.floor(customer * ratio)\n }\n }\n\n return { partner, customer }\n }\n\n // totalCommission.value drives the total pool (percentage of item total)\n let totalPotCents = Math.floor((itemTotalCents * commissionValue) / 100)\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n if (totalPotCents > maxPotForLine) totalPotCents = maxPotForLine\n }\n\n const splits = getRuleSplits(rule)\n if (!splits) return null\n\n return {\n partner: Math.floor((totalPotCents * splits.partnerSplit) / 100),\n customer: Math.floor((totalPotCents * splits.customerSplit) / 100),\n }\n }\n\n // ── Fixed pool mode: totalCommission.value is a per-unit currency amount\n {\n const splits = getRuleSplits(rule)\n if (!splits) return null\n\n // totalCommission.value is a per-unit currency amount → convert to cents\n let totalPotCents = toCents(rule.totalCommission.value) * quantity\n\n if (resolvedMaxAmountCents != null) {\n const maxPotForLine = resolvedMaxAmountCents * quantity\n if (totalPotCents > maxPotForLine) totalPotCents = maxPotForLine\n }\n\n return {\n partner: Math.floor((totalPotCents * splits.partnerSplit) / 100),\n customer: Math.floor((totalPotCents * splits.customerSplit) / 100),\n }\n }\n }\n\n // ── Legacy direct-reward model (migration compatibility) ─────────────────\n if (rule.referrerReward && rule.refereeReward) {\n let partner = 0\n if (rule.referrerReward.type === 'percentage') {\n partner = Math.floor((itemTotalCents * rule.referrerReward.value) / 100)\n } else {\n partner = toCents(rule.referrerReward.value) * quantity\n }\n if (rule.referrerReward.maxReward != null) {\n const maxCents = toCents(rule.referrerReward.maxReward)\n if (partner > maxCents) partner = maxCents\n }\n\n let customer = 0\n if (rule.refereeReward.type === 'percentage') {\n customer = Math.floor((itemTotalCents * rule.refereeReward.value) / 100)\n } else {\n customer = toCents(rule.refereeReward.value) * quantity\n }\n if (rule.refereeReward.maxReward != null) {\n const maxCents = toCents(rule.refereeReward.maxReward)\n if (customer > maxCents) customer = maxCents\n }\n\n return { partner, customer }\n }\n\n return null\n}\n\n// ---------------------------------------------------------------------------\n// Rule selection helpers\n// ---------------------------------------------------------------------------\n\nfunction getItemCategoryIds(item: any): Array<string | number> {\n const productCategories = Array.isArray(item?.product?.categories)\n ? normalizeIds(item.product.categories)\n : []\n const singleCategory = relationId(item?.category ?? item?.product?.category)\n return [...productCategories, ...(singleCategory != null ? [singleCategory] : [])]\n}\n\nfunction getItemTagIds(item: any): Array<string | number> {\n return Array.isArray(item?.product?.tags) ? normalizeIds(item.product.tags) : []\n}\n\nfunction selectBestRuleForItem({\n rules,\n item,\n itemTotalCents,\n quantity,\n cartTotalCents,\n minOrderAmountCents,\n allowedTotalCommissionTypes,\n}: {\n rules: any[]\n item: any\n itemTotalCents: number\n quantity: number\n cartTotalCents: number\n minOrderAmountCents?: number | null\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { rule: any; reward: { partner: number; customer: number } } | null {\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n const eligibleRules = rules.filter((rule: any) => {\n const hasSharedType = rule?.totalCommission?.type\n ? allowedTypes.has(rule.totalCommission.type)\n : true\n if (!hasSharedType) return false\n\n // minOrderAmount on the rule itself is stored in normal currency → cents\n const resolvedMinCents =\n minOrderAmountCents != null && Number.isFinite(minOrderAmountCents)\n ? minOrderAmountCents\n : typeof rule?.minOrderAmount === 'number' && Number.isFinite(rule.minOrderAmount)\n ? toCents(rule.minOrderAmount)\n : null\n\n if (resolvedMinCents != null) {\n return cartTotalCents >= resolvedMinCents\n }\n return true\n })\n\n const productId = relationId(item.product)\n const itemCategoryIds = new Set(getItemCategoryIds(item))\n const itemTagIds = new Set(getItemTagIds(item))\n\n const productCandidates = eligibleRules.filter(\n (r: any) =>\n r.appliesTo === 'products' &&\n normalizeIds(r.products).some((id) => productId != null && id === productId),\n )\n\n const segmentCategoryCandidates = eligibleRules.filter((r: any) => {\n const isSegment = r.appliesTo === 'segments' || r.appliesTo === 'categories'\n if (!isSegment) return false\n return normalizeIds(r.categories).some((id) => itemCategoryIds.has(id))\n })\n\n const segmentTagCandidates = eligibleRules.filter((r: any) => {\n if (r.appliesTo !== 'segments') return false\n return normalizeIds(r.tags).some((id) => itemTagIds.has(id))\n })\n\n const allCandidates = eligibleRules.filter((r: any) => r.appliesTo === 'all')\n\n const levels = [productCandidates, segmentCategoryCandidates, segmentTagCandidates, allCandidates]\n const candidates = levels.find((level) => level.length > 0) ?? []\n if (!candidates.length) return null\n\n let best: { rule: any; reward: { partner: number; customer: number } } | null = null\n\n for (const rule of candidates) {\n const reward = calculateItemRewardByRule({\n rule,\n itemTotalCents,\n quantity,\n allowedTotalCommissionTypes,\n })\n if (!reward) continue\n\n if (!best) {\n best = { rule, reward }\n continue\n }\n\n if (reward.customer > best.reward.customer) {\n best = { rule, reward }\n continue\n }\n\n if (reward.customer === best.reward.customer && reward.partner > best.reward.partner) {\n best = { rule, reward }\n }\n }\n\n return best\n}\n\n// ---------------------------------------------------------------------------\n// Public: minimum order amount\n// ---------------------------------------------------------------------------\n\n/**\n * Returns the effective minimum order amount for a program in NORMAL CURRENCY.\n * Returns null if there is no minimum.\n */\nexport function getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes,\n}: {\n program: any\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): number | null {\n if (typeof program?.minOrderAmount === 'number' && Number.isFinite(program.minOrderAmount)) {\n return program.minOrderAmount\n }\n\n const rules = Array.isArray(program?.commissionRules) ? program.commissionRules : []\n if (!rules.length) return null\n\n const allowedTypes = allowedCommissionTypesSet(allowedTotalCommissionTypes)\n\n const minValues = rules\n .filter((rule: any) => {\n if (rule?.totalCommission?.type) return allowedTypes.has(rule.totalCommission.type)\n return true\n })\n .map((rule: any) => rule?.minOrderAmount)\n .filter(\n (value: unknown): value is number => typeof value === 'number' && Number.isFinite(value),\n )\n\n if (!minValues.length) return null\n return Math.min(...minValues)\n}\n\n// ---------------------------------------------------------------------------\n// Public: commission + discount calculation\n// ---------------------------------------------------------------------------\n\n/**\n * Calculate total partner commission and customer discount for a cart.\n *\n * All monetary inputs are in NORMAL CURRENCY.\n * Returns results in NORMAL CURRENCY (2 dp).\n */\nexport function calculateCommissionAndDiscount({\n cartItems,\n program,\n currencyCode = 'AED',\n cartTotal = 0,\n allowedTotalCommissionTypes,\n}: {\n cartItems: any[]\n program: any\n currencyCode?: string\n cartTotal?: number\n allowedTotalCommissionTypes?: Array<'fixed' | 'percentage'>\n}): { partnerCommission: number; customerDiscount: number } {\n const rules = Array.isArray(program?.commissionRules) ? program.commissionRules : []\n\n if (!rules.length) {\n return { partnerCommission: 0, customerDiscount: 0 }\n }\n\n // Scale cart total to cents for eligibility checks\n const cartTotalCents = toCents(cartTotal)\n\n // Scale program-level minOrderAmount to cents (if present) for rule filtering\n const programMinOrderAmountCents =\n typeof program?.minOrderAmount === 'number' && Number.isFinite(program.minOrderAmount)\n ? toCents(program.minOrderAmount)\n : null\n\n let totalPartnerCents = 0\n let totalCustomerCents = 0\n\n for (const item of cartItems) {\n const product = typeof item.product === 'object' ? item.product : {}\n const variant = typeof item.variant === 'object' ? item.variant : {}\n\n // Unit price from DB is in normal currency → convert to cents\n const itemPriceCurrency = getCartItemUnitPrice({\n item,\n product,\n variant,\n currencyCode,\n })\n\n const quantity = item.quantity ?? 1\n const itemTotalCents = toCents(itemPriceCurrency) * quantity\n\n const bestMatch = selectBestRuleForItem({\n rules,\n item: { ...item, product },\n itemTotalCents,\n quantity,\n cartTotalCents,\n minOrderAmountCents: programMinOrderAmountCents,\n allowedTotalCommissionTypes,\n })\n\n if (!bestMatch) continue\n\n totalPartnerCents += bestMatch.reward.partner\n totalCustomerCents += bestMatch.reward.customer\n }\n\n // Per-order caps are stored in normal currency → convert to cents for comparison\n const maxPartnerCents =\n typeof program?.maxPartnerCommissionPerOrder === 'number' &&\n Number.isFinite(program.maxPartnerCommissionPerOrder)\n ? toCents(program.maxPartnerCommissionPerOrder)\n : null\n\n const maxCustomerCents =\n typeof program?.maxCustomerDiscountPerOrder === 'number' &&\n Number.isFinite(program.maxCustomerDiscountPerOrder)\n ? toCents(program.maxCustomerDiscountPerOrder)\n : null\n\n if (maxPartnerCents != null) {\n totalPartnerCents = Math.min(totalPartnerCents, maxPartnerCents)\n }\n if (maxCustomerCents != null) {\n totalCustomerCents = Math.min(totalCustomerCents, maxCustomerCents)\n }\n\n // Convert back to normal currency for return\n return {\n partnerCommission: fromCents(totalPartnerCents),\n customerDiscount: fromCents(totalCustomerCents),\n }\n}\n","/**\n * Rounds a number to 2 decimal places (standard for monetary values).\n */\nexport function roundTo2(value: number): number {\n return Math.round(value * 100) / 100\n}\n","import { addDataAndFileToRequest, type Endpoint, type PayloadHandler } from 'payload'\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n calculateCouponDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction writeField(doc: Record<string, unknown>, field: string, value: unknown): void {\n doc[field] = value\n}\n\nfunction normalizeCode(value: unknown): string {\n return typeof value === 'string' ? value.trim().toUpperCase() : ''\n}\n\nasync function ensureRequestData(req: any): Promise<Record<string, unknown>> {\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n await addDataAndFileToRequest(req)\n } catch {\n // Fallback for non-standard test/mocked requests where payload parser cannot run.\n }\n\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n const parsed = await req?.json?.()\n if (parsed && typeof parsed === 'object') {\n req.data = parsed\n return parsed as Record<string, unknown>\n }\n } catch {\n // Ignore malformed/empty body; validation below will return proper 400 errors.\n }\n\n req.data = {}\n return req.data\n}\n\nasync function findByNormalizedCode({\n payload,\n collection,\n normalizedCode,\n}: {\n payload: any\n collection: string\n normalizedCode: string\n}): Promise<any | null> {\n const exactQuery = await payload.find({\n collection,\n where: {\n normalizedCode: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n if (exactQuery?.docs?.[0]) return exactQuery.docs[0]\n\n const lowerQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toLowerCase() },\n },\n limit: 1,\n })\n\n if (lowerQuery?.docs?.[0]) return lowerQuery.docs[0]\n\n const upperQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toUpperCase() },\n },\n limit: 1,\n })\n\n if (upperQuery?.docs?.[0]) return upperQuery.docs[0]\n\n const exactCodeQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n return exactCodeQuery?.docs?.[0] ?? null\n}\n\nexport const applyCouponHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload } = req\n const fields = pluginConfig.integration.fields\n const collections = pluginConfig.integration.collections\n const data = await ensureRequestData(req)\n\n const rawCode = data?.code\n const cartIDRaw = data?.cartID\n const cartID = typeof cartIDRaw === 'string' || typeof cartIDRaw === 'number' ? cartIDRaw : null\n const customerEmail = typeof data?.customerEmail === 'string' ? data.customerEmail : undefined\n\n const normalizedCode = normalizeCode(rawCode)\n\n if (!normalizedCode || !cartID) {\n return Response.json(\n {\n success: false,\n error: `${pluginConfig.enableReferrals ? 'Referral code' : 'Coupon code'} and cart ID are required`,\n },\n { status: 400 },\n )\n }\n\n const allowCoupon = await Promise.resolve(\n pluginConfig.policies.canApplyCoupon({ req, user: req?.user, payload }),\n )\n\n const allowReferral = await Promise.resolve(\n pluginConfig.policies.canApplyReferral({ req, user: req?.user, payload }),\n )\n\n if (!allowCoupon && !(pluginConfig.enableReferrals && allowReferral)) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n try {\n const cart = await payload.findByID({\n collection: collections.cartsSlug,\n id: cartID,\n depth: 2,\n })\n\n if (!cart) {\n return Response.json({ success: false, error: 'Cart not found' }, { status: 404 })\n }\n\n const cartAppliedCoupon = relationId(\n readField(cart, fields.cartAppliedCouponField) as RelationValue,\n )\n const cartAppliedReferral = relationId(\n readField(cart, fields.cartAppliedReferralCodeField) as RelationValue,\n )\n\n if (\n pluginConfig.referralConfig.singleCodePerCart &&\n (cartAppliedCoupon || cartAppliedReferral)\n ) {\n return Response.json(\n {\n success: false,\n error:\n 'A code has already been applied to this cart. Only one code can be used per order.',\n },\n { status: 400 },\n )\n }\n\n if (pluginConfig.enableReferrals && allowReferral) {\n const referralResult = await handleReferralCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n pluginConfig,\n })\n\n if (\n !referralResult.ok &&\n referralResult.status === 404 &&\n pluginConfig.referralConfig.allowBothSystems &&\n allowCoupon\n ) {\n return await handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n })\n }\n\n return referralResult\n }\n\n if (!allowCoupon) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n })\n } catch (error) {\n console.error('Code application error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n }\n\nasync function handleCouponCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n customerEmail,\n pluginConfig,\n}: {\n payload: any\n cart: any\n cartID: string | number\n normalizedCode: string\n customerEmail?: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const resolvers = pluginConfig.integration.resolvers\n const coupon = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.couponsSlug,\n normalizedCode,\n })\n\n if (!coupon) {\n return Response.json({ success: false, error: 'Invalid coupon code' }, { status: 404 })\n }\n\n const now = new Date()\n const activeFrom = coupon.activeFrom ? new Date(coupon.activeFrom) : null\n const activeUntil = coupon.activeUntil ? new Date(coupon.activeUntil) : null\n\n if (activeFrom && now < activeFrom) {\n return Response.json({ success: false, error: 'Coupon is not yet active' }, { status: 400 })\n }\n\n if (activeUntil && now > activeUntil) {\n return Response.json({ success: false, error: 'Coupon has expired' }, { status: 400 })\n }\n\n if (coupon.usageLimit && coupon.usageCount >= coupon.usageLimit) {\n return Response.json({ success: false, error: 'Coupon usage limit exceeded' }, { status: 400 })\n }\n\n if (coupon.perCustomerLimit != null && coupon.perCustomerLimit > 0) {\n const email = typeof customerEmail === 'string' ? customerEmail.trim() : ''\n if (!email) {\n return Response.json(\n {\n success: false,\n error: 'Customer email is required for this coupon.',\n },\n { status: 400 },\n )\n }\n\n const ordersQuery = await payload.find({\n collection: pluginConfig.orderIntegration.ordersSlug,\n where: {\n and: [\n { [fields.orderAppliedCouponField]: { equals: coupon.id } },\n {\n [pluginConfig.orderIntegration.orderCustomerEmailField]: {\n equals: email,\n },\n },\n {\n [pluginConfig.orderIntegration.orderPaymentStatusField]: {\n equals: pluginConfig.orderIntegration.orderPaidStatusValue,\n },\n },\n ],\n },\n limit: 0,\n })\n\n if (ordersQuery.totalDocs >= coupon.perCustomerLimit) {\n return Response.json(\n {\n success: false,\n error: 'You have reached the maximum uses for this coupon.',\n },\n { status: 400 },\n )\n }\n }\n\n const existingCouponId = relationId(\n readField(cart, fields.cartAppliedCouponField) as RelationValue,\n )\n if (existingCouponId === coupon.id) {\n return Response.json(\n { success: false, error: 'Coupon already applied to this cart' },\n { status: 400 },\n )\n }\n\n // Use cartSubtotal (pre-discount baseline) for all order-value eligibility checks\n // and discount calculations. cartTotal may already reflect a previously applied\n // discount and would produce an inconsistent baseline for min/max enforcement.\n const cartSubtotal = Number(resolvers.getCartSubtotal(cart)) || 0\n const cartTotal = Number(resolvers.getCartTotal(cart)) || cartSubtotal || 0\n\n if (coupon.minOrderValue && cartSubtotal < coupon.minOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${coupon.minOrderValue} ${pluginConfig.defaultCurrency} required`,\n },\n { status: 400 },\n )\n }\n\n if (coupon.maxOrderValue && cartSubtotal > coupon.maxOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Maximum order value of ${coupon.maxOrderValue} ${pluginConfig.defaultCurrency} exceeded`,\n },\n { status: 400 },\n )\n }\n\n const discountAmount = calculateCouponDiscount({ coupon, cartTotal: cartSubtotal })\n const nextTotal = roundTo2(Math.max(0, cartTotal - discountAmount))\n\n const data: Record<string, unknown> = {}\n writeField(data, fields.cartAppliedCouponField, coupon.id)\n writeField(data, fields.cartDiscountAmountField, discountAmount)\n writeField(data, fields.cartTotalField, nextTotal)\n\n await payload.update({\n collection: pluginConfig.integration.collections.cartsSlug,\n id: cartID,\n data,\n })\n\n return Response.json({\n success: true,\n message: 'Coupon applied successfully',\n coupon: {\n code: coupon.code,\n type: coupon.type,\n value: coupon.value,\n },\n discount: discountAmount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nasync function handleReferralCode({\n payload,\n cart,\n cartID,\n normalizedCode,\n pluginConfig,\n}: {\n payload: any\n cart: any\n cartID: string | number\n normalizedCode: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const resolvers = pluginConfig.integration.resolvers\n\n const referralCode = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.referralCodesSlug,\n normalizedCode,\n })\n\n if (!referralCode) {\n return Response.json({ success: false, error: 'Invalid referral code' }, { status: 404 })\n }\n\n if (!referralCode.isActive) {\n return Response.json({ success: false, error: 'Referral code is not active' }, { status: 400 })\n }\n\n if (referralCode.expiresAt && new Date() > new Date(referralCode.expiresAt)) {\n return Response.json({ success: false, error: 'Referral code has expired' }, { status: 400 })\n }\n\n if (referralCode.usageLimit && referralCode.usageCount >= referralCode.usageLimit) {\n return Response.json(\n { success: false, error: 'Referral code usage limit exceeded' },\n { status: 400 },\n )\n }\n\n const programId =\n typeof referralCode.program === 'string' || typeof referralCode.program === 'number'\n ? referralCode.program\n : referralCode.program?.id\n\n const program = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n\n if (!program || !program.isActive) {\n return Response.json(\n { success: false, error: 'Referral program is not active' },\n { status: 400 },\n )\n }\n\n const existingReferralId = relationId(\n readField(cart, fields.cartAppliedReferralCodeField) as RelationValue,\n )\n\n if (existingReferralId === referralCode.id) {\n return Response.json(\n { success: false, error: 'Referral code already applied to this cart' },\n { status: 400 },\n )\n }\n\n const cartItems = resolvers.getCartItems(cart)\n // Use cartSubtotal as the pre-discount baseline for min-order enforcement and\n // commission/discount calculations, matching the recalculateCart hook policy.\n const cartSubtotal = Number(resolvers.getCartSubtotal(cart)) || 0\n const cartTotal = Number(resolvers.getCartTotal(cart)) || cartSubtotal || 0\n\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && cartSubtotal < minOrderAmount) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderAmount} ${pluginConfig.defaultCurrency} required for this referral program`,\n },\n { status: 400 },\n )\n }\n\n const { partnerCommission, customerDiscount } = calculateCommissionAndDiscount({\n cartItems,\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal: cartSubtotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const roundedPartnerCommission = roundTo2(partnerCommission)\n const roundedCustomerDiscount = roundTo2(customerDiscount)\n const nextTotal = roundTo2(Math.max(0, cartTotal - roundedCustomerDiscount))\n\n const data: Record<string, unknown> = {}\n writeField(data, fields.cartAppliedReferralCodeField, referralCode.id)\n writeField(data, fields.cartPartnerCommissionField, roundedPartnerCommission)\n writeField(data, fields.cartCustomerDiscountField, roundedCustomerDiscount)\n writeField(data, fields.cartTotalField, nextTotal)\n\n await payload.update({\n collection: pluginConfig.integration.collections.cartsSlug,\n id: cartID,\n data,\n })\n\n return Response.json({\n success: true,\n message: 'Referral code applied successfully',\n referralCode: {\n code: referralCode.code,\n },\n partnerCommission: roundedPartnerCommission,\n customerDiscount: roundedCustomerDiscount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nexport const applyCouponEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.applyCoupon,\n method: 'post',\n handler: applyCouponHandler({ pluginConfig }),\n})\n","import type { Endpoint, PayloadHandler } from 'payload'\n\nimport type { PartnerDashboardData, PartnerStats, SanitizedCouponPluginOptions } from '../types'\nimport { isAdminUser, isPartnerUser } from '../utilities/userRoles'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction asNumber(value: unknown): number {\n return typeof value === 'number' && Number.isFinite(value) ? value : 0\n}\n\nfunction asString(value: unknown): string {\n return typeof value === 'string' ? value : ''\n}\n\nfunction toStatsStatus(value: unknown): 'pending' | 'paid' | 'cancelled' {\n if (value === 'paid') return 'paid'\n if (value === 'cancelled') return 'cancelled'\n return 'pending'\n}\n\nexport const partnerStatsHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload, user } = req\n const fields = pluginConfig.integration.fields\n const collections = pluginConfig.integration.collections\n\n if (!user) {\n return Response.json({ success: false, error: 'Authentication required' }, { status: 401 })\n }\n\n const typedUser = user as { id?: string | number }\n const userID = pluginConfig.integration.resolvers.getUserID({ req, user })\n\n if (userID == null) {\n return Response.json(\n { success: false, error: 'Unable to resolve user identity' },\n { status: 403 },\n )\n }\n\n const isPartner =\n isPartnerUser({ user: typedUser, roleConfig: pluginConfig.roleConfig }) ||\n (await Promise.resolve(pluginConfig.access.isPartner?.({ req } as any)))\n\n const isAdmin =\n isAdminUser({ user: typedUser, roleConfig: pluginConfig.roleConfig }) ||\n (await Promise.resolve(pluginConfig.access.isAdmin?.({ req } as any)))\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canViewPartnerStats({\n req,\n user,\n payload,\n requestedPartnerID: userID,\n }),\n )\n\n if (!policyAllowed && !isAdmin && !isPartner) {\n return Response.json({ success: false, error: 'Partner access required' }, { status: 403 })\n }\n\n try {\n const referralCodesQuery = await payload.find({\n collection: pluginConfig.collections.referralCodesSlug,\n where: {\n partner: { equals: userID },\n },\n limit: 100,\n })\n\n const referralCodes = Array.isArray(referralCodesQuery?.docs) ? referralCodesQuery.docs : []\n\n let totalEarnings = 0\n let pendingEarnings = 0\n let paidEarnings = 0\n let totalReferrals = 0\n let successfulReferrals = 0\n\n const referralCodeData = referralCodes.map((code: any) => {\n totalEarnings += asNumber(code?.totalEarnings)\n pendingEarnings += asNumber(code?.pendingEarnings)\n paidEarnings += asNumber(code?.paidEarnings)\n totalReferrals += asNumber(code?.usageCount)\n successfulReferrals += asNumber(code?.successfulReferralsCount)\n\n return {\n id: String(code?.id ?? ''),\n code: asString(code?.code),\n usageCount: asNumber(code?.usageCount),\n totalEarnings: asNumber(code?.totalEarnings),\n isActive: Boolean(code?.isActive),\n }\n })\n\n const conversionRate = totalReferrals > 0 ? (successfulReferrals / totalReferrals) * 100 : 0\n const recentReferrals: PartnerStats['recentReferrals'] = []\n\n try {\n const referralCodeIDs = referralCodes\n .map((c: any) => relationId(c?.id as RelationValue))\n .filter((id): id is string | number => id != null)\n\n if (referralCodeIDs.length > 0) {\n const ordersQuery = await payload.find({\n collection: collections.ordersSlug,\n where: {\n [fields.orderAppliedReferralCodeField]: {\n in: referralCodeIDs,\n },\n },\n limit: 10,\n sort: `-${fields.orderCreatedAtField}`,\n })\n\n for (const order of (ordersQuery?.docs || []) as any[]) {\n const orderReferralRelation = readField(order, fields.orderAppliedReferralCodeField)\n const orderReferralID = relationId(orderReferralRelation as RelationValue)\n\n const matchedCode = referralCodes.find(\n (c: any) => relationId(c?.id as RelationValue) === orderReferralID,\n )\n\n const paymentStatus = readField(order, fields.orderPaymentStatusField)\n const createdAt = readField(order, fields.orderCreatedAtField)\n\n recentReferrals.push({\n id: String(order?.id ?? ''),\n code: asString(matchedCode?.code),\n orderValue: asNumber(readField(order, fields.cartTotalField) ?? order?.total),\n commission: asNumber(readField(order, fields.orderPartnerCommissionField)),\n date: asString(createdAt),\n status: toStatsStatus(paymentStatus),\n })\n }\n }\n } catch {\n // Host app may not expose expected order structure.\n }\n\n const monthlyEarnings: PartnerStats['monthlyEarnings'] = []\n const now = new Date()\n\n for (let i = 5; i >= 0; i--) {\n const monthDate = new Date(now.getFullYear(), now.getMonth() - i, 1)\n const monthName = monthDate.toLocaleString('default', { month: 'short', year: 'numeric' })\n\n monthlyEarnings.push({\n month: monthName,\n earnings: 0,\n referrals: 0,\n })\n }\n\n let program: PartnerDashboardData['program'] = null\n\n if (referralCodes.length > 0) {\n const firstCode = referralCodes[0] as any\n const programID = relationId(firstCode?.program as RelationValue)\n\n if (programID != null) {\n try {\n const programData = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programID,\n })\n\n if (programData) {\n const typedProgram = programData as any\n const firstRule = typedProgram?.commissionRules?.[0]\n\n const partnerSplit =\n asNumber(firstRule?.partnerSplit) ||\n asNumber(firstRule?.referrerSplit) ||\n asNumber(firstRule?.split?.partnerPercentage)\n\n const customerSplit =\n asNumber(firstRule?.customerSplit) ||\n asNumber(firstRule?.refereeSplit) ||\n asNumber(firstRule?.split?.customerPercentage) ||\n Math.max(0, 100 - partnerSplit)\n\n program = {\n name: asString(typedProgram?.name),\n commissionRate: partnerSplit,\n customerDiscount: customerSplit,\n }\n }\n } catch {\n // Program lookup failed or removed.\n }\n }\n }\n\n const stats: PartnerStats = {\n totalEarnings,\n pendingEarnings,\n paidEarnings,\n totalReferrals,\n successfulReferrals,\n conversionRate: Math.round(conversionRate * 100) / 100,\n recentReferrals,\n monthlyEarnings,\n }\n\n const dashboardData: PartnerDashboardData = {\n stats,\n referralCodes: referralCodeData,\n program,\n }\n\n return Response.json({\n success: true,\n data: dashboardData,\n currency: pluginConfig.defaultCurrency,\n })\n } catch (error) {\n console.error('Partner stats error:', error)\n return Response.json(\n { success: false, error: 'Failed to fetch partner stats' },\n { status: 500 },\n )\n }\n }\n\nexport const partnerStatsEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.partnerStats,\n method: 'get',\n handler: partnerStatsHandler({ pluginConfig }),\n})\n","import { addDataAndFileToRequest, type Endpoint, type PayloadHandler } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype Args = {\n pluginConfig: SanitizedCouponPluginOptions\n}\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction normalizeCode(value: unknown): string {\n return typeof value === 'string' ? value.trim().toUpperCase() : ''\n}\n\nasync function ensureRequestData(req: any): Promise<Record<string, unknown>> {\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n await addDataAndFileToRequest(req)\n } catch {\n // Fallback for non-standard test/mocked requests where payload parser cannot run.\n }\n\n if (req?.data && typeof req.data === 'object') return req.data as Record<string, unknown>\n\n try {\n const parsed = await req?.json?.()\n if (parsed && typeof parsed === 'object') {\n req.data = parsed\n return parsed as Record<string, unknown>\n }\n } catch {\n // Ignore malformed/empty body; validation below will return proper 400 errors.\n }\n\n req.data = {}\n return req.data\n}\n\nasync function findByNormalizedCode({\n payload,\n collection,\n normalizedCode,\n}: {\n payload: any\n collection: string\n normalizedCode: string\n}): Promise<any | null> {\n const normalizedQuery = await payload.find({\n collection,\n where: {\n normalizedCode: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n if (normalizedQuery?.docs?.length) return normalizedQuery.docs[0]\n\n const lowerQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toLowerCase() },\n },\n limit: 1,\n })\n\n if (lowerQuery?.docs?.length) return lowerQuery.docs[0]\n\n const upperQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode.toUpperCase() },\n },\n limit: 1,\n })\n\n if (upperQuery?.docs?.length) return upperQuery.docs[0]\n\n const exactQuery = await payload.find({\n collection,\n where: {\n code: { equals: normalizedCode },\n },\n limit: 1,\n })\n\n return exactQuery?.docs?.[0] ?? null\n}\n\nexport const validateCouponHandler =\n ({ pluginConfig }: Args): PayloadHandler =>\n async (req) => {\n const { payload } = req\n const data = await ensureRequestData(req)\n\n const rawCode = data?.code\n const cartValue = typeof data?.cartValue === 'number' ? data.cartValue : undefined\n const cartIDRaw = data?.cartID\n const cartID =\n typeof cartIDRaw === 'string' || typeof cartIDRaw === 'number' ? cartIDRaw : undefined\n const customerEmail = typeof data?.customerEmail === 'string' ? data.customerEmail : undefined\n\n const normalizedCode = normalizeCode(rawCode)\n\n if (!normalizedCode) {\n return Response.json(\n {\n success: false,\n error: 'Code is required',\n },\n { status: 400 },\n )\n }\n\n try {\n if (pluginConfig.enableReferrals) {\n const canApplyReferral = await Promise.resolve(\n pluginConfig.policies.canApplyReferral({\n req,\n user: req?.user,\n payload,\n }),\n )\n\n if (!canApplyReferral) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await validateReferralCode({\n payload,\n normalizedCode,\n cartID,\n pluginConfig,\n })\n }\n\n const canApplyCoupon = await Promise.resolve(\n pluginConfig.policies.canApplyCoupon({ req, user: req?.user, payload }),\n )\n\n if (!canApplyCoupon) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n return await validateCouponCode({\n payload,\n normalizedCode,\n cartValue,\n customerEmail,\n pluginConfig,\n })\n } catch (error) {\n console.error('Code validation error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n }\n\nasync function validateCouponCode({\n payload,\n normalizedCode,\n cartValue,\n customerEmail,\n pluginConfig,\n}: {\n payload: any\n normalizedCode: string\n cartValue?: number\n customerEmail?: string\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const fields = pluginConfig.integration.fields\n const couponData = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.couponsSlug,\n normalizedCode,\n })\n\n if (!couponData) {\n return Response.json({ success: false, error: 'Invalid coupon code' }, { status: 404 })\n }\n\n const now = new Date()\n const activeFrom = couponData.activeFrom ? new Date(couponData.activeFrom) : null\n const activeUntil = couponData.activeUntil ? new Date(couponData.activeUntil) : null\n\n if (activeFrom && now < activeFrom) {\n return Response.json({ success: false, error: 'Coupon is not yet active' }, { status: 400 })\n }\n\n if (activeUntil && now > activeUntil) {\n return Response.json({ success: false, error: 'Coupon has expired' }, { status: 400 })\n }\n\n if (couponData.usageLimit && couponData.usageCount >= couponData.usageLimit) {\n return Response.json({ success: false, error: 'Coupon usage limit exceeded' }, { status: 400 })\n }\n\n if (\n couponData.perCustomerLimit != null &&\n couponData.perCustomerLimit > 0 &&\n typeof customerEmail === 'string' &&\n customerEmail.trim().length > 0\n ) {\n const email = customerEmail.trim()\n const ordersQuery = await payload.find({\n collection: pluginConfig.orderIntegration.ordersSlug,\n where: {\n and: [\n { [fields.orderAppliedCouponField]: { equals: couponData.id } },\n {\n [pluginConfig.orderIntegration.orderCustomerEmailField]: {\n equals: email,\n },\n },\n {\n [pluginConfig.orderIntegration.orderPaymentStatusField]: {\n equals: pluginConfig.orderIntegration.orderPaidStatusValue,\n },\n },\n ],\n },\n limit: 0,\n })\n\n if (ordersQuery.totalDocs >= couponData.perCustomerLimit) {\n return Response.json(\n {\n success: false,\n error: 'You have reached the maximum uses for this coupon.',\n },\n { status: 400 },\n )\n }\n }\n\n // cartValue is the caller-supplied cart subtotal (pre-discount baseline).\n // Using the subtotal ensures min/max checks and discount calculations are\n // always based on the original item total, not a post-discount total.\n if (cartValue !== undefined) {\n const minOrderValue = couponData.minOrderValue\n const maxOrderValue = couponData.maxOrderValue\n\n if (minOrderValue && cartValue < minOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderValue} ${pluginConfig.defaultCurrency} required`,\n },\n { status: 400 },\n )\n }\n\n if (maxOrderValue && cartValue > maxOrderValue) {\n return Response.json(\n {\n success: false,\n error: `Maximum order value of ${maxOrderValue} ${pluginConfig.defaultCurrency} exceeded`,\n },\n { status: 400 },\n )\n }\n }\n\n let discount = 0\n if (cartValue !== undefined) {\n if (couponData.type === 'percentage') {\n discount = roundTo2((cartValue * couponData.value) / 100)\n if (couponData.maxDiscountAmount != null && discount > couponData.maxDiscountAmount) {\n discount = roundTo2(couponData.maxDiscountAmount)\n }\n } else if (couponData.type === 'fixed') {\n discount = roundTo2(couponData.value)\n if (discount > cartValue) discount = roundTo2(cartValue)\n }\n }\n\n return Response.json({\n success: true,\n coupon: {\n code: couponData.code,\n type: couponData.type,\n value: couponData.value,\n description: couponData.description,\n },\n discount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nasync function validateReferralCode({\n payload,\n normalizedCode,\n cartID,\n pluginConfig,\n}: {\n payload: any\n normalizedCode: string\n cartID?: string | number\n pluginConfig: SanitizedCouponPluginOptions\n}) {\n const collections = pluginConfig.integration.collections\n const resolvers = pluginConfig.integration.resolvers\n\n const referralData = await findByNormalizedCode({\n payload,\n collection: pluginConfig.collections.referralCodesSlug,\n normalizedCode,\n })\n\n if (!referralData) {\n return Response.json({ success: false, error: 'Referral code not found' }, { status: 404 })\n }\n\n if (!referralData.isActive) {\n return Response.json({ success: false, error: 'Referral code is not active' }, { status: 400 })\n }\n\n if (referralData.expiresAt && new Date() > new Date(referralData.expiresAt)) {\n return Response.json({ success: false, error: 'Referral code has expired' }, { status: 400 })\n }\n\n if (referralData.usageLimit && referralData.usageCount >= referralData.usageLimit) {\n return Response.json(\n { success: false, error: 'Referral code usage limit exceeded' },\n { status: 400 },\n )\n }\n\n const programId = relationId(referralData.program as RelationValue)\n if (programId == null) {\n return Response.json({ success: false, error: 'Referral program not found' }, { status: 404 })\n }\n\n const program = await payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n\n if (!program || !program.isActive) {\n return Response.json(\n { success: false, error: 'Referral program is not active' },\n { status: 400 },\n )\n }\n\n const cart = cartID\n ? await payload.findByID({\n collection: collections.cartsSlug,\n id: cartID,\n depth: 2,\n })\n : null\n\n // Use cartSubtotal (pre-discount baseline) for min-order enforcement and\n // commission calculations, consistent with applyCoupon and recalculateCart.\n const cartSubtotal = cart\n ? Number(resolvers.getCartSubtotal(cart)) || Number(resolvers.getCartTotal(cart)) || 0\n : 0\n const cartTotal = cartSubtotal\n\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && cartTotal < minOrderAmount) {\n return Response.json(\n {\n success: false,\n error: `Minimum order value of ${minOrderAmount} ${pluginConfig.defaultCurrency} required for this referral program`,\n },\n { status: 400 },\n )\n }\n\n const { partnerCommission, customerDiscount } = calculateCommissionAndDiscount({\n cartItems: cart ? resolvers.getCartItems(cart) : [],\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const cappedCustomerDiscount =\n cartTotal > 0 ? Math.min(customerDiscount, cartTotal) : customerDiscount\n const roundedPartnerCommission = roundTo2(partnerCommission)\n const roundedCustomerDiscount = roundTo2(cappedCustomerDiscount)\n\n return Response.json({\n success: true,\n referralCode: {\n code: referralData.code,\n description: `Get ${roundedCustomerDiscount.toFixed(2)} discount with this referral code`,\n },\n partnerCommission: roundedPartnerCommission,\n customerDiscount: roundedCustomerDiscount,\n currency: pluginConfig.defaultCurrency,\n })\n}\n\nexport const validateCouponEndpoint = ({ pluginConfig }: Args): Endpoint => ({\n path: pluginConfig.endpoints.validateCoupon,\n method: 'post',\n handler: validateCouponHandler({ pluginConfig }),\n})\n","import type { CollectionBeforeChangeHook } from 'payload'\nimport type { SanitizedCouponPluginOptions } from '../types'\nimport {\n calculateCommissionAndDiscount,\n calculateCouponDiscount,\n getProgramMinimumOrderAmount,\n} from '../utilities/calculateValues'\nimport { roundTo2 } from '../utilities/roundTo2'\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nfunction writeField(doc: Record<string, unknown>, field: string, value: unknown): void {\n doc[field] = value\n}\n\nfunction clearCouponFields(\n target: Record<string, unknown>,\n fields: SanitizedCouponPluginOptions['integration']['fields'],\n): void {\n writeField(target, fields.cartAppliedCouponField, null)\n writeField(target, fields.cartDiscountAmountField, 0)\n}\n\nfunction clearReferralFields(\n target: Record<string, unknown>,\n fields: SanitizedCouponPluginOptions['integration']['fields'],\n): void {\n writeField(target, fields.cartAppliedReferralCodeField, null)\n writeField(target, fields.cartPartnerCommissionField, 0)\n writeField(target, fields.cartCustomerDiscountField, 0)\n}\n\nexport const recalculateCartHook =\n (pluginConfig: SanitizedCouponPluginOptions): CollectionBeforeChangeHook =>\n async ({ data, req, originalDoc }) => {\n if (!req.payload) return data\n\n const integration = pluginConfig.integration || ({} as any)\n const collections = integration.collections || {\n cartsSlug: 'carts',\n ordersSlug: 'orders',\n productsSlug: 'products',\n usersSlug: 'users',\n categoriesSlug: 'categories',\n tagsSlug: 'tags',\n }\n const fields = integration.fields || {\n cartItemsField: 'items',\n cartSubtotalField: 'subtotal',\n cartTotalField: 'total',\n cartAppliedCouponField: 'appliedCoupon',\n cartAppliedReferralCodeField: 'appliedReferralCode',\n cartDiscountAmountField: 'discountAmount',\n cartCustomerDiscountField: 'customerDiscount',\n cartPartnerCommissionField: 'partnerCommission',\n orderAppliedCouponField: 'appliedCoupon',\n orderAppliedReferralCodeField: 'appliedReferralCode',\n orderDiscountAmountField: 'discountAmount',\n orderCustomerDiscountField: 'customerDiscount',\n orderPartnerCommissionField: 'partnerCommission',\n orderCustomerEmailField: 'customerEmail',\n orderPaymentStatusField: 'paymentStatus',\n orderCreatedAtField: 'createdAt',\n productPriceField: 'price',\n productCurrencyCodeField: 'currencyCode',\n }\n const resolvers = integration.resolvers || {\n getUserID: ({ user }: { user?: unknown }) => {\n if (!user || typeof user !== 'object') return null\n const id = (user as Record<string, unknown>).id\n if (typeof id === 'string' || typeof id === 'number') return id\n return null\n },\n getCartItems: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return []\n const value = (cart as Record<string, unknown>)[fields.cartItemsField]\n return Array.isArray(value) ? value : []\n },\n getCartSubtotal: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[fields.cartSubtotalField]\n return typeof value === 'number' ? value : 0\n },\n getCartTotal: (cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[fields.cartTotalField]\n return typeof value === 'number' ? value : 0\n },\n isOrderPaid: (_order: unknown) => false,\n getProductUnitPrice: ({ item, product, variant, currencyCode }: any) => {\n if (item && typeof item === 'object') {\n const itemPrice = (item as Record<string, unknown>).price\n if (typeof itemPrice === 'number') return itemPrice\n const unitPrice = (item as Record<string, unknown>).unitPrice\n if (typeof unitPrice === 'number') return unitPrice\n }\n\n const readPrice = (entity: unknown, code?: string) => {\n if (!entity || typeof entity !== 'object') return undefined\n const map = entity as Record<string, unknown>\n if (code && typeof code === 'string') {\n const key = `priceIn${code.toUpperCase()}`\n const value = map[key]\n if (typeof value === 'number') return value\n }\n const base = map.price\n return typeof base === 'number' ? base : undefined\n }\n\n return readPrice(variant, currencyCode) ?? readPrice(product, currencyCode) ?? 0\n },\n }\n\n const mutableData = (data || {}) as Record<string, unknown>\n const original = (originalDoc || {}) as Record<string, unknown>\n\n const effectiveItems =\n readField<any[]>(mutableData, fields.cartItemsField) ??\n readField<any[]>(original, fields.cartItemsField) ??\n []\n\n const effectiveAppliedReferral =\n readField<RelationValue>(mutableData, fields.cartAppliedReferralCodeField) !== undefined\n ? readField<RelationValue>(mutableData, fields.cartAppliedReferralCodeField)\n : readField<RelationValue>(original, fields.cartAppliedReferralCodeField)\n\n const effectiveAppliedCoupon =\n readField<RelationValue>(mutableData, fields.cartAppliedCouponField) !== undefined\n ? readField<RelationValue>(mutableData, fields.cartAppliedCouponField)\n : readField<RelationValue>(original, fields.cartAppliedCouponField)\n\n if (!Array.isArray(effectiveItems) || effectiveItems.length === 0) {\n clearReferralFields(mutableData, fields)\n clearCouponFields(mutableData, fields)\n writeField(mutableData, fields.cartTotalField, 0)\n return mutableData\n }\n\n const getRelationID = (value: unknown): string | number | null =>\n relationId(value as RelationValue)\n\n const productIds = effectiveItems\n .map((item: any) => getRelationID(item?.product))\n .filter((id: string | number | null): id is string | number => id != null)\n\n let productsMap = new Map<string, any>()\n if (productIds.length > 0) {\n const productsQuery = await req.payload.find({\n collection: collections.productsSlug,\n where: {\n id: { in: productIds },\n },\n limit: productIds.length,\n })\n productsMap = new Map((productsQuery?.docs || []).map((p: any) => [String(p.id), p]))\n }\n\n let calculatedSubtotal = 0\n const enrichedItems = effectiveItems.map((item: any) => {\n const pid = getRelationID(item?.product)\n const product = pid != null ? productsMap.get(String(pid)) || {} : {}\n const variant = typeof item?.variant === 'object' ? item.variant : undefined\n\n const unitPrice = Number(\n resolvers.getProductUnitPrice({\n item,\n product,\n variant,\n currencyCode: pluginConfig.defaultCurrency,\n }),\n )\n\n const quantity =\n typeof item?.quantity === 'number' && Number.isFinite(item.quantity) ? item.quantity : 1\n const safeUnitPrice = Number.isFinite(unitPrice) ? unitPrice : 0\n calculatedSubtotal += safeUnitPrice * quantity\n\n return {\n ...item,\n product,\n price: safeUnitPrice,\n quantity,\n }\n })\n\n writeField(mutableData, fields.cartSubtotalField, roundTo2(calculatedSubtotal))\n\n let customerDiscount = 0\n let couponDiscount = 0\n\n const appliedReferralID = relationId(effectiveAppliedReferral)\n if (pluginConfig.enableReferrals && appliedReferralID != null) {\n const referralQuery = await req.payload.find({\n collection: pluginConfig.collections.referralCodesSlug,\n where: { id: { equals: appliedReferralID } },\n limit: 1,\n depth: 1,\n })\n\n const referralCode = referralQuery?.docs?.[0]\n if (!referralCode || referralCode.isActive === false) {\n clearReferralFields(mutableData, fields)\n } else {\n const programId = relationId(referralCode.program as RelationValue)\n const program =\n typeof referralCode.program === 'object'\n ? referralCode.program\n : programId != null\n ? await req.payload.findByID({\n collection: pluginConfig.collections.referralProgramsSlug,\n id: programId,\n })\n : null\n\n if (!program || program.isActive === false) {\n clearReferralFields(mutableData, fields)\n } else {\n const minOrderAmount = getProgramMinimumOrderAmount({\n program,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n if (typeof minOrderAmount === 'number' && calculatedSubtotal < minOrderAmount) {\n clearReferralFields(mutableData, fields)\n } else {\n const result = calculateCommissionAndDiscount({\n cartItems: enrichedItems,\n program,\n currencyCode: pluginConfig.defaultCurrency,\n cartTotal: calculatedSubtotal,\n allowedTotalCommissionTypes: pluginConfig.referralConfig.allowedTotalCommissionTypes,\n })\n\n const roundedPartnerCommission = roundTo2(result.partnerCommission)\n const roundedCustomerDiscount = roundTo2(Math.max(0, result.customerDiscount))\n\n writeField(mutableData, fields.cartPartnerCommissionField, roundedPartnerCommission)\n writeField(mutableData, fields.cartCustomerDiscountField, roundedCustomerDiscount)\n customerDiscount = roundedCustomerDiscount\n }\n }\n }\n } else {\n if (readField(mutableData, fields.cartAppliedReferralCodeField) === null) {\n clearReferralFields(mutableData, fields)\n }\n }\n\n const appliedCouponID = relationId(effectiveAppliedCoupon)\n const canUseCouponWithReferral =\n !pluginConfig.enableReferrals ||\n pluginConfig.referralConfig.allowBothSystems ||\n relationId(readField(mutableData, fields.cartAppliedReferralCodeField) as RelationValue) ==\n null\n\n if (appliedCouponID != null && canUseCouponWithReferral) {\n const couponQuery = await req.payload.find({\n collection: pluginConfig.collections.couponsSlug,\n where: { id: { equals: appliedCouponID } },\n limit: 1,\n })\n\n const coupon = couponQuery?.docs?.[0]\n if (!coupon) {\n clearCouponFields(mutableData, fields)\n } else {\n const now = new Date()\n const activeFrom = coupon.activeFrom ? new Date(coupon.activeFrom) : null\n const activeUntil = coupon.activeUntil ? new Date(coupon.activeUntil) : null\n const isValidDate =\n (!activeFrom || now >= activeFrom) && (!activeUntil || now <= activeUntil)\n const underUsage =\n !coupon.usageLimit || Number(coupon.usageCount || 0) < Number(coupon.usageLimit || 0)\n\n if (!isValidDate || !underUsage) {\n clearCouponFields(mutableData, fields)\n } else {\n couponDiscount = roundTo2(\n calculateCouponDiscount({ coupon, cartTotal: calculatedSubtotal }),\n )\n writeField(mutableData, fields.cartDiscountAmountField, couponDiscount)\n }\n }\n } else {\n if (readField(mutableData, fields.cartAppliedCouponField) === null) {\n clearCouponFields(mutableData, fields)\n writeField(mutableData, fields.cartCustomerDiscountField, 0)\n writeField(mutableData, fields.cartPartnerCommissionField, 0)\n writeField(\n mutableData,\n fields.cartTotalField,\n roundTo2(Number(resolvers.getCartSubtotal(mutableData)) || calculatedSubtotal),\n )\n return mutableData\n }\n }\n\n const nextTotal = roundTo2(Math.max(0, calculatedSubtotal - customerDiscount - couponDiscount))\n writeField(mutableData, fields.cartTotalField, nextTotal)\n\n return mutableData\n }\n","import type { Payload } from 'payload'\n\nimport type { SanitizedCouponPluginOptions } from '../types'\n\ntype RelationValue = string | number | { id?: string | number } | null | undefined\n\nexport type OrderWithCouponFields = {\n id?: string | number\n [key: string]: unknown\n}\n\nexport type RecordCouponUsageResult = {\n recordedCoupon: boolean\n recordedReferral: boolean\n alreadyRecorded: boolean\n}\n\nconst USAGE_MARKER_FIELD = 'couponUsageRecordedAt'\n\nfunction relationId(value: RelationValue): string | number | null {\n if (value == null) return null\n if (typeof value === 'string' || typeof value === 'number') return value\n if (typeof value === 'object' && (typeof value.id === 'string' || typeof value.id === 'number')) {\n return value.id\n }\n return null\n}\n\nfunction asNumber(value: unknown): number {\n return typeof value === 'number' && Number.isFinite(value) ? value : 0\n}\n\nfunction readField<T = unknown>(doc: unknown, field: string): T | undefined {\n if (!doc || typeof doc !== 'object') return undefined\n return (doc as Record<string, unknown>)[field] as T | undefined\n}\n\nasync function markOrderUsageRecorded({\n payload,\n pluginConfig,\n orderID,\n}: {\n payload: Payload\n pluginConfig: SanitizedCouponPluginOptions\n orderID: string | number\n}): Promise<boolean> {\n const ordersSlug = pluginConfig.integration.collections.ordersSlug\n\n const latestOrder = await payload.findByID({\n collection: ordersSlug as any,\n id: orderID,\n depth: 0,\n })\n\n if (!latestOrder) return false\n\n const alreadyRecorded = Boolean(readField(latestOrder, USAGE_MARKER_FIELD))\n if (alreadyRecorded) return false\n\n await payload.update({\n collection: ordersSlug as any,\n id: orderID,\n data: {\n [USAGE_MARKER_FIELD]: new Date().toISOString(),\n },\n })\n\n return true\n}\n\n/**\n * Record coupon and referral usage when an order is placed successfully.\n * This function is idempotent and integration-field aware.\n *\n * Behavior:\n * - Uses configured order field names from `pluginConfig.integration.fields`\n * - Uses configured paid-order resolver (`integration.resolvers.isOrderPaid`)\n * - Marks the order with `couponUsageRecordedAt` before mutating counters to avoid duplicate counting\n * - If marker already exists, returns `alreadyRecorded: true` and performs no increments\n */\nexport async function recordCouponUsageForOrder(\n payload: Payload,\n order: OrderWithCouponFields,\n pluginConfig: SanitizedCouponPluginOptions,\n): Promise<RecordCouponUsageResult> {\n const result: RecordCouponUsageResult = {\n recordedCoupon: false,\n recordedReferral: false,\n alreadyRecorded: false,\n }\n\n const orderID = order.id\n if (orderID == null) return result\n\n const isPaid = await Promise.resolve(pluginConfig.integration.resolvers.isOrderPaid(order))\n if (!isPaid) return result\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canRecordOrderUsage({\n req: {},\n user: undefined,\n payload,\n order,\n }),\n )\n\n if (!policyAllowed) return result\n\n const acquiredMarker = await markOrderUsageRecorded({\n payload,\n pluginConfig,\n orderID,\n })\n\n if (!acquiredMarker) {\n result.alreadyRecorded = true\n return result\n }\n\n const fields = pluginConfig.integration.fields\n\n const couponField = fields.orderAppliedCouponField\n const referralField = fields.orderAppliedReferralCodeField\n const partnerCommissionField = fields.orderPartnerCommissionField\n\n const couponId = relationId(readField(order, couponField) as RelationValue)\n const referralCodeId = relationId(readField(order, referralField) as RelationValue)\n const commission = asNumber(readField(order, partnerCommissionField))\n\n if (couponId) {\n const coupon = await payload.findByID({\n collection: pluginConfig.collections.couponsSlug as any,\n id: couponId,\n depth: 0,\n })\n\n if (coupon) {\n const currentUsage = asNumber((coupon as Record<string, unknown>).usageCount)\n await payload.update({\n collection: pluginConfig.collections.couponsSlug as any,\n id: couponId,\n data: {\n usageCount: currentUsage + 1,\n },\n })\n result.recordedCoupon = true\n }\n }\n\n if (referralCodeId) {\n const referralCode = await payload.findByID({\n collection: pluginConfig.collections.referralCodesSlug as any,\n id: referralCodeId,\n depth: 0,\n })\n\n if (referralCode) {\n const rc = referralCode as Record<string, unknown>\n const currentTotal = asNumber(rc.totalEarnings)\n const currentPending = asNumber(rc.pendingEarnings)\n const currentUsage = asNumber(rc.usageCount)\n const currentSuccessful = asNumber(rc.successfulReferralsCount)\n\n await payload.update({\n collection: pluginConfig.collections.referralCodesSlug as any,\n id: referralCodeId,\n data: {\n usageCount: currentUsage + 1,\n successfulReferralsCount: currentSuccessful + 1,\n totalEarnings: currentTotal + commission,\n pendingEarnings: currentPending + commission,\n },\n })\n\n result.recordedReferral = true\n }\n }\n\n return result\n}\n","import type { CouponPluginOptions, SanitizedCouponPluginOptions } from '../types'\nimport { getCartItemUnitPrice } from './pricing'\nimport { isAdminUser, isPartnerUser } from './userRoles'\n\nconst toCleanStringArray = (value: unknown): string[] =>\n Array.isArray(value)\n ? value\n .filter((item): item is string => typeof item === 'string')\n .map((item) => item.trim())\n .filter(Boolean)\n : []\n\nconst toBoolean = (value: unknown, fallback: boolean): boolean => {\n if (typeof value === 'boolean') return value\n if (typeof value === 'number') {\n if (value === 1) return true\n if (value === 0) return false\n }\n if (typeof value === 'string') {\n const normalized = value.trim().toLowerCase()\n if (\n normalized === 'true' ||\n normalized === '1' ||\n normalized === 'yes' ||\n normalized === 'on'\n ) {\n return true\n }\n if (\n normalized === 'false' ||\n normalized === '0' ||\n normalized === 'no' ||\n normalized === 'off'\n ) {\n return false\n }\n }\n return fallback\n}\n\nexport const sanitizePluginConfig = ({\n pluginConfig,\n}: {\n pluginConfig: CouponPluginOptions\n}): SanitizedCouponPluginOptions => {\n const roleConfig = {\n roleFieldPaths:\n toCleanStringArray(pluginConfig?.roleConfig?.roleFieldPaths).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.roleFieldPaths)\n : ['role', 'roles'],\n adminRoleValues:\n toCleanStringArray(pluginConfig?.roleConfig?.adminRoleValues).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.adminRoleValues)\n : ['admin'],\n partnerRoleValues:\n toCleanStringArray(pluginConfig?.roleConfig?.partnerRoleValues).length > 0\n ? toCleanStringArray(pluginConfig?.roleConfig?.partnerRoleValues)\n : ['partner'],\n customRoleResolver:\n typeof pluginConfig?.roleConfig?.customRoleResolver === 'function'\n ? pluginConfig.roleConfig.customRoleResolver\n : undefined,\n }\n\n const normalizedAllowedTotalCommissionTypes = Array.isArray(\n pluginConfig?.referralConfig?.allowedTotalCommissionTypes,\n )\n ? [\n ...new Set(\n pluginConfig.referralConfig.allowedTotalCommissionTypes.filter(\n (value): value is 'fixed' | 'percentage' => value === 'fixed' || value === 'percentage',\n ),\n ),\n ]\n : []\n\n const integrationCollections = {\n cartsSlug:\n typeof pluginConfig?.integration?.collections?.cartsSlug === 'string' &&\n pluginConfig.integration.collections.cartsSlug.trim().length > 0\n ? pluginConfig.integration.collections.cartsSlug.trim()\n : 'carts',\n ordersSlug:\n typeof pluginConfig?.integration?.collections?.ordersSlug === 'string' &&\n pluginConfig.integration.collections.ordersSlug.trim().length > 0\n ? pluginConfig.integration.collections.ordersSlug.trim()\n : 'orders',\n productsSlug:\n typeof pluginConfig?.integration?.collections?.productsSlug === 'string' &&\n pluginConfig.integration.collections.productsSlug.trim().length > 0\n ? pluginConfig.integration.collections.productsSlug.trim()\n : 'products',\n usersSlug:\n typeof pluginConfig?.integration?.collections?.usersSlug === 'string' &&\n pluginConfig.integration.collections.usersSlug.trim().length > 0\n ? pluginConfig.integration.collections.usersSlug.trim()\n : 'users',\n categoriesSlug:\n typeof pluginConfig?.integration?.collections?.categoriesSlug === 'string' &&\n pluginConfig.integration.collections.categoriesSlug.trim().length > 0\n ? pluginConfig.integration.collections.categoriesSlug.trim()\n : 'categories',\n tagsSlug:\n typeof pluginConfig?.integration?.collections?.tagsSlug === 'string' &&\n pluginConfig.integration.collections.tagsSlug.trim().length > 0\n ? pluginConfig.integration.collections.tagsSlug.trim()\n : 'tags',\n }\n\n const integrationFields = {\n cartItemsField: pluginConfig?.integration?.fields?.cartItemsField?.trim() || 'items',\n cartSubtotalField: pluginConfig?.integration?.fields?.cartSubtotalField?.trim() || 'subtotal',\n cartTotalField: pluginConfig?.integration?.fields?.cartTotalField?.trim() || 'total',\n cartAppliedCouponField:\n pluginConfig?.integration?.fields?.cartAppliedCouponField?.trim() || 'appliedCoupon',\n cartAppliedReferralCodeField:\n pluginConfig?.integration?.fields?.cartAppliedReferralCodeField?.trim() ||\n 'appliedReferralCode',\n cartDiscountAmountField:\n pluginConfig?.integration?.fields?.cartDiscountAmountField?.trim() || 'discountAmount',\n cartCustomerDiscountField:\n pluginConfig?.integration?.fields?.cartCustomerDiscountField?.trim() || 'customerDiscount',\n cartPartnerCommissionField:\n pluginConfig?.integration?.fields?.cartPartnerCommissionField?.trim() || 'partnerCommission',\n orderAppliedCouponField:\n pluginConfig?.integration?.fields?.orderAppliedCouponField?.trim() || 'appliedCoupon',\n orderAppliedReferralCodeField:\n pluginConfig?.integration?.fields?.orderAppliedReferralCodeField?.trim() ||\n 'appliedReferralCode',\n orderDiscountAmountField:\n pluginConfig?.integration?.fields?.orderDiscountAmountField?.trim() || 'discountAmount',\n orderCustomerDiscountField:\n pluginConfig?.integration?.fields?.orderCustomerDiscountField?.trim() || 'customerDiscount',\n orderPartnerCommissionField:\n pluginConfig?.integration?.fields?.orderPartnerCommissionField?.trim() || 'partnerCommission',\n orderCustomerEmailField:\n pluginConfig?.integration?.fields?.orderCustomerEmailField?.trim() ||\n pluginConfig?.orderIntegration?.orderCustomerEmailField?.trim() ||\n 'customerEmail',\n orderPaymentStatusField:\n pluginConfig?.integration?.fields?.orderPaymentStatusField?.trim() ||\n pluginConfig?.orderIntegration?.orderPaymentStatusField?.trim() ||\n 'paymentStatus',\n orderCreatedAtField:\n pluginConfig?.integration?.fields?.orderCreatedAtField?.trim() || 'createdAt',\n productPriceField: pluginConfig?.integration?.fields?.productPriceField?.trim() || 'price',\n productCurrencyCodeField:\n pluginConfig?.integration?.fields?.productCurrencyCodeField?.trim() || 'currencyCode',\n }\n\n const integrationResolvers = {\n getUserID:\n pluginConfig?.integration?.resolvers?.getUserID ||\n (({ user }: { user?: unknown }) => {\n if (!user || typeof user !== 'object') return null\n const id = (user as Record<string, unknown>).id\n if (typeof id === 'string' || typeof id === 'number') return id\n return null\n }),\n getCartItems:\n pluginConfig?.integration?.resolvers?.getCartItems ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return []\n const value = (cart as Record<string, unknown>)[integrationFields.cartItemsField]\n return Array.isArray(value) ? value : []\n }),\n getCartSubtotal:\n pluginConfig?.integration?.resolvers?.getCartSubtotal ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[integrationFields.cartSubtotalField]\n return typeof value === 'number' ? value : 0\n }),\n getCartTotal:\n pluginConfig?.integration?.resolvers?.getCartTotal ||\n ((cart: unknown) => {\n if (!cart || typeof cart !== 'object') return 0\n const value = (cart as Record<string, unknown>)[integrationFields.cartTotalField]\n return typeof value === 'number' ? value : 0\n }),\n isOrderPaid:\n pluginConfig?.integration?.resolvers?.isOrderPaid ||\n ((order: unknown) => {\n if (!order || typeof order !== 'object') return false\n const status = (order as Record<string, unknown>)[integrationFields.orderPaymentStatusField]\n return status === (pluginConfig?.orderIntegration?.orderPaidStatusValue ?? 'paid')\n }),\n getProductUnitPrice:\n pluginConfig?.integration?.resolvers?.getProductUnitPrice ||\n ((args: { item: unknown; product: unknown; variant?: unknown; currencyCode?: string }) =>\n getCartItemUnitPrice({\n item: (args.item as any) ?? null,\n product: (args.product as any) ?? null,\n variant: (args.variant as any) ?? null,\n currencyCode: args.currencyCode || 'USD',\n })),\n }\n\n return {\n enabled: toBoolean(pluginConfig?.enabled, true),\n enableReferrals: toBoolean(pluginConfig?.enableReferrals, false),\n allowStackWithOtherCoupons: toBoolean(pluginConfig?.allowStackWithOtherCoupons, false),\n defaultCurrency:\n typeof pluginConfig?.defaultCurrency === 'string' &&\n pluginConfig.defaultCurrency.length > 0 &&\n pluginConfig.defaultCurrency.length <= 3\n ? pluginConfig.defaultCurrency\n : 'USD',\n collections: {\n couponsSlug:\n typeof pluginConfig?.collections?.couponsSlug === 'string' &&\n pluginConfig.collections.couponsSlug.trim().length > 0 &&\n pluginConfig.collections.couponsSlug.length <= 100\n ? pluginConfig.collections.couponsSlug\n : 'coupons',\n referralProgramsSlug:\n typeof pluginConfig?.collections?.referralProgramsSlug === 'string' &&\n pluginConfig.collections.referralProgramsSlug.trim().length > 0 &&\n pluginConfig.collections.referralProgramsSlug.length <= 100\n ? pluginConfig.collections.referralProgramsSlug\n : 'referral-programs',\n referralCodesSlug:\n typeof pluginConfig?.collections?.referralCodesSlug === 'string' &&\n pluginConfig.collections.referralCodesSlug.trim().length > 0 &&\n pluginConfig.collections.referralCodesSlug.length <= 100\n ? pluginConfig.collections.referralCodesSlug\n : 'referral-codes',\n referralPartnersSlug:\n typeof pluginConfig?.collections?.referralPartnersSlug === 'string' &&\n pluginConfig.collections.referralPartnersSlug.trim().length > 0 &&\n pluginConfig.collections.referralPartnersSlug.length <= 100\n ? pluginConfig.collections.referralPartnersSlug\n : 'referral-partners',\n },\n endpoints: {\n applyCoupon:\n typeof pluginConfig?.endpoints?.applyCoupon === 'string' &&\n pluginConfig.endpoints.applyCoupon.trim().length > 0\n ? pluginConfig.endpoints.applyCoupon.trim()\n : '/coupons/apply',\n validateCoupon:\n typeof pluginConfig?.endpoints?.validateCoupon === 'string' &&\n pluginConfig.endpoints.validateCoupon.trim().length > 0\n ? pluginConfig.endpoints.validateCoupon.trim()\n : '/coupons/validate',\n partnerStats:\n typeof pluginConfig?.endpoints?.partnerStats === 'string' &&\n pluginConfig.endpoints.partnerStats.trim().length > 0\n ? pluginConfig.endpoints.partnerStats.trim()\n : '/referrals/partner-stats',\n recordOrderUsage:\n typeof pluginConfig?.endpoints?.recordOrderUsage === 'string' &&\n pluginConfig.endpoints.recordOrderUsage.trim().length > 0\n ? pluginConfig.endpoints.recordOrderUsage.trim()\n : '/coupons/record-order-usage',\n },\n autoIntegrate: pluginConfig?.autoIntegrate !== false,\n access: {\n canUseCoupons:\n typeof pluginConfig?.access?.canUseCoupons === 'function'\n ? pluginConfig.access.canUseCoupons\n : () => true,\n canUseReferrals:\n typeof pluginConfig?.access?.canUseReferrals === 'function'\n ? pluginConfig.access.canUseReferrals\n : () => false,\n isAdmin:\n typeof pluginConfig?.access?.isAdmin === 'function'\n ? pluginConfig.access.isAdmin\n : ({ req }) => isAdminUser({ user: req?.user, roleConfig }),\n isPartner:\n typeof pluginConfig?.access?.isPartner === 'function'\n ? pluginConfig.access.isPartner\n : ({ req }) => isPartnerUser({ user: req?.user, roleConfig }),\n },\n policies: {\n canApplyCoupon:\n typeof pluginConfig?.policies?.canApplyCoupon === 'function'\n ? pluginConfig.policies.canApplyCoupon\n : () => true,\n canApplyReferral:\n typeof pluginConfig?.policies?.canApplyReferral === 'function'\n ? pluginConfig.policies.canApplyReferral\n : () => true,\n canViewPartnerStats:\n typeof pluginConfig?.policies?.canViewPartnerStats === 'function'\n ? pluginConfig.policies.canViewPartnerStats\n : ({ req }) =>\n isPartnerUser({ user: (req as any)?.user, roleConfig }) ||\n isAdminUser({ user: (req as any)?.user, roleConfig }),\n canRecordOrderUsage:\n typeof pluginConfig?.policies?.canRecordOrderUsage === 'function'\n ? pluginConfig.policies.canRecordOrderUsage\n : () => true,\n },\n integration: {\n collections: integrationCollections,\n fields: integrationFields,\n resolvers: integrationResolvers,\n },\n referralConfig: {\n allowBothSystems: pluginConfig?.referralConfig?.allowBothSystems ?? false,\n singleCodePerCart: pluginConfig?.referralConfig?.singleCodePerCart ?? true,\n defaultPartnerSplit: pluginConfig?.referralConfig?.defaultPartnerSplit ?? 70,\n defaultCustomerSplit: pluginConfig?.referralConfig?.defaultCustomerSplit ?? 30,\n allowedTotalCommissionTypes:\n normalizedAllowedTotalCommissionTypes.length > 0\n ? normalizedAllowedTotalCommissionTypes\n : ['fixed', 'percentage'],\n },\n adminGroups: {\n couponsGroup: pluginConfig?.adminGroups?.couponsGroup ?? 'Coupons',\n referralsGroup: pluginConfig?.adminGroups?.referralsGroup ?? 'Referrals',\n },\n partnerDashboard: {\n enabled: pluginConfig?.partnerDashboard?.enabled ?? true,\n showEarningsSummary: pluginConfig?.partnerDashboard?.showEarningsSummary ?? true,\n showReferralPerformance: pluginConfig?.partnerDashboard?.showReferralPerformance ?? true,\n showRecentReferrals: pluginConfig?.partnerDashboard?.showRecentReferrals ?? true,\n showCommissionBreakdown: pluginConfig?.partnerDashboard?.showCommissionBreakdown ?? true,\n },\n orderIntegration: {\n ordersSlug:\n typeof pluginConfig?.orderIntegration?.ordersSlug === 'string' &&\n pluginConfig.orderIntegration.ordersSlug.trim().length > 0\n ? pluginConfig.orderIntegration.ordersSlug\n : integrationCollections.ordersSlug,\n orderCustomerEmailField: integrationFields.orderCustomerEmailField,\n orderPaymentStatusField: integrationFields.orderPaymentStatusField,\n orderPaidStatusValue:\n typeof pluginConfig?.orderIntegration?.orderPaidStatusValue === 'string'\n ? pluginConfig.orderIntegration.orderPaidStatusValue\n : 'paid',\n },\n roleConfig,\n }\n}\n","import type { Config } from 'payload'\n\nimport { createCouponsCollection } from './collections/createCouponsCollection'\nimport { createReferralCodesCollection } from './collections/createReferralCodesCollection'\nimport { createReferralProgramsCollection } from './collections/createReferralProgramsCollection'\nimport { applyCouponEndpoint } from './endpoints/applyCoupon'\nimport { partnerStatsEndpoint } from './endpoints/partnerStats'\nimport { validateCouponEndpoint } from './endpoints/validateCoupon'\nimport { recalculateCartHook } from './hooks/recalculateCart'\nimport { CouponPluginOptions } from './types'\nimport { recordCouponUsageForOrder } from './utilities/recordCouponUsageForOrder'\nimport { sanitizePluginConfig } from './utilities/sanitizePluginConfig'\n\nconst RECALCULATE_HOOK_KEY = '__payloadEcommerceCouponRecalculateHook__'\n\ntype GenericCollection = {\n slug: string\n fields?: Array<Record<string, unknown>>\n endpoints?: Array<Record<string, unknown>>\n hooks?: {\n beforeChange?: Array<unknown>\n [key: string]: unknown\n }\n [key: string]: unknown\n}\n\nconst asArray = <T>(value: T[] | null | undefined): T[] => (Array.isArray(value) ? value : [])\n\nconst hasNamedField = (collection: GenericCollection, fieldName: string): boolean =>\n asArray(collection.fields).some((f) => f?.name === fieldName)\n\nconst normalizePath = (path: string): string => {\n if (!path) return '/'\n const withLeadingSlash = path.startsWith('/') ? path : `/${path}`\n return withLeadingSlash.replace(/\\/+$/, '') || '/'\n}\n\nconst toCollectionEndpointPath = ({\n endpointPath,\n collectionSlug,\n}: {\n endpointPath: string\n collectionSlug: string\n}): string | null => {\n const normalizedEndpointPath = normalizePath(endpointPath)\n const normalizedCollectionPath = normalizePath(`/${collectionSlug}`)\n\n if (!normalizedEndpointPath.startsWith(`${normalizedCollectionPath}/`)) return null\n const relative = normalizedEndpointPath.slice(normalizedCollectionPath.length)\n return relative.startsWith('/') ? relative : `/${relative}`\n}\n\nconst ensureCouponCollectionEndpoints = ({\n collection,\n pluginConfig,\n}: {\n collection: GenericCollection\n pluginConfig: ReturnType<typeof sanitizePluginConfig>\n}): GenericCollection => {\n const couponsSlug = pluginConfig.collections.couponsSlug\n const applyPath = toCollectionEndpointPath({\n endpointPath: pluginConfig.endpoints.applyCoupon,\n collectionSlug: couponsSlug,\n })\n const validatePath = toCollectionEndpointPath({\n endpointPath: pluginConfig.endpoints.validateCoupon,\n collectionSlug: couponsSlug,\n })\n\n if (!applyPath && !validatePath) return collection\n\n const endpoints = asArray(collection.endpoints)\n const endpointKeys = new Set(\n endpoints.map((e: any) => `${(e?.method || 'get').toLowerCase()}:${e?.path || ''}`),\n )\n\n if (validatePath) {\n const validateKey = `post:${validatePath}`\n if (!endpointKeys.has(validateKey)) {\n endpointKeys.add(validateKey)\n endpoints.push({\n path: validatePath,\n method: 'post',\n handler: validateCouponEndpoint({ pluginConfig }).handler,\n })\n }\n }\n\n if (applyPath) {\n const applyKey = `post:${applyPath}`\n if (!endpointKeys.has(applyKey)) {\n endpointKeys.add(applyKey)\n endpoints.push({\n path: applyPath,\n method: 'post',\n handler: applyCouponEndpoint({ pluginConfig }).handler,\n })\n }\n }\n\n collection.endpoints = endpoints\n return collection\n}\n\nconst addFieldsToCollection = (\n config: Config,\n targetSlug: string,\n newFields: Array<Record<string, unknown>>,\n): void => {\n const collections = asArray(config.collections as GenericCollection[])\n const idx = collections.findIndex((c) => c.slug === targetSlug)\n if (idx === -1) return\n\n const collection = collections[idx]\n collection.fields = asArray(collection.fields)\n\n for (const field of newFields) {\n const name = typeof field.name === 'string' ? field.name : ''\n if (!name) continue\n if (!hasNamedField(collection, name)) {\n collection.fields.push(field)\n }\n }\n\n collections[idx] = collection\n config.collections = collections as any\n}\n\ntype RecalculateHookFn = (...args: any[]) => any\n\nconst markHook = <T extends RecalculateHookFn>(fn: T): T => {\n ;(fn as any)[RECALCULATE_HOOK_KEY] = true\n return fn\n}\n\nconst hasMarkedHook = (hook: unknown): boolean =>\n Boolean(hook && typeof hook === 'function' && (hook as any)[RECALCULATE_HOOK_KEY])\n\nconst createRecordOrderUsageEndpoint = ({\n pluginConfig,\n}: {\n pluginConfig: ReturnType<typeof sanitizePluginConfig>\n}) => ({\n path: pluginConfig.endpoints.recordOrderUsage,\n method: 'post' as const,\n handler: async (req: any) => {\n try {\n const payload = req?.payload\n const orderId = req?.data?.orderId ?? req?.json?.orderId\n\n if (!payload) {\n return Response.json(\n { success: false, error: 'Payload instance is required' },\n { status: 500 },\n )\n }\n\n if (!orderId) {\n return Response.json({ success: false, error: 'orderId is required' }, { status: 400 })\n }\n\n const policyAllowed = await Promise.resolve(\n pluginConfig.policies.canRecordOrderUsage({\n req,\n user: req?.user,\n payload,\n order: { id: orderId },\n }),\n )\n\n if (!policyAllowed) {\n return Response.json({ success: false, error: 'Forbidden' }, { status: 403 })\n }\n\n const order = await payload.findByID({\n collection: pluginConfig.integration.collections.ordersSlug as any,\n id: orderId,\n })\n\n if (!order) {\n return Response.json({ success: false, error: 'Order not found' }, { status: 404 })\n }\n\n const result = await recordCouponUsageForOrder(payload, order, pluginConfig)\n\n return Response.json({\n success: true,\n result,\n })\n } catch (error) {\n console.error('record-order-usage endpoint error:', error)\n return Response.json({ success: false, error: 'Internal server error' }, { status: 500 })\n }\n },\n})\n\nexport const payloadEcommerceCouponPlugin =\n (pluginOptions: CouponPluginOptions = {}) =>\n async (incomingConfig: Config): Promise<Config> => {\n const pluginConfig = sanitizePluginConfig({ pluginConfig: pluginOptions })\n\n if (!pluginConfig.enabled) return incomingConfig || {}\n\n if (!incomingConfig) {\n incomingConfig = { collections: [], endpoints: [] } as any\n }\n\n incomingConfig.collections = asArray(incomingConfig.collections as any) as Config['collections']\n incomingConfig.endpoints = asArray(incomingConfig.endpoints as any) as Config['endpoints']\n\n const collectionsToAdd: GenericCollection[] = []\n\n if (pluginConfig.enableReferrals) {\n let referralProgramsCollection = createReferralProgramsCollection(\n pluginConfig,\n ) as GenericCollection\n let referralCodesCollection = createReferralCodesCollection(pluginConfig) as GenericCollection\n\n if (pluginOptions.collections?.referralProgramsCollectionOverride) {\n referralProgramsCollection =\n await pluginOptions.collections.referralProgramsCollectionOverride({\n defaultCollection: referralProgramsCollection,\n })\n }\n\n if (pluginOptions.collections?.referralCodesCollectionOverride) {\n referralCodesCollection = await pluginOptions.collections.referralCodesCollectionOverride({\n defaultCollection: referralCodesCollection,\n })\n }\n\n collectionsToAdd.push(referralProgramsCollection, referralCodesCollection)\n\n if (pluginConfig.referralConfig.allowBothSystems) {\n let couponsCollection = createCouponsCollection(pluginConfig) as GenericCollection\n if (pluginOptions.collections?.couponsCollectionOverride) {\n couponsCollection = await pluginOptions.collections.couponsCollectionOverride({\n defaultCollection: couponsCollection,\n })\n }\n couponsCollection = ensureCouponCollectionEndpoints({\n collection: couponsCollection,\n pluginConfig,\n })\n collectionsToAdd.push(couponsCollection)\n }\n } else {\n let couponsCollection = createCouponsCollection(pluginConfig) as GenericCollection\n if (pluginOptions.collections?.couponsCollectionOverride) {\n couponsCollection = await pluginOptions.collections.couponsCollectionOverride({\n defaultCollection: couponsCollection,\n })\n }\n couponsCollection = ensureCouponCollectionEndpoints({\n collection: couponsCollection,\n pluginConfig,\n })\n collectionsToAdd.push(couponsCollection)\n }\n\n const existingSlugs = new Set(\n asArray(incomingConfig.collections as GenericCollection[]).map((c) => c.slug),\n )\n const toAppend = collectionsToAdd.filter((c) => !existingSlugs.has(c.slug))\n incomingConfig.collections = [\n ...asArray(incomingConfig.collections as GenericCollection[]),\n ...toAppend,\n ] as any\n\n const endpointPaths = new Set(\n asArray(incomingConfig.endpoints as any[]).map(\n (e: any) => `${e?.method || 'get'}:${e?.path || ''}`,\n ),\n )\n\n const maybePushEndpoint = (endpoint: any) => {\n const key = `${endpoint?.method || 'get'}:${endpoint?.path || ''}`\n if (!endpointPaths.has(key)) {\n endpointPaths.add(key)\n ;(incomingConfig.endpoints as any[]).push(endpoint)\n }\n }\n\n maybePushEndpoint(validateCouponEndpoint({ pluginConfig }))\n maybePushEndpoint(applyCouponEndpoint({ pluginConfig }))\n\n if (pluginOptions.endpoints?.recordOrderUsage) {\n maybePushEndpoint(createRecordOrderUsageEndpoint({ pluginConfig }))\n }\n\n if (pluginConfig.enableReferrals) {\n maybePushEndpoint(partnerStatsEndpoint({ pluginConfig }))\n }\n\n if (pluginConfig.autoIntegrate) {\n const allSlugs = new Set(\n asArray(incomingConfig.collections as GenericCollection[]).map((c) => c.slug),\n )\n\n const cartsSlug = pluginConfig.integration.collections.cartsSlug\n const ordersSlug = pluginConfig.integration.collections.ordersSlug\n\n const {\n cartAppliedReferralCodeField,\n cartPartnerCommissionField,\n cartCustomerDiscountField,\n cartAppliedCouponField,\n cartDiscountAmountField,\n orderAppliedReferralCodeField,\n orderPartnerCommissionField,\n orderCustomerDiscountField,\n orderAppliedCouponField,\n orderDiscountAmountField,\n } = pluginConfig.integration.fields\n\n if (\n pluginConfig.enableReferrals &&\n allSlugs.has(pluginConfig.collections.referralCodesSlug)\n ) {\n const cartReferralFields: Array<Record<string, unknown>> = [\n {\n name: cartAppliedReferralCodeField,\n type: 'relationship',\n relationTo: pluginConfig.collections.referralCodesSlug,\n admin: { description: 'Referral code applied to this cart' },\n },\n {\n name: cartPartnerCommissionField,\n type: 'number',\n admin: { description: 'Partner commission amount for this cart' },\n },\n {\n name: cartCustomerDiscountField,\n type: 'number',\n admin: { description: 'Customer discount amount for this cart' },\n },\n ]\n\n if (\n pluginConfig.referralConfig.allowBothSystems &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n cartReferralFields.push(\n {\n name: cartAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: { description: 'Coupon applied to this cart' },\n },\n {\n name: cartDiscountAmountField,\n type: 'number',\n admin: { description: 'Discount amount from coupon' },\n },\n )\n }\n\n addFieldsToCollection(incomingConfig, cartsSlug, cartReferralFields)\n\n const orderReferralFields: Array<Record<string, unknown>> = [\n {\n name: orderAppliedReferralCodeField,\n type: 'relationship',\n relationTo: pluginConfig.collections.referralCodesSlug,\n admin: {\n description: 'Referral code applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderPartnerCommissionField,\n type: 'number',\n admin: {\n description: 'Partner commission amount for this order',\n readOnly: true,\n },\n },\n {\n name: orderCustomerDiscountField,\n type: 'number',\n admin: {\n description: 'Customer discount amount for this order',\n readOnly: true,\n },\n },\n ]\n\n if (\n pluginConfig.referralConfig.allowBothSystems &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n orderReferralFields.push(\n {\n name: orderAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: {\n description: 'Coupon applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderDiscountAmountField,\n type: 'number',\n admin: {\n description: 'Discount amount from coupon',\n readOnly: true,\n },\n },\n )\n }\n\n addFieldsToCollection(incomingConfig, ordersSlug, orderReferralFields)\n } else if (\n !pluginConfig.enableReferrals &&\n allSlugs.has(pluginConfig.collections.couponsSlug)\n ) {\n const cartCouponFields: Array<Record<string, unknown>> = [\n {\n name: cartAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: { description: 'Coupon applied to this cart' },\n },\n {\n name: cartDiscountAmountField,\n type: 'number',\n admin: { description: 'Discount amount from coupon' },\n },\n ]\n\n const orderCouponFields: Array<Record<string, unknown>> = [\n {\n name: orderAppliedCouponField,\n type: 'relationship',\n relationTo: pluginConfig.collections.couponsSlug,\n admin: {\n description: 'Coupon applied to this order',\n readOnly: true,\n },\n },\n {\n name: orderDiscountAmountField,\n type: 'number',\n admin: {\n description: 'Discount amount from coupon',\n readOnly: true,\n },\n },\n ]\n\n addFieldsToCollection(incomingConfig, cartsSlug, cartCouponFields)\n addFieldsToCollection(incomingConfig, ordersSlug, orderCouponFields)\n }\n }\n\n const cartsSlug = pluginConfig.integration.collections.cartsSlug\n const cartIndex = asArray(incomingConfig.collections as GenericCollection[]).findIndex(\n (c) => c.slug === cartsSlug,\n )\n\n if (cartIndex > -1) {\n const collection = (incomingConfig.collections as GenericCollection[])[cartIndex]\n const beforeChangeHooks = asArray(collection.hooks?.beforeChange)\n\n const alreadyAdded = beforeChangeHooks.some((h) => hasMarkedHook(h))\n if (!alreadyAdded) {\n const hook = markHook(recalculateCartHook(pluginConfig))\n collection.hooks = {\n ...(collection.hooks || {}),\n beforeChange: [...beforeChangeHooks, hook],\n }\n }\n\n ;(incomingConfig.collections as GenericCollection[])[cartIndex] = collection\n }\n\n return incomingConfig\n }\n","import type { ApplyCouponHook, ApplyCouponResponse, PartnerDashboardData } from '../types'\n\ntype EndpointInput =\n | string\n | {\n applyCoupon?: string\n validateCoupon?: string\n partnerStats?: string\n baseURL?: string\n }\n\nconst DEFAULT_ENDPOINTS = {\n applyCoupon: '/api/coupons/apply',\n validateCoupon: '/api/coupons/validate',\n partnerStats: '/api/referrals/partner-stats',\n} as const\n\nfunction normalizePath(path: string): string {\n if (!path) return ''\n return path.startsWith('/') ? path : `/${path}`\n}\n\nfunction withBaseURL(path: string, baseURL?: string): string {\n if (!baseURL) return path\n const trimmedBase = baseURL.endsWith('/') ? baseURL.slice(0, -1) : baseURL\n const normalizedPath = normalizePath(path)\n return `${trimmedBase}${normalizedPath}`\n}\n\nfunction resolveEndpoints(input?: EndpointInput) {\n if (typeof input === 'string') {\n return {\n ...DEFAULT_ENDPOINTS,\n partnerStats: input,\n }\n }\n\n const baseURL = input?.baseURL\n return {\n applyCoupon: withBaseURL(input?.applyCoupon || DEFAULT_ENDPOINTS.applyCoupon, baseURL),\n validateCoupon: withBaseURL(input?.validateCoupon || DEFAULT_ENDPOINTS.validateCoupon, baseURL),\n partnerStats: withBaseURL(input?.partnerStats || DEFAULT_ENDPOINTS.partnerStats, baseURL),\n }\n}\n\n/**\n * Apply a coupon/referral code to a cart\n * @param options - Code, cart ID, and optional customerEmail\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function useCouponCode(\n options: ApplyCouponHook,\n endpointConfig?: EndpointInput,\n): Promise<ApplyCouponResponse> {\n const { code, cartID, customerEmail } = options\n\n if (!code) {\n return {\n success: false,\n message: 'Coupon code is required',\n error: 'Code is missing',\n }\n }\n\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.applyCoupon, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ code, cartID, customerEmail }),\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n message: (data.error as string) || 'Failed to apply coupon',\n error: data.error as string,\n }\n }\n\n const couponData = data.coupon as Record<string, unknown> | undefined\n const referralData = data.referralCode as Record<string, unknown> | undefined\n\n return {\n success: Boolean(data.success),\n message: (data.message as string) || 'Code applied',\n discount: (data.discount as number) || (data.customerDiscount as number),\n partnerCommission: data.partnerCommission as number,\n customerDiscount: data.customerDiscount as number,\n\n coupon: couponData\n ? {\n code: (couponData.code as string) || '',\n type: (couponData.type as 'percentage' | 'fixed') || 'percentage',\n value: (couponData.value as number) || 0,\n }\n : undefined,\n referralCode: referralData\n ? {\n code: (referralData.code as string) || '',\n }\n : undefined,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, message, error: message }\n }\n}\n\n/**\n * Validate a coupon/referral code without applying it\n * @param code - Code to validate\n * @param cartValue - Optional cart value\n * @param cartID - Optional cart ID\n * @param customerEmail - Optional customer email (for per-customer limits)\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function validateCouponCode(\n code: string,\n cartValue?: number,\n cartID?: string,\n customerEmail?: string,\n endpointConfig?: EndpointInput,\n): Promise<ApplyCouponResponse> {\n if (!code) {\n return {\n success: false,\n message: 'Code required',\n error: 'Code missing',\n }\n }\n\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.validateCoupon, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ code, cartValue, cartID, customerEmail }),\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n message: (data.error as string) || 'Invalid coupon',\n error: data.error as string,\n }\n }\n\n const couponData = data.coupon as Record<string, unknown> | undefined\n const referralData = data.referralCode as Record<string, unknown> | undefined\n\n return {\n success: Boolean(data.success),\n message: (data.message as string) || 'Code is valid',\n coupon: couponData\n ? {\n code: (couponData.code as string) || '',\n type: (couponData.type as 'percentage' | 'fixed') || 'percentage',\n value: (couponData.value as number) || 0,\n }\n : undefined,\n referralCode: referralData\n ? {\n code: (referralData.code as string) || '',\n }\n : undefined,\n discount: data.discount as number,\n partnerCommission: data.partnerCommission as number,\n customerDiscount: data.customerDiscount as number,\n currency: data.currency as string,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, message, error: message }\n }\n}\n\nexport type PartnerStatsResponse = {\n success: boolean\n data?: PartnerDashboardData\n currency?: string\n error?: string\n}\n\n/**\n * Fetch partner dashboard statistics\n * @param endpointConfig - Optional endpoint override config\n */\nexport async function usePartnerStats(\n endpointConfig?: EndpointInput,\n): Promise<PartnerStatsResponse> {\n const endpoints = resolveEndpoints(endpointConfig)\n\n try {\n const response = await fetch(endpoints.partnerStats, {\n method: 'GET',\n headers: { 'Content-Type': 'application/json' },\n credentials: 'include',\n })\n\n const data = (await response.json()) as Record<string, unknown>\n\n if (!response.ok) {\n return {\n success: false,\n error: (data.error as string) || 'Failed to fetch partner stats',\n }\n }\n\n return {\n success: Boolean(data.success),\n data: data.data as PartnerDashboardData,\n currency: data.currency as string,\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : 'Network error'\n return { success: false, error: message }\n }\n}\n","/**\n * @license React\n * react.production.js\n *\n * Copyright (c) Meta Platforms, Inc. and affiliates.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n */\n\n\"use strict\";\nvar REACT_ELEMENT_TYPE = Symbol.for(\"react.transitional.element\"),\n REACT_PORTAL_TYPE = Symbol.for(\"react.portal\"),\n REACT_FRAGMENT_TYPE = Symbol.for(\"react.fragment\"),\n REACT_STRICT_MODE_TYPE = Symbol.for(\"react.strict_mode\"),\n REACT_PROFILER_TYPE = Symbol.for(\"react.profiler\"),\n REACT_CONSUMER_TYPE = Symbol.for(\"react.consumer\"),\n REACT_CONTEXT_TYPE = Symbol.for(\"react.context\"),\n REACT_FORWARD_REF_TYPE = Symbol.for(\"react.forward_ref\"),\n REACT_SUSPENSE_TYPE = Symbol.for(\"react.suspense\"),\n REACT_MEMO_TYPE = Symbol.for(\"react.memo\"),\n REACT_LAZY_TYPE = Symbol.for(\"react.lazy\"),\n REACT_ACTIVITY_TYPE = Symbol.for(\"react.activity\"),\n MAYBE_ITERATOR_SYMBOL = Symbol.iterator;\nfunction getIteratorFn(maybeIterable) {\n if (null === maybeIterable || \"object\" !== typeof maybeIterable) return null;\n maybeIterable =\n (MAYBE_ITERATOR_SYMBOL && maybeIterable[MAYBE_ITERATOR_SYMBOL]) ||\n maybeIterable[\"@@iterator\"];\n return \"function\" === typeof maybeIterable ? maybeIterable : null;\n}\nvar ReactNoopUpdateQueue = {\n isMounted: function () {\n return !1;\n },\n enqueueForceUpdate: function () {},\n enqueueReplaceState: function () {},\n enqueueSetState: function () {}\n },\n assign = Object.assign,\n emptyObject = {};\nfunction Component(props, context, updater) {\n this.props = props;\n this.context = context;\n this.refs = emptyObject;\n this.updater = updater || ReactNoopUpdateQueue;\n}\nComponent.prototype.isReactComponent = {};\nComponent.prototype.setState = function (partialState, callback) {\n if (\n \"object\" !== typeof partialState &&\n \"function\" !== typeof partialState &&\n null != partialState\n )\n throw Error(\n \"takes an object of state variables to update or a function which returns an object of state variables.\"\n );\n this.updater.enqueueSetState(this, partialState, callback, \"setState\");\n};\nComponent.prototype.forceUpdate = function (callback) {\n this.updater.enqueueForceUpdate(this, callback, \"forceUpdate\");\n};\nfunction ComponentDummy() {}\nComponentDummy.prototype = Component.prototype;\nfunction PureComponent(props, context, updater) {\n this.props = props;\n this.context = context;\n this.refs = emptyObject;\n this.updater = updater || ReactNoopUpdateQueue;\n}\nvar pureComponentPrototype = (PureComponent.prototype = new ComponentDummy());\npureComponentPrototype.constructor = PureComponent;\nassign(pureComponentPrototype, Component.prototype);\npureComponentPrototype.isPureReactComponent = !0;\nvar isArrayImpl = Array.isArray;\nfunction noop() {}\nvar ReactSharedInternals = { H: null, A: null, T: null, S: null },\n hasOwnProperty = Object.prototype.hasOwnProperty;\nfunction ReactElement(type, key, props) {\n var refProp = props.ref;\n return {\n $$typeof: REACT_ELEMENT_TYPE,\n type: type,\n key: key,\n ref: void 0 !== refProp ? refProp : null,\n props: props\n };\n}\nfunction cloneAndReplaceKey(oldElement, newKey) {\n return ReactElement(oldElement.type, newKey, oldElement.props);\n}\nfunction isValidElement(object) {\n return (\n \"object\" === typeof object &&\n null !== object &&\n object.$$typeof === REACT_ELEMENT_TYPE\n );\n}\nfunction escape(key) {\n var escaperLookup = { \"=\": \"=0\", \":\": \"=2\" };\n return (\n \"$\" +\n key.replace(/[=:]/g, function (match) {\n return escaperLookup[match];\n })\n );\n}\nvar userProvidedKeyEscapeRegex = /\\/+/g;\nfunction getElementKey(element, index) {\n return \"object\" === typeof element && null !== element && null != element.key\n ? escape(\"\" + element.key)\n : index.toString(36);\n}\nfunction resolveThenable(thenable) {\n switch (thenable.status) {\n case \"fulfilled\":\n return thenable.value;\n case \"rejected\":\n throw thenable.reason;\n default:\n switch (\n (\"string\" === typeof thenable.status\n ? thenable.then(noop, noop)\n : ((thenable.status = \"pending\"),\n thenable.then(\n function (fulfilledValue) {\n \"pending\" === thenable.status &&\n ((thenable.status = \"fulfilled\"),\n (thenable.value = fulfilledValue));\n },\n function (error) {\n \"pending\" === thenable.status &&\n ((thenable.status = \"rejected\"), (thenable.reason = error));\n }\n )),\n thenable.status)\n ) {\n case \"fulfilled\":\n return thenable.value;\n case \"rejected\":\n throw thenable.reason;\n }\n }\n throw thenable;\n}\nfunction mapIntoArray(children, array, escapedPrefix, nameSoFar, callback) {\n var type = typeof children;\n if (\"undefined\" === type || \"boolean\" === type) children = null;\n var invokeCallback = !1;\n if (null === children) invokeCallback = !0;\n else\n switch (type) {\n case \"bigint\":\n case \"string\":\n case \"number\":\n invokeCallback = !0;\n break;\n case \"object\":\n switch (children.$$typeof) {\n case REACT_ELEMENT_TYPE:\n case REACT_PORTAL_TYPE:\n invokeCallback = !0;\n break;\n case REACT_LAZY_TYPE:\n return (\n (invokeCallback = children._init),\n mapIntoArray(\n invokeCallback(children._payload),\n array,\n escapedPrefix,\n nameSoFar,\n callback\n )\n );\n }\n }\n if (invokeCallback)\n return (\n (callback = callback(children)),\n (invokeCallback =\n \"\" === nameSoFar ? \".\" + getElementKey(children, 0) : nameSoFar),\n isArrayImpl(callback)\n ? ((escapedPrefix = \"\"),\n null != invokeCallback &&\n (escapedPrefix =\n invokeCallback.replace(userProvidedKeyEscapeRegex, \"$&/\") + \"/\"),\n mapIntoArray(callback, array, escapedPrefix, \"\", function (c) {\n return c;\n }))\n : null != callback &&\n (isValidElement(callback) &&\n (callback = cloneAndReplaceKey(\n callback,\n escapedPrefix +\n (null == callback.key ||\n (children && children.key === callback.key)\n ? \"\"\n : (\"\" + callback.key).replace(\n userProvidedKeyEscapeRegex,\n \"$&/\"\n ) + \"/\") +\n invokeCallback\n )),\n array.push(callback)),\n 1\n );\n invokeCallback = 0;\n var nextNamePrefix = \"\" === nameSoFar ? \".\" : nameSoFar + \":\";\n if (isArrayImpl(children))\n for (var i = 0; i < children.length; i++)\n (nameSoFar = children[i]),\n (type = nextNamePrefix + getElementKey(nameSoFar, i)),\n (invokeCallback += mapIntoArray(\n nameSoFar,\n array,\n escapedPrefix,\n type,\n callback\n ));\n else if (((i = getIteratorFn(children)), \"function\" === typeof i))\n for (\n children = i.call(children), i = 0;\n !(nameSoFar = children.next()).done;\n\n )\n (nameSoFar = nameSoFar.value),\n (type = nextNamePrefix + getElementKey(nameSoFar, i++)),\n (invokeCallback += mapIntoArray(\n nameSoFar,\n array,\n escapedPrefix,\n type,\n callback\n ));\n else if (\"object\" === type) {\n if (\"function\" === typeof children.then)\n return mapIntoArray(\n resolveThenable(children),\n array,\n escapedPrefix,\n nameSoFar,\n callback\n );\n array = String(children);\n throw Error(\n \"Objects are not valid as a React child (found: \" +\n (\"[object Object]\" === array\n ? \"object with keys {\" + Object.keys(children).join(\", \") + \"}\"\n : array) +\n \"). If you meant to render a collection of children, use an array instead.\"\n );\n }\n return invokeCallback;\n}\nfunction mapChildren(children, func, context) {\n if (null == children) return children;\n var result = [],\n count = 0;\n mapIntoArray(children, result, \"\", \"\", function (child) {\n return func.call(context, child, count++);\n });\n return result;\n}\nfunction lazyInitializer(payload) {\n if (-1 === payload._status) {\n var ctor = payload._result;\n ctor = ctor();\n ctor.then(\n function (moduleObject) {\n if (0 === payload._status || -1 === payload._status)\n (payload._status = 1), (payload._result = moduleObject);\n },\n function (error) {\n if (0 === payload._status || -1 === payload._status)\n (payload._status = 2), (payload._result = error);\n }\n );\n -1 === payload._status && ((payload._status = 0), (payload._result = ctor));\n }\n if (1 === payload._status) return payload._result.default;\n throw payload._result;\n}\nvar reportGlobalError =\n \"function\" === typeof reportError\n ? reportError\n : function (error) {\n if (\n \"object\" === typeof window &&\n \"function\" === typeof window.ErrorEvent\n ) {\n var event = new window.ErrorEvent(\"error\", {\n bubbles: !0,\n cancelable: !0,\n message:\n \"object\" === typeof error &&\n null !== error &&\n \"string\" === typeof error.message\n ? String(error.message)\n : String(error),\n error: error\n });\n if (!window.dispatchEvent(event)) return;\n } else if (\n \"object\" === typeof process &&\n \"function\" === typeof process.emit\n ) {\n process.emit(\"uncaughtException\", error);\n return;\n }\n console.error(error);\n },\n Children = {\n map: mapChildren,\n forEach: function (children, forEachFunc, forEachContext) {\n mapChildren(\n children,\n function () {\n forEachFunc.apply(this, arguments);\n },\n forEachContext\n );\n },\n count: function (children) {\n var n = 0;\n mapChildren(children, function () {\n n++;\n });\n return n;\n },\n toArray: function (children) {\n return (\n mapChildren(children, function (child) {\n return child;\n }) || []\n );\n },\n only: function (children) {\n if (!isValidElement(children))\n throw Error(\n \"React.Children.only expected to receive a single React element child.\"\n );\n return children;\n }\n };\nexports.Activity = REACT_ACTIVITY_TYPE;\nexports.Children = Children;\nexports.Component = Component;\nexports.Fragment = REACT_FRAGMENT_TYPE;\nexports.Profiler = REACT_PROFILER_TYPE;\nexports.PureComponent = PureComponent;\nexports.StrictMode = REACT_STRICT_MODE_TYPE;\nexports.Suspense = REACT_SUSPENSE_TYPE;\nexports.__CLIENT_INTERNALS_DO_NOT_USE_OR_WARN_USERS_THEY_CANNOT_UPGRADE =\n ReactSharedInternals;\nexports.__COMPILER_RUNTIME = {\n __proto__: null,\n c: function (size) {\n return ReactSharedInternals.H.useMemoCache(size);\n }\n};\nexports.cache = function (fn) {\n return function () {\n return fn.apply(null, arguments);\n };\n};\nexports.cacheSignal = function () {\n return null;\n};\nexports.cloneElement = function (element, config, children) {\n if (null === element || void 0 === element)\n throw Error(\n \"The argument must be a React element, but you passed \" + element + \".\"\n );\n var props = assign({}, element.props),\n key = element.key;\n if (null != config)\n for (propName in (void 0 !== config.key && (key = \"\" + config.key), config))\n !hasOwnProperty.call(config, propName) ||\n \"key\" === propName ||\n \"__self\" === propName ||\n \"__source\" === propName ||\n (\"ref\" === propName && void 0 === config.ref) ||\n (props[propName] = config[propName]);\n var propName = arguments.length - 2;\n if (1 === propName) props.children = children;\n else if (1 < propName) {\n for (var childArray = Array(propName), i = 0; i < propName; i++)\n childArray[i] = arguments[i + 2];\n props.children = childArray;\n }\n return ReactElement(element.type, key, props);\n};\nexports.createContext = function (defaultValue) {\n defaultValue = {\n $$typeof: REACT_CONTEXT_TYPE,\n _currentValue: defaultValue,\n _currentValue2: defaultValue,\n _threadCount: 0,\n Provider: null,\n Consumer: null\n };\n defaultValue.Provider = defaultValue;\n defaultValue.Consumer = {\n $$typeof: REACT_CONSUMER_TYPE,\n _context: defaultValue\n };\n return defaultValue;\n};\nexports.createElement = function (type, config, children) {\n var propName,\n props = {},\n key = null;\n if (null != config)\n for (propName in (void 0 !== config.key && (key = \"\" + config.key), config))\n hasOwnProperty.call(config, propName) &&\n \"key\" !== propName &&\n \"__self\" !== propName &&\n \"__source\" !== propName &&\n (props[propName] = config[propName]);\n var childrenLength = arguments.length - 2;\n if (1 === childrenLength) props.children = children;\n else if (1 < childrenLength) {\n for (var childArray = Array(childrenLength), i = 0; i < childrenLength; i++)\n childArray[i] = arguments[i + 2];\n props.children = childArray;\n }\n if (type && type.defaultProps)\n for (propName in ((childrenLength = type.defaultProps), childrenLength))\n void 0 === props[propName] &&\n (props[propName] = childrenLength[propName]);\n return ReactElement(type, key, props);\n};\nexports.createRef = function () {\n return { current: null };\n};\nexports.forwardRef = function (render) {\n return { $$typeof: REACT_FORWARD_REF_TYPE, render: render };\n};\nexports.isValidElement = isValidElement;\nexports.lazy = function (ctor) {\n return {\n $$typeof: REACT_LAZY_TYPE,\n _payload: { _status: -1, _result: ctor },\n _init: lazyInitializer\n };\n};\nexports.memo = function (type, compare) {\n return {\n $$typeof: REACT_MEMO_TYPE,\n type: type,\n compare: void 0 === compare ? null : compare\n };\n};\nexports.startTransition = function (scope) {\n var prevTransition = ReactSharedInternals.T,\n currentTransition = {};\n ReactSharedInternals.T = currentTransition;\n try {\n var returnValue = scope(),\n onStartTransitionFinish = ReactSharedInternals.S;\n null !== onStartTransitionFinish &&\n onStartTransitionFinish(currentTransition, returnValue);\n \"object\" === typeof returnValue &&\n null !== returnValue &&\n \"function\" === typeof returnValue.then &&\n returnValue.then(noop, reportGlobalError);\n } catch (error) {\n reportGlobalError(error);\n } finally {\n null !== prevTransition &&\n null !== currentTransition.types &&\n (prevTransition.types = currentTransition.types),\n (ReactSharedInternals.T = prevTransition);\n }\n};\nexports.unstable_useCacheRefresh = function () {\n return ReactSharedInternals.H.useCacheRefresh();\n};\nexports.use = function (usable) {\n return ReactSharedInternals.H.use(usable);\n};\nexports.useActionState = function (action, initialState, permalink) {\n return ReactSharedInternals.H.useActionState(action, initialState, permalink);\n};\nexports.useCallback = function (callback, deps) {\n return ReactSharedInternals.H.useCallback(callback, deps);\n};\nexports.useContext = function (Context) {\n return ReactSharedInternals.H.useContext(Context);\n};\nexports.useDebugValue = function () {};\nexports.useDeferredValue = function (value, initialValue) {\n return ReactSharedInternals.H.useDeferredValue(value, initialValue);\n};\nexports.useEffect = function (create, deps) {\n return ReactSharedInternals.H.useEffect(create, deps);\n};\nexports.useEffectEvent = function (callback) {\n return ReactSharedInternals.H.useEffectEvent(callback);\n};\nexports.useId = function () {\n return ReactSharedInternals.H.useId();\n};\nexports.useImperativeHandle = function (ref, create, deps) {\n return ReactSharedInternals.H.useImperativeHandle(ref, create, deps);\n};\nexports.useInsertionEffect = function (create, deps) {\n return ReactSharedInternals.H.useInsertionEffect(create, deps);\n};\nexports.useLayoutEffect = function (create, deps) {\n return ReactSharedInternals.H.useLayoutEffect(create, deps);\n};\nexports.useMemo = function (create, deps) {\n return ReactSharedInternals.H.useMemo(create, deps);\n};\nexports.useOptimistic = function (passthrough, reducer) {\n return ReactSharedInternals.H.useOptimistic(passthrough, reducer);\n};\nexports.useReducer = function (reducer, initialArg, init) {\n return ReactSharedInternals.H.useReducer(reducer, initialArg, init);\n};\nexports.useRef = function (initialValue) {\n return ReactSharedInternals.H.useRef(initialValue);\n};\nexports.useState = function (initialState) {\n return ReactSharedInternals.H.useState(initialState);\n};\nexports.useSyncExternalStore = function (\n subscribe,\n getSnapshot,\n getServerSnapshot\n) {\n return ReactSharedInternals.H.useSyncExternalStore(\n subscribe,\n getSnapshot,\n getServerSnapshot\n );\n};\nexports.useTransition = function () {\n return ReactSharedInternals.H.useTransition();\n};\nexports.version = \"19.2.4\";\n","'use strict';\n\nif (process.env.NODE_ENV === 'production') {\n module.exports = require('./cjs/react.production.js');\n} else {\n module.exports = require('./cjs/react.development.js');\n}\n","/**\n * @license React\n * react-jsx-runtime.production.js\n *\n * Copyright (c) Meta Platforms, Inc. and affiliates.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n */\n\n\"use strict\";\nvar REACT_ELEMENT_TYPE = Symbol.for(\"react.transitional.element\"),\n REACT_FRAGMENT_TYPE = Symbol.for(\"react.fragment\");\nfunction jsxProd(type, config, maybeKey) {\n var key = null;\n void 0 !== maybeKey && (key = \"\" + maybeKey);\n void 0 !== config.key && (key = \"\" + config.key);\n if (\"key\" in config) {\n maybeKey = {};\n for (var propName in config)\n \"key\" !== propName && (maybeKey[propName] = config[propName]);\n } else maybeKey = config;\n config = maybeKey.ref;\n return {\n $$typeof: REACT_ELEMENT_TYPE,\n type: type,\n key: key,\n ref: void 0 !== config ? config : null,\n props: maybeKey\n };\n}\nexports.Fragment = REACT_FRAGMENT_TYPE;\nexports.jsx = jsxProd;\nexports.jsxs = jsxProd;\n","'use strict';\n\nif (process.env.NODE_ENV === 'production') {\n module.exports = require('./cjs/react-jsx-runtime.production.js');\n} else {\n module.exports = require('./cjs/react-jsx-runtime.development.js');\n}\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type EarningsSummaryProps = {\n stats: PartnerStats\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nexport const EarningsSummary: React.FC<EarningsSummaryProps> = ({ stats, currency }) => {\n return (\n <div className=\"partner-widget partner-widget--earnings\">\n <h3 className=\"partner-widget__title\">Earnings Summary</h3>\n <div className=\"partner-widget__content\">\n <div className=\"earnings-grid\">\n <div className=\"earnings-card earnings-card--total\">\n <span className=\"earnings-card__label\">Total Earnings</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.totalEarnings, currency)}\n </span>\n </div>\n <div className=\"earnings-card earnings-card--pending\">\n <span className=\"earnings-card__label\">Pending</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.pendingEarnings, currency)}\n </span>\n </div>\n <div className=\"earnings-card earnings-card--paid\">\n <span className=\"earnings-card__label\">Paid Out</span>\n <span className=\"earnings-card__value\">\n {formatCurrency(stats.paidEarnings, currency)}\n </span>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default EarningsSummary\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type ReferralPerformanceProps = {\n stats: PartnerStats\n}\n\nexport const ReferralPerformance: React.FC<ReferralPerformanceProps> = ({ stats }) => {\n return (\n <div className=\"partner-widget partner-widget--performance\">\n <h3 className=\"partner-widget__title\">Referral Performance</h3>\n <div className=\"partner-widget__content\">\n <div className=\"performance-grid\">\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.totalReferrals}</span>\n <span className=\"performance-stat__label\">Total Referrals</span>\n </div>\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.successfulReferrals}</span>\n <span className=\"performance-stat__label\">Successful</span>\n </div>\n <div className=\"performance-stat\">\n <span className=\"performance-stat__value\">{stats.conversionRate.toFixed(1)}%</span>\n <span className=\"performance-stat__label\">Conversion Rate</span>\n </div>\n </div>\n\n {stats.monthlyEarnings.length > 0 && (\n <div className=\"performance-chart\">\n <h4 className=\"performance-chart__title\">Monthly Trend</h4>\n <div className=\"performance-chart__bars\">\n {stats.monthlyEarnings.map((month, index) => {\n const maxEarnings = Math.max(...stats.monthlyEarnings.map((m) => m.earnings), 1)\n const heightPercent = (month.earnings / maxEarnings) * 100\n\n return (\n <div key={index} className=\"performance-chart__bar-container\">\n <div\n className=\"performance-chart__bar\"\n style={{ height: `${Math.max(heightPercent, 5)}%` }}\n title={`${month.month}: ${month.referrals} referrals`}\n />\n <span className=\"performance-chart__label\">{month.month.split(' ')[0]}</span>\n </div>\n )\n })}\n </div>\n </div>\n )}\n </div>\n </div>\n )\n}\n\nexport default ReferralPerformance\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type RecentReferralsProps = {\n referrals: PartnerStats['recentReferrals']\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nconst formatDate = (dateString: string): string => {\n return new Date(dateString).toLocaleDateString('en-US', {\n month: 'short',\n day: 'numeric',\n year: 'numeric',\n })\n}\n\nexport const RecentReferrals: React.FC<RecentReferralsProps> = ({ referrals, currency }) => {\n return (\n <div className=\"partner-widget partner-widget--recent\">\n <h3 className=\"partner-widget__title\">Recent Referrals</h3>\n <div className=\"partner-widget__content\">\n <div className=\"referrals-table\">\n <div className=\"referrals-table__header\">\n <span>Code</span>\n <span>Order Value</span>\n <span>Commission</span>\n <span>Date</span>\n <span>Status</span>\n </div>\n <div className=\"referrals-table__body\">\n {referrals.map((referral) => (\n <div key={referral.id} className=\"referrals-table__row\">\n <span className=\"referrals-table__code\">{referral.code}</span>\n <span className=\"referrals-table__value\">\n {formatCurrency(referral.orderValue, currency)}\n </span>\n <span className=\"referrals-table__commission\">\n {formatCurrency(referral.commission, currency)}\n </span>\n <span className=\"referrals-table__date\">{formatDate(referral.date)}</span>\n <span\n className={`referrals-table__status referrals-table__status--${referral.status}`}\n >\n {referral.status}\n </span>\n </div>\n ))}\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default RecentReferrals\n","'use client'\n\nimport React, { useState } from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nexport type ReferralCodesProps = {\n codes: PartnerDashboardData['referralCodes']\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nexport const ReferralCodes: React.FC<ReferralCodesProps> = ({ codes, currency }) => {\n const [copiedCode, setCopiedCode] = useState<string | null>(null)\n\n const copyToClipboard = async (code: string) => {\n try {\n await navigator.clipboard.writeText(code)\n setCopiedCode(code)\n setTimeout(() => setCopiedCode(null), 2000)\n } catch {\n // Fallback for older browsers\n const textArea = document.createElement('textarea')\n textArea.value = code\n document.body.appendChild(textArea)\n textArea.select()\n document.execCommand('copy')\n document.body.removeChild(textArea)\n setCopiedCode(code)\n setTimeout(() => setCopiedCode(null), 2000)\n }\n }\n\n return (\n <div className=\"partner-widget partner-widget--codes\">\n <h3 className=\"partner-widget__title\">Your Referral Codes</h3>\n <div className=\"partner-widget__content\">\n <div className=\"codes-list\">\n {codes.map((codeData) => (\n <div\n key={codeData.id}\n className={`code-card ${!codeData.isActive ? 'code-card--inactive' : ''}`}\n >\n <div className=\"code-card__header\">\n <span className=\"code-card__code\">{codeData.code}</span>\n <button\n type=\"button\"\n className=\"code-card__copy\"\n onClick={() => copyToClipboard(codeData.code)}\n title=\"Copy code\"\n >\n {copiedCode === codeData.code ? '✓ Copied' : 'Copy'}\n </button>\n </div>\n <div className=\"code-card__stats\">\n <div className=\"code-card__stat\">\n <span className=\"code-card__stat-value\">{codeData.usageCount}</span>\n <span className=\"code-card__stat-label\">Uses</span>\n </div>\n <div className=\"code-card__stat\">\n <span className=\"code-card__stat-value\">\n {formatCurrency(codeData.totalEarnings, currency)}\n </span>\n <span className=\"code-card__stat-label\">Earnings</span>\n </div>\n </div>\n {!codeData.isActive && (\n <span className=\"code-card__badge code-card__badge--inactive\">Inactive</span>\n )}\n </div>\n ))}\n </div>\n </div>\n </div>\n )\n}\n\nexport default ReferralCodes\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nexport type ProgramOverviewProps = {\n program: PartnerDashboardData['program']\n}\n\nexport const ProgramOverview: React.FC<ProgramOverviewProps> = ({ program }) => {\n if (!program) {\n return (\n <div className=\"partner-widget partner-widget--program-overview\">\n <h3 className=\"partner-widget__title\">Program Overview</h3>\n <div className=\"partner-widget__content\">\n <div className=\"program-overview program-overview--empty\">\n <p>No referral program details are available yet.</p>\n </div>\n </div>\n </div>\n )\n }\n\n return (\n <div className=\"partner-widget partner-widget--program-overview\">\n <h3 className=\"partner-widget__title\">Program Overview</h3>\n <div className=\"partner-widget__content\">\n <div className=\"program-overview\">\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Program Name</span>\n <span className=\"program-overview__value\">{program.name}</span>\n </div>\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Commission Rate</span>\n <span className=\"program-overview__value\">{program.commissionRate}%</span>\n </div>\n <div className=\"program-overview__row\">\n <span className=\"program-overview__label\">Customer Discount</span>\n <span className=\"program-overview__value\">{program.customerDiscount}%</span>\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default ProgramOverview\n","'use client'\n\nimport React from 'react'\n\nimport type { PartnerStats } from '../../types'\n\nexport type CommissionBreakdownProps = {\n stats: PartnerStats\n currency: string\n}\n\nconst formatCurrency = (amount: number, currency: string): string => {\n return new Intl.NumberFormat('en-US', {\n style: 'currency',\n currency,\n minimumFractionDigits: 2,\n }).format(amount)\n}\n\nconst formatPercent = (value: number): string => {\n if (!Number.isFinite(value)) return '0%'\n return `${value.toFixed(1)}%`\n}\n\nexport const CommissionBreakdown: React.FC<CommissionBreakdownProps> = ({ stats, currency }) => {\n const totalEarnings = Math.max(stats.totalEarnings || 0, 0)\n const paidEarnings = Math.max(stats.paidEarnings || 0, 0)\n const pendingEarnings = Math.max(stats.pendingEarnings || 0, 0)\n\n const paidPercent = totalEarnings > 0 ? (paidEarnings / totalEarnings) * 100 : 0\n const pendingPercent = totalEarnings > 0 ? (pendingEarnings / totalEarnings) * 100 : 0\n const unpaidOrOther = Math.max(totalEarnings - (paidEarnings + pendingEarnings), 0)\n const otherPercent = totalEarnings > 0 ? (unpaidOrOther / totalEarnings) * 100 : 0\n\n return (\n <div className=\"partner-widget partner-widget--commission-breakdown\">\n <h3 className=\"partner-widget__title\">Commission Breakdown</h3>\n\n <div className=\"partner-widget__content\">\n <div className=\"commission-breakdown\">\n <div className=\"commission-breakdown__summary\">\n <span className=\"commission-breakdown__total-label\">Total Commission</span>\n <span className=\"commission-breakdown__total-value\">\n {formatCurrency(totalEarnings, currency)}\n </span>\n </div>\n\n <div\n className=\"commission-breakdown__bars\"\n role=\"img\"\n aria-label=\"Commission distribution\"\n >\n <div className=\"commission-breakdown__track\">\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--paid\"\n style={{ width: `${Math.max(Math.min(paidPercent, 100), 0)}%` }}\n title={`Paid: ${formatCurrency(paidEarnings, currency)} (${formatPercent(paidPercent)})`}\n />\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--pending\"\n style={{ width: `${Math.max(Math.min(pendingPercent, 100), 0)}%` }}\n title={`Pending: ${formatCurrency(pendingEarnings, currency)} (${formatPercent(pendingPercent)})`}\n />\n {otherPercent > 0 && (\n <div\n className=\"commission-breakdown__segment commission-breakdown__segment--other\"\n style={{ width: `${Math.max(Math.min(otherPercent, 100), 0)}%` }}\n title={`Other: ${formatCurrency(unpaidOrOther, currency)} (${formatPercent(otherPercent)})`}\n />\n )}\n </div>\n </div>\n\n <div className=\"commission-breakdown__legend\">\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--paid\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Paid</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(paidEarnings, currency)} · {formatPercent(paidPercent)}\n </span>\n </div>\n </div>\n\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--pending\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Pending</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(pendingEarnings, currency)} · {formatPercent(pendingPercent)}\n </span>\n </div>\n </div>\n\n {otherPercent > 0 && (\n <div className=\"commission-breakdown__item\">\n <span className=\"commission-breakdown__dot commission-breakdown__dot--other\" />\n <div className=\"commission-breakdown__meta\">\n <span className=\"commission-breakdown__label\">Other</span>\n <span className=\"commission-breakdown__value\">\n {formatCurrency(unpaidOrOther, currency)} · {formatPercent(otherPercent)}\n </span>\n </div>\n </div>\n )}\n </div>\n\n <div className=\"commission-breakdown__footnote\">\n Based on current referral payouts and pending settlements.\n </div>\n </div>\n </div>\n </div>\n )\n}\n\nexport default CommissionBreakdown\n","'use client'\n\nimport React, { useEffect, useState } from 'react'\n\nimport type { PartnerDashboardData } from '../../types'\n\nimport { EarningsSummary } from './EarningsSummary'\nimport { ReferralPerformance } from './ReferralPerformance'\nimport { RecentReferrals } from './RecentReferrals'\nimport { ReferralCodes } from './ReferralCodes'\nimport { ProgramOverview } from './ProgramOverview'\nimport { CommissionBreakdown } from './CommissionBreakdown'\n\nexport type PartnerDashboardProps = {\n showEarningsSummary?: boolean\n showReferralPerformance?: boolean\n showRecentReferrals?: boolean\n showReferralCodes?: boolean\n showProgramOverview?: boolean\n showCommissionBreakdown?: boolean\n apiEndpoint?: string\n}\n\nexport const PartnerDashboard: React.FC<PartnerDashboardProps> = ({\n showEarningsSummary = true,\n showReferralPerformance = true,\n showRecentReferrals = true,\n showReferralCodes = true,\n showProgramOverview = true,\n showCommissionBreakdown = true,\n apiEndpoint = '/api/referrals/partner-stats',\n}) => {\n const [data, setData] = useState<PartnerDashboardData | null>(null)\n const [loading, setLoading] = useState(true)\n const [error, setError] = useState<string | null>(null)\n const [currency, setCurrency] = useState('USD')\n\n useEffect(() => {\n const fetchData = async () => {\n try {\n setLoading(true)\n const response = await fetch(apiEndpoint, {\n credentials: 'include',\n })\n\n if (!response.ok) {\n if (response.status === 401) {\n setError('Please log in to view your partner dashboard')\n return\n }\n if (response.status === 403) {\n setError('Partner access required')\n return\n }\n throw new Error('Failed to fetch partner data')\n }\n\n const result = (await response.json()) as {\n success: boolean\n data?: PartnerDashboardData\n currency?: string\n error?: string\n }\n\n if (result.success) {\n setData(result.data ?? null)\n setCurrency(result.currency ?? 'USD')\n } else {\n setError(result.error ?? 'Failed to load dashboard data')\n }\n } catch (err) {\n setError(err instanceof Error ? err.message : 'An error occurred')\n } finally {\n setLoading(false)\n }\n }\n\n fetchData()\n }, [apiEndpoint])\n\n if (loading) {\n return (\n <div className=\"partner-dashboard partner-dashboard--loading\">\n <div className=\"partner-dashboard__loader\">\n <div className=\"partner-dashboard__spinner\" />\n <p>Loading partner dashboard...</p>\n </div>\n </div>\n )\n }\n\n if (error) {\n return (\n <div className=\"partner-dashboard partner-dashboard--error\">\n <div className=\"partner-dashboard__error\">\n <h3>Unable to load dashboard</h3>\n <p>{error}</p>\n </div>\n </div>\n )\n }\n\n if (!data) {\n return (\n <div className=\"partner-dashboard partner-dashboard--empty\">\n <div className=\"partner-dashboard__empty\">\n <h3>No data available</h3>\n <p>Your partner dashboard will appear here once you have referral activity.</p>\n </div>\n </div>\n )\n }\n\n return (\n <div className=\"partner-dashboard\">\n <div className=\"partner-dashboard__header\">\n <h2>Partner Dashboard</h2>\n {data.program && (\n <div className=\"partner-dashboard__program\">\n <span className=\"partner-dashboard__program-name\">{data.program.name}</span>\n <span className=\"partner-dashboard__program-rate\">\n {data.program.commissionRate}% commission\n </span>\n </div>\n )}\n </div>\n\n <div className=\"partner-dashboard__grid\">\n {showEarningsSummary && <EarningsSummary stats={data.stats} currency={currency} />}\n\n {showReferralPerformance && <ReferralPerformance stats={data.stats} />}\n\n {showRecentReferrals && data.stats.recentReferrals.length > 0 && (\n <RecentReferrals referrals={data.stats.recentReferrals} currency={currency} />\n )}\n\n {showReferralCodes && data.referralCodes.length > 0 && (\n <ReferralCodes codes={data.referralCodes} currency={currency} />\n )}\n\n {showProgramOverview && <ProgramOverview program={data.program} />}\n\n {showCommissionBreakdown && <CommissionBreakdown stats={data.stats} currency={currency} />}\n </div>\n </div>\n )\n}\n\nexport default PartnerDashboard\n","import { roundTo2 } from './roundTo2'\n\nexport type CartLike = {\n subtotal?: number\n total?: number\n discountAmount?: number\n customerDiscount?: number\n}\n\n/**\n * Computes the cart total after applying plugin discounts.\n * Use this in your host app's cart beforeChange (or wherever you compute total)\n * so the amount always reflects coupon/referral discounts and is not overwritten incorrectly.\n *\n * Formula: subtotal - discountAmount - customerDiscount (each defaulting to 0).\n */\nexport function getCartTotalWithDiscounts(cart: CartLike): number {\n const subtotal = cart.subtotal ?? cart.total ?? 0\n const discountAmount = cart.discountAmount ?? 0\n const customerDiscount = cart.customerDiscount ?? 0\n return roundTo2(Math.max(0, subtotal - discountAmount - customerDiscount))\n}\n"],"x_google_ignoreList":[15,16,17,18],"mappings":"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aAAa,gCACd;IACF;GACD;IACE,MAAM;IACN,MAAM;IACN,cAAc;IACd,OAAO,EACL,aAAa,qDACd;IACF;GACD;IACE,MAAM;IACN,MAAM;IACN,KAAK;IACL,OAAO,EACL,aACE,oHACH;IACF;GACD;IACE,MAAM;IACN,MAAM;IACN,KAAK;IACL,OAAO,EACL,aACE,+GACH;IACF;GACD;IACE,MAAM;IACN,MAAM;IACN,KAAK;IACL,OAAO,EACL,aACE,iIACH;IACF;GACD;IACE,MAAM;IACN,MAAM;IACN,UAAU;IACV,SAAS;IACT,OAAO,EACL,aAAa,qEACd;IACD,QAAQ;KACN;MACE,MAAM;MACN,MAAM;MACN,UAAU;MACV,SAAS;OACP;QAAE,OAAO;QAAgB,OAAO;QAAO;OACvC;QAAE,OAAO;QAAqB,OAAO;QAAY;OACjD;QAAE,OAAO;QAAuB,OAAO;QAAY;OACpD;MACD,cAAc;MACf;KACD;MACE,MAAM;MACN,MAAM;MACN,YAAY,cAAc;MAC1B,SAAS;MACT,OAAO;OACL,YAAY,GAAY,gBACtB,aAAa,cAAc;OAC7B,aAAa;OACd;MACF;KACD;MACE,MAAM;MACN,MAAM;MACN,YAAY,cAAc;MAC1B,SAAS;MACT,OAAO;OACL,YAAY,GAAY,gBACtB,aAAa,cAAc;OAC7B,aAAa;OACd;MACF;KACD;MACE,MAAM;MACN,MAAM;MACN,YAAY,cAAc;MAC1B,SAAS;MACT,OAAO;OACL,YAAY,GAAY,gBACtB,aAAa,cAAc;OAC7B,aAAa;OACd;MACF;KACD;MACE,MAAM;MACN,MAAM;MACN,OAAO,EACL,aAAa,uCACd;MACD,QAAQ,CACN;OACE,MAAM;OACN,MAAM;OACN,UAAU;OACV,SAAS,4BAA4B,KAAK,WAAW;QACnD,OAAO,UAAU,UAAU,iBAAiB;QAC5C;QACD,EAAE;OACH,cAAc,4BAA4B,SAAS,QAAQ,GACvD,UACA;OACL,CACF;MACF;KACD;MACE,MAAM;MACN,MAAM;MACN,KAAK;MACL,KAAK;MACL,OAAO;OACL,YAAY,GAAY,gBACtB,aAAa,iBAAiB,SAAS;OACzC,aAAa;OACd;MACF;K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"}