From dbde1748a2cef3d5bb9ea983f9ed3015368acc30 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Tue, 20 Aug 2024 10:17:08 +0530 Subject: [PATCH 01/20] feat: coupon code schemas --- schemas/app/AccountingSettings.json | 7 +++ schemas/app/CouponCode.json | 71 +++++++++++++++++++++++++++++ 2 files changed, 78 insertions(+) create mode 100644 schemas/app/CouponCode.json diff --git a/schemas/app/AccountingSettings.json b/schemas/app/AccountingSettings.json index 80fd8312..e7d09d60 100644 --- a/schemas/app/AccountingSettings.json +++ b/schemas/app/AccountingSettings.json @@ -79,6 +79,13 @@ "default": false, "section": "Features" }, + { + "fieldname": "enableCouponCode", + "label": "Enable Coupon Code", + "fieldtype": "Check", + "default": false, + "section": "Features" + }, { "fieldname": "enablePriceList", "label": "Enable Price List", diff --git a/schemas/app/CouponCode.json b/schemas/app/CouponCode.json new file mode 100644 index 00000000..10c36ac2 --- /dev/null +++ b/schemas/app/CouponCode.json @@ -0,0 +1,71 @@ +{ + "name": "CouponCode", + "label": "Coupon Code", + "naming": "manual", + + "fields": [ + { + "fieldname": "name", + "label": "Name", + "fieldtype": "Data", + "required": true, + "placeholder": "Coupon Name", + "section": "Default" + }, + { + "fieldname": "couponCode", + "label": "Coupon Code", + "fieldtype": "Data", + "required": true + }, + { + "fieldname": "isEnabled", + "label": "Is Enabled", + "fieldtype": "Check", + "default": true, + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "validFrom", + "label": "Valid From", + "fieldtype": "Date", + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "validTo", + "label": "Valid To", + "fieldtype": "Date", + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "maximumUse", + "label": "Maximum Use", + "fieldtype": "Int", + "default":0, + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "used", + "label": "Used", + "fieldtype": "Int", + "default":0, + "required": true, + "readOnly": true, + "section": "Validity and Usage" + } + ], + "quickEditFields": [ + "name", + "couponCode", + "validFrom", + "validTo", + "maximumUse", + "used" + ], + "keywordFields": ["name"] + } + \ No newline at end of file From e278f9aa329b388e33c9c5b70555ceb1357c87ac Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Tue, 20 Aug 2024 10:38:50 +0530 Subject: [PATCH 02/20] feat: added coupon codes to sidebar and formatted --- schemas/app/CouponCode.json | 137 ++++++++++++++++++------------------ src/utils/sidebarConfig.ts | 7 ++ 2 files changed, 75 insertions(+), 69 deletions(-) diff --git a/schemas/app/CouponCode.json b/schemas/app/CouponCode.json index 10c36ac2..a31b2372 100644 --- a/schemas/app/CouponCode.json +++ b/schemas/app/CouponCode.json @@ -1,71 +1,70 @@ { - "name": "CouponCode", - "label": "Coupon Code", - "naming": "manual", + "name": "CouponCode", + "label": "Coupon Code", + "naming": "manual", - "fields": [ - { - "fieldname": "name", - "label": "Name", - "fieldtype": "Data", - "required": true, - "placeholder": "Coupon Name", - "section": "Default" - }, - { - "fieldname": "couponCode", - "label": "Coupon Code", - "fieldtype": "Data", - "required": true - }, - { - "fieldname": "isEnabled", - "label": "Is Enabled", - "fieldtype": "Check", - "default": true, - "required": true, - "section": "Validity and Usage" - }, - { - "fieldname": "validFrom", - "label": "Valid From", - "fieldtype": "Date", - "required": true, - "section": "Validity and Usage" - }, - { - "fieldname": "validTo", - "label": "Valid To", - "fieldtype": "Date", - "required": true, - "section": "Validity and Usage" - }, - { - "fieldname": "maximumUse", - "label": "Maximum Use", - "fieldtype": "Int", - "default":0, - "required": true, - "section": "Validity and Usage" - }, - { - "fieldname": "used", - "label": "Used", - "fieldtype": "Int", - "default":0, - "required": true, - "readOnly": true, - "section": "Validity and Usage" - } - ], - "quickEditFields": [ - "name", - "couponCode", - "validFrom", - "validTo", - "maximumUse", - "used" - ], - "keywordFields": ["name"] - } - \ No newline at end of file + "fields": [ + { + "fieldname": "name", + "label": "Name", + "fieldtype": "Data", + "required": true, + "placeholder": "Coupon Name", + "section": "Default" + }, + { + "fieldname": "couponCode", + "label": "Coupon Code", + "fieldtype": "Data", + "required": true + }, + { + "fieldname": "isEnabled", + "label": "Is Enabled", + "fieldtype": "Check", + "default": true, + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "validFrom", + "label": "Valid From", + "fieldtype": "Date", + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "validTo", + "label": "Valid To", + "fieldtype": "Date", + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "maximumUse", + "label": "Maximum Use", + "fieldtype": "Int", + "default": 0, + "required": true, + "section": "Validity and Usage" + }, + { + "fieldname": "used", + "label": "Used", + "fieldtype": "Int", + "default": 0, + "required": true, + "readOnly": true, + "section": "Validity and Usage" + } + ], + "quickEditFields": [ + "name", + "couponCode", + "validFrom", + "validTo", + "maximumUse", + "used" + ], + "keywordFields": ["name"] +} diff --git a/src/utils/sidebarConfig.ts b/src/utils/sidebarConfig.ts index 64738381..d506bc07 100644 --- a/src/utils/sidebarConfig.ts +++ b/src/utils/sidebarConfig.ts @@ -216,6 +216,13 @@ function getCompleteSidebar(): SidebarConfig { schemaName: 'Lead', hidden: () => !fyo.singles.AccountingSettings?.enableLead, }, + { + label: t`Coupon Code`, + name: 'coupon-code', + route: `/list/CouponCode/CouponCode`, + schemaName: 'CouponCode', + hidden: () => !fyo.singles.AccountingSettings?.enableCouponCode, + }, ] as SidebarItem[], }, { From f7175337309da52bbb3f02a70ac7f94ec1257d49 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Tue, 20 Aug 2024 15:10:26 +0530 Subject: [PATCH 03/20] feat: register schemas for coupon codes --- schemas/app/AccountingSettings.json | 15 +++++++------- schemas/app/CouponCode.json | 32 +++++++++++++++++++++++------ schemas/schemas.ts | 2 ++ 3 files changed, 36 insertions(+), 13 deletions(-) diff --git a/schemas/app/AccountingSettings.json b/schemas/app/AccountingSettings.json index e7d09d60..377bbde2 100644 --- a/schemas/app/AccountingSettings.json +++ b/schemas/app/AccountingSettings.json @@ -79,13 +79,6 @@ "default": false, "section": "Features" }, - { - "fieldname": "enableCouponCode", - "label": "Enable Coupon Code", - "fieldtype": "Check", - "default": false, - "section": "Features" - }, { "fieldname": "enablePriceList", "label": "Enable Price List", @@ -128,6 +121,14 @@ "default": false, "section": "Features" }, + { + + "fieldname": "enableCouponCode", + "label": "Enable Coupon Code", + "fieldtype": "Check", + "default": false, + "section": "Features" + }, { "fieldname": "fiscalYearStart", "label": "Fiscal Year Start Date", diff --git a/schemas/app/CouponCode.json b/schemas/app/CouponCode.json index a31b2372..91ee1621 100644 --- a/schemas/app/CouponCode.json +++ b/schemas/app/CouponCode.json @@ -19,12 +19,23 @@ "required": true }, { - "fieldname": "isEnabled", - "label": "Is Enabled", - "fieldtype": "Check", - "default": true, - "required": true, - "section": "Validity and Usage" + "fieldname": "pricingRule", + "label": "Pricing Rule", + "fieldtype": "Link", + "target": "PricingRule", + "required": true + }, + { + "fieldname": "minAmount", + "label": "Min Amount", + "fieldtype": "Currency", + "section": "Amount" + }, + { + "fieldname": "maxAmount", + "label": "Max Amount", + "fieldtype": "Currency", + "section": "Amount" }, { "fieldname": "validFrom", @@ -40,6 +51,14 @@ "required": true, "section": "Validity and Usage" }, + { + "fieldname": "isEnabled", + "label": "Is Enabled", + "fieldtype": "Check", + "default": true, + "required": true, + "section": "Validity and Usage" + }, { "fieldname": "maximumUse", "label": "Maximum Use", @@ -61,6 +80,7 @@ "quickEditFields": [ "name", "couponCode", + "pricingRule", "validFrom", "validTo", "maximumUse", diff --git a/schemas/schemas.ts b/schemas/schemas.ts index aef9ff6d..d6dbd45d 100644 --- a/schemas/schemas.ts +++ b/schemas/schemas.ts @@ -19,6 +19,7 @@ import Lead from './app/Lead.json'; import LoyaltyProgram from './app/LoyaltyProgram.json'; import LoyaltyPointEntry from './app/LoyaltyPointEntry.json'; import CollectionRulesItems from './app/CollectionRulesItems.json'; +import CouponCode from './app/CouponCode.json'; import Payment from './app/Payment.json'; import PaymentFor from './app/PaymentFor.json'; import PriceList from './app/PriceList.json'; @@ -128,6 +129,7 @@ export const appSchemas: Schema[] | SchemaStub[] = [ SalesInvoiceItem as SchemaStub, PurchaseInvoiceItem as SchemaStub, SalesQuoteItem as SchemaStub, + CouponCode as Schema, PriceList as Schema, PriceListItem as SchemaStub, From 1e3078432411ad574736ba035a03dc076b3ab69d Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 21 Aug 2024 11:11:43 +0530 Subject: [PATCH 04/20] feat: Coupon Codes --- .../AccountingSettings/AccountingSettings.ts | 3 ++ models/baseModels/CouponCode/CouponCode.ts | 32 +++++++++++++++++++ models/index.ts | 2 ++ models/types.ts | 1 + schemas/app/CouponCode.json | 12 +++---- 5 files changed, 44 insertions(+), 6 deletions(-) create mode 100644 models/baseModels/CouponCode/CouponCode.ts diff --git a/models/baseModels/AccountingSettings/AccountingSettings.ts b/models/baseModels/AccountingSettings/AccountingSettings.ts index a600f35b..86ce363c 100644 --- a/models/baseModels/AccountingSettings/AccountingSettings.ts +++ b/models/baseModels/AccountingSettings/AccountingSettings.ts @@ -16,6 +16,7 @@ export class AccountingSettings extends Doc { enableInventory?: boolean; enablePriceList?: boolean; enableLead?: boolean; + enableCouponCode?: boolean; enableFormCustomization?: boolean; enableInvoiceReturns?: boolean; enableLoyaltyProgram?: boolean; @@ -67,6 +68,8 @@ export class AccountingSettings extends Doc { gstin: () => this.fyo.singles.SystemSettings?.countryCode !== 'in', enablePricingRule: () => !this.fyo.singles.AccountingSettings?.enableDiscounting, + enableCouponCode: () => + !this.fyo.singles.AccountingSettings?.enablePricingRule, }; async change(ch: ChangeArg) { diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts new file mode 100644 index 00000000..aef79f37 --- /dev/null +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -0,0 +1,32 @@ +import { Doc } from 'fyo/model/doc'; +import { FormulaMap, ListViewSettings, ValidationMap } from 'fyo/model/types'; +import { Money } from 'pesa'; + +export class CouponCode extends Doc { + name?: string; + couponName?: string; + pricingRule?: string; + + validFrom?: Date; + validTo?: Date; + + minAmount?: Money; + maxAmount?: Money; + + formulas: FormulaMap = { + name: { + formula: () => { + return this.couponName?.replace(/\s+/g, '').toUpperCase().slice(0, 8); + }, + dependsOn: ['couponName'], + }, + }; + + validations: ValidationMap = {}; + + static getListViewSettings(): ListViewSettings { + return { + columns: ['name', 'couponName', 'pricingRule', 'maximumUse', 'used'], + }; + } +} diff --git a/models/index.ts b/models/index.ts index ace9901e..d7eb222a 100644 --- a/models/index.ts +++ b/models/index.ts @@ -13,6 +13,7 @@ import { LoyaltyProgram } from './baseModels/LoyaltyProgram/LoyaltyProgram'; import { LoyaltyPointEntry } from './baseModels/LoyaltyPointEntry/LoyaltyPointEntry'; import { CollectionRulesItems } from './baseModels/CollectionRulesItems/CollectionRulesItems'; import { Lead } from './baseModels/Lead/Lead'; +import { CouponCode } from './baseModels/CouponCode/CouponCode'; import { Payment } from './baseModels/Payment/Payment'; import { PaymentFor } from './baseModels/PaymentFor/PaymentFor'; import { PriceList } from './baseModels/PriceList/PriceList'; @@ -64,6 +65,7 @@ export const models = { LoyaltyProgram, LoyaltyPointEntry, CollectionRulesItems, + CouponCode, Payment, PaymentFor, PrintSettings, diff --git a/models/types.ts b/models/types.ts index 9b127d90..5392865a 100644 --- a/models/types.ts +++ b/models/types.ts @@ -22,6 +22,7 @@ export enum ModelNameEnum { LoyaltyProgram = 'LoyaltyProgram', LoyaltyPointEntry = 'LoyaltyPointEntry', CollectionRulesItems = 'CollectionRulesItems', + CouponCode = 'CouponCode', Payment = 'Payment', PaymentFor = 'PaymentFor', PriceList = 'PriceList', diff --git a/schemas/app/CouponCode.json b/schemas/app/CouponCode.json index 91ee1621..85ccc115 100644 --- a/schemas/app/CouponCode.json +++ b/schemas/app/CouponCode.json @@ -6,18 +6,18 @@ "fields": [ { "fieldname": "name", + "label": "Coupon Code", + "fieldtype": "Data", + "required": true + }, + { + "fieldname": "couponName", "label": "Name", "fieldtype": "Data", "required": true, "placeholder": "Coupon Name", "section": "Default" }, - { - "fieldname": "couponCode", - "label": "Coupon Code", - "fieldtype": "Data", - "required": true - }, { "fieldname": "pricingRule", "label": "Pricing Rule", From 11f9b0d46f1b662f58c4a6b7da272d61cc2840ad Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 26 Aug 2024 11:14:40 +0530 Subject: [PATCH 05/20] feat: coupon codes validations and filters --- models/baseModels/CouponCode/CouponCode.ts | 71 +++++++++++++++++++++- 1 file changed, 69 insertions(+), 2 deletions(-) diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts index aef79f37..40e00c12 100644 --- a/models/baseModels/CouponCode/CouponCode.ts +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -1,5 +1,13 @@ +import { DocValue } from 'fyo/core/types'; import { Doc } from 'fyo/model/doc'; -import { FormulaMap, ListViewSettings, ValidationMap } from 'fyo/model/types'; +import { + FiltersMap, + FormulaMap, + ListViewSettings, + ValidationMap, +} from 'fyo/model/types'; +import { ValidationError } from 'fyo/utils/errors'; +import { t } from 'fyo'; import { Money } from 'pesa'; export class CouponCode extends Doc { @@ -22,7 +30,66 @@ export class CouponCode extends Doc { }, }; - validations: ValidationMap = {}; + validations: ValidationMap = { + minAmount: (value: DocValue) => { + if (!value || !this.maxAmount) { + return; + } + + if ((value as Money).isZero() && this.maxAmount.isZero()) { + return; + } + + if ((value as Money).gte(this.maxAmount)) { + throw new ValidationError( + t`Minimum Amount should be less than the Maximum Amount.` + ); + } + }, + maxAmount: (value: DocValue) => { + if (!this.minAmount || !value) { + return; + } + + if (this.minAmount.isZero() && (value as Money).isZero()) { + return; + } + + if ((value as Money).lte(this.minAmount)) { + throw new ValidationError( + t`Maximum Amount should be greater than the Minimum Amount.` + ); + } + }, + validFrom: (value: DocValue) => { + if (!value || !this.validTo) { + return; + } + + if ((value as Date).toISOString() > this.validTo.toISOString()) { + throw new ValidationError( + t`Valid From Date should be less than Valid To Date.` + ); + } + }, + validTo: (value: DocValue) => { + if (!this.validFrom || !value) { + return; + } + + if ((value as Date).toISOString() < this.validFrom.toISOString()) { + throw new ValidationError( + t`Valid To Date should be greater than Valid From Date.` + ); + } + }, + }; + + static filters: FiltersMap = { + pricingRule: () => ({ + isCouponCodeBased: true, + }), + }; static getListViewSettings(): ListViewSettings { return { From f200f775cfecb2d28178a54bc1ba5204497579c6 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 11 Sep 2024 16:02:29 +0530 Subject: [PATCH 06/20] feat: update coupon schemas --- schemas/app/AppliedCouponCodes.json | 15 +++++++++++++++ schemas/app/PricingRule.json | 12 ++++++++++++ schemas/app/SalesInvoice.json | 7 +++++++ schemas/schemas.ts | 2 ++ 4 files changed, 36 insertions(+) create mode 100644 schemas/app/AppliedCouponCodes.json diff --git a/schemas/app/AppliedCouponCodes.json b/schemas/app/AppliedCouponCodes.json new file mode 100644 index 00000000..a5acdace --- /dev/null +++ b/schemas/app/AppliedCouponCodes.json @@ -0,0 +1,15 @@ +{ + "name": "AppliedCouponCodes", + "label": "Applied Coupon Codes", + "isChild": true, + "fields": [ + { + "fieldname": "coupons", + "label": "Coupons", + "fieldtype": "Link", + "target": "CouponCode" + } + ], + "tableFields": ["coupons"], + "quickEditFields": ["coupons"] +} diff --git a/schemas/app/PricingRule.json b/schemas/app/PricingRule.json index 9d474c77..cefc335f 100644 --- a/schemas/app/PricingRule.json +++ b/schemas/app/PricingRule.json @@ -54,6 +54,18 @@ } ] }, + { + "fieldname": "isCouponCodeBased", + "label": "Is Coupon Code Based", + "fieldtype": "Check", + "default": false + }, + { + "fieldname": "isMultiple", + "label": "Is Multiple", + "fieldtype": "Check", + "default": false + }, { "fieldname": "priceDiscountType", "label": "Price Discount Type", diff --git a/schemas/app/SalesInvoice.json b/schemas/app/SalesInvoice.json index b196f2b8..2086377f 100644 --- a/schemas/app/SalesInvoice.json +++ b/schemas/app/SalesInvoice.json @@ -24,6 +24,13 @@ "required": true, "section": "Default" }, + { + "fieldname": "coupons", + "label": "Coupons", + "fieldtype": "Table", + "target": "AppliedCouponCodes", + "section": "Coupons" + }, { "fieldname": "backReference", "label": "Back Reference", diff --git a/schemas/schemas.ts b/schemas/schemas.ts index d6dbd45d..f3d56c23 100644 --- a/schemas/schemas.ts +++ b/schemas/schemas.ts @@ -20,6 +20,7 @@ import LoyaltyProgram from './app/LoyaltyProgram.json'; import LoyaltyPointEntry from './app/LoyaltyPointEntry.json'; import CollectionRulesItems from './app/CollectionRulesItems.json'; import CouponCode from './app/CouponCode.json'; +import AppliedCouponCodes from './app/AppliedCouponCodes.json'; import Payment from './app/Payment.json'; import PaymentFor from './app/PaymentFor.json'; import PriceList from './app/PriceList.json'; @@ -130,6 +131,7 @@ export const appSchemas: Schema[] | SchemaStub[] = [ PurchaseInvoiceItem as SchemaStub, SalesQuoteItem as SchemaStub, CouponCode as Schema, + AppliedCouponCodes as Schema, PriceList as Schema, PriceListItem as SchemaStub, From a6f6269ee7f9b5a103d6582de9504cd864b46ca9 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 11 Sep 2024 16:06:54 +0530 Subject: [PATCH 07/20] feat: add coupon code support in sales invoice --- models/baseModels/Invoice/Invoice.ts | 85 +++++++++++++++++++- models/baseModels/PricingRule/PricingRule.ts | 2 + 2 files changed, 86 insertions(+), 1 deletion(-) diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts index 6cccedef..6bb72341 100644 --- a/models/baseModels/Invoice/Invoice.ts +++ b/models/baseModels/Invoice/Invoice.ts @@ -13,6 +13,7 @@ import { ValidationError } from 'fyo/utils/errors'; import { Transactional } from 'models/Transactional/Transactional'; import { addItem, + canApplyCouponCode, canApplyPricingRule, createLoyaltyPointEntry, filterPricingRules, @@ -42,6 +43,8 @@ import { PricingRule } from '../PricingRule/PricingRule'; import { ApplicablePricingRules } from './types'; import { PricingRuleDetail } from '../PricingRuleDetail/PricingRuleDetail'; import { LoyaltyProgram } from '../LoyaltyProgram/LoyaltyProgram'; +import { AppliedCouponCodes } from '../AppliedCouponCodes/AppliedCouponCodes'; +import { CouponCode } from '../CouponCode/CouponCode'; export type TaxDetail = { account: string; @@ -61,6 +64,7 @@ export abstract class Invoice extends Transactional { taxes?: TaxSummary[]; items?: InvoiceItem[]; + coupons?: AppliedCouponCodes[]; party?: string; account?: string; currency?: string; @@ -727,7 +731,7 @@ export abstract class Invoice extends Transactional { await this.appendPricingRuleDetail(pricingRule); return !!pricingRule; }, - dependsOn: ['items'], + dependsOn: ['items', 'coupons'], }, }; @@ -1298,6 +1302,41 @@ export abstract class Invoice extends Transactional { }) ).map((doc) => doc.parent) as string[]; + if (!pricingRuleDocNames.length) { + continue; + } + + if (this.coupons?.length) { + for (const coupon of this.coupons) { + const couponCodeDatas = await this.fyo.db.getAll( + ModelNameEnum.CouponCode, + { + fields: ['*'], + filters: { + name: coupon?.coupons as string, + isEnabled: true, + }, + } + ); + + const couponPricingRuleDocNames = couponCodeDatas + .map((doc) => doc.pricingRule) + .filter((val) => + pricingRuleDocNames.includes(val as string) + ) as string[]; + + const filtered = canApplyCouponCode( + couponCodeDatas[0] as CouponCode, + this.grandTotal as Money, + this.date as Date + ); + + if (filtered) { + pricingRuleDocNames.push(...couponPricingRuleDocNames); + } + } + } + const pricingRuleDocsForItem = (await this.fyo.db.getAll( ModelNameEnum.PricingRule, { @@ -1311,6 +1350,50 @@ export abstract class Invoice extends Transactional { } )) as PricingRule[]; + if (pricingRuleDocsForItem[0].isCouponCodeBased) { + if (!this.coupons?.length) { + continue; + } + + const data = await Promise.allSettled( + this.coupons?.map(async (val) => { + if (!val.coupons) { + return false; + } + + const [pricingRule] = ( + await this.fyo.db.getAll(ModelNameEnum.CouponCode, { + fields: ['pricingRule'], + filters: { + name: val?.coupons, + }, + }) + ).map((doc) => doc.pricingRule); + + if (!pricingRule) { + return false; + } + + if (pricingRuleDocsForItem[0].name === pricingRule) { + return pricingRule; + } + + return false; + }) + ); + + const fulfilledData = data + .filter( + (result): result is PromiseFulfilledResult => + result.status === 'fulfilled' + ) + .map((result) => result.value as string); + + if (!fulfilledData[0] && !fulfilledData.filter((val) => val).length) { + continue; + } + } + const filtered = filterPricingRules( pricingRuleDocsForItem, this.date as Date, diff --git a/models/baseModels/PricingRule/PricingRule.ts b/models/baseModels/PricingRule/PricingRule.ts index 31888d6f..88b6f207 100644 --- a/models/baseModels/PricingRule/PricingRule.ts +++ b/models/baseModels/PricingRule/PricingRule.ts @@ -23,6 +23,8 @@ export class PricingRule extends Doc { discountPercentage?: number; discountAmount?: Money; + isCouponCodeBased?: boolean; + forPriceList?: string; freeItem?: string; From 64a1551abbfca4388005de5b9bb1dcc21aa3251e Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 11 Sep 2024 16:08:41 +0530 Subject: [PATCH 08/20] feat: Applied Coupon Codes --- .../baseModels/AppliedCouponCodes/AppliedCouponCodes.ts | 8 ++++++++ models/index.ts | 2 ++ models/types.ts | 2 ++ 3 files changed, 12 insertions(+) create mode 100644 models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts diff --git a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts new file mode 100644 index 00000000..397753cb --- /dev/null +++ b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts @@ -0,0 +1,8 @@ +import { ValidationMap } from 'fyo/model/types'; +import { InvoiceItem } from '../InvoiceItem/InvoiceItem'; + +export class AppliedCouponCodes extends InvoiceItem { + coupons?: string; + + validations: ValidationMap = {}; +} diff --git a/models/index.ts b/models/index.ts index d7eb222a..ed4973b3 100644 --- a/models/index.ts +++ b/models/index.ts @@ -13,6 +13,7 @@ import { LoyaltyProgram } from './baseModels/LoyaltyProgram/LoyaltyProgram'; import { LoyaltyPointEntry } from './baseModels/LoyaltyPointEntry/LoyaltyPointEntry'; import { CollectionRulesItems } from './baseModels/CollectionRulesItems/CollectionRulesItems'; import { Lead } from './baseModels/Lead/Lead'; +import { AppliedCouponCodes } from './baseModels/AppliedCouponCodes/AppliedCouponCodes'; import { CouponCode } from './baseModels/CouponCode/CouponCode'; import { Payment } from './baseModels/Payment/Payment'; import { PaymentFor } from './baseModels/PaymentFor/PaymentFor'; @@ -77,6 +78,7 @@ export const models = { PurchaseInvoiceItem, SalesInvoice, SalesInvoiceItem, + AppliedCouponCodes, SalesQuote, SalesQuoteItem, SerialNumber, diff --git a/models/types.ts b/models/types.ts index 5392865a..2420b8fe 100644 --- a/models/types.ts +++ b/models/types.ts @@ -23,6 +23,8 @@ export enum ModelNameEnum { LoyaltyPointEntry = 'LoyaltyPointEntry', CollectionRulesItems = 'CollectionRulesItems', CouponCode = 'CouponCode', + + AppliedCouponCodes = 'AppliedCouponCodes', Payment = 'Payment', PaymentFor = 'PaymentFor', PriceList = 'PriceList', From b4b0994690d9d3ff17ccc8c34a8a188a0e5a5ade Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 11 Sep 2024 16:11:16 +0530 Subject: [PATCH 09/20] feat: validate applied coupon code --- .../AppliedCouponCodes/AppliedCouponCodes.ts | 71 ++++++++++++++++++- models/helpers.ts | 45 +++++++++++- 2 files changed, 113 insertions(+), 3 deletions(-) diff --git a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts index 397753cb..570d2a79 100644 --- a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts +++ b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts @@ -1,8 +1,77 @@ +import { DocValue } from 'fyo/core/types'; import { ValidationMap } from 'fyo/model/types'; +import { ValidationError } from 'fyo/utils/errors'; +import { ModelNameEnum } from 'models/types'; +import { Money } from 'pesa'; import { InvoiceItem } from '../InvoiceItem/InvoiceItem'; export class AppliedCouponCodes extends InvoiceItem { coupons?: string; - validations: ValidationMap = {}; + validations: ValidationMap = { + coupons: async (value: DocValue) => { + if (!value) { + return; + } + + const coupon = await this.fyo.db.getAll(ModelNameEnum.CouponCode, { + fields: [ + 'minAmount', + 'maxAmount', + 'pricingRule', + 'validFrom', + 'validTO', + ], + filters: { name: value as string }, + }); + + const couponExist = this.parentdoc?.coupons?.some( + (coupon) => coupon?.coupons === value + ); + + if (couponExist) { + throw new ValidationError( + this.fyo.t`${value as string} already applied.` + ); + } + + if ( + (coupon[0].minAmount as Money).gte( + this.parentdoc?.grandTotal as Money + ) && + !(coupon[0].minAmount as Money).isZero() + ) { + throw new ValidationError( + this.fyo.t`The Grand Total must exceed ${ + (coupon[0].minAmount as Money).float + } to apply the coupon ${value as string}.` + ); + } + + if ( + (coupon[0].maxAmount as Money).lte( + this.parentdoc?.grandTotal as Money + ) && + !(coupon[0].maxAmount as Money).isZero() + ) { + throw new ValidationError( + this.fyo.t`The Grand Total must be less than ${ + (coupon[0].maxAmount as Money).float + } to apply this coupon.` + ); + } + + if ((coupon[0].validFrom as Date) > (this.parentdoc?.date as Date)) { + throw new ValidationError( + this.fyo.t`Valid From Date should be less than Valid To Date.` + ); + } + + if ((coupon[0].validTo as Date) < (this.parentdoc?.date as Date)) { + throw new ValidationError( + this.fyo.t`Valid To Date should be greater than Valid From Date.` + ); + } + }, + }; } diff --git a/models/helpers.ts b/models/helpers.ts index 167096d3..9f094369 100644 --- a/models/helpers.ts +++ b/models/helpers.ts @@ -28,6 +28,7 @@ import { LoyaltyProgram } from './baseModels/LoyaltyProgram/LoyaltyProgram'; import { CollectionRulesItems } from './baseModels/CollectionRulesItems/CollectionRulesItems'; import { isPesa } from 'fyo/utils'; import { Party } from './baseModels/Party/Party'; +import { CouponCode } from './baseModels/CouponCode/CouponCode'; export function getQuoteActions( fyo: Fyo, @@ -584,8 +585,6 @@ export async function getExchangeRate({ }; exchangeRate = data.rates[toCurrency]; } catch (error) { - // eslint-disable-next-line no-console - console.error(error); exchangeRate ??= 1; } @@ -916,6 +915,7 @@ export function canApplyPricingRule( ) { return false; } + if ( pricingRuleDoc.validTo && new Date(sinvDate.setHours(0, 0, 0, 0)).toISOString() > @@ -925,6 +925,47 @@ export function canApplyPricingRule( } return true; } + +export function canApplyCouponCode( + couponCodeData: CouponCode, + amount: Money, + sinvDate: Date +): boolean { + // Filter by Amount + if ( + !couponCodeData.minAmount?.isZero() && + amount.lte(couponCodeData.minAmount as Money) + ) { + return false; + } + + if ( + !couponCodeData.maxAmount?.isZero() && + amount.gte(couponCodeData.maxAmount as Money) + ) { + return false; + } + + // Filter by Validity + if ( + couponCodeData.validFrom && + new Date(sinvDate.setHours(0, 0, 0, 0)).toISOString() < + couponCodeData.validFrom.toISOString() + ) { + return false; + } + + if ( + couponCodeData.validTo && + new Date(sinvDate.setHours(0, 0, 0, 0)).toISOString() > + couponCodeData.validTo.toISOString() + ) { + return false; + } + + return true; +} + export function getPricingRulesConflicts( pricingRules: PricingRule[] ): undefined | boolean { From a61f734a26567fa2e1e81494bc9e6e77501febe1 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Tue, 17 Sep 2024 12:23:49 +0530 Subject: [PATCH 10/20] feat: remove unused coupon codes after saving SINV --- .../AppliedCouponCodes/AppliedCouponCodes.ts | 17 ++++- models/baseModels/CouponCode/CouponCode.ts | 16 +++++ models/baseModels/Invoice/Invoice.ts | 63 ++++++++++++++++--- models/baseModels/Invoice/types.ts | 5 ++ .../baseModels/tests/testPricingRule.spec.ts | 6 +- models/helpers.ts | 36 +++++++++++ 6 files changed, 127 insertions(+), 16 deletions(-) diff --git a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts index 570d2a79..318ec9f4 100644 --- a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts +++ b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts @@ -4,6 +4,8 @@ import { ValidationError } from 'fyo/utils/errors'; import { ModelNameEnum } from 'models/types'; import { Money } from 'pesa'; import { InvoiceItem } from '../InvoiceItem/InvoiceItem'; +import { getApplicableCouponCodesName } from 'models/helpers'; +import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; export class AppliedCouponCodes extends InvoiceItem { coupons?: string; @@ -20,11 +22,24 @@ export class AppliedCouponCodes extends InvoiceItem { 'maxAmount', 'pricingRule', 'validFrom', - 'validTO', + 'validTo', ], filters: { name: value as string }, }); + const applicableCouponCodesNames = await getApplicableCouponCodesName( + value as string, + this.parentdoc as SalesInvoice + ); + + if (!applicableCouponCodesNames?.length) { + throw new ValidationError( + this.fyo.t`Coupon ${ + value as string + } is not applicable for applied items.` + ); + } + const couponExist = this.parentdoc?.coupons?.some( (coupon) => coupon?.coupons === value ); diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts index 40e00c12..d7a96957 100644 --- a/models/baseModels/CouponCode/CouponCode.ts +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -9,6 +9,8 @@ import { import { ValidationError } from 'fyo/utils/errors'; import { t } from 'fyo'; import { Money } from 'pesa'; +import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; +import { ApplicableCouponCodes } from '../Invoice/types'; export class CouponCode extends Doc { name?: string; @@ -21,6 +23,20 @@ export class CouponCode extends Doc { minAmount?: Money; maxAmount?: Money; + removeUnusedCoupons(coupons: ApplicableCouponCodes[], sinvDoc: SalesInvoice) { + if (!coupons.length) { + sinvDoc.coupons = []; + + return; + } + + sinvDoc.coupons = sinvDoc.coupons!.filter((coupon) => { + return coupons.find((c: ApplicableCouponCodes) => + coupon?.coupons?.includes(c?.coupon) + ); + }); + } + formulas: FormulaMap = { name: { formula: () => { diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts index 6bb72341..c1862544 100644 --- a/models/baseModels/Invoice/Invoice.ts +++ b/models/baseModels/Invoice/Invoice.ts @@ -18,6 +18,7 @@ import { createLoyaltyPointEntry, filterPricingRules, getAddedLPWithGrandTotal, + getApplicableCouponCodesName, getExchangeRate, getNumberSeries, getPricingRulesConflicts, @@ -40,11 +41,14 @@ import { TaxSummary } from '../TaxSummary/TaxSummary'; import { ReturnDocItem } from 'models/inventory/types'; import { AccountFieldEnum, PaymentTypeEnum } from '../Payment/types'; import { PricingRule } from '../PricingRule/PricingRule'; -import { ApplicablePricingRules } from './types'; +import { ApplicableCouponCodes, ApplicablePricingRules } from './types'; import { PricingRuleDetail } from '../PricingRuleDetail/PricingRuleDetail'; import { LoyaltyProgram } from '../LoyaltyProgram/LoyaltyProgram'; import { AppliedCouponCodes } from '../AppliedCouponCodes/AppliedCouponCodes'; import { CouponCode } from '../CouponCode/CouponCode'; +import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; +import { SalesInvoiceItem } from '../SalesInvoiceItem/SalesInvoiceItem'; +import { PriceListItem } from '../PriceList/PriceListItem'; export type TaxDetail = { account: string; @@ -1039,6 +1043,31 @@ export abstract class Invoice extends Transactional { } else { this.clearFreeItems(); } + + if (!this.coupons?.length) { + return; + } + + const applicableCouponCodes = await Promise.all( + this.coupons?.map(async (coupon) => { + return await getApplicableCouponCodesName( + coupon.coupons as string, + this as SalesInvoice + ); + }) + ); + + const flattedApplicableCouponCodes = applicableCouponCodes?.flat(); + + const couponCodeDoc = (await this.fyo.doc.getDoc( + ModelNameEnum.CouponCode, + this.coupons[0].coupons + )) as CouponCode; + + couponCodeDoc.removeUnusedCoupons( + flattedApplicableCouponCodes as ApplicableCouponCodes[], + this as SalesInvoice + ); } async beforeCancel(): Promise { @@ -1263,6 +1292,21 @@ export abstract class Invoice extends Transactional { } } + async getPricingRuleDocNames( + item: SalesInvoiceItem, + sinvDoc: SalesInvoice + ): Promise { + const docs = (await sinvDoc.fyo.db.getAll(ModelNameEnum.PricingRuleItem, { + fields: ['parent'], + filters: { + item: item.item as string, + unit: item.unit as string, + }, + })) as PriceListItem[]; + + return docs.map((doc) => doc.parent) as string[]; + } + async getPricingRule(): Promise { if (!this.isSales || !this.items) { return; @@ -1292,15 +1336,10 @@ export abstract class Invoice extends Transactional { continue; } - const pricingRuleDocNames = ( - await this.fyo.db.getAll(ModelNameEnum.PricingRuleItem, { - fields: ['parent'], - filters: { - item: item.item as string, - unit: item.unit as string, - }, - }) - ).map((doc) => doc.parent) as string[]; + const pricingRuleDocNames = await this.getPricingRuleDocNames( + item, + this as SalesInvoice + ); if (!pricingRuleDocNames.length) { continue; @@ -1325,6 +1364,10 @@ export abstract class Invoice extends Transactional { pricingRuleDocNames.includes(val as string) ) as string[]; + if (!couponPricingRuleDocNames.length) { + continue; + } + const filtered = canApplyCouponCode( couponCodeDatas[0] as CouponCode, this.grandTotal as Money, diff --git a/models/baseModels/Invoice/types.ts b/models/baseModels/Invoice/types.ts index 50362b21..80e1d8f8 100644 --- a/models/baseModels/Invoice/types.ts +++ b/models/baseModels/Invoice/types.ts @@ -4,3 +4,8 @@ export interface ApplicablePricingRules { applyOnItem: string; pricingRule: PricingRule; } + +export interface ApplicableCouponCodes { + pricingRule: string; + coupon: string; +} diff --git a/models/baseModels/tests/testPricingRule.spec.ts b/models/baseModels/tests/testPricingRule.spec.ts index dfffab3c..515c6e99 100644 --- a/models/baseModels/tests/testPricingRule.spec.ts +++ b/models/baseModels/tests/testPricingRule.spec.ts @@ -2,10 +2,6 @@ import test from 'tape'; import { closeTestFyo, getTestFyo, setupTestFyo } from 'tests/helpers'; import { ModelNameEnum } from 'models/types'; import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; -import { - assertDoesNotThrow, - assertThrows, -} from 'backend/database/tests/helpers'; import { getItem } from 'models/inventory/tests/helpers'; import { PricingRule } from '../PricingRule/PricingRule'; @@ -104,7 +100,7 @@ test('disabled pricing rule is not applied', async (t) => { await sinv.append('items', { item: itemMap.Jacket.name, quantity: 5 }); await sinv.runFormulas(); - t.equal(sinv.pricingRuleDetail?.length, undefined); + t.equal(sinv.pricingRuleDetail?.length, 0); }); test('pricing rule is applied when filtered by min and max qty', async (t) => { diff --git a/models/helpers.ts b/models/helpers.ts index 9f094369..81813b3d 100644 --- a/models/helpers.ts +++ b/models/helpers.ts @@ -29,6 +29,7 @@ import { CollectionRulesItems } from './baseModels/CollectionRulesItems/Collecti import { isPesa } from 'fyo/utils'; import { Party } from './baseModels/Party/Party'; import { CouponCode } from './baseModels/CouponCode/CouponCode'; +import { SalesInvoice } from './baseModels/SalesInvoice/SalesInvoice'; export function getQuoteActions( fyo: Fyo, @@ -966,6 +967,41 @@ export function canApplyCouponCode( return true; } +export async function getApplicableCouponCodesName( + couponName: string, + sinvDoc: SalesInvoice +) { + const couponCodeDatas = (await sinvDoc.fyo.db.getAll( + ModelNameEnum.CouponCode, + { + fields: ['*'], + filters: { + name: couponName, + isEnabled: true, + }, + } + )) as CouponCode[]; + + if (!couponCodeDatas || couponCodeDatas.length === 0) { + return []; + } + + const applicablePricingRules = await getPricingRule(sinvDoc); + + if (!applicablePricingRules?.length) { + return []; + } + + return applicablePricingRules + ?.filter( + (rule) => rule?.pricingRule?.name === couponCodeDatas[0].pricingRule + ) + .map((rule) => ({ + pricingRule: rule.pricingRule.name, + coupon: couponCodeDatas[0].name, + })); +} + export function getPricingRulesConflicts( pricingRules: PricingRule[] ): undefined | boolean { From 8697e7ebf9644f3eb79a45a96470a3f4099285e7 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Tue, 17 Sep 2024 12:49:10 +0530 Subject: [PATCH 11/20] feat: validate coupon code --- models/baseModels/CouponCode/CouponCode.ts | 72 +++++++++++++++++++--- 1 file changed, 62 insertions(+), 10 deletions(-) diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts index d7a96957..8b4c915b 100644 --- a/models/baseModels/CouponCode/CouponCode.ts +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -9,6 +9,7 @@ import { import { ValidationError } from 'fyo/utils/errors'; import { t } from 'fyo'; import { Money } from 'pesa'; +import { ModelNameEnum } from 'models/types'; import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; import { ApplicableCouponCodes } from '../Invoice/types'; @@ -46,54 +47,105 @@ export class CouponCode extends Doc { }, }; + async pricingRuleData() { + return await this.fyo.db.getAll(ModelNameEnum.PricingRule, { + fields: ['minAmount', 'maxAmount', 'validFrom', 'validFrom'], + filters: { + name: this.pricingRule as string, + }, + }); + } + validations: ValidationMap = { - minAmount: (value: DocValue) => { - if (!value || !this.maxAmount) { + minAmount: async (value: DocValue) => { + if (!value || !this.maxAmount || !this.pricingRule) { return; } + const [pricingRuleData] = await this.pricingRuleData(); + const { minAmount } = pricingRuleData; + if ((value as Money).isZero() && this.maxAmount.isZero()) { return; } + if ((value as Money).lt(minAmount as Money)) { + throw new ValidationError( + t`Minimum Amount should be greather than the Pricing Rule's Minimum Amount.` + ); + } + if ((value as Money).gte(this.maxAmount)) { throw new ValidationError( t`Minimum Amount should be less than the Maximum Amount.` ); } }, - maxAmount: (value: DocValue) => { - if (!this.minAmount || !value) { + maxAmount: async (value: DocValue) => { + if (!this.minAmount || !value || !this.pricingRule) { return; } + const [pricingRuleData] = await this.pricingRuleData(); + const { maxAmount } = pricingRuleData; + if (this.minAmount.isZero() && (value as Money).isZero()) { return; } + if ((value as Money).gt(maxAmount as Money)) { + throw new ValidationError( + t`Maximum Amount should be lesser than Pricing Rule's Maximum Amount` + ); + } + if ((value as Money).lte(this.minAmount)) { throw new ValidationError( t`Maximum Amount should be greater than the Minimum Amount.` ); } }, - validFrom: (value: DocValue) => { - if (!value || !this.validTo) { + validFrom: async (value: DocValue) => { + if (!value || !this.validTo || !this.pricingRule) { return; } - if ((value as Date).toISOString() > this.validTo.toISOString()) { + const [pricingRuleData] = await this.pricingRuleData(); + const { validFrom } = pricingRuleData; + + if ( + validFrom && + (value as Date).toISOString() < (validFrom as Date).toISOString() + ) { throw new ValidationError( t`Valid From Date should be less than Valid To Date.` ); } + + if ((value as Date).toISOString() >= this.validTo.toISOString()) { + throw new ValidationError( + t`Valid From Date should be greather than Pricing Rule's Valid From Date.` + ); + } }, - validTo: (value: DocValue) => { - if (!this.validFrom || !value) { + validTo: async (value: DocValue) => { + if (!this.validFrom || !value || !this.pricingRule) { return; } - if ((value as Date).toISOString() < this.validFrom.toISOString()) { + const [pricingRuleData] = await this.pricingRuleData(); + const { validTo } = pricingRuleData; + + if ( + validTo && + (value as Date).toISOString() > (validTo as Date).toISOString() + ) { + throw new ValidationError( + t`Valid To Date should be lesser than Pricing Rule's Valid To Date.` + ); + } + + if ((value as Date).toISOString() <= this.validFrom.toISOString()) { throw new ValidationError( t`Valid To Date should be greater than Valid From Date.` ); From 41a67cc864145ddc2ee5bc4147a67e67e32d609a Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 18 Sep 2024 10:41:15 +0530 Subject: [PATCH 12/20] feat: prevent coupon usage if maximum limit is exceeded --- .../AppliedCouponCodes/AppliedCouponCodes.ts | 8 ++++++++ models/baseModels/Invoice/Invoice.ts | 15 +++++++++++++++ 2 files changed, 23 insertions(+) diff --git a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts index 318ec9f4..9a4b7490 100644 --- a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts +++ b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts @@ -23,10 +23,18 @@ export class AppliedCouponCodes extends InvoiceItem { 'pricingRule', 'validFrom', 'validTo', + 'maximumUse', + 'used', ], filters: { name: value as string }, }); + if ((coupon[0]?.maximumUse as number) <= (coupon[0]?.used as number)) { + throw new ValidationError( + 'Coupon code has been used maximum number of times' + ); + } + const applicableCouponCodesNames = await getApplicableCouponCodesName( value as string, this.parentdoc as SalesInvoice diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts index c1862544..7452696e 100644 --- a/models/baseModels/Invoice/Invoice.ts +++ b/models/baseModels/Invoice/Invoice.ts @@ -236,6 +236,10 @@ export abstract class Invoice extends Transactional { await this._updateIsItemsReturned(); await this._createLoyaltyPointEntry(); + + if (this.schemaName === ModelNameEnum.SalesInvoice) { + this.updateUsedCountOfCoupons(); + } } async afterCancel() { @@ -553,6 +557,17 @@ export abstract class Invoice extends Transactional { return newReturnDoc; } + updateUsedCountOfCoupons() { + this.coupons?.map(async (coupon) => { + const couponDoc = await this.fyo.doc.getDoc( + ModelNameEnum.CouponCode, + coupon.coupons + ); + + await couponDoc.setAndSync({ used: (couponDoc.used as number) + 1 }); + }); + } + async _updateIsItemsReturned() { if (!this.isReturn || !this.returnAgainst || this.isQuote) { return; From 02adcb4610edb9971e64aeb663c5a377b65ead5e Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 18 Sep 2024 11:09:44 +0530 Subject: [PATCH 13/20] fix: formatted --- models/helpers.ts | 2 +- schemas/app/AccountingSettings.json | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/models/helpers.ts b/models/helpers.ts index 81813b3d..80f8ab82 100644 --- a/models/helpers.ts +++ b/models/helpers.ts @@ -916,7 +916,7 @@ export function canApplyPricingRule( ) { return false; } - + if ( pricingRuleDoc.validTo && new Date(sinvDate.setHours(0, 0, 0, 0)).toISOString() > diff --git a/schemas/app/AccountingSettings.json b/schemas/app/AccountingSettings.json index 377bbde2..4493af67 100644 --- a/schemas/app/AccountingSettings.json +++ b/schemas/app/AccountingSettings.json @@ -122,7 +122,6 @@ "section": "Features" }, { - "fieldname": "enableCouponCode", "label": "Enable Coupon Code", "fieldtype": "Check", From 53289ef55a84cef006fc226872a35ca8ad2a30b3 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 23 Sep 2024 10:15:52 +0530 Subject: [PATCH 14/20] fix: allow flexible min/max amount for coupon codes when pricing rule limits are zero --- models/baseModels/CouponCode/CouponCode.ts | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts index 8b4c915b..71779263 100644 --- a/models/baseModels/CouponCode/CouponCode.ts +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -63,6 +63,14 @@ export class CouponCode extends Doc { } const [pricingRuleData] = await this.pricingRuleData(); + + if ( + (pricingRuleData?.minAmount as Money).isZero() && + (pricingRuleData.maxAmount as Money).isZero() + ) { + return; + } + const { minAmount } = pricingRuleData; if ((value as Money).isZero() && this.maxAmount.isZero()) { @@ -87,6 +95,14 @@ export class CouponCode extends Doc { } const [pricingRuleData] = await this.pricingRuleData(); + + if ( + (pricingRuleData?.minAmount as Money).isZero() && + (pricingRuleData.maxAmount as Money).isZero() + ) { + return; + } + const { maxAmount } = pricingRuleData; if (this.minAmount.isZero() && (value as Money).isZero()) { From 71b195491527411469d4a7cdb901f308452dd7ff Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 25 Sep 2024 09:17:39 +0530 Subject: [PATCH 15/20] fix: validations --- .../AppliedCouponCodes/AppliedCouponCodes.ts | 7 +++++++ models/baseModels/CouponCode/CouponCode.ts | 17 +++++++++++++---- 2 files changed, 20 insertions(+), 4 deletions(-) diff --git a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts index 9a4b7490..f7158c1b 100644 --- a/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts +++ b/models/baseModels/AppliedCouponCodes/AppliedCouponCodes.ts @@ -25,10 +25,17 @@ export class AppliedCouponCodes extends InvoiceItem { 'validTo', 'maximumUse', 'used', + 'isEnabled', ], filters: { name: value as string }, }); + if (!coupon[0].isEnabled) { + throw new ValidationError( + 'Coupon code cannot be applied as it is not enabled' + ); + } + if ((coupon[0]?.maximumUse as number) <= (coupon[0]?.used as number)) { throw new ValidationError( 'Coupon code has been used maximum number of times' diff --git a/models/baseModels/CouponCode/CouponCode.ts b/models/baseModels/CouponCode/CouponCode.ts index 71779263..7f26efdc 100644 --- a/models/baseModels/CouponCode/CouponCode.ts +++ b/models/baseModels/CouponCode/CouponCode.ts @@ -49,7 +49,7 @@ export class CouponCode extends Doc { async pricingRuleData() { return await this.fyo.db.getAll(ModelNameEnum.PricingRule, { - fields: ['minAmount', 'maxAmount', 'validFrom', 'validFrom'], + fields: ['minAmount', 'maxAmount', 'validFrom', 'validTo'], filters: { name: this.pricingRule as string, }, @@ -127,20 +127,24 @@ export class CouponCode extends Doc { } const [pricingRuleData] = await this.pricingRuleData(); - const { validFrom } = pricingRuleData; + if (!pricingRuleData?.validFrom && !pricingRuleData.validTo) { + return; + } + + const { validFrom } = pricingRuleData; if ( validFrom && (value as Date).toISOString() < (validFrom as Date).toISOString() ) { throw new ValidationError( - t`Valid From Date should be less than Valid To Date.` + t`Valid From Date should be greather than Pricing Rule's Valid From Date.` ); } if ((value as Date).toISOString() >= this.validTo.toISOString()) { throw new ValidationError( - t`Valid From Date should be greather than Pricing Rule's Valid From Date.` + t`Valid From Date should be less than Valid To Date.` ); } }, @@ -150,6 +154,11 @@ export class CouponCode extends Doc { } const [pricingRuleData] = await this.pricingRuleData(); + + if (!pricingRuleData?.validFrom && !pricingRuleData.validTo) { + return; + } + const { validTo } = pricingRuleData; if ( From 5f86cc8ff265c16b2c586ee5140ef29d99dba706 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Wed, 25 Sep 2024 09:18:08 +0530 Subject: [PATCH 16/20] test: coupon code --- .../baseModels/tests/testCouponCodes.spec.ts | 298 ++++++++++++++++++ 1 file changed, 298 insertions(+) create mode 100644 models/baseModels/tests/testCouponCodes.spec.ts diff --git a/models/baseModels/tests/testCouponCodes.spec.ts b/models/baseModels/tests/testCouponCodes.spec.ts new file mode 100644 index 00000000..a7d26956 --- /dev/null +++ b/models/baseModels/tests/testCouponCodes.spec.ts @@ -0,0 +1,298 @@ +import test from 'tape'; +import { closeTestFyo, getTestFyo, setupTestFyo } from 'tests/helpers'; +import { ModelNameEnum } from 'models/types'; +import { getItem } from 'models/inventory/tests/helpers'; +import { SalesInvoice } from '../SalesInvoice/SalesInvoice'; +import { PricingRule } from '../PricingRule/PricingRule'; +import { assertThrows } from 'backend/database/tests/helpers'; + +const fyo = getTestFyo(); +setupTestFyo(fyo, __filename); + +const itemMap = { + Jacket: { + name: 'Jacket', + rate: 1000, + unit: 'Unit', + }, + Cap: { + name: 'Cap', + rate: 100, + unit: 'Unit', + }, +}; + +const partyMap = { + partyOne: { + name: 'Daisy', + email: 'daisy@alien.com', + account: 'Debtors', + }, +}; + +const pricingRuleMap = [ + { + name: 'PRLE-1001', + title: 'JKT PDR Offer', + isCouponCodeBased: true, + appliedItems: [{ item: itemMap.Jacket.name }], + discountType: 'Price Discount', + priceDiscountType: 'rate', + discountRate: 800, + minQuantity: 4, + maxQuantity: 6, + minAmount: fyo.pesa(4000), + maxAmount: fyo.pesa(6000), + validFrom: '2024-02-01', + validTo: '2024-02-29', + priority: '1', + }, + { + name: 'PRLE-1002', + title: 'CAP PDR Offer', + appliedItems: [{ item: itemMap.Cap.name }], + discountType: 'Product Discount', + freeItem: 'Cap', + freeItemQuantity: 1, + freeItemUnit: 'Unit', + freeItemRate: 0, + minQuantity: 4, + maxQuantity: 6, + minAmount: 200, + maxAmount: 1000, + validFrom: '2024-02-01', + validTo: '2024-02-29', + priority: '1', + }, +]; +const couponCodesMap = [ + { + name: 'COUPON1', + isEnabled: true, + couponName: 'coupon1', + pricingRule: pricingRuleMap[0].name, + maximumUse: 5, + used: 0, + minAmount: fyo.pesa(4000), + maxAmount: fyo.pesa(6000), + validFrom: '2024-02-01', + validTo: '2024-02-29', + }, + { + name: 'COUPON2', + couponName: 'coupon2', + pricingRule: pricingRuleMap[1].name, + maximumUse: 1, + used: 0, + minAmount: 200, + maxAmount: 1000, + validFrom: '2024-02-01', + validTo: '2024-02-29', + }, +]; + +test(' Coupon Codes: create dummy item, party, pricing rules, coupon codes', async (t) => { + // Create Items + for (const { name, rate } of Object.values(itemMap)) { + const item = getItem(name, rate, false); + + await fyo.doc.getNewDoc(ModelNameEnum.Item, item).sync(); + + t.ok(await fyo.db.exists(ModelNameEnum.Item, name), `Item: ${name} exists`); + } + + // Create Party + await fyo.doc.getNewDoc(ModelNameEnum.Party, partyMap.partyOne).sync(); + + t.ok( + await fyo.db.exists(ModelNameEnum.Party, partyMap.partyOne.name), + `Party: ${partyMap.partyOne.name} exists` + ); + + // Create Pricing Rules + for (const pricingRule of Object.values(pricingRuleMap)) { + await fyo.doc.getNewDoc(ModelNameEnum.PricingRule, pricingRule).sync(); + + t.ok( + await fyo.db.exists(ModelNameEnum.PricingRule, pricingRule.name), + `Pricing Rule: ${pricingRule.name} exists` + ); + } + + await fyo.singles.AccountingSettings?.set('enablePricingRule', true); + + t.ok(fyo.singles.AccountingSettings?.enablePricingRule); + + // Create Coupon Codes + for (const couponCodes of Object.values(couponCodesMap)) { + await fyo.doc.getNewDoc(ModelNameEnum.CouponCode, couponCodes).sync(); + + t.ok( + await fyo.db.exists(ModelNameEnum.CouponCode, couponCodes.name), + `Coupoon Code: ${couponCodes.name} exists` + ); + } + + await fyo.singles.AccountingSettings?.set('enableCouponCode', true); + + t.ok(fyo.singles.AccountingSettings?.enableCouponCode); +}); + +test('disabled coupon codes is not applied', async (t) => { + const sinv = fyo.doc.getNewDoc(ModelNameEnum.SalesInvoice, { + date: '2024-01-20T18:30:00.000Z', + party: partyMap.partyOne.name, + account: partyMap.partyOne.account, + }) as SalesInvoice; + + await sinv.append('items', { + item: itemMap.Jacket.name, + quantity: 5, + rate: itemMap.Jacket.rate, + }); + + await sinv.append('coupons', { + coupons: couponCodesMap[0].name, + }); + + await sinv.runFormulas(); + + t.equal(sinv.pricingRuleDetail?.length, undefined); +}); + +test('Coupon code not created: coupons min amount must be lesser than coupons max.', async (t) => { + couponCodesMap[0].minAmount = fyo.pesa(7000); + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + 'Minimum Amount should be less than the Maximum Amount' + ); +}); + +test('Coupon code not created: pricing rules max amount is lower than the coupons min.', async (t) => { + couponCodesMap[0].minAmount = fyo.pesa(3000); + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + "Minimum Amount should be greather than the Pricing Rule's Minimum Amount" + ); +}); + +test('coupon code not created: pricing rules max amount is lower than the coupons max.', async (t) => { + couponCodesMap[0].maxAmount = fyo.pesa(7000); + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + "Maximum Amount should be lesser than Pricing Rule's Maximum Amount" + ); +}); + +test("coupon code is not applied when coupon's validfrom date < coupon's validTo date", async (t) => { + couponCodesMap[0].minAmount = fyo.pesa(4000); + couponCodesMap[0].maxAmount = fyo.pesa(6000); + couponCodesMap[0].validTo = '2024-01-10'; + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + 'Valid From Date should be less than Valid To Date' + ); +}); + +test("coupon code is not applied when coupon's validFrom date < pricing rule's validFrom date", async (t) => { + couponCodesMap[0].validFrom = '2024-01-01'; + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + "Valid From Date should be greather than Pricing Rule's Valid From Date" + ); +}); + +test("coupon code is not applied when coupon's validTo date > pricing rule's validTo date", async (t) => { + couponCodesMap[0].validFrom = '2024-02-01'; + couponCodesMap[0].validTo = '2024-03-01'; + + const ccodeDoc = fyo.doc.getNewDoc( + ModelNameEnum.CouponCode, + couponCodesMap[0] + ) as PricingRule; + + await assertThrows( + async () => await ccodeDoc.sync(), + "Valid To Date should be lesser than Pricing Rule's Valid To Date" + ); +}); + +test('apply coupon code', async (t) => { + couponCodesMap[0].name = 'COUPON1'; + couponCodesMap[0].validTo = '2024-02-29'; + + const sinv = fyo.doc.getNewDoc(ModelNameEnum.SalesInvoice, { + date: '2024-02-10', + party: partyMap.partyOne.name, + }) as SalesInvoice; + + await sinv.append('items', { + item: itemMap.Jacket.name, + quantity: 5, + rate: itemMap.Jacket.rate, + }); + + await sinv.append('coupons', { coupons: couponCodesMap[0].name }); + await sinv.runFormulas(); + + t.equal(sinv.pricingRuleDetail?.length, 1); + + t.equal( + sinv.pricingRuleDetail![0].referenceName, + pricingRuleMap[0].name, + 'Pricing Rule is applied' + ); +}); + +test('Coupon not applied: incorrect items added.', async (t) => { + const sinv = fyo.doc.getNewDoc(ModelNameEnum.SalesInvoice, { + date: '2024-02-10', + party: partyMap.partyOne.name, + account: partyMap.partyOne.account, + }) as SalesInvoice; + + await sinv.append('items', { + item: itemMap.Cap.name, + quantity: 5, + rate: itemMap.Cap.rate, + }); + + await sinv.append('coupons', { coupons: couponCodesMap[0].name }); + + await sinv.runFormulas(); + await sinv.sync(); + + t.equal(sinv.coupons?.length, 0, 'coupon code is not applied'); +}); + +closeTestFyo(fyo, __filename); From 301a04a63851f2ea8081967b41c560bbd796e90b Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 30 Sep 2024 14:57:09 +0530 Subject: [PATCH 17/20] feat: reset used coupon count after canceling or returning an invoice --- models/baseModels/Invoice/Invoice.ts | 25 ++++++++++++++++++++++--- models/helpers.ts | 4 ++++ 2 files changed, 26 insertions(+), 3 deletions(-) diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts index 7452696e..820f854f 100644 --- a/models/baseModels/Invoice/Invoice.ts +++ b/models/baseModels/Invoice/Invoice.ts @@ -195,6 +195,9 @@ export abstract class Invoice extends Transactional { if (this.isReturn) { await this._removeLoyaltyPointEntry(); + await this.reduceUsedCountOfCoupons(); + + return; } if (this.isQuote) { @@ -248,12 +251,14 @@ export abstract class Invoice extends Transactional { await this._updatePartyOutStanding(); await this._updateIsItemsReturned(); await this._removeLoyaltyPointEntry(); + await this._reduceUsedCountOfCoupon(); + } + + async _reduceUsedCountOfCoupon() { + await this.reduceUsedCountOfCoupons(); } async _removeLoyaltyPointEntry() { - if (!this.loyaltyProgram) { - return; - } await removeLoyaltyPoint(this); } @@ -567,6 +572,20 @@ export abstract class Invoice extends Transactional { await couponDoc.setAndSync({ used: (couponDoc.used as number) + 1 }); }); } + async reduceUsedCountOfCoupons() { + if (!this.coupons?.length) { + return; + } + + this.coupons?.map(async (coupon) => { + const couponDoc = await this.fyo.doc.getDoc( + ModelNameEnum.CouponCode, + coupon.coupons + ); + + await couponDoc.setAndSync({ used: (couponDoc.used as number) - 1 }); + }); + } async _updateIsItemsReturned() { if (!this.isReturn || !this.returnAgainst || this.isQuote) { diff --git a/models/helpers.ts b/models/helpers.ts index 80f8ab82..141ecc9c 100644 --- a/models/helpers.ts +++ b/models/helpers.ts @@ -762,6 +762,10 @@ export function getLoyaltyProgramTier( } export async function removeLoyaltyPoint(doc: Doc) { + if (!doc.loyaltyProgram) { + return; + } + const data = (await doc.fyo.db.getAll(ModelNameEnum.LoyaltyPointEntry, { fields: ['name', 'loyaltyPoints', 'expiryDate'], filters: { From ea5293bb8b224c578b10d01e7e0c124ee7e0a769 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 30 Sep 2024 15:08:00 +0530 Subject: [PATCH 18/20] fix: resolve linting issues --- models/baseModels/Invoice/Invoice.ts | 10 +++------- 1 file changed, 3 insertions(+), 7 deletions(-) diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts index 820f854f..28cb2e2f 100644 --- a/models/baseModels/Invoice/Invoice.ts +++ b/models/baseModels/Invoice/Invoice.ts @@ -195,7 +195,7 @@ export abstract class Invoice extends Transactional { if (this.isReturn) { await this._removeLoyaltyPointEntry(); - await this.reduceUsedCountOfCoupons(); + this.reduceUsedCountOfCoupons(); return; } @@ -251,11 +251,7 @@ export abstract class Invoice extends Transactional { await this._updatePartyOutStanding(); await this._updateIsItemsReturned(); await this._removeLoyaltyPointEntry(); - await this._reduceUsedCountOfCoupon(); - } - - async _reduceUsedCountOfCoupon() { - await this.reduceUsedCountOfCoupons(); + this.reduceUsedCountOfCoupons(); } async _removeLoyaltyPointEntry() { @@ -572,7 +568,7 @@ export abstract class Invoice extends Transactional { await couponDoc.setAndSync({ used: (couponDoc.used as number) + 1 }); }); } - async reduceUsedCountOfCoupons() { + reduceUsedCountOfCoupons() { if (!this.coupons?.length) { return; } From 02a366f45b1d8a8f60a23ee74e1949a2972bcda5 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 30 Sep 2024 15:27:36 +0530 Subject: [PATCH 19/20] fix: corrected routing for CouponCode --- src/utils/sidebarConfig.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/utils/sidebarConfig.ts b/src/utils/sidebarConfig.ts index d506bc07..c040569f 100644 --- a/src/utils/sidebarConfig.ts +++ b/src/utils/sidebarConfig.ts @@ -219,7 +219,7 @@ function getCompleteSidebar(): SidebarConfig { { label: t`Coupon Code`, name: 'coupon-code', - route: `/list/CouponCode/CouponCode`, + route: `/list/CouponCode`, schemaName: 'CouponCode', hidden: () => !fyo.singles.AccountingSettings?.enableCouponCode, }, From 872f967a5b7a38cf3c27451bbe0952c8593ec7d6 Mon Sep 17 00:00:00 2001 From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com> Date: Mon, 30 Sep 2024 17:20:25 +0530 Subject: [PATCH 20/20] fix: set isEnabled field in default section --- schemas/app/CouponCode.json | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/schemas/app/CouponCode.json b/schemas/app/CouponCode.json index 85ccc115..6a59d77c 100644 --- a/schemas/app/CouponCode.json +++ b/schemas/app/CouponCode.json @@ -18,6 +18,13 @@ "placeholder": "Coupon Name", "section": "Default" }, + { + "fieldname": "isEnabled", + "label": "Is Enabled", + "fieldtype": "Check", + "default": true, + "required": true + }, { "fieldname": "pricingRule", "label": "Pricing Rule", @@ -51,14 +58,6 @@ "required": true, "section": "Validity and Usage" }, - { - "fieldname": "isEnabled", - "label": "Is Enabled", - "fieldtype": "Check", - "default": true, - "required": true, - "section": "Validity and Usage" - }, { "fieldname": "maximumUse", "label": "Maximum Use",