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

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

300 lines
6.6 KiB
TypeScript
Raw Normal View History

2022-11-09 18:42:00 +00:00
import { Fyo } from 'fyo';
import { StockQueue } from 'models/inventory/stockQueue';
import { ValuationMethod } from 'models/inventory/types';
2022-11-09 18:42:00 +00:00
import { ModelNameEnum } from 'models/types';
import { safeParseFloat, safeParseInt } from 'utils/index';
import type {
2022-11-25 09:21:58 +00:00
ComputedStockLedgerEntry,
RawStockLedgerEntry,
StockBalanceEntry,
} from './types';
import type { QueryFilter } from 'utils/db/types';
import type { StockTransfer } from 'models/inventory/StockTransfer';
2022-11-25 09:21:58 +00:00
type Item = string;
type Location = string;
2023-02-28 06:01:04 +00:00
type Batch = string;
2022-11-09 18:42:00 +00:00
export async function getRawStockLedgerEntries(
fyo: Fyo,
filters: QueryFilter = {}
) {
2022-11-09 18:42:00 +00:00
const fieldnames = [
'name',
'date',
'item',
2023-02-28 06:01:04 +00:00
'batch',
2023-05-04 10:45:12 +00:00
'serialNumber',
2022-11-09 18:42:00 +00:00
'rate',
'quantity',
'location',
'referenceName',
'referenceType',
];
return (await fyo.db.getAllRaw(ModelNameEnum.StockLedgerEntry, {
fields: fieldnames,
filters,
2023-01-05 05:44:05 +00:00
orderBy: ['date', 'created', 'name'],
2022-11-09 18:42:00 +00:00
order: 'asc',
})) as RawStockLedgerEntry[];
}
export async function getShipmentCOGSAmountFromSLEs(
stockTransfer: StockTransfer
) {
const fyo = stockTransfer.fyo;
const date = stockTransfer.date ?? new Date();
const items = (stockTransfer.items ?? []).filter((i) => i.item);
const itemNames = Array.from(new Set(items.map((i) => i.item))) as string[];
type Item = string;
type Batch = string;
type Location = string;
type Queues = Record<Item, Record<Location, Record<Batch, StockQueue>>>;
const rawSles = await getRawStockLedgerEntries(fyo, {
item: ['in', itemNames],
date: ['<=', date.toISOString()],
});
const q: Queues = {};
for (const sle of rawSles) {
const i = sle.item;
const l = sle.location;
const b = sle.batch ?? '-';
q[i] ??= {};
q[i][l] ??= {};
q[i][l][b] ??= new StockQueue();
const sq = q[i][l][b];
if (sle.quantity > 0) {
const rate = fyo.pesa(sle.rate);
sq.inward(rate.float, sle.quantity);
} else {
sq.outward(-sle.quantity);
}
}
let total = fyo.pesa(0);
for (const item of items) {
const i = item.item ?? '-';
const l = item.location ?? '-';
const b = item.batch ?? '-';
const sq = q[i][l][b];
const stAmount = item.amount ?? 0;
if (!sq) {
total = total.add(stAmount);
}
const stRate = item.rate?.float ?? 0;
const stQuantity = item.quantity ?? 0;
const rate = sq.outward(stQuantity) ?? stRate;
const amount = rate * stQuantity;
total = total.add(amount);
}
return total;
}
2022-11-09 18:42:00 +00:00
export function getStockLedgerEntries(
rawSLEs: RawStockLedgerEntry[],
valuationMethod: ValuationMethod
2022-11-09 18:42:00 +00:00
): ComputedStockLedgerEntry[] {
const computedSLEs: ComputedStockLedgerEntry[] = [];
const stockQueues: Record<
Item,
2023-02-28 06:01:04 +00:00
Record<Location, Record<Batch, StockQueue>>
> = {};
2022-11-09 18:42:00 +00:00
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 06:01:04 +00:00
const batch = sle.batch ?? '';
2023-05-04 10:45:12 +00:00
const serialNumber = sle.serialNumber ?? '';
2022-11-09 18:42:00 +00:00
if (quantity === 0) {
continue;
}
stockQueues[item] ??= {};
stockQueues[item][location] ??= {};
2023-02-28 06:01:04 +00:00
stockQueues[item][location][batch] ??= new StockQueue();
2022-11-09 18:42:00 +00:00
2023-02-28 06:01:04 +00:00
const q = stockQueues[item][location][batch];
2022-11-09 18:42:00 +00:00
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-09 18:42:00 +00:00
const balanceValue = q.value;
const valueChange = balanceValue - initialValue;
const csle: ComputedStockLedgerEntry = {
name,
date,
item,
location,
2023-02-28 06:01:04 +00:00
batch,
2023-05-04 10:45:12 +00:00
serialNumber,
2023-02-17 05:23:16 +00:00
2022-11-09 18:42:00 +00:00
quantity,
balanceQuantity,
incomingRate,
valuationRate,
balanceValue,
valueChange,
referenceName,
referenceType,
};
computedSLEs.push(csle);
}
return computedSLEs;
}
2022-11-25 09:21:58 +00:00
export function getStockBalanceEntries(
computedSLEs: ComputedStockLedgerEntry[],
filters: {
item?: string;
location?: string;
fromDate?: string;
toDate?: string;
batch?: string;
2022-11-25 09:21:58 +00:00
}
): StockBalanceEntry[] {
const sbeMap: Record<
Item,
2023-02-28 06:01:04 +00:00
Record<Location, Record<Batch, StockBalanceEntry>>
> = {};
2022-11-25 09:21:58 +00:00
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 06:01:04 +00:00
const batch = sle.batch || '';
2022-11-25 09:21:58 +00:00
sbeMap[sle.item] ??= {};
sbeMap[sle.item][sle.location] ??= {};
2023-02-28 06:01:04 +00:00
sbeMap[sle.item][sle.location][batch] ??= getSBE(
2023-01-18 12:07:31 +00:00
sle.item,
sle.location,
2023-02-28 06:01:04 +00:00
batch
2023-01-18 12:07:31 +00:00
);
2022-11-25 09:21:58 +00:00
const date = sle.date.valueOf();
if (fromDate && date < fromDate) {
2023-02-28 06:01:04 +00:00
const sbe = sbeMap[sle.item][sle.location][batch];
2022-11-25 09:21:58 +00:00
updateOpeningBalances(sbe, sle);
continue;
}
if (toDate && date > toDate) {
continue;
}
2023-02-28 06:01:04 +00:00
const sbe = sbeMap[sle.item][sle.location][batch];
2022-11-25 09:21:58 +00:00
updateCurrentBalances(sbe, sle);
}
return Object.values(sbeMap)
.map((sbeBatched) =>
Object.values(sbeBatched).map((sbes) => Object.values(sbes))
)
.flat(2);
2022-11-25 09:21:58 +00:00
}
2023-01-18 12:07:31 +00:00
function getSBE(
item: string,
location: string,
2023-02-28 06:01:04 +00:00
batch: string
2023-01-18 12:07:31 +00:00
): StockBalanceEntry {
2022-11-25 09:21:58 +00:00
return {
name: 0,
item,
location,
2023-02-28 06:01:04 +00:00
batch,
2022-11-25 09:21:58 +00:00
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;
}