mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-19 14:20:31 +00:00
fix: sync inventory items quantity with services.
This commit is contained in:
@@ -29,4 +29,9 @@ export interface IInventoryLotCost {
|
|||||||
transactionType: string,
|
transactionType: string,
|
||||||
transactionId: number,
|
transactionId: number,
|
||||||
entryId: number
|
entryId: number
|
||||||
}
|
};
|
||||||
|
|
||||||
|
export interface IItemsQuantityChanges {
|
||||||
|
itemId: number,
|
||||||
|
balanceChange: number,
|
||||||
|
};
|
||||||
@@ -7,19 +7,16 @@ import 'subscribers/manualJournals';
|
|||||||
import 'subscribers/expenses';
|
import 'subscribers/expenses';
|
||||||
|
|
||||||
import 'subscribers/Bills';
|
import 'subscribers/Bills';
|
||||||
import 'subscribers/Bills/SyncItemsQuantity';
|
|
||||||
import 'subscribers/Bills/SyncVendorsBalances';
|
import 'subscribers/Bills/SyncVendorsBalances';
|
||||||
import 'subscribers/Bills/WriteJournalEntries';
|
import 'subscribers/Bills/WriteJournalEntries';
|
||||||
import 'subscribers/Bills/WriteInventoryTransactions';
|
import 'subscribers/Bills/WriteInventoryTransactions';
|
||||||
|
|
||||||
import 'subscribers/SaleInvoices';
|
import 'subscribers/SaleInvoices';
|
||||||
import 'subscribers/SaleInvoices/SyncCustomersBalance';
|
import 'subscribers/SaleInvoices/SyncCustomersBalance';
|
||||||
import 'subscribers/SaleInvoices/SyncItemsQuantity';
|
|
||||||
import 'subscribers/SaleInvoices/WriteInventoryTransactions';
|
import 'subscribers/SaleInvoices/WriteInventoryTransactions';
|
||||||
import 'subscribers/SaleInvoices/WriteJournalEntries';
|
import 'subscribers/SaleInvoices/WriteJournalEntries';
|
||||||
|
|
||||||
import 'subscribers/SaleReceipt';
|
import 'subscribers/SaleReceipt';
|
||||||
import 'subscribers/SaleReceipt/SyncItemsQuantity';
|
|
||||||
import 'subscribers/SaleReceipt/WriteInventoryTransactions';
|
import 'subscribers/SaleReceipt/WriteInventoryTransactions';
|
||||||
import 'subscribers/SaleReceipt/WriteJournalEntries';
|
import 'subscribers/SaleReceipt/WriteJournalEntries';
|
||||||
|
|
||||||
|
|||||||
@@ -183,7 +183,7 @@ export default class InventoryAdjustmentService {
|
|||||||
inventoryAdjustmentId: number
|
inventoryAdjustmentId: number
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
// Retrieve the inventory adjustment or throw not found service error.
|
// Retrieve the inventory adjustment or throw not found service error.
|
||||||
const adjustment = await this.getInventoryAdjustmentOrThrowError(
|
const oldInventoryAdjustment = await this.getInventoryAdjustmentOrThrowError(
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryAdjustmentId
|
inventoryAdjustmentId
|
||||||
);
|
);
|
||||||
@@ -208,6 +208,7 @@ export default class InventoryAdjustmentService {
|
|||||||
await this.eventDispatcher.dispatch(events.inventoryAdjustment.onDeleted, {
|
await this.eventDispatcher.dispatch(events.inventoryAdjustment.onDeleted, {
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryAdjustmentId,
|
inventoryAdjustmentId,
|
||||||
|
oldInventoryAdjustment
|
||||||
});
|
});
|
||||||
this.logger.info(
|
this.logger.info(
|
||||||
'[inventory_adjustment] the adjustment deleted successfully.',
|
'[inventory_adjustment] the adjustment deleted successfully.',
|
||||||
@@ -317,7 +318,7 @@ export default class InventoryAdjustmentService {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
// Saves the given inventory transactions to the storage.
|
// Saves the given inventory transactions to the storage.
|
||||||
this.inventoryService.recordInventoryTransactions(
|
await this.inventoryService.recordInventoryTransactions(
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryTransactions,
|
inventoryTransactions,
|
||||||
override
|
override
|
||||||
|
|||||||
92
server/src/services/Inventory/InventoryItemsQuantitySync.ts
Normal file
92
server/src/services/Inventory/InventoryItemsQuantitySync.ts
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
import { Inject, Service } from 'typedi';
|
||||||
|
import { toSafeInteger } from 'lodash';
|
||||||
|
import { IInventoryTransaction, IItemsQuantityChanges } from 'interfaces';
|
||||||
|
import HasTenancyService from 'services/Tenancy/TenancyService';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Syncs the inventory transactions with inventory items quantity.
|
||||||
|
*/
|
||||||
|
@Service()
|
||||||
|
export default class InventoryItemsQuantitySync {
|
||||||
|
@Inject()
|
||||||
|
tenancy: HasTenancyService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverse the given inventory transactions.
|
||||||
|
* @param {IInventoryTransaction[]} inventroyTransactions
|
||||||
|
* @return {IInventoryTransaction[]}
|
||||||
|
*/
|
||||||
|
reverseInventoryTransactions(
|
||||||
|
inventroyTransactions: IInventoryTransaction[]
|
||||||
|
): IInventoryTransaction[] {
|
||||||
|
return inventroyTransactions.map((transaction) => ({
|
||||||
|
...transaction,
|
||||||
|
direction: transaction.direction === 'OUT' ? 'IN' : 'OUT',
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverses the inventory transactions.
|
||||||
|
* @param {IInventoryTransaction[]} inventroyTransactions -
|
||||||
|
* @return {IItemsQuantityChanges[]}
|
||||||
|
*/
|
||||||
|
getReverseItemsQuantityChanges(
|
||||||
|
inventroyTransactions: IInventoryTransaction[]
|
||||||
|
): IItemsQuantityChanges[] {
|
||||||
|
const reversedTransactions = this.reverseInventoryTransactions(
|
||||||
|
inventroyTransactions
|
||||||
|
);
|
||||||
|
return this.getItemsQuantityChanges(reversedTransactions);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve the items quantity changes from the given inventory transactions.
|
||||||
|
* @param {IInventoryTransaction[]} inventroyTransactions - Inventory transactions.
|
||||||
|
* @return {IItemsQuantityChanges[]}
|
||||||
|
*/
|
||||||
|
getItemsQuantityChanges(
|
||||||
|
inventroyTransactions: IInventoryTransaction[]
|
||||||
|
): IItemsQuantityChanges[] {
|
||||||
|
const balanceMap: { [itemId: number]: number } = {};
|
||||||
|
|
||||||
|
inventroyTransactions.forEach(
|
||||||
|
(inventoryTransaction: IInventoryTransaction) => {
|
||||||
|
const { itemId, direction, quantity } = inventoryTransaction;
|
||||||
|
|
||||||
|
if (!balanceMap[itemId]) {
|
||||||
|
balanceMap[itemId] = 0;
|
||||||
|
}
|
||||||
|
balanceMap[itemId] += direction === 'IN' ? quantity : 0;
|
||||||
|
balanceMap[itemId] -= direction === 'OUT' ? quantity : 0;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return Object.entries(balanceMap).map(([itemId, balanceChange]) => ({
|
||||||
|
itemId: toSafeInteger(itemId),
|
||||||
|
balanceChange,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes the items quantity changes.
|
||||||
|
* @param {IItemsQuantityChanges[]} itemsQuantity - Items quantity changes.
|
||||||
|
* @return {Promise<void>}
|
||||||
|
*/
|
||||||
|
async changeItemsQuantity(
|
||||||
|
tenantId: number,
|
||||||
|
itemsQuantity: IItemsQuantityChanges[]
|
||||||
|
): Promise<void> {
|
||||||
|
const { itemRepository } = this.tenancy.repositories(tenantId);
|
||||||
|
const opers = [];
|
||||||
|
|
||||||
|
itemsQuantity.forEach((itemQuantity: IItemsQuantityChanges) => {
|
||||||
|
const changeQuantityOper = itemRepository.changeNumber(
|
||||||
|
{ id: itemQuantity.itemId, type: 'inventory' },
|
||||||
|
'quantityOnHand',
|
||||||
|
itemQuantity.balanceChange
|
||||||
|
);
|
||||||
|
opers.push(changeQuantityOper);
|
||||||
|
});
|
||||||
|
await Promise.all(opers);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
import { Container } from 'typedi';
|
|
||||||
import { EventSubscriber, On } from 'event-dispatch';
|
|
||||||
import events from 'subscribers/events';
|
|
||||||
import TenancyService from 'services/Tenancy/TenancyService';
|
|
||||||
import ItemsEntriesService from 'services/Items/ItemsEntriesService';
|
|
||||||
|
|
||||||
@EventSubscriber()
|
|
||||||
export default class BillSubscriber {
|
|
||||||
tenancy: TenancyService;
|
|
||||||
logger: any;
|
|
||||||
itemsEntriesService: ItemsEntriesService;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor method.
|
|
||||||
*/
|
|
||||||
constructor() {
|
|
||||||
this.tenancy = Container.get(TenancyService);
|
|
||||||
this.logger = Container.get('logger');
|
|
||||||
this.itemsEntriesService = Container.get(ItemsEntriesService);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Increments the sale invoice items once the invoice created.
|
|
||||||
*/
|
|
||||||
@On(events.bill.onCreated)
|
|
||||||
public async handleDecrementSaleInvoiceItemsQuantity({ tenantId, bill }) {
|
|
||||||
await this.itemsEntriesService.incrementItemsEntries(
|
|
||||||
tenantId,
|
|
||||||
bill.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrements the sale invoice items once the invoice deleted.
|
|
||||||
*/
|
|
||||||
@On(events.bill.onDeleted)
|
|
||||||
public async handleIncrementSaleInvoiceItemsQuantity({ tenantId, oldBill }) {
|
|
||||||
await this.itemsEntriesService.decrementItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
oldBill.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle increment/decrement the different items quantity once the sale invoice be edited.
|
|
||||||
*/
|
|
||||||
@On(events.bill.onEdited)
|
|
||||||
public async handleChangeSaleInvoiceItemsQuantityOnEdit({
|
|
||||||
tenantId,
|
|
||||||
bill,
|
|
||||||
oldBill,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.changeItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
bill.entries,
|
|
||||||
oldBill.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,16 +3,21 @@ import { EventSubscriber, On } from 'event-dispatch';
|
|||||||
import { map, head } from 'lodash';
|
import { map, head } from 'lodash';
|
||||||
import events from 'subscribers/events';
|
import events from 'subscribers/events';
|
||||||
import SaleInvoicesCost from 'services/Sales/SalesInvoicesCost';
|
import SaleInvoicesCost from 'services/Sales/SalesInvoicesCost';
|
||||||
|
import InventoryItemsQuantitySync from 'services/Inventory/InventoryItemsQuantitySync';
|
||||||
|
import { InventoryTransaction } from 'models';
|
||||||
|
|
||||||
@EventSubscriber()
|
@EventSubscriber()
|
||||||
export class InventorySubscriber {
|
export class InventorySubscriber {
|
||||||
depends: number = 0;
|
depends: number = 0;
|
||||||
startingDate: Date;
|
startingDate: Date;
|
||||||
saleInvoicesCost: SaleInvoicesCost;
|
saleInvoicesCost: SaleInvoicesCost;
|
||||||
|
|
||||||
|
itemsQuantitySync: InventoryItemsQuantitySync;
|
||||||
agenda: any;
|
agenda: any;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.saleInvoicesCost = Container.get(SaleInvoicesCost);
|
this.saleInvoicesCost = Container.get(SaleInvoicesCost);
|
||||||
|
this.itemsQuantitySync = Container.get(InventoryItemsQuantitySync);
|
||||||
this.agenda = Container.get('agenda');
|
this.agenda = Container.get('agenda');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -30,20 +35,54 @@ export class InventorySubscriber {
|
|||||||
if (dependsComputeJobs.length === 0) {
|
if (dependsComputeJobs.length === 0) {
|
||||||
this.startingDate = null;
|
this.startingDate = null;
|
||||||
|
|
||||||
await this.saleInvoicesCost.scheduleWriteJournalEntries(
|
await this.saleInvoicesCost.scheduleWriteJournalEntries(
|
||||||
tenantId,
|
tenantId,
|
||||||
startingDate
|
startingDate
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sync inventory items quantity once inventory transactions created.
|
||||||
|
*/
|
||||||
|
@On(events.inventory.onInventoryTransactionsCreated)
|
||||||
|
async syncItemsQuantityOnceInventoryTransactionsCreated({
|
||||||
|
tenantId,
|
||||||
|
inventoryTransactions,
|
||||||
|
}) {
|
||||||
|
const itemsQuantityChanges = this.itemsQuantitySync.getItemsQuantityChanges(
|
||||||
|
inventoryTransactions
|
||||||
|
);
|
||||||
|
await this.itemsQuantitySync.changeItemsQuantity(
|
||||||
|
tenantId,
|
||||||
|
itemsQuantityChanges
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sync inventory items quantity once inventory transactions deleted.
|
||||||
|
*/
|
||||||
|
@On(events.inventory.onInventoryTransactionsDeleted)
|
||||||
|
async syncItemsQuantityOnceInventoryTransactionsDeleted({
|
||||||
|
tenantId,
|
||||||
|
oldInventoryTransactions,
|
||||||
|
}) {
|
||||||
|
const itemsQuantityChanges = this.itemsQuantitySync.getReverseItemsQuantityChanges(
|
||||||
|
oldInventoryTransactions
|
||||||
|
);
|
||||||
|
await this.itemsQuantitySync.changeItemsQuantity(
|
||||||
|
tenantId,
|
||||||
|
itemsQuantityChanges
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@On(events.inventory.onInventoryTransactionsCreated)
|
@On(events.inventory.onInventoryTransactionsCreated)
|
||||||
async handleScheduleItemsCostOnInventoryTransactionsCreated({
|
async handleScheduleItemsCostOnInventoryTransactionsCreated({
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryTransactions
|
inventoryTransactions,
|
||||||
}) {
|
}) {
|
||||||
const inventoryItemsIds = map(inventoryTransactions, 'itemId');
|
const inventoryItemsIds = map(inventoryTransactions, 'itemId');
|
||||||
|
|
||||||
@@ -61,7 +100,7 @@ export class InventorySubscriber {
|
|||||||
tenantId,
|
tenantId,
|
||||||
transactionType,
|
transactionType,
|
||||||
transactionId,
|
transactionId,
|
||||||
oldInventoryTransactions
|
oldInventoryTransactions,
|
||||||
}) {
|
}) {
|
||||||
// Ignore compute item cost with theses transaction types.
|
// Ignore compute item cost with theses transaction types.
|
||||||
const ignoreWithTransactionTypes = ['OpeningItem'];
|
const ignoreWithTransactionTypes = ['OpeningItem'];
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export default class InventoryAdjustmentsSubscriber {
|
|||||||
await this.inventoryAdjustment.writeInventoryTransactions(
|
await this.inventoryAdjustment.writeInventoryTransactions(
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryAdjustment
|
inventoryAdjustment
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -43,10 +43,10 @@ export default class InventoryAdjustmentsSubscriber {
|
|||||||
async handleRevertInventoryTransactionsOnceDeleted({
|
async handleRevertInventoryTransactionsOnceDeleted({
|
||||||
tenantId,
|
tenantId,
|
||||||
inventoryAdjustmentId,
|
inventoryAdjustmentId,
|
||||||
oldInventoryTransaction,
|
oldInventoryAdjustment,
|
||||||
}) {
|
}) {
|
||||||
// Can't continue if the inventory adjustment is not published.
|
// Can't continue if the inventory adjustment is not published.
|
||||||
if (!oldInventoryTransaction.isPublished) { return; }
|
if (!oldInventoryAdjustment.isPublished) { return; }
|
||||||
|
|
||||||
await this.inventoryAdjustment.revertInventoryTransactions(
|
await this.inventoryAdjustment.revertInventoryTransactions(
|
||||||
tenantId,
|
tenantId,
|
||||||
|
|||||||
@@ -1,71 +0,0 @@
|
|||||||
import { Container } from 'typedi';
|
|
||||||
import { On, EventSubscriber } from 'event-dispatch';
|
|
||||||
import events from 'subscribers/events';
|
|
||||||
import TenancyService from 'services/Tenancy/TenancyService';
|
|
||||||
import ItemsEntriesService from 'services/Items/ItemsEntriesService';
|
|
||||||
|
|
||||||
@EventSubscriber()
|
|
||||||
export default class SyncItemsQuantityWithInvoices {
|
|
||||||
logger: any;
|
|
||||||
tenancy: TenancyService;
|
|
||||||
itemsEntriesService: ItemsEntriesService;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor method.
|
|
||||||
*/
|
|
||||||
constructor() {
|
|
||||||
this.logger = Container.get('logger');
|
|
||||||
this.tenancy = Container.get(TenancyService);
|
|
||||||
this.itemsEntriesService = Container.get(ItemsEntriesService);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Increments the sale invoice items once the invoice created.
|
|
||||||
*/
|
|
||||||
@On(events.saleInvoice.onCreated)
|
|
||||||
public async handleDecrementSaleInvoiceItemsQuantity({
|
|
||||||
tenantId,
|
|
||||||
saleInvoice,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.decrementItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
saleInvoice.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrements the sale invoice items once the invoice deleted.
|
|
||||||
*/
|
|
||||||
@On(events.saleInvoice.onDeleted)
|
|
||||||
public async handleIncrementSaleInvoiceItemsQuantity({
|
|
||||||
tenantId,
|
|
||||||
oldSaleInvoice,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.incrementItemsEntries(
|
|
||||||
tenantId,
|
|
||||||
oldSaleInvoice.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle increment/decrement the different items quantity once the sale invoice be edited.
|
|
||||||
*/
|
|
||||||
@On(events.saleInvoice.onEdited)
|
|
||||||
public async handleChangeSaleInvoiceItemsQuantityOnEdit({
|
|
||||||
tenantId,
|
|
||||||
saleInvoice,
|
|
||||||
oldSaleInvoice,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.changeItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
saleInvoice.entries.map((entry) => ({
|
|
||||||
...entry,
|
|
||||||
quantity: entry.quantity * -1,
|
|
||||||
})),
|
|
||||||
oldSaleInvoice.entries.map((entry) => ({
|
|
||||||
...entry,
|
|
||||||
quantity: entry.quantity * -1,
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
import { Container } from 'typedi';
|
|
||||||
import { On, EventSubscriber } from 'event-dispatch';
|
|
||||||
import events from 'subscribers/events';
|
|
||||||
import TenancyService from 'services/Tenancy/TenancyService';
|
|
||||||
import ItemsEntriesService from 'services/Items/ItemsEntriesService';
|
|
||||||
import SalesInvoicesCost from 'services/Sales/SalesInvoicesCost';
|
|
||||||
|
|
||||||
@EventSubscriber()
|
|
||||||
export default class SaleReceiptSubscriber {
|
|
||||||
logger: any;
|
|
||||||
tenancy: TenancyService;
|
|
||||||
itemsEntriesService: ItemsEntriesService;
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this.logger = Container.get('logger');
|
|
||||||
this.tenancy = Container.get(TenancyService);
|
|
||||||
this.itemsEntriesService = Container.get(ItemsEntriesService);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Increments the sale receipt items once be created.
|
|
||||||
*/
|
|
||||||
@On(events.saleReceipt.onCreated)
|
|
||||||
public async handleDecremenReceiptItemsQuantity({ tenantId, saleReceipt }) {
|
|
||||||
await this.itemsEntriesService.decrementItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
saleReceipt.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrements the sale receipt items once be deleted.
|
|
||||||
*/
|
|
||||||
@On(events.saleReceipt.onDeleted)
|
|
||||||
public async handleIncrementReceiptItemsQuantity({
|
|
||||||
tenantId,
|
|
||||||
oldSaleReceipt,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.incrementItemsEntries(
|
|
||||||
tenantId,
|
|
||||||
oldSaleReceipt.entries
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handle increment/decrement the different items quantity once
|
|
||||||
* the sale receipt be edited.
|
|
||||||
*/
|
|
||||||
@On(events.saleReceipt.onEdited)
|
|
||||||
public async handleChangeSaleInvoiceItemsQuantityOnEdit({
|
|
||||||
tenantId,
|
|
||||||
saleReceipt,
|
|
||||||
oldSaleReceipt,
|
|
||||||
}) {
|
|
||||||
await this.itemsEntriesService.changeItemsQuantity(
|
|
||||||
tenantId,
|
|
||||||
saleReceipt.entries.map((entry) => ({
|
|
||||||
...entry,
|
|
||||||
quantity: entry.quantity * -1,
|
|
||||||
})),
|
|
||||||
oldSaleReceipt.entries.map((entry) => ({
|
|
||||||
...entry,
|
|
||||||
quantity: entry.quantity * -1,
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user