From 46719ef361cdaca9ebd231d8736bc3e6beba92d7 Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Sun, 8 Dec 2024 14:20:11 +0200 Subject: [PATCH 1/8] fix: discount transactions GL entries --- .../src/database/seeds/data/accounts.js | 60 ++++++-- packages/server/src/interfaces/SaleInvoice.ts | 5 + packages/server/src/models/Bill.ts | 12 +- packages/server/src/models/CreditNote.ts | 2 +- packages/server/src/models/SaleEstimate.ts | 2 +- packages/server/src/models/SaleInvoice.ts | 14 +- packages/server/src/models/SaleReceipt.ts | 2 +- packages/server/src/models/VendorCredit.ts | 2 +- .../src/repositories/AccountRepository.ts | 133 ++++++++++++++++-- .../services/Purchases/Bills/BillGLEntries.ts | 87 +++++++++++- .../Sales/Invoices/InvoiceGLEntries.ts | 90 +++++++++++- .../Purchases/Bills/BillForm/utils.tsx | 2 +- .../CreditNotes/CreditNoteForm/utils.tsx | 5 +- .../CreditNotes/CreditNoteForm/utils.tsx | 5 +- .../Sales/Estimates/EstimateForm/utils.tsx | 5 +- .../Sales/Invoices/InvoiceForm/utils.tsx | 2 +- .../Sales/Receipts/ReceiptForm/utils.tsx | 5 +- 17 files changed, 375 insertions(+), 58 deletions(-) diff --git a/packages/server/src/database/seeds/data/accounts.js b/packages/server/src/database/seeds/data/accounts.js index 5d21686c31..35430bbed5 100644 --- a/packages/server/src/database/seeds/data/accounts.js +++ b/packages/server/src/database/seeds/data/accounts.js @@ -1,3 +1,14 @@ +export const OtherExpensesAccount = { + name: 'Other Expenses', + slug: 'other-expenses', + account_type: 'other-expense', + code: '40011', + description: '', + active: 1, + index: 1, + predefined: 1, +}; + export const TaxPayableAccount = { name: 'Tax Payable', slug: 'tax-payable', @@ -39,8 +50,38 @@ export const StripeClearingAccount = { code: '100020', active: true, index: 1, - predefined: true, -} + predefined: true, +}; + +export const DiscountExpenseAccount = { + name: 'Discount', + slug: 'discount', + account_type: 'other-income', + code: '40008', + active: true, + index: 1, + predefined: true, +}; + +export const PurchaseDiscountAccount = { + name: 'Purchase Discount', + slug: 'purchase-discount', + account_type: 'other-expense', + code: '40009', + active: true, + index: 1, + predefined: true, +}; + +export const OtherChargesAccount = { + name: 'Other Charges', + slug: 'other-charges', + account_type: 'other-income', + code: '40010', + active: true, + index: 1, + predefined: true, +}; export default [ { @@ -231,17 +272,7 @@ export default [ }, // Expenses - { - name: 'Other Expenses', - slug: 'other-expenses', - account_type: 'other-expense', - parent_account_id: null, - code: '40001', - description: '', - active: 1, - index: 1, - predefined: 1, - }, + OtherExpensesAccount, { name: 'Cost of Goods Sold', slug: 'cost-of-goods-sold', @@ -358,4 +389,7 @@ export default [ }, UnearnedRevenueAccount, PrepardExpenses, + DiscountExpenseAccount, + PurchaseDiscountAccount, + OtherChargesAccount, ]; diff --git a/packages/server/src/interfaces/SaleInvoice.ts b/packages/server/src/interfaces/SaleInvoice.ts index 897ac84990..c97153d8d2 100644 --- a/packages/server/src/interfaces/SaleInvoice.ts +++ b/packages/server/src/interfaces/SaleInvoice.ts @@ -80,6 +80,11 @@ export interface ISaleInvoice { pdfTemplateId?: number; paymentMethods?: Array; + + adjustment?: number; + + discount?: number; + discountAmount?: number; } export enum DiscountType { diff --git a/packages/server/src/models/Bill.ts b/packages/server/src/models/Bill.ts index 41e6364d4b..2d6299ce74 100644 --- a/packages/server/src/models/Bill.ts +++ b/packages/server/src/models/Bill.ts @@ -1,6 +1,7 @@ import { Model, raw, mixin } from 'objection'; import { castArray, defaultTo, difference } from 'lodash'; import moment from 'moment'; +import * as R from 'ramda'; import TenantModel from 'models/TenantModel'; import BillSettings from './Bill.Settings'; import ModelSetting from './ModelSetting'; @@ -133,12 +134,11 @@ export default class Bill extends mixin(TenantModel, [ get total() { const adjustmentAmount = defaultTo(this.adjustment, 0); - return this.isInclusiveTax - ? this.subtotal - this.discountAmount - adjustmentAmount - : this.subtotal + - this.taxAmountWithheld - - this.discountAmount - - adjustmentAmount; + return R.compose( + R.add(adjustmentAmount), + R.subtract(this.discountAmount), + R.when(R.always(this.isInclusiveTax), R.add(this.taxAmountWithheld)) + )(this.subtotal); } /** diff --git a/packages/server/src/models/CreditNote.ts b/packages/server/src/models/CreditNote.ts index b2cee85722..cdbbf90525 100644 --- a/packages/server/src/models/CreditNote.ts +++ b/packages/server/src/models/CreditNote.ts @@ -107,7 +107,7 @@ export default class CreditNote extends mixin(TenantModel, [ * @returns {number} */ get total() { - return this.subtotal - this.discountAmount - this.adjustment; + return this.subtotal - this.discountAmount + this.adjustment; } /** diff --git a/packages/server/src/models/SaleEstimate.ts b/packages/server/src/models/SaleEstimate.ts index 83df38c1e0..fce0f334b2 100644 --- a/packages/server/src/models/SaleEstimate.ts +++ b/packages/server/src/models/SaleEstimate.ts @@ -116,7 +116,7 @@ export default class SaleEstimate extends mixin(TenantModel, [ get total() { const adjustmentAmount = defaultTo(this.adjustment, 0); - return this.subtotal - this.discountAmount - adjustmentAmount; + return this.subtotal - this.discountAmount + adjustmentAmount; } /** diff --git a/packages/server/src/models/SaleInvoice.ts b/packages/server/src/models/SaleInvoice.ts index 7e874d2dee..098d4a20df 100644 --- a/packages/server/src/models/SaleInvoice.ts +++ b/packages/server/src/models/SaleInvoice.ts @@ -1,4 +1,5 @@ import { mixin, Model, raw } from 'objection'; +import * as R from 'ramda'; import { castArray, defaultTo, takeWhile } from 'lodash'; import moment from 'moment'; import TenantModel from 'models/TenantModel'; @@ -147,9 +148,7 @@ export default class SaleInvoice extends mixin(TenantModel, [ * @returns {number | null} */ get discountPercentage(): number | null { - return this.discountType === DiscountType.Percentage - ? this.discount - : null; + return this.discountType === DiscountType.Percentage ? this.discount : null; } /** @@ -158,11 +157,12 @@ export default class SaleInvoice extends mixin(TenantModel, [ */ get total() { const adjustmentAmount = defaultTo(this.adjustment, 0); - const differencies = this.discountAmount + adjustmentAmount; - return this.isInclusiveTax - ? this.subtotal - differencies - : this.subtotal + this.taxAmountWithheld - differencies; + return R.compose( + R.add(adjustmentAmount), + R.subtract(R.__, this.discountAmount), + R.when(R.always(this.isInclusiveTax), R.add(this.taxAmountWithheld)) + )(this.subtotal); } /** diff --git a/packages/server/src/models/SaleReceipt.ts b/packages/server/src/models/SaleReceipt.ts index 8bef046ed8..220bb4d839 100644 --- a/packages/server/src/models/SaleReceipt.ts +++ b/packages/server/src/models/SaleReceipt.ts @@ -108,7 +108,7 @@ export default class SaleReceipt extends mixin(TenantModel, [ get total() { const adjustmentAmount = defaultTo(this.adjustment, 0); - return this.subtotal - this.discountAmount - adjustmentAmount; + return this.subtotal - this.discountAmount + adjustmentAmount; } /** diff --git a/packages/server/src/models/VendorCredit.ts b/packages/server/src/models/VendorCredit.ts index 24f96afa83..0c4b3d4235 100644 --- a/packages/server/src/models/VendorCredit.ts +++ b/packages/server/src/models/VendorCredit.ts @@ -73,7 +73,7 @@ export default class VendorCredit extends mixin(TenantModel, [ * @returns {number} */ get total() { - return this.subtotal - this.discountAmount - this.adjustment; + return this.subtotal - this.discountAmount + this.adjustment; } /** diff --git a/packages/server/src/repositories/AccountRepository.ts b/packages/server/src/repositories/AccountRepository.ts index 53b26e284e..43e320b229 100644 --- a/packages/server/src/repositories/AccountRepository.ts +++ b/packages/server/src/repositories/AccountRepository.ts @@ -3,7 +3,11 @@ import TenantRepository from '@/repositories/TenantRepository'; import { IAccount } from '@/interfaces'; import { Knex } from 'knex'; import { + DiscountExpenseAccount, + OtherChargesAccount, + OtherExpensesAccount, PrepardExpenses, + PurchaseDiscountAccount, StripeClearingAccount, TaxPayableAccount, UnearnedRevenueAccount, @@ -188,9 +192,9 @@ export default class AccountRepository extends TenantRepository { /** * Finds or creates the unearned revenue. - * @param {Record} extraAttrs - * @param {Knex.Transaction} trx - * @returns + * @param {Record} extraAttrs + * @param {Knex.Transaction} trx + * @returns */ public async findOrCreateUnearnedRevenue( extraAttrs: Record = {}, @@ -219,9 +223,9 @@ export default class AccountRepository extends TenantRepository { /** * Finds or creates the prepard expenses account. - * @param {Record} extraAttrs - * @param {Knex.Transaction} trx - * @returns + * @param {Record} extraAttrs + * @param {Knex.Transaction} trx + * @returns */ public async findOrCreatePrepardExpenses( extraAttrs: Record = {}, @@ -249,12 +253,11 @@ export default class AccountRepository extends TenantRepository { return result; } - /** * Finds or creates the stripe clearing account. - * @param {Record} extraAttrs - * @param {Knex.Transaction} trx - * @returns + * @param {Record} extraAttrs + * @param {Knex.Transaction} trx + * @returns */ public async findOrCreateStripeClearing( extraAttrs: Record = {}, @@ -281,4 +284,114 @@ export default class AccountRepository extends TenantRepository { } return result; } + + /** + * Finds or creates the discount expense account. + * @param {Record} extraAttrs + * @param {Knex.Transaction} trx + * @returns + */ + public async findOrCreateDiscountAccount( + extraAttrs: Record = {}, + trx?: Knex.Transaction + ) { + // Retrieves the given tenant metadata. + const tenantMeta = await TenantMetadata.query().findOne({ + tenantId: this.tenantId, + }); + const _extraAttrs = { + currencyCode: tenantMeta.baseCurrency, + ...extraAttrs, + }; + + let result = await this.model + .query(trx) + .findOne({ slug: DiscountExpenseAccount.slug, ..._extraAttrs }); + + if (!result) { + result = await this.model.query(trx).insertAndFetch({ + ...DiscountExpenseAccount, + ..._extraAttrs, + }); + } + return result; + } + + public async findOrCreatePurchaseDiscountAccount( + extraAttrs: Record = {}, + trx?: Knex.Transaction + ) { + // Retrieves the given tenant metadata. + const tenantMeta = await TenantMetadata.query().findOne({ + tenantId: this.tenantId, + }); + const _extraAttrs = { + currencyCode: tenantMeta.baseCurrency, + ...extraAttrs, + }; + + let result = await this.model + .query(trx) + .findOne({ slug: PurchaseDiscountAccount.slug, ..._extraAttrs }); + + if (!result) { + result = await this.model.query(trx).insertAndFetch({ + ...PurchaseDiscountAccount, + ..._extraAttrs, + }); + } + return result; + } + + public async findOrCreateOtherChargesAccount( + extraAttrs: Record = {}, + trx?: Knex.Transaction + ) { + // Retrieves the given tenant metadata. + const tenantMeta = await TenantMetadata.query().findOne({ + tenantId: this.tenantId, + }); + const _extraAttrs = { + currencyCode: tenantMeta.baseCurrency, + ...extraAttrs, + }; + + let result = await this.model + .query(trx) + .findOne({ slug: OtherChargesAccount.slug, ..._extraAttrs }); + + if (!result) { + result = await this.model.query(trx).insertAndFetch({ + ...OtherChargesAccount, + ..._extraAttrs, + }); + } + return result; + } + + public async findOrCreateOtherExpensesAccount( + extraAttrs: Record = {}, + trx?: Knex.Transaction + ) { + // Retrieves the given tenant metadata. + const tenantMeta = await TenantMetadata.query().findOne({ + tenantId: this.tenantId, + }); + const _extraAttrs = { + currencyCode: tenantMeta.baseCurrency, + ...extraAttrs, + }; + + let result = await this.model + .query(trx) + .findOne({ slug: OtherExpensesAccount.slug, ..._extraAttrs }); + + if (!result) { + result = await this.model.query(trx).insertAndFetch({ + ...OtherExpensesAccount, + ..._extraAttrs, + }); + } + return result; + } } diff --git a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts index 9c1352e9a6..e023a562eb 100644 --- a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts +++ b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts @@ -52,10 +52,22 @@ export class BillGLEntries { {}, trx ); + // Find or create other expenses account. + const otherExpensesAccount = await accountRepository.findOrCreateOtherExpensesAccount( + {}, + trx + ); + // Find or create purchase discount account. + const purchaseDiscountAccount = await accountRepository.findOrCreatePurchaseDiscountAccount( + {}, + trx + ); const billLedger = this.getBillLedger( bill, APAccount.id, - taxPayableAccount.id + taxPayableAccount.id, + purchaseDiscountAccount.id, + otherExpensesAccount.id ); // Commit the GL enties on the storage. await this.ledgerStorage.commit(tenantId, billLedger, trx); @@ -240,6 +252,51 @@ export class BillGLEntries { return nonZeroTaxEntries.map(transformTaxEntry); }; + /** + * Retrieves the purchase discount GL entry. + * @param {IBill} bill + * @param {number} purchaseDiscountAccountId + * @returns {ILedgerEntry} + */ + private getPurchaseDiscountEntry = ( + bill: IBill, + purchaseDiscountAccountId: number + ) => { + const commonEntry = this.getBillCommonEntry(bill); + + return { + ...commonEntry, + credit: bill.discountAmount, + accountId: purchaseDiscountAccountId, + accountNormal: AccountNormal.DEBIT, + index: 1, + indexGroup: 40, + }; + }; + + /** + * Retrieves the purchase other charges GL entry. + * @param {IBill} bill + * @param {number} otherChargesAccountId + * @returns {ILedgerEntry} + */ + private getAdjustmentEntry = ( + bill: IBill, + otherExpensesAccountId: number + ) => { + const commonEntry = this.getBillCommonEntry(bill); + + return { + ...commonEntry, + debit: bill.adjustment < 0 ? bill.adjustment : 0, + credit: bill.adjustment > 0 ? bill.adjustment : 0, + accountId: otherExpensesAccountId, + accountNormal: AccountNormal.DEBIT, + index: 1, + indexGroup: 40, + }; + }; + /** * Retrieves the given bill GL entries. * @param {IBill} bill @@ -249,7 +306,9 @@ export class BillGLEntries { private getBillGLEntries = ( bill: IBill, payableAccountId: number, - taxPayableAccountId: number + taxPayableAccountId: number, + purchaseDiscountAccountId: number, + otherExpensesAccountId: number ): ILedgerEntry[] => { const payableEntry = this.getBillPayableEntry(payableAccountId, bill); @@ -262,8 +321,21 @@ export class BillGLEntries { ); const taxEntries = this.getBillTaxEntries(bill, taxPayableAccountId); + const purchaseDiscountEntry = this.getPurchaseDiscountEntry( + bill, + purchaseDiscountAccountId + ); + const adjustmentEntry = this.getAdjustmentEntry(bill, otherExpensesAccountId); + // Allocate cost entries journal entries. - return [payableEntry, ...itemsEntries, ...landedCostEntries, ...taxEntries]; + return [ + payableEntry, + ...itemsEntries, + ...landedCostEntries, + ...taxEntries, + purchaseDiscountEntry, + adjustmentEntry, + ]; }; /** @@ -275,14 +347,17 @@ export class BillGLEntries { private getBillLedger = ( bill: IBill, payableAccountId: number, - taxPayableAccountId: number + taxPayableAccountId: number, + purchaseDiscountAccountId: number, + otherExpensesAccountId: number ) => { const entries = this.getBillGLEntries( bill, payableAccountId, - taxPayableAccountId + taxPayableAccountId, + purchaseDiscountAccountId, + otherExpensesAccountId ); - return new Ledger(entries); }; } diff --git a/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts b/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts index 61c575f678..6c671e861e 100644 --- a/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts +++ b/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts @@ -44,18 +44,31 @@ export class SaleInvoiceGLEntries { // Find or create the A/R account. const ARAccount = await accountRepository.findOrCreateAccountReceivable( - saleInvoice.currencyCode, {}, trx + saleInvoice.currencyCode, + {}, + trx ); // Find or create tax payable account. const taxPayableAccount = await accountRepository.findOrCreateTaxPayable( {}, trx ); + // Find or create the discount expense account. + const discountAccount = await accountRepository.findOrCreateDiscountAccount( + {}, + trx + ); + // Find or create the other charges account. + const otherChargesAccount = + await accountRepository.findOrCreateOtherChargesAccount({}, trx); + // Retrieves the ledger of the invoice. const ledger = this.getInvoiceGLedger( saleInvoice, ARAccount.id, - taxPayableAccount.id + taxPayableAccount.id, + discountAccount.id, + otherChargesAccount.id ); // Commits the ledger entries to the storage as UOW. await this.ledegrRepository.commit(tenantId, ledger, trx); @@ -107,12 +120,16 @@ export class SaleInvoiceGLEntries { public getInvoiceGLedger = ( saleInvoice: ISaleInvoice, ARAccountId: number, - taxPayableAccountId: number + taxPayableAccountId: number, + discountAccountId: number, + otherChargesAccountId: number ): ILedger => { const entries = this.getInvoiceGLEntries( saleInvoice, ARAccountId, - taxPayableAccountId + taxPayableAccountId, + discountAccountId, + otherChargesAccountId ); return new Ledger(entries); }; @@ -127,6 +144,7 @@ export class SaleInvoiceGLEntries { ): Partial => ({ credit: 0, debit: 0, + currencyCode: saleInvoice.currencyCode, exchangeRate: saleInvoice.exchangeRate, @@ -249,6 +267,50 @@ export class SaleInvoiceGLEntries { return nonZeroTaxEntries.map(transformTaxEntry); }; + /** + * Retrieves the invoice discount GL entry. + * @param {ISaleInvoice} saleInvoice + * @param {number} discountAccountId + * @returns {ILedgerEntry} + */ + private getInvoiceDiscountEntry = ( + saleInvoice: ISaleInvoice, + discountAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getInvoiceGLCommonEntry(saleInvoice); + + return { + ...commonEntry, + debit: saleInvoice.discountAmount, + accountId: discountAccountId, + accountNormal: AccountNormal.CREDIT, + index: 1, + } as ILedgerEntry; + }; + + /** + * Retrieves the invoice adjustment GL entry. + * @param {ISaleInvoice} saleInvoice + * @param {number} adjustmentAccountId + * @returns {ILedgerEntry} + */ + private getAdjustmentEntry = ( + saleInvoice: ISaleInvoice, + otherChargesAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getInvoiceGLCommonEntry(saleInvoice); + const adjustmentAmount = Math.abs(saleInvoice.adjustment); + + return { + ...commonEntry, + debit: saleInvoice.adjustment < 0 ? adjustmentAmount : 0, + credit: saleInvoice.adjustment > 0 ? adjustmentAmount : 0, + accountId: otherChargesAccountId, + accountNormal: AccountNormal.CREDIT, + index: 1, + }; + }; + /** * Retrieves the invoice GL entries. * @param {ISaleInvoice} saleInvoice @@ -258,7 +320,9 @@ export class SaleInvoiceGLEntries { public getInvoiceGLEntries = ( saleInvoice: ISaleInvoice, ARAccountId: number, - taxPayableAccountId: number + taxPayableAccountId: number, + discountAccountId: number, + otherChargesAccountId: number ): ILedgerEntry[] => { const receivableEntry = this.getInvoiceReceivableEntry( saleInvoice, @@ -271,6 +335,20 @@ export class SaleInvoiceGLEntries { saleInvoice, taxPayableAccountId ); - return [receivableEntry, ...creditEntries, ...taxEntries]; + const discountEntry = this.getInvoiceDiscountEntry( + saleInvoice, + discountAccountId + ); + const adjustmentEntry = this.getAdjustmentEntry( + saleInvoice, + otherChargesAccountId + ); + return [ + receivableEntry, + ...creditEntries, + ...taxEntries, + discountEntry, + adjustmentEntry, + ]; }; } diff --git a/packages/webapp/src/containers/Purchases/Bills/BillForm/utils.tsx b/packages/webapp/src/containers/Purchases/Bills/BillForm/utils.tsx index 35296368a8..b570b132ca 100644 --- a/packages/webapp/src/containers/Purchases/Bills/BillForm/utils.tsx +++ b/packages/webapp/src/containers/Purchases/Bills/BillForm/utils.tsx @@ -415,7 +415,7 @@ export const useBillTotal = () => { return R.compose( R.when(R.always(isExclusiveTax), R.add(totalTaxAmount)), R.subtract(R.__, discountAmount), - R.subtract(R.__, adjustmentAmount), + R.add(R.__, adjustmentAmount), )(subtotal); }; diff --git a/packages/webapp/src/containers/Purchases/CreditNotes/CreditNoteForm/utils.tsx b/packages/webapp/src/containers/Purchases/CreditNotes/CreditNoteForm/utils.tsx index e796d85d31..7724629be1 100644 --- a/packages/webapp/src/containers/Purchases/CreditNotes/CreditNoteForm/utils.tsx +++ b/packages/webapp/src/containers/Purchases/CreditNotes/CreditNoteForm/utils.tsx @@ -259,7 +259,10 @@ export const useVendorCreditTotal = () => { const discountAmount = useVendorCreditDiscountAmount(); const adjustment = useVendorCreditAdjustment(); - return subtotal - discountAmount - adjustment; + return R.compose( + R.subtract(R.__, discountAmount), + R.add(R.__, adjustment), + )(subtotal); }; /** diff --git a/packages/webapp/src/containers/Sales/CreditNotes/CreditNoteForm/utils.tsx b/packages/webapp/src/containers/Sales/CreditNotes/CreditNoteForm/utils.tsx index 8a772f4ce4..470cc3f5cf 100644 --- a/packages/webapp/src/containers/Sales/CreditNotes/CreditNoteForm/utils.tsx +++ b/packages/webapp/src/containers/Sales/CreditNotes/CreditNoteForm/utils.tsx @@ -263,7 +263,10 @@ export const useCreditNoteTotal = () => { const discountAmount = useCreditNoteDiscountAmount(); const adjustmentAmount = useCreditNoteAdjustmentAmount(); - return subtotal - discountAmount - adjustmentAmount; + return R.compose( + R.subtract(R.__, discountAmount), + R.add(R.__, adjustmentAmount), + )(subtotal); }; /** diff --git a/packages/webapp/src/containers/Sales/Estimates/EstimateForm/utils.tsx b/packages/webapp/src/containers/Sales/Estimates/EstimateForm/utils.tsx index 3669bf868a..1bca08342c 100644 --- a/packages/webapp/src/containers/Sales/Estimates/EstimateForm/utils.tsx +++ b/packages/webapp/src/containers/Sales/Estimates/EstimateForm/utils.tsx @@ -299,7 +299,10 @@ export const useEstimateTotal = () => { const discount = useEstimateDiscount(); const adjustment = useEstimateAdjustment(); - return subtotal - discount - adjustment; + return R.compose( + R.subtract(R.__, discount), + R.add(R.__, adjustment), + )(subtotal); }; /** diff --git a/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx b/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx index 93fb588416..530a70dd41 100644 --- a/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx +++ b/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx @@ -455,7 +455,7 @@ export const useInvoiceTotal = () => { return R.compose( R.when(R.always(isExclusiveTax), R.add(totalTaxAmount)), R.subtract(R.__, discountAmount), - R.subtract(R.__, adjustmentAmount), + R.add(R.__, adjustmentAmount), )(subtotal); }; diff --git a/packages/webapp/src/containers/Sales/Receipts/ReceiptForm/utils.tsx b/packages/webapp/src/containers/Sales/Receipts/ReceiptForm/utils.tsx index c7365c6805..bca99c9f6a 100644 --- a/packages/webapp/src/containers/Sales/Receipts/ReceiptForm/utils.tsx +++ b/packages/webapp/src/containers/Sales/Receipts/ReceiptForm/utils.tsx @@ -284,7 +284,10 @@ export const useReceiptTotal = () => { const adjustmentAmount = useReceiptAdjustmentAmount(); const discountAmount = useReceiptDiscountAmount(); - return subtotal - discountAmount - adjustmentAmount; + return R.compose( + R.add(R.__, adjustmentAmount), + R.subtract(R.__, discountAmount), + )(subtotal); }; /** From 11d7a40326dbf0a57423be126d7c0be6df3af2b5 Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Sun, 8 Dec 2024 14:47:03 +0200 Subject: [PATCH 2/8] fix: display adjustment in minues --- packages/server/src/models/Bill.ts | 2 +- packages/server/src/models/SaleReceipt.ts | 11 +++++++++++ .../VendorCredits/VendorCreditTransformer.ts | 10 ++++++++++ .../Drawers/BillDrawer/BillDetailTableFooter.tsx | 2 +- .../CreditNoteDetailTableFooter.tsx | 2 +- .../ReceiptDetailDrawer/ReceiptDetailTableFooter.tsx | 2 +- .../VendorCreditDetailDrawerFooter.tsx | 2 +- .../VendorCreditDetailHeader.tsx | 4 ++-- .../containers/Sales/Invoices/InvoiceForm/utils.tsx | 2 +- 9 files changed, 29 insertions(+), 8 deletions(-) diff --git a/packages/server/src/models/Bill.ts b/packages/server/src/models/Bill.ts index 2d6299ce74..c2e6eb3f43 100644 --- a/packages/server/src/models/Bill.ts +++ b/packages/server/src/models/Bill.ts @@ -136,7 +136,7 @@ export default class Bill extends mixin(TenantModel, [ return R.compose( R.add(adjustmentAmount), - R.subtract(this.discountAmount), + R.subtract(R.__, this.discountAmount), R.when(R.always(this.isInclusiveTax), R.add(this.taxAmountWithheld)) )(this.subtotal); } diff --git a/packages/server/src/models/SaleReceipt.ts b/packages/server/src/models/SaleReceipt.ts index 220bb4d839..7c60634f3f 100644 --- a/packages/server/src/models/SaleReceipt.ts +++ b/packages/server/src/models/SaleReceipt.ts @@ -49,6 +49,9 @@ export default class SaleReceipt extends mixin(TenantModel, [ 'total', 'totalLocal', + 'adjustment', + 'adjustmentLocal', + 'discountAmount', 'discountPercentage', @@ -119,6 +122,14 @@ export default class SaleReceipt extends mixin(TenantModel, [ return this.total * this.exchangeRate; } + /** + * Adjustment amount in local currency. + * @returns {number} + */ + get adjustmentLocal() { + return this.adjustment * this.exchangeRate; + } + /** * Detarmine whether the sale receipt closed. * @return {boolean} diff --git a/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts b/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts index b42c90ea0e..8cc36ba7f2 100644 --- a/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts +++ b/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts @@ -20,6 +20,7 @@ export class VendorCreditTransformer extends Transformer { 'discountAmountFormatted', 'discountPercentageFormatted', 'adjustmentFormatted', + 'totalFormatted', 'entries', 'attachments', ]; @@ -118,6 +119,15 @@ export class VendorCreditTransformer extends Transformer { }); }; + /** + * Retrieves the formatted total. + * @param {IVendorCredit} credit + * @returns {string} + */ + protected totalFormatted = (credit) => { + return formatNumber(credit.total, { currencyCode: credit.currencyCode }); + }; + /** * Retrieves the entries of the bill. * @param {IVendorCredit} vendorCredit diff --git a/packages/webapp/src/containers/Drawers/BillDrawer/BillDetailTableFooter.tsx b/packages/webapp/src/containers/Drawers/BillDrawer/BillDetailTableFooter.tsx index 558d0f1ca0..eb6846afbc 100644 --- a/packages/webapp/src/containers/Drawers/BillDrawer/BillDetailTableFooter.tsx +++ b/packages/webapp/src/containers/Drawers/BillDrawer/BillDetailTableFooter.tsx @@ -42,7 +42,7 @@ export function BillDetailTableFooter() { textStyle={TotalLineTextStyle.Regular} /> )} - {bill.adjustment > 0 && ( + {bill.adjustment_formatted && ( )} - {creditNote.adjustment > 0 && ( + {creditNote.adjustment_formatted && ( )} - {receipt.adjustment > 0 && ( + {receipt.adjustment_formatted && ( } - value={vendorCredit.formatted_amount} + value={vendorCredit.total_formatted} borderStyle={TotalLineBorderStyle.DoubleDark} textStyle={TotalLineTextStyle.Bold} /> diff --git a/packages/webapp/src/containers/Drawers/VendorCreditDetailDrawer/VendorCreditDetailHeader.tsx b/packages/webapp/src/containers/Drawers/VendorCreditDetailDrawer/VendorCreditDetailHeader.tsx index 989b1b245e..fac1f84944 100644 --- a/packages/webapp/src/containers/Drawers/VendorCreditDetailDrawer/VendorCreditDetailHeader.tsx +++ b/packages/webapp/src/containers/Drawers/VendorCreditDetailDrawer/VendorCreditDetailHeader.tsx @@ -5,7 +5,6 @@ import styled from 'styled-components'; import { defaultTo } from 'lodash'; import { - FormatDate, T, Row, Col, @@ -29,13 +28,14 @@ export default function VendorCreditDetailHeader() { - {vendorCredit.formatted_amount} + {vendorCredit.total_formatted} + diff --git a/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx b/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx index 530a70dd41..c069fd903d 100644 --- a/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx +++ b/packages/webapp/src/containers/Sales/Invoices/InvoiceForm/utils.tsx @@ -455,7 +455,7 @@ export const useInvoiceTotal = () => { return R.compose( R.when(R.always(isExclusiveTax), R.add(totalTaxAmount)), R.subtract(R.__, discountAmount), - R.add(R.__, adjustmentAmount), + R.add(adjustmentAmount), )(subtotal); }; From 0a5115fc207baef328f532b7f4a400cde45688b0 Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Sun, 8 Dec 2024 17:26:52 +0200 Subject: [PATCH 3/8] feat: add paid amount attr and formatting to SaleReceipt transformer --- packages/server/src/models/SaleReceipt.ts | 19 +++++++++++++++++++ .../Sales/Receipts/SaleReceiptTransformer.ts | 13 +++++++++++++ .../ReceiptDetailHeader.tsx | 3 +-- .../ReceiptDetailTableFooter.tsx | 5 ++--- 4 files changed, 35 insertions(+), 5 deletions(-) diff --git a/packages/server/src/models/SaleReceipt.ts b/packages/server/src/models/SaleReceipt.ts index 7c60634f3f..939f462a14 100644 --- a/packages/server/src/models/SaleReceipt.ts +++ b/packages/server/src/models/SaleReceipt.ts @@ -55,6 +55,9 @@ export default class SaleReceipt extends mixin(TenantModel, [ 'discountAmount', 'discountPercentage', + 'paid', + 'paidLocal', + 'isClosed', 'isDraft', ]; @@ -130,6 +133,22 @@ export default class SaleReceipt extends mixin(TenantModel, [ return this.adjustment * this.exchangeRate; } + /** + * Receipt paid amount. + * @returns {number} + */ + get paid() { + return this.total; + } + + /** + * Receipt paid amount in local currency. + * @returns {number} + */ + get paidLocal() { + return this.paid * this.exchangeRate; + } + /** * Detarmine whether the sale receipt closed. * @return {boolean} diff --git a/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts b/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts index adafab5d82..52d59821e1 100644 --- a/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts +++ b/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts @@ -24,6 +24,7 @@ export class SaleReceiptTransformer extends Transformer { 'formattedReceiptDate', 'formattedClosedAtDate', 'formattedCreatedAt', + 'paidFormatted', 'entries', 'attachments', ]; @@ -130,6 +131,18 @@ export class SaleReceiptTransformer extends Transformer { return receipt.discountPercentage ? `${receipt.discountPercentage}%` : ''; }; + /** + * Retrieves formatted paid amount. + * @param receipt + * @returns {string} + */ + protected paidFormatted = (receipt: ISaleReceipt): string => { + return formatNumber(receipt.paid, { + currencyCode: receipt.currencyCode, + excerptZero: true, + }); + }; + /** * Retrieves formatted adjustment amount. * @param receipt diff --git a/packages/webapp/src/containers/Drawers/ReceiptDetailDrawer/ReceiptDetailHeader.tsx b/packages/webapp/src/containers/Drawers/ReceiptDetailDrawer/ReceiptDetailHeader.tsx index 21bd0a9232..78f969031a 100644 --- a/packages/webapp/src/containers/Drawers/ReceiptDetailDrawer/ReceiptDetailHeader.tsx +++ b/packages/webapp/src/containers/Drawers/ReceiptDetailDrawer/ReceiptDetailHeader.tsx @@ -5,14 +5,12 @@ import styled from 'styled-components'; import { defaultTo } from 'lodash'; import { - ButtonLink, CustomerDrawerLink, CommercialDocHeader, CommercialDocTopHeader, ExchangeRateDetailItem, Row, Col, - FormatDate, DetailsMenu, DetailItem, } from '@/components'; @@ -66,6 +64,7 @@ export default function ReceiptDetailHeader() { /> + } - value={receipt.formatted_amount} - borderStyle={TotalLineBorderStyle.DoubleDark} + value={receipt.paid_formatted} + borderStyle={TotalLineBorderStyle.SingleDark} /> } From 994c441bb8d13facc339557bcf8d887b9e9a3afd Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Sun, 8 Dec 2024 18:11:03 +0200 Subject: [PATCH 4/8] feat: add local discount and adjustment calculations to financial models and transformers - Introduced `discountAmountLocal` and `adjustmentLocal` properties across Bill, CreditNote, SaleInvoice, SaleReceipt, and VendorCredit models to calculate amounts in local currency. - Updated transformers for CreditNote, PurchaseInvoice, and VendorCredit to include formatted representations of local discount and adjustment amounts. - Enhanced GL entry services to handle discount and adjustment entries for SaleReceipt and CreditNote, ensuring accurate ledger entries. - Improved overall consistency in handling financial calculations across various models and services. --- packages/server/src/models/Bill.ts | 20 +++++ packages/server/src/models/CreditNote.ts | 23 ++++- packages/server/src/models/SaleInvoice.ts | 18 ++++ packages/server/src/models/SaleReceipt.ts | 13 ++- packages/server/src/models/VendorCredit.ts | 19 ++++ .../CreditNotes/CreditNoteGLEntries.ts | 83 ++++++++++++++++-- .../CreditNotes/CreditNoteTransformer.ts | 47 ++++++++-- .../Bills/PurchaseInvoiceTransformer.ts | 36 +++++++- .../VendorCredits/VendorCreditTransformer.ts | 29 +++++++ .../Sales/Receipts/SaleReceiptGLEntries.ts | 87 +++++++++++++++++-- .../Sales/Receipts/SaleReceiptTransformer.ts | 6 ++ 11 files changed, 351 insertions(+), 30 deletions(-) diff --git a/packages/server/src/models/Bill.ts b/packages/server/src/models/Bill.ts index c2e6eb3f43..e813ac52e9 100644 --- a/packages/server/src/models/Bill.ts +++ b/packages/server/src/models/Bill.ts @@ -56,8 +56,11 @@ export default class Bill extends mixin(TenantModel, [ 'amountLocal', 'discountAmount', + 'discountAmountLocal', 'discountPercentage', + 'adjustmentLocal', + 'subtotal', 'subtotalLocal', 'subtotalExludingTax', @@ -119,6 +122,15 @@ export default class Bill extends mixin(TenantModel, [ : this.subtotal * (this.discount / 100); } + /** + * Discount amount in local currency. + * @returns {number | null} + */ + get discountAmountLocal() { + return this.discountAmount ? this.discountAmount * this.exchangeRate : null; + } + + /** /** * Discount percentage. * @returns {number | null} @@ -127,6 +139,14 @@ export default class Bill extends mixin(TenantModel, [ return this.discountType === DiscountType.Percentage ? this.discount : null; } + /** + * Adjustment amount in local currency. + * @returns {number | null} + */ + get adjustmentLocal() { + return this.adjustment ? this.adjustment * this.exchangeRate : null; + } + /** * Invoice total. (Tax included) * @returns {number} diff --git a/packages/server/src/models/CreditNote.ts b/packages/server/src/models/CreditNote.ts index cdbbf90525..ab5f6c878e 100644 --- a/packages/server/src/models/CreditNote.ts +++ b/packages/server/src/models/CreditNote.ts @@ -51,10 +51,13 @@ export default class CreditNote extends mixin(TenantModel, [ 'subtotalLocal', 'discountAmount', + 'discountAmountLocal', 'discountPercentage', 'total', 'totalLocal', + + 'adjustmentLocal', ]; } @@ -92,14 +95,28 @@ export default class CreditNote extends mixin(TenantModel, [ : this.subtotal * (this.discount / 100); } + /** + * Discount amount in local currency. + * @returns {number} + */ + get discountAmountLocal() { + return this.discountAmount ? this.discountAmount * this.exchangeRate : null; + } + /** * Discount percentage. * @returns {number | null} */ get discountPercentage(): number | null { - return this.discountType === DiscountType.Percentage - ? this.discount - : null; + return this.discountType === DiscountType.Percentage ? this.discount : null; + } + + /** + * Adjustment amount in local currency. + * @returns {number} + */ + get adjustmentLocal() { + return this.adjustment ? this.adjustment * this.exchangeRate : null; } /** diff --git a/packages/server/src/models/SaleInvoice.ts b/packages/server/src/models/SaleInvoice.ts index 098d4a20df..dfcb0a9542 100644 --- a/packages/server/src/models/SaleInvoice.ts +++ b/packages/server/src/models/SaleInvoice.ts @@ -73,12 +73,14 @@ export default class SaleInvoice extends mixin(TenantModel, [ 'taxAmountWithheldLocal', 'discountAmount', + 'discountAmountLocal', 'discountPercentage', 'total', 'totalLocal', 'writtenoffAmountLocal', + 'adjustmentLocal', ]; } @@ -143,6 +145,14 @@ export default class SaleInvoice extends mixin(TenantModel, [ : this.subtotal * (this.discount / 100); } + /** + * Local discount amount. + * @returns {number | null} + */ + get discountAmountLocal() { + return this.discountAmount ? this.discountAmount * this.exchangeRate : null; + } + /** * Discount percentage. * @returns {number | null} @@ -151,6 +161,14 @@ export default class SaleInvoice extends mixin(TenantModel, [ return this.discountType === DiscountType.Percentage ? this.discount : null; } + /** + * Adjustment amount in local currency. + * @returns {number | null} + */ + get adjustmentLocal(): number | null { + return this.adjustment ? this.adjustment * this.exchangeRate : null; + } + /** * Invoice total. (Tax included) * @returns {number} diff --git a/packages/server/src/models/SaleReceipt.ts b/packages/server/src/models/SaleReceipt.ts index 939f462a14..95abf74304 100644 --- a/packages/server/src/models/SaleReceipt.ts +++ b/packages/server/src/models/SaleReceipt.ts @@ -53,6 +53,7 @@ export default class SaleReceipt extends mixin(TenantModel, [ 'adjustmentLocal', 'discountAmount', + 'discountAmountLocal', 'discountPercentage', 'paid', @@ -97,14 +98,20 @@ export default class SaleReceipt extends mixin(TenantModel, [ : this.subtotal * (this.discount / 100); } + /** + * Discount amount in local currency. + * @returns {number | null} + */ + get discountAmountLocal() { + return this.discountAmount ? this.discountAmount * this.exchangeRate : null; + } + /** * Discount percentage. * @returns {number | null} */ get discountPercentage(): number | null { - return this.discountType === DiscountType.Percentage - ? this.discount - : null; + return this.discountType === DiscountType.Percentage ? this.discount : null; } /** diff --git a/packages/server/src/models/VendorCredit.ts b/packages/server/src/models/VendorCredit.ts index 0c4b3d4235..75473060ec 100644 --- a/packages/server/src/models/VendorCredit.ts +++ b/packages/server/src/models/VendorCredit.ts @@ -60,6 +60,14 @@ export default class VendorCredit extends mixin(TenantModel, [ : this.subtotal * (this.discount / 100); } + /** + * Discount amount in local currency. + * @returns {number | null} + */ + get discountAmountLocal() { + return this.discountAmount ? this.discountAmount * this.exchangeRate : null; + } + /** * Discount percentage. * @returns {number | null} @@ -68,6 +76,14 @@ export default class VendorCredit extends mixin(TenantModel, [ return this.discountType === DiscountType.Percentage ? this.discount : null; } + /** + * Adjustment amount in local currency. + * @returns {number | null} + */ + get adjustmentLocal() { + return this.adjustment ? this.adjustment * this.exchangeRate : null; + } + /** * Vendor credit total. * @returns {number} @@ -180,8 +196,11 @@ export default class VendorCredit extends mixin(TenantModel, [ 'localAmount', 'discountAmount', + 'discountAmountLocal', 'discountPercentage', + 'adjustmentLocal', + 'total', 'totalLocal', ]; diff --git a/packages/server/src/services/CreditNotes/CreditNoteGLEntries.ts b/packages/server/src/services/CreditNotes/CreditNoteGLEntries.ts index 697072f3cb..08734d37a3 100644 --- a/packages/server/src/services/CreditNotes/CreditNoteGLEntries.ts +++ b/packages/server/src/services/CreditNotes/CreditNoteGLEntries.ts @@ -12,6 +12,7 @@ import { import HasTenancyService from '@/services/Tenancy/TenancyService'; import Ledger from '@/services/Accounting/Ledger'; import LedgerStorageService from '@/services/Accounting/LedgerStorageService'; +import { SaleReceipt } from '@/models'; @Service() export default class CreditNoteGLEntries { @@ -29,11 +30,15 @@ export default class CreditNoteGLEntries { */ private getCreditNoteGLedger = ( creditNote: ICreditNote, - receivableAccount: number + receivableAccount: number, + discountAccount: number, + adjustmentAccount: number ): Ledger => { const ledgerEntries = this.getCreditNoteGLEntries( creditNote, - receivableAccount + receivableAccount, + discountAccount, + adjustmentAccount ); return new Ledger(ledgerEntries); }; @@ -49,9 +54,16 @@ export default class CreditNoteGLEntries { tenantId: number, creditNote: ICreditNote, payableAccount: number, + discountAccount: number, + adjustmentAccount: number, trx?: Knex.Transaction ): Promise => { - const ledger = this.getCreditNoteGLedger(creditNote, payableAccount); + const ledger = this.getCreditNoteGLedger( + creditNote, + payableAccount, + discountAccount, + adjustmentAccount + ); await this.ledgerStorage.commit(tenantId, ledger, trx); }; @@ -98,11 +110,18 @@ export default class CreditNoteGLEntries { const ARAccount = await accountRepository.findOrCreateAccountReceivable( creditNoteWithItems.currencyCode ); + const discountAccount = await accountRepository.findOrCreateDiscountAccount( + {} + ); + const adjustmentAccount = + await accountRepository.findOrCreateOtherChargesAccount({}); // Saves the credit note GL entries. await this.saveCreditNoteGLEntries( tenantId, creditNoteWithItems, ARAccount.id, + discountAccount.id, + adjustmentAccount.id, trx ); }; @@ -169,7 +188,7 @@ export default class CreditNoteGLEntries { return { ...commonEntry, - credit: creditNote.localAmount, + credit: creditNote.totalLocal, accountId: ARAccountId, contactId: creditNote.customerId, index: 1, @@ -206,6 +225,50 @@ export default class CreditNoteGLEntries { } ); + /** + * Retrieves the credit note discount entry. + * @param {ICreditNote} creditNote + * @param {number} discountAccountId + * @returns {ILedgerEntry} + */ + private getDiscountEntry = ( + creditNote: ICreditNote, + discountAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getCreditNoteCommonEntry(creditNote); + + return { + ...commonEntry, + credit: creditNote.discountAmountLocal, + accountId: discountAccountId, + index: 1, + accountNormal: AccountNormal.CREDIT, + }; + }; + + /** + * Retrieves the credit note adjustment entry. + * @param {ICreditNote} creditNote + * @param {number} adjustmentAccountId + * @returns {ILedgerEntry} + */ + private getAdjustmentEntry = ( + creditNote: ICreditNote, + adjustmentAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getCreditNoteCommonEntry(creditNote); + const adjustmentAmount = Math.abs(creditNote.adjustmentLocal); + + return { + ...commonEntry, + credit: creditNote.adjustmentLocal < 0 ? adjustmentAmount : 0, + debit: creditNote.adjustmentLocal > 0 ? adjustmentAmount : 0, + accountId: adjustmentAccountId, + accountNormal: AccountNormal.CREDIT, + index: 1, + }; + }; + /** * Retrieve the credit note GL entries. * @param {ICreditNote} creditNote - Credit note. @@ -214,13 +277,21 @@ export default class CreditNoteGLEntries { */ public getCreditNoteGLEntries = ( creditNote: ICreditNote, - ARAccountId: number + ARAccountId: number, + discountAccountId: number, + adjustmentAccountId: number ): ILedgerEntry[] => { const AREntry = this.getCreditNoteAREntry(creditNote, ARAccountId); const getItemEntry = this.getCreditNoteItemEntry(creditNote); const itemsEntries = creditNote.entries.map(getItemEntry); - return [AREntry, ...itemsEntries]; + const discountEntry = this.getDiscountEntry(creditNote, discountAccountId); + const adjustmentEntry = this.getAdjustmentEntry( + creditNote, + adjustmentAccountId + ); + + return [AREntry, discountEntry, adjustmentEntry, ...itemsEntries]; }; } diff --git a/packages/server/src/services/CreditNotes/CreditNoteTransformer.ts b/packages/server/src/services/CreditNotes/CreditNoteTransformer.ts index 611d006ade..86e84681df 100644 --- a/packages/server/src/services/CreditNotes/CreditNoteTransformer.ts +++ b/packages/server/src/services/CreditNotes/CreditNoteTransformer.ts @@ -18,11 +18,18 @@ export class CreditNoteTransformer extends Transformer { 'formattedAmount', 'formattedCreditsUsed', 'formattedSubtotal', + 'discountAmountFormatted', + 'discountAmountLocalFormatted', + 'discountPercentageFormatted', + 'adjustmentFormatted', + 'adjustmentLocalFormatted', + 'totalFormatted', 'totalLocalFormatted', + 'entries', 'attachments', ]; @@ -39,7 +46,7 @@ export class CreditNoteTransformer extends Transformer { /** * Retrieve formatted created at date. - * @param credit + * @param credit * @returns {string} */ protected formattedCreatedAt = (credit): string => { @@ -90,7 +97,7 @@ export class CreditNoteTransformer extends Transformer { /** * Retrieves formatted discount amount. - * @param credit + * @param credit * @returns {string} */ protected discountAmountFormatted = (credit): string => { @@ -100,20 +107,30 @@ export class CreditNoteTransformer extends Transformer { }); }; + /** + * Retrieves the formatted discount amount in local currency. + * @param {ICreditNote} credit + * @returns {string} + */ + protected discountAmountLocalFormatted = (credit): string => { + return formatNumber(credit.discountAmountLocal, { + currencyCode: credit.currencyCode, + excerptZero: true, + }); + }; + /** * Retrieves formatted discount percentage. - * @param credit + * @param credit * @returns {string} */ protected discountPercentageFormatted = (credit): string => { - return credit.discountPercentage - ? `${credit.discountPercentage}%` - : ''; + return credit.discountPercentage ? `${credit.discountPercentage}%` : ''; }; /** * Retrieves formatted adjustment amount. - * @param credit + * @param credit * @returns {string} */ protected adjustmentFormatted = (credit): string => { @@ -123,9 +140,21 @@ export class CreditNoteTransformer extends Transformer { }); }; + /** + * Retrieves the formatted adjustment amount in local currency. + * @param {ICreditNote} credit + * @returns {string} + */ + protected adjustmentLocalFormatted = (credit): string => { + return formatNumber(credit.adjustmentLocal, { + currencyCode: this.context.organization.baseCurrency, + excerptZero: true, + }); + }; + /** * Retrieves the formatted total. - * @param credit + * @param credit * @returns {string} */ protected totalFormatted = (credit): string => { @@ -136,7 +165,7 @@ export class CreditNoteTransformer extends Transformer { /** * Retrieves the formatted total in local currency. - * @param credit + * @param credit * @returns {string} */ protected totalLocalFormatted = (credit): string => { diff --git a/packages/server/src/services/Purchases/Bills/PurchaseInvoiceTransformer.ts b/packages/server/src/services/Purchases/Bills/PurchaseInvoiceTransformer.ts index a6766a0dbe..05aa0311db 100644 --- a/packages/server/src/services/Purchases/Bills/PurchaseInvoiceTransformer.ts +++ b/packages/server/src/services/Purchases/Bills/PurchaseInvoiceTransformer.ts @@ -20,13 +20,21 @@ export class PurchaseInvoiceTransformer extends Transformer { 'formattedBalance', 'formattedDueAmount', 'formattedExchangeRate', + 'subtotalFormatted', 'subtotalLocalFormatted', + 'subtotalExcludingTaxFormatted', 'taxAmountWithheldLocalFormatted', + 'discountAmountFormatted', + 'discountAmountLocalFormatted', + 'discountPercentageFormatted', + 'adjustmentFormatted', + 'adjustmentLocalFormatted', + 'totalFormatted', 'totalLocalFormatted', 'taxes', @@ -175,15 +183,25 @@ export class PurchaseInvoiceTransformer extends Transformer { }); }; + /** + * Retrieves the formatted discount amount in local currency. + * @param {IBill} bill + * @returns {string} + */ + protected discountAmountLocalFormatted = (bill): string => { + return formatNumber(bill.discountAmountLocal, { + currencyCode: this.context.organization.baseCurrency, + excerptZero: true, + }); + }; + /** * Retrieves the formatted discount percentage. * @param {IBill} bill * @returns {string} */ protected discountPercentageFormatted = (bill): string => { - return bill.discountPercentage - ? `${bill.discountPercentage}%` - : ''; + return bill.discountPercentage ? `${bill.discountPercentage}%` : ''; }; /** @@ -198,6 +216,18 @@ export class PurchaseInvoiceTransformer extends Transformer { }); }; + /** + * Retrieves the formatted adjustment amount in local currency. + * @param {IBill} bill + * @returns {string} + */ + protected adjustmentLocalFormatted = (bill): string => { + return formatNumber(bill.adjustmentLocal, { + currencyCode: this.context.organization.baseCurrency, + excerptZero: true, + }); + }; + /** * Retrieves the total formatted. * @param {IBill} bill diff --git a/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts b/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts index 8cc36ba7f2..61e4d67369 100644 --- a/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts +++ b/packages/server/src/services/Purchases/VendorCredits/VendorCreditTransformer.ts @@ -17,9 +17,14 @@ export class VendorCreditTransformer extends Transformer { 'formattedCreatedAt', 'formattedCreditsRemaining', 'formattedInvoicedAmount', + 'discountAmountFormatted', 'discountPercentageFormatted', + 'discountAmountLocalFormatted', + 'adjustmentFormatted', + 'adjustmentLocalFormatted', + 'totalFormatted', 'entries', 'attachments', @@ -87,6 +92,18 @@ export class VendorCreditTransformer extends Transformer { }); }; + /** + * Retrieves the formatted discount amount in local currency. + * @param {IVendorCredit} credit + * @returns {string} + */ + protected discountAmountLocalFormatted = (credit): string => { + return formatNumber(credit.discountAmountLocal, { + currencyCode: this.context.organization.baseCurrency, + excerptZero: true, + }); + }; + /** * Retrieves the formatted discount percentage. * @param {IVendorCredit} credit @@ -108,6 +125,18 @@ export class VendorCreditTransformer extends Transformer { }); }; + /** + * Retrieves the formatted adjustment amount in local currency. + * @param {IVendorCredit} credit + * @returns {string} + */ + protected adjustmentLocalFormatted = (credit): string => { + return formatNumber(credit.adjustmentLocal, { + currencyCode: this.context.organization.baseCurrency, + excerptZero: true, + }); + }; + /** * Retrieves the formatted invoiced amount. * @param credit diff --git a/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts b/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts index d354141e98..5f45d6f8db 100644 --- a/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts +++ b/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts @@ -31,13 +31,27 @@ export class SaleReceiptGLEntries { trx?: Knex.Transaction ): Promise => { const { SaleReceipt } = this.tenancy.models(tenantId); + const { accountRepository } = this.tenancy.repositories(tenantId); const saleReceipt = await SaleReceipt.query(trx) .findById(saleReceiptId) .withGraphFetched('entries.item'); + // Find or create the discount expense account. + const discountAccount = await accountRepository.findOrCreateDiscountAccount( + {}, + trx + ); + // Find or create the other charges account. + const otherChargesAccount = + await accountRepository.findOrCreateOtherChargesAccount({}, trx); + // Retrieve the income entries ledger. - const incomeLedger = this.getIncomeEntriesLedger(saleReceipt); + const incomeLedger = this.getIncomeEntriesLedger( + saleReceipt, + discountAccount.id, + otherChargesAccount.id + ); // Commits the ledger entries to the storage. await this.ledgerStorage.commit(tenantId, incomeLedger, trx); @@ -87,8 +101,16 @@ export class SaleReceiptGLEntries { * @param {ISaleReceipt} saleReceipt * @returns {Ledger} */ - private getIncomeEntriesLedger = (saleReceipt: ISaleReceipt): Ledger => { - const entries = this.getIncomeGLEntries(saleReceipt); + private getIncomeEntriesLedger = ( + saleReceipt: ISaleReceipt, + discountAccountId: number, + otherChargesAccountId: number + ): Ledger => { + const entries = this.getIncomeGLEntries( + saleReceipt, + discountAccountId, + otherChargesAccountId + ); return new Ledger(entries); }; @@ -161,24 +183,77 @@ export class SaleReceiptGLEntries { return { ...commonEntry, - debit: saleReceipt.localAmount, + debit: saleReceipt.totalLocal, accountId: saleReceipt.depositAccountId, index: 1, accountNormal: AccountNormal.DEBIT, }; }; + /** + * Retrieves the discount GL entry. + * @param {ISaleReceipt} saleReceipt + * @param {number} discountAccountId + * @returns {ILedgerEntry} + */ + private getDiscountEntry = ( + saleReceipt: ISaleReceipt, + discountAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getIncomeGLCommonEntry(saleReceipt); + + return { + ...commonEntry, + debit: saleReceipt.discountAmount, + accountId: discountAccountId, + index: 1, + accountNormal: AccountNormal.CREDIT, + }; + }; + + /** + * Retrieves the adjustment GL entry. + * @param {ISaleReceipt} saleReceipt + * @param {number} adjustmentAccountId + * @returns {ILedgerEntry} + */ + private getAdjustmentEntry = ( + saleReceipt: ISaleReceipt, + adjustmentAccountId: number + ): ILedgerEntry => { + const commonEntry = this.getIncomeGLCommonEntry(saleReceipt); + const adjustmentAmount = Math.abs(saleReceipt.adjustment); + + return { + ...commonEntry, + debit: saleReceipt.adjustment < 0 ? adjustmentAmount : 0, + credit: saleReceipt.adjustment > 0 ? adjustmentAmount : 0, + accountId: adjustmentAccountId, + accountNormal: AccountNormal.CREDIT, + index: 1, + }; + }; + /** * Retrieves the income GL entries. * @param {ISaleReceipt} saleReceipt - * @returns {ILedgerEntry[]} */ - private getIncomeGLEntries = (saleReceipt: ISaleReceipt): ILedgerEntry[] => { + private getIncomeGLEntries = ( + saleReceipt: ISaleReceipt, + discountAccountId: number, + otherChargesAccountId: number + ): ILedgerEntry[] => { const getItemEntry = this.getReceiptIncomeItemEntry(saleReceipt); const creditEntries = saleReceipt.entries.map(getItemEntry); const depositEntry = this.getReceiptDepositEntry(saleReceipt); + const discountEntry = this.getDiscountEntry(saleReceipt, discountAccountId); + const adjustmentEntry = this.getAdjustmentEntry( + saleReceipt, + otherChargesAccountId + ); - return [depositEntry, ...creditEntries]; + return [depositEntry, ...creditEntries, discountEntry, adjustmentEntry]; }; } diff --git a/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts b/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts index 52d59821e1..eb833b3442 100644 --- a/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts +++ b/packages/server/src/services/Sales/Receipts/SaleReceiptTransformer.ts @@ -15,11 +15,17 @@ export class SaleReceiptTransformer extends Transformer { return [ 'discountAmountFormatted', 'discountPercentageFormatted', + 'discountAmountLocalFormatted', + 'subtotalFormatted', 'subtotalLocalFormatted', + 'totalFormatted', 'totalLocalFormatted', + 'adjustmentFormatted', + 'adjustmentLocalFormatted', + 'formattedAmount', 'formattedReceiptDate', 'formattedClosedAtDate', From b9963aa241343d7e687f82fd757126a9bb86bc70 Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Mon, 9 Dec 2024 00:01:42 +0200 Subject: [PATCH 5/8] feat: filter ledger blank entries --- .../server/src/services/Accounting/Ledger.ts | 1 + .../services/Accounting/LedgerEntriesStorage.ts | 17 ++++++++++------- 2 files changed, 11 insertions(+), 7 deletions(-) diff --git a/packages/server/src/services/Accounting/Ledger.ts b/packages/server/src/services/Accounting/Ledger.ts index 155a61e104..6e7519bc8a 100644 --- a/packages/server/src/services/Accounting/Ledger.ts +++ b/packages/server/src/services/Accounting/Ledger.ts @@ -238,6 +238,7 @@ export default class Ledger implements ILedger { return { credit: defaultTo(entry.credit, 0), debit: defaultTo(entry.debit, 0), + exchangeRate: entry.exchangeRate, currencyCode: entry.currencyCode, diff --git a/packages/server/src/services/Accounting/LedgerEntriesStorage.ts b/packages/server/src/services/Accounting/LedgerEntriesStorage.ts index dd192b8115..8f789e3cf7 100644 --- a/packages/server/src/services/Accounting/LedgerEntriesStorage.ts +++ b/packages/server/src/services/Accounting/LedgerEntriesStorage.ts @@ -9,15 +9,18 @@ import { import HasTenancyService from '@/services/Tenancy/TenancyService'; import { transformLedgerEntryToTransaction } from './utils'; +// Filter the blank entries. +const filterBlankEntry = (entry: ILedgerEntry) => Boolean(entry.credit || entry.debit); + @Service() export class LedgerEntriesStorage { @Inject() - tenancy: HasTenancyService; + private tenancy: HasTenancyService; /** * Saves entries of the given ledger. - * @param {number} tenantId - * @param {ILedger} ledger - * @param {Knex.Transaction} knex + * @param {number} tenantId + * @param {ILedger} ledger + * @param {Knex.Transaction} knex * @returns {Promise} */ public saveEntries = async ( @@ -26,7 +29,7 @@ export class LedgerEntriesStorage { trx?: Knex.Transaction ) => { const saveEntryQueue = async.queue(this.saveEntryTask, 10); - const entries = ledger.getEntries(); + const entries = ledger.filter(filterBlankEntry).getEntries(); entries.forEach((entry) => { saveEntryQueue.push({ tenantId, entry, trx }); @@ -57,8 +60,8 @@ export class LedgerEntriesStorage { /** * Saves the ledger entry to the account transactions repository. - * @param {number} tenantId - * @param {ILedgerEntry} entry + * @param {number} tenantId + * @param {ILedgerEntry} entry * @returns {Promise} */ private saveEntry = async ( From 477da0e7c083b4de03963a3f0e84527345f26d8b Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Mon, 9 Dec 2024 00:19:22 +0200 Subject: [PATCH 6/8] feat: add local adjustment and discount properties to SaleInvoice and SaleReceipt interfaces. --- packages/server/src/interfaces/SaleInvoice.ts | 2 ++ packages/server/src/interfaces/SaleReceipt.ts | 3 +++ .../services/Purchases/Bills/BillGLEntries.ts | 23 +++++++++---------- .../Sales/Invoices/InvoiceGLEntries.ts | 8 +++---- .../Sales/Receipts/SaleReceiptGLEntries.ts | 9 ++++---- 5 files changed, 24 insertions(+), 21 deletions(-) diff --git a/packages/server/src/interfaces/SaleInvoice.ts b/packages/server/src/interfaces/SaleInvoice.ts index c97153d8d2..b7f208ac09 100644 --- a/packages/server/src/interfaces/SaleInvoice.ts +++ b/packages/server/src/interfaces/SaleInvoice.ts @@ -82,9 +82,11 @@ export interface ISaleInvoice { paymentMethods?: Array; adjustment?: number; + adjustmentLocal?: number | null; discount?: number; discountAmount?: number; + discountAmountLocal?: number | null; } export enum DiscountType { diff --git a/packages/server/src/interfaces/SaleReceipt.ts b/packages/server/src/interfaces/SaleReceipt.ts index ec0d2a7264..8d854d46b3 100644 --- a/packages/server/src/interfaces/SaleReceipt.ts +++ b/packages/server/src/interfaces/SaleReceipt.ts @@ -39,6 +39,9 @@ export interface ISaleReceipt { discountPercentage?: number | null; adjustment?: number; + adjustmentLocal?: number | null; + + discountAmountLocal?: number | null; } export interface ISalesReceiptsFilter { diff --git a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts index e023a562eb..0f621ca124 100644 --- a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts +++ b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts @@ -53,15 +53,11 @@ export class BillGLEntries { trx ); // Find or create other expenses account. - const otherExpensesAccount = await accountRepository.findOrCreateOtherExpensesAccount( - {}, - trx - ); + const otherExpensesAccount = + await accountRepository.findOrCreateOtherExpensesAccount({}, trx); // Find or create purchase discount account. - const purchaseDiscountAccount = await accountRepository.findOrCreatePurchaseDiscountAccount( - {}, - trx - ); + const purchaseDiscountAccount = + await accountRepository.findOrCreatePurchaseDiscountAccount({}, trx); const billLedger = this.getBillLedger( bill, APAccount.id, @@ -266,7 +262,7 @@ export class BillGLEntries { return { ...commonEntry, - credit: bill.discountAmount, + credit: bill.discountAmountLocal, accountId: purchaseDiscountAccountId, accountNormal: AccountNormal.DEBIT, index: 1, @@ -288,8 +284,8 @@ export class BillGLEntries { return { ...commonEntry, - debit: bill.adjustment < 0 ? bill.adjustment : 0, - credit: bill.adjustment > 0 ? bill.adjustment : 0, + debit: bill.adjustmentLocal < 0 ? bill.adjustmentLocal : 0, + credit: bill.adjustmentLocal > 0 ? bill.adjustmentLocal : 0, accountId: otherExpensesAccountId, accountNormal: AccountNormal.DEBIT, index: 1, @@ -325,7 +321,10 @@ export class BillGLEntries { bill, purchaseDiscountAccountId ); - const adjustmentEntry = this.getAdjustmentEntry(bill, otherExpensesAccountId); + const adjustmentEntry = this.getAdjustmentEntry( + bill, + otherExpensesAccountId + ); // Allocate cost entries journal entries. return [ diff --git a/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts b/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts index 6c671e861e..c5ca66946b 100644 --- a/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts +++ b/packages/server/src/services/Sales/Invoices/InvoiceGLEntries.ts @@ -281,7 +281,7 @@ export class SaleInvoiceGLEntries { return { ...commonEntry, - debit: saleInvoice.discountAmount, + debit: saleInvoice.discountAmountLocal, accountId: discountAccountId, accountNormal: AccountNormal.CREDIT, index: 1, @@ -299,12 +299,12 @@ export class SaleInvoiceGLEntries { otherChargesAccountId: number ): ILedgerEntry => { const commonEntry = this.getInvoiceGLCommonEntry(saleInvoice); - const adjustmentAmount = Math.abs(saleInvoice.adjustment); + const adjustmentAmount = Math.abs(saleInvoice.adjustmentLocal); return { ...commonEntry, - debit: saleInvoice.adjustment < 0 ? adjustmentAmount : 0, - credit: saleInvoice.adjustment > 0 ? adjustmentAmount : 0, + debit: saleInvoice.adjustmentLocal < 0 ? adjustmentAmount : 0, + credit: saleInvoice.adjustmentLocal > 0 ? adjustmentAmount : 0, accountId: otherChargesAccountId, accountNormal: AccountNormal.CREDIT, index: 1, diff --git a/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts b/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts index 5f45d6f8db..8dc7bf8038 100644 --- a/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts +++ b/packages/server/src/services/Sales/Receipts/SaleReceiptGLEntries.ts @@ -204,7 +204,7 @@ export class SaleReceiptGLEntries { return { ...commonEntry, - debit: saleReceipt.discountAmount, + debit: saleReceipt.discountAmountLocal, accountId: discountAccountId, index: 1, accountNormal: AccountNormal.CREDIT, @@ -222,12 +222,12 @@ export class SaleReceiptGLEntries { adjustmentAccountId: number ): ILedgerEntry => { const commonEntry = this.getIncomeGLCommonEntry(saleReceipt); - const adjustmentAmount = Math.abs(saleReceipt.adjustment); + const adjustmentAmount = Math.abs(saleReceipt.adjustmentLocal); return { ...commonEntry, - debit: saleReceipt.adjustment < 0 ? adjustmentAmount : 0, - credit: saleReceipt.adjustment > 0 ? adjustmentAmount : 0, + debit: saleReceipt.adjustmentLocal < 0 ? adjustmentAmount : 0, + credit: saleReceipt.adjustmentLocal > 0 ? adjustmentAmount : 0, accountId: adjustmentAccountId, accountNormal: AccountNormal.CREDIT, index: 1, @@ -253,7 +253,6 @@ export class SaleReceiptGLEntries { saleReceipt, otherChargesAccountId ); - return [depositEntry, ...creditEntries, discountEntry, adjustmentEntry]; }; } From 1d54947764c172cc3c3e162c6d575fc010c76fdf Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Mon, 9 Dec 2024 00:44:50 +0200 Subject: [PATCH 7/8] fix: correct debit and credit calculations for local adjustments in BillGLEntries --- .../server/src/services/Purchases/Bills/BillGLEntries.ts | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts index 0f621ca124..a735b98334 100644 --- a/packages/server/src/services/Purchases/Bills/BillGLEntries.ts +++ b/packages/server/src/services/Purchases/Bills/BillGLEntries.ts @@ -110,6 +110,7 @@ export class BillGLEntries { return { debit: 0, credit: 0, + currencyCode: bill.currencyCode, exchangeRate: bill.exchangeRate || 1, @@ -281,11 +282,12 @@ export class BillGLEntries { otherExpensesAccountId: number ) => { const commonEntry = this.getBillCommonEntry(bill); + const adjustmentAmount = Math.abs(bill.adjustmentLocal); return { ...commonEntry, - debit: bill.adjustmentLocal < 0 ? bill.adjustmentLocal : 0, - credit: bill.adjustmentLocal > 0 ? bill.adjustmentLocal : 0, + debit: bill.adjustmentLocal > 0 ? adjustmentAmount : 0, + credit: bill.adjustmentLocal < 0 ? adjustmentAmount : 0, accountId: otherExpensesAccountId, accountNormal: AccountNormal.DEBIT, index: 1, From c633fa8522998871262f80c3a388e694462f3703 Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Mon, 9 Dec 2024 11:06:42 +0200 Subject: [PATCH 8/8] feat: add vendor credit discount and adjustment GL entries --- .../VendorCredits/VendorCreditGLEntries.ts | 71 +++++++++++++++++-- 1 file changed, 67 insertions(+), 4 deletions(-) diff --git a/packages/server/src/services/Purchases/VendorCredits/VendorCreditGLEntries.ts b/packages/server/src/services/Purchases/VendorCredits/VendorCreditGLEntries.ts index fe269cef54..0747a5845e 100644 --- a/packages/server/src/services/Purchases/VendorCredits/VendorCreditGLEntries.ts +++ b/packages/server/src/services/Purchases/VendorCredits/VendorCreditGLEntries.ts @@ -56,7 +56,7 @@ export default class VendorCreditGLEntries { return { ...commonEntity, - debit: vendorCredit.localAmount, + debit: vendorCredit.totalLocal, accountId: APAccountId, contactId: vendorCredit.vendorId, accountNormal: AccountNormal.CREDIT, @@ -94,6 +94,52 @@ export default class VendorCreditGLEntries { } ); + /** + * Retrieves the vendor credit discount GL entry. + * @param {IVendorCredit} vendorCredit + * @param {number} discountAccountId + * @returns {ILedgerEntry} + */ + public getDiscountEntry = ( + vendorCredit: IVendorCredit, + purchaseDiscountAccountId: number + ) => { + const commonEntry = this.getVendorCreditGLCommonEntry(vendorCredit); + + return { + ...commonEntry, + debit: vendorCredit.discountAmountLocal, + accountId: purchaseDiscountAccountId, + accountNormal: AccountNormal.DEBIT, + index: 1, + indexGroup: 40, + }; + }; + + /** + * Retrieves the vendor credit adjustment GL entry. + * @param {IVendorCredit} vendorCredit + * @param {number} adjustmentAccountId + * @returns {ILedgerEntry} + */ + public getAdjustmentEntry = ( + vendorCredit: IVendorCredit, + otherExpensesAccountId: number + ) => { + const commonEntry = this.getVendorCreditGLCommonEntry(vendorCredit); + const adjustmentAmount = Math.abs(vendorCredit.adjustmentLocal); + + return { + ...commonEntry, + credit: vendorCredit.adjustmentLocal > 0 ? adjustmentAmount : 0, + debit: vendorCredit.adjustmentLocal < 0 ? adjustmentAmount : 0, + accountId: otherExpensesAccountId, + accountNormal: AccountNormal.DEBIT, + index: 1, + indexGroup: 40, + }; + }; + /** * Retrieve the vendor credit GL entries. * @param {IVendorCredit} vendorCredit - @@ -102,7 +148,9 @@ export default class VendorCreditGLEntries { */ public getVendorCreditGLEntries = ( vendorCredit: IVendorCredit, - payableAccountId: number + payableAccountId: number, + purchaseDiscountAccountId: number, + otherExpensesAccountId: number ): ILedgerEntry[] => { const payableEntry = this.getVendorCreditPayableGLEntry( vendorCredit, @@ -111,7 +159,15 @@ export default class VendorCreditGLEntries { const getItemEntry = this.getVendorCreditGLItemEntry(vendorCredit); const itemsEntries = vendorCredit.entries.map(getItemEntry); - return [payableEntry, ...itemsEntries]; + const discountEntry = this.getDiscountEntry( + vendorCredit, + purchaseDiscountAccountId + ); + const adjustmentEntry = this.getAdjustmentEntry( + vendorCredit, + otherExpensesAccountId + ); + return [payableEntry, discountEntry, adjustmentEntry, ...itemsEntries]; }; /** @@ -158,10 +214,17 @@ export default class VendorCreditGLEntries { {}, trx ); + const purchaseDiscountAccount = + await accountRepository.findOrCreatePurchaseDiscountAccount({}, trx); + + const otherExpensesAccount = + await accountRepository.findOrCreateOtherExpensesAccount({}, trx); // Saves the vendor credit GL entries. const ledgerEntries = this.getVendorCreditGLEntries( vendorCredit, - APAccount.id + APAccount.id, + purchaseDiscountAccount.id, + otherExpensesAccount.id ); const ledger = new Ledger(ledgerEntries);