2
0
mirror of https://github.com/frappe/books.git synced 2025-01-23 15:18:24 +00:00
books/reports/inventory/helpers.ts

233 lines
5.0 KiB
TypeScript
Raw Permalink Normal View History

2022-11-10 00:12:00 +05:30
import { Fyo } from 'fyo';
import { StockQueue } from 'models/inventory/stockQueue';
import { ValuationMethod } from 'models/inventory/types';
2022-11-10 00:12:00 +05:30
import { ModelNameEnum } from 'models/types';
import { safeParseFloat, safeParseInt } from 'utils/index';
2022-11-25 14:51:58 +05:30
import {
ComputedStockLedgerEntry,
RawStockLedgerEntry,
StockBalanceEntry,
} from './types';
type Item = string;
type Location = string;
2023-02-28 11:31:04 +05:30
type Batch = string;
2022-11-10 00:12:00 +05:30
export async function getRawStockLedgerEntries(fyo: Fyo) {
const fieldnames = [
'name',
'date',
'item',
2023-02-28 11:31:04 +05:30
'batch',
2023-05-04 16:15:12 +05:30
'serialNumber',
2022-11-10 00:12:00 +05:30
'rate',
'quantity',
'location',
'referenceName',
'referenceType',
];
return (await fyo.db.getAllRaw(ModelNameEnum.StockLedgerEntry, {
fields: fieldnames,
2023-01-05 11:14:05 +05:30
orderBy: ['date', 'created', 'name'],
2022-11-10 00:12:00 +05:30
order: 'asc',
})) as RawStockLedgerEntry[];
}
export function getStockLedgerEntries(
rawSLEs: RawStockLedgerEntry[],
valuationMethod: ValuationMethod
2022-11-10 00:12:00 +05:30
): ComputedStockLedgerEntry[] {
const computedSLEs: ComputedStockLedgerEntry[] = [];
const stockQueues: Record<
Item,
2023-02-28 11:31:04 +05:30
Record<Location, Record<Batch, StockQueue>>
> = {};
2022-11-10 00:12:00 +05:30
for (const sle of rawSLEs) {
const name = safeParseInt(sle.name);
const date = new Date(sle.date);
const rate = safeParseFloat(sle.rate);
const { item, location, quantity, referenceName, referenceType } = sle;
2023-02-28 11:31:04 +05:30
const batch = sle.batch ?? '';
2023-05-04 16:15:12 +05:30
const serialNumber = sle.serialNumber ?? '';
2022-11-10 00:12:00 +05:30
if (quantity === 0) {
continue;
}
stockQueues[item] ??= {};
stockQueues[item][location] ??= {};
2023-02-28 11:31:04 +05:30
stockQueues[item][location][batch] ??= new StockQueue();
2022-11-10 00:12:00 +05:30
2023-02-28 11:31:04 +05:30
const q = stockQueues[item][location][batch];
2022-11-10 00:12:00 +05:30
const initialValue = q.value;
let incomingRate: number | null;
if (quantity > 0) {
incomingRate = q.inward(rate, quantity);
} else {
incomingRate = q.outward(-quantity);
}
if (incomingRate === null) {
continue;
}
const balanceQuantity = q.quantity;
let valuationRate = q.fifo;
if (valuationMethod === ValuationMethod.MovingAverage) {
valuationRate = q.movingAverage;
}
2022-11-10 00:12:00 +05:30
const balanceValue = q.value;
const valueChange = balanceValue - initialValue;
const csle: ComputedStockLedgerEntry = {
name,
date,
item,
location,
2023-02-28 11:31:04 +05:30
batch,
2023-05-04 16:15:12 +05:30
serialNumber,
2023-02-17 10:53:16 +05:30
2022-11-10 00:12:00 +05:30
quantity,
balanceQuantity,
incomingRate,
valuationRate,
balanceValue,
valueChange,
referenceName,
referenceType,
};
computedSLEs.push(csle);
}
return computedSLEs;
}
2022-11-25 14:51:58 +05:30
export function getStockBalanceEntries(
computedSLEs: ComputedStockLedgerEntry[],
filters: {
item?: string;
location?: string;
fromDate?: string;
toDate?: string;
batch?: string;
2022-11-25 14:51:58 +05:30
}
): StockBalanceEntry[] {
const sbeMap: Record<
Item,
2023-02-28 11:31:04 +05:30
Record<Location, Record<Batch, StockBalanceEntry>>
> = {};
2022-11-25 14:51:58 +05:30
const fromDate = filters.fromDate ? Date.parse(filters.fromDate) : null;
const toDate = filters.toDate ? Date.parse(filters.toDate) : null;
for (const sle of computedSLEs) {
if (filters.item && sle.item !== filters.item) {
continue;
}
if (filters.location && sle.location !== filters.location) {
continue;
}
if (filters.batch && sle.batch !== filters.batch) {
continue;
}
2023-02-28 11:31:04 +05:30
const batch = sle.batch || '';
2022-11-25 14:51:58 +05:30
sbeMap[sle.item] ??= {};
sbeMap[sle.item][sle.location] ??= {};
2023-02-28 11:31:04 +05:30
sbeMap[sle.item][sle.location][batch] ??= getSBE(
2023-01-18 17:37:31 +05:30
sle.item,
sle.location,
2023-02-28 11:31:04 +05:30
batch
2023-01-18 17:37:31 +05:30
);
2022-11-25 14:51:58 +05:30
const date = sle.date.valueOf();
if (fromDate && date < fromDate) {
2023-02-28 11:31:04 +05:30
const sbe = sbeMap[sle.item][sle.location][batch];
2022-11-25 14:51:58 +05:30
updateOpeningBalances(sbe, sle);
continue;
}
if (toDate && date > toDate) {
continue;
}
2023-02-28 11:31:04 +05:30
const sbe = sbeMap[sle.item][sle.location][batch];
2022-11-25 14:51:58 +05:30
updateCurrentBalances(sbe, sle);
}
return Object.values(sbeMap)
.map((sbeBatched) =>
Object.values(sbeBatched).map((sbes) => Object.values(sbes))
)
.flat(2);
2022-11-25 14:51:58 +05:30
}
2023-01-18 17:37:31 +05:30
function getSBE(
item: string,
location: string,
2023-02-28 11:31:04 +05:30
batch: string
2023-01-18 17:37:31 +05:30
): StockBalanceEntry {
2022-11-25 14:51:58 +05:30
return {
name: 0,
item,
location,
2023-02-28 11:31:04 +05:30
batch,
2022-11-25 14:51:58 +05:30
balanceQuantity: 0,
balanceValue: 0,
openingQuantity: 0,
openingValue: 0,
incomingQuantity: 0,
incomingValue: 0,
outgoingQuantity: 0,
outgoingValue: 0,
valuationRate: 0,
};
}
function updateOpeningBalances(
sbe: StockBalanceEntry,
sle: ComputedStockLedgerEntry
) {
sbe.openingQuantity += sle.quantity;
sbe.openingValue += sle.valueChange;
sbe.balanceQuantity += sle.quantity;
sbe.balanceValue += sle.valueChange;
}
function updateCurrentBalances(
sbe: StockBalanceEntry,
sle: ComputedStockLedgerEntry
) {
sbe.balanceQuantity += sle.quantity;
sbe.balanceValue += sle.valueChange;
if (sle.quantity > 0) {
sbe.incomingQuantity += sle.quantity;
sbe.incomingValue += sle.valueChange;
} else {
sbe.outgoingQuantity -= sle.quantity;
sbe.outgoingValue -= sle.valueChange;
}
sbe.valuationRate = sle.valuationRate;
}