2022-11-21 12:05:05 +05:30
|
|
|
import { Fyo, t } from 'fyo';
|
2022-11-14 14:00:11 +05:30
|
|
|
import { Attachment } from 'fyo/core/types';
|
|
|
|
import { Doc } from 'fyo/model/doc';
|
2023-02-21 12:12:06 +05:30
|
|
|
import {
|
|
|
|
Action,
|
2023-03-29 11:44:59 +05:30
|
|
|
ChangeArg,
|
2023-02-21 12:12:06 +05:30
|
|
|
DefaultMap,
|
|
|
|
FiltersMap,
|
|
|
|
FormulaMap,
|
|
|
|
HiddenMap,
|
|
|
|
} from 'fyo/model/types';
|
2022-11-21 12:05:05 +05:30
|
|
|
import { ValidationError } from 'fyo/utils/errors';
|
2023-05-04 16:15:12 +05:30
|
|
|
import { LedgerPosting } from 'models/Transactional/LedgerPosting';
|
2022-11-18 23:01:50 +05:30
|
|
|
import { Defaults } from 'models/baseModels/Defaults/Defaults';
|
2022-11-22 14:42:49 +05:30
|
|
|
import { Invoice } from 'models/baseModels/Invoice/Invoice';
|
2023-01-16 15:08:02 +05:30
|
|
|
import { addItem, getLedgerLinkAction, getNumberSeries } from 'models/helpers';
|
2022-11-18 23:01:50 +05:30
|
|
|
import { ModelNameEnum } from 'models/types';
|
|
|
|
import { Money } from 'pesa';
|
2023-03-29 11:44:59 +05:30
|
|
|
import { TargetField } from 'schemas/types';
|
2023-05-05 11:13:04 +05:30
|
|
|
import { SerialNumber } from './SerialNumber';
|
2022-11-18 23:01:50 +05:30
|
|
|
import { StockTransferItem } from './StockTransferItem';
|
|
|
|
import { Transfer } from './Transfer';
|
2023-05-04 16:15:12 +05:30
|
|
|
import {
|
2023-05-05 11:13:04 +05:30
|
|
|
canValidateSerialNumber,
|
2023-05-04 16:15:12 +05:30
|
|
|
getSerialNumberFromDoc,
|
|
|
|
updateSerialNumbers,
|
|
|
|
validateBatch,
|
|
|
|
validateSerialNumber,
|
|
|
|
} from './helpers';
|
2022-11-14 14:00:11 +05:30
|
|
|
|
2022-11-18 23:01:50 +05:30
|
|
|
export abstract class StockTransfer extends Transfer {
|
2022-11-14 14:00:11 +05:30
|
|
|
name?: string;
|
2022-11-18 23:01:50 +05:30
|
|
|
date?: Date;
|
2022-11-14 14:00:11 +05:30
|
|
|
party?: string;
|
|
|
|
terms?: string;
|
|
|
|
attachment?: Attachment;
|
2022-11-18 23:01:50 +05:30
|
|
|
grandTotal?: Money;
|
2022-11-22 14:42:49 +05:30
|
|
|
backReference?: string;
|
2022-11-18 23:01:50 +05:30
|
|
|
items?: StockTransferItem[];
|
|
|
|
|
|
|
|
get isSales() {
|
|
|
|
return this.schemaName === ModelNameEnum.Shipment;
|
|
|
|
}
|
|
|
|
|
|
|
|
formulas: FormulaMap = {
|
|
|
|
grandTotal: {
|
|
|
|
formula: () => this.getSum('items', 'amount', false),
|
|
|
|
dependsOn: ['items'],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2023-02-21 12:12:06 +05:30
|
|
|
hidden: HiddenMap = {
|
|
|
|
backReference: () =>
|
|
|
|
!(this.backReference || !(this.isSubmitted || this.isCancelled)),
|
|
|
|
terms: () => !(this.terms || !(this.isSubmitted || this.isCancelled)),
|
|
|
|
attachment: () =>
|
|
|
|
!(this.attachment || !(this.isSubmitted || this.isCancelled)),
|
|
|
|
};
|
|
|
|
|
2022-11-18 23:01:50 +05:30
|
|
|
static defaults: DefaultMap = {
|
|
|
|
numberSeries: (doc) => getNumberSeries(doc.schemaName, doc.fyo),
|
|
|
|
terms: (doc) => {
|
|
|
|
const defaults = doc.fyo.singles.Defaults as Defaults | undefined;
|
|
|
|
if (doc.schemaName === ModelNameEnum.Shipment) {
|
|
|
|
return defaults?.shipmentTerms ?? '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return defaults?.purchaseReceiptTerms ?? '';
|
|
|
|
},
|
2022-12-05 15:31:31 +05:30
|
|
|
date: () => new Date(),
|
2022-11-18 23:01:50 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
static filters: FiltersMap = {
|
|
|
|
party: (doc: Doc) => ({
|
|
|
|
role: ['in', [doc.isSales ? 'Customer' : 'Supplier', 'Both']],
|
|
|
|
}),
|
|
|
|
numberSeries: (doc: Doc) => ({ referenceType: doc.schemaName }),
|
2023-03-29 11:44:59 +05:30
|
|
|
backReference: () => ({
|
2023-03-29 12:15:47 +05:30
|
|
|
stockNotTransferred: ['!=', 0],
|
2023-03-29 11:44:59 +05:30
|
|
|
submitted: true,
|
|
|
|
cancelled: false,
|
|
|
|
}),
|
2022-11-18 23:01:50 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
override _getTransferDetails() {
|
|
|
|
return (this.items ?? []).map((row) => {
|
|
|
|
let fromLocation = undefined;
|
|
|
|
let toLocation = undefined;
|
|
|
|
|
|
|
|
if (this.isSales) {
|
|
|
|
fromLocation = row.location;
|
|
|
|
} else {
|
|
|
|
toLocation = row.location;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
item: row.item!,
|
|
|
|
rate: row.rate!,
|
|
|
|
quantity: row.quantity!,
|
2023-02-28 11:31:04 +05:30
|
|
|
batch: row.batch!,
|
2023-05-04 16:15:12 +05:30
|
|
|
serialNumber: row.serialNumber!,
|
2022-11-18 23:01:50 +05:30
|
|
|
fromLocation,
|
|
|
|
toLocation,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
override async getPosting(): Promise<LedgerPosting | null> {
|
|
|
|
await this.validateAccounts();
|
|
|
|
const stockInHand = (await this.fyo.getValue(
|
|
|
|
ModelNameEnum.InventorySettings,
|
|
|
|
'stockInHand'
|
|
|
|
)) as string;
|
|
|
|
|
|
|
|
const amount = this.grandTotal ?? this.fyo.pesa(0);
|
|
|
|
const posting = new LedgerPosting(this, this.fyo);
|
|
|
|
|
|
|
|
if (this.isSales) {
|
|
|
|
const costOfGoodsSold = (await this.fyo.getValue(
|
|
|
|
ModelNameEnum.InventorySettings,
|
|
|
|
'costOfGoodsSold'
|
|
|
|
)) as string;
|
|
|
|
|
|
|
|
await posting.debit(costOfGoodsSold, amount);
|
|
|
|
await posting.credit(stockInHand, amount);
|
|
|
|
} else {
|
|
|
|
const stockReceivedButNotBilled = (await this.fyo.getValue(
|
|
|
|
ModelNameEnum.InventorySettings,
|
|
|
|
'stockReceivedButNotBilled'
|
|
|
|
)) as string;
|
|
|
|
|
|
|
|
await posting.debit(stockInHand, amount);
|
|
|
|
await posting.credit(stockReceivedButNotBilled, amount);
|
|
|
|
}
|
|
|
|
|
2022-11-21 12:05:05 +05:30
|
|
|
await posting.makeRoundOffEntry();
|
2022-11-18 23:01:50 +05:30
|
|
|
return posting;
|
|
|
|
}
|
|
|
|
|
|
|
|
async validateAccounts() {
|
|
|
|
const settings: string[] = ['stockInHand'];
|
|
|
|
if (this.isSales) {
|
|
|
|
settings.push('costOfGoodsSold');
|
|
|
|
} else {
|
|
|
|
settings.push('stockReceivedButNotBilled');
|
|
|
|
}
|
|
|
|
|
|
|
|
const messages: string[] = [];
|
|
|
|
for (const setting of settings) {
|
|
|
|
const value = this.fyo.singles.InventorySettings?.[setting] as
|
|
|
|
| string
|
|
|
|
| undefined;
|
|
|
|
const field = this.fyo.getField(ModelNameEnum.InventorySettings, setting);
|
|
|
|
if (!value) {
|
|
|
|
messages.push(t`${field.label} account not set in Inventory Settings.`);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const exists = await this.fyo.db.exists(ModelNameEnum.Account, value);
|
|
|
|
if (!exists) {
|
|
|
|
messages.push(t`Account ${value} does not exist.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (messages.length) {
|
|
|
|
throw new ValidationError(messages.join(' '));
|
|
|
|
}
|
|
|
|
}
|
2022-11-21 12:05:05 +05:30
|
|
|
|
2023-02-27 18:39:18 +05:30
|
|
|
override async validate(): Promise<void> {
|
|
|
|
await super.validate();
|
2023-02-28 11:31:04 +05:30
|
|
|
await validateBatch(this);
|
2023-05-04 16:15:12 +05:30
|
|
|
await validateSerialNumber(this);
|
|
|
|
await validateSerialNumberStatus(this);
|
2023-02-27 18:39:18 +05:30
|
|
|
}
|
|
|
|
|
2022-11-21 12:05:05 +05:30
|
|
|
static getActions(fyo: Fyo): Action[] {
|
|
|
|
return [getLedgerLinkAction(fyo, false), getLedgerLinkAction(fyo, true)];
|
|
|
|
}
|
2022-11-22 14:42:49 +05:30
|
|
|
|
|
|
|
async afterSubmit() {
|
|
|
|
await super.afterSubmit();
|
2023-05-04 16:15:12 +05:30
|
|
|
await updateSerialNumbers(this, false);
|
2022-11-22 14:42:49 +05:30
|
|
|
await this._updateBackReference();
|
|
|
|
}
|
|
|
|
|
|
|
|
async afterCancel(): Promise<void> {
|
|
|
|
await super.afterCancel();
|
2023-05-04 16:15:12 +05:30
|
|
|
await updateSerialNumbers(this, true);
|
2022-11-22 14:42:49 +05:30
|
|
|
await this._updateBackReference();
|
|
|
|
}
|
|
|
|
|
|
|
|
async _updateBackReference() {
|
|
|
|
if (!this.isCancelled && !this.isSubmitted) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.backReference) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const schemaName = this.isSales
|
|
|
|
? ModelNameEnum.SalesInvoice
|
|
|
|
: ModelNameEnum.PurchaseInvoice;
|
|
|
|
|
|
|
|
const invoice = (await this.fyo.doc.getDoc(
|
|
|
|
schemaName,
|
|
|
|
this.backReference
|
|
|
|
)) as Invoice;
|
|
|
|
const transferMap = this._getTransferMap();
|
|
|
|
|
|
|
|
for (const row of invoice.items ?? []) {
|
|
|
|
const item = row.item!;
|
|
|
|
const quantity = row.quantity!;
|
|
|
|
const notTransferred = (row.stockNotTransferred as number) ?? 0;
|
|
|
|
|
|
|
|
const transferred = transferMap[item];
|
2022-11-23 13:47:23 +05:30
|
|
|
if (
|
|
|
|
typeof transferred !== 'number' ||
|
|
|
|
typeof notTransferred !== 'number'
|
|
|
|
) {
|
2022-11-22 14:42:49 +05:30
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.isCancelled) {
|
|
|
|
await row.set(
|
|
|
|
'stockNotTransferred',
|
|
|
|
Math.min(notTransferred + transferred, quantity)
|
|
|
|
);
|
|
|
|
transferMap[item] = Math.max(
|
|
|
|
transferred + notTransferred - quantity,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
await row.set(
|
|
|
|
'stockNotTransferred',
|
|
|
|
Math.max(notTransferred - transferred, 0)
|
|
|
|
);
|
|
|
|
transferMap[item] = Math.max(transferred - notTransferred, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const notTransferred = invoice.getStockNotTransferred();
|
|
|
|
await invoice.setAndSync('stockNotTransferred', notTransferred);
|
|
|
|
}
|
|
|
|
|
|
|
|
_getTransferMap() {
|
|
|
|
return (this.items ?? []).reduce((acc, item) => {
|
|
|
|
if (!item.item) {
|
|
|
|
return acc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!item.quantity) {
|
|
|
|
return acc;
|
|
|
|
}
|
|
|
|
|
|
|
|
acc[item.item] ??= 0;
|
|
|
|
acc[item.item] += item.quantity;
|
|
|
|
|
|
|
|
return acc;
|
|
|
|
}, {} as Record<string, number>);
|
|
|
|
}
|
2022-11-23 13:47:23 +05:30
|
|
|
|
|
|
|
override duplicate(): Doc {
|
|
|
|
const doc = super.duplicate() as StockTransfer;
|
|
|
|
doc.backReference = undefined;
|
|
|
|
return doc;
|
|
|
|
}
|
2022-12-01 14:01:23 +05:30
|
|
|
|
|
|
|
static createFilters: FiltersMap = {
|
|
|
|
party: (doc: Doc) => ({
|
|
|
|
role: doc.isSales ? 'Customer' : 'Supplier',
|
|
|
|
}),
|
|
|
|
};
|
2023-01-16 15:08:02 +05:30
|
|
|
|
|
|
|
async addItem(name: string) {
|
|
|
|
return await addItem(name, this);
|
|
|
|
}
|
2023-03-29 11:44:59 +05:30
|
|
|
|
|
|
|
override async change({ doc, changed }: ChangeArg): Promise<void> {
|
|
|
|
if (doc.name === this.name && changed === 'backReference') {
|
|
|
|
await this.setFieldsFromBackReference();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async setFieldsFromBackReference() {
|
|
|
|
const backReference = this.backReference;
|
|
|
|
const { target } = this.fyo.getField(
|
|
|
|
this.schemaName,
|
|
|
|
'backReference'
|
|
|
|
) as TargetField;
|
|
|
|
|
|
|
|
if (!backReference || !target) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const brDoc = await this.fyo.doc.getDoc(target, backReference);
|
|
|
|
if (!(brDoc instanceof Invoice)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const stDoc = await brDoc.getStockTransfer();
|
|
|
|
if (!stDoc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await this.set('party', stDoc.party);
|
|
|
|
await this.set('terms', stDoc.terms);
|
|
|
|
await this.set('date', stDoc.date);
|
|
|
|
await this.set('items', stDoc.items);
|
|
|
|
}
|
2022-11-14 14:00:11 +05:30
|
|
|
}
|
2023-05-04 16:15:12 +05:30
|
|
|
|
|
|
|
async function validateSerialNumberStatus(doc: StockTransfer) {
|
2023-05-08 15:07:13 +05:30
|
|
|
if (doc.isCancelled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-05-05 11:13:04 +05:30
|
|
|
for (const { serialNumber, item } of getSerialNumberFromDoc(doc)) {
|
|
|
|
const cannotValidate = !(await canValidateSerialNumber(item, serialNumber));
|
|
|
|
if (cannotValidate) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-05-04 16:15:12 +05:30
|
|
|
const snDoc = await doc.fyo.doc.getDoc(
|
|
|
|
ModelNameEnum.SerialNumber,
|
|
|
|
serialNumber
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!(snDoc instanceof SerialNumber)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const status = snDoc.status ?? 'Inactive';
|
|
|
|
|
|
|
|
if (
|
|
|
|
doc.schemaName === ModelNameEnum.PurchaseReceipt &&
|
|
|
|
status !== 'Inactive'
|
|
|
|
) {
|
|
|
|
throw new ValidationError(
|
|
|
|
t`Serial Number ${serialNumber} is not Inactive`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (doc.schemaName === ModelNameEnum.Shipment && status !== 'Active') {
|
|
|
|
throw new ValidationError(
|
|
|
|
t`Serial Number ${serialNumber} is not Active.`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|