mirror of
https://github.com/frappe/books.git
synced 2025-01-25 08:08:37 +00:00
942 lines
25 KiB
TypeScript
942 lines
25 KiB
TypeScript
import {
|
|
assertDoesNotThrow,
|
|
assertThrows,
|
|
} from 'backend/database/tests/helpers';
|
|
import { Invoice } from 'models/baseModels/Invoice/Invoice';
|
|
import { ModelNameEnum } from 'models/types';
|
|
import { RawValue } from 'schemas/types';
|
|
import test from 'tape';
|
|
import { closeTestFyo, getTestFyo, setupTestFyo } from 'tests/helpers';
|
|
import { InventorySettings } from '../InventorySettings';
|
|
import { Shipment } from '../Shipment';
|
|
import { StockTransfer } from '../StockTransfer';
|
|
import { MovementTypeEnum, ValuationMethod } from '../types';
|
|
import { getALEs, getItem, getSLEs, getStockTransfer } from './helpers';
|
|
import { PurchaseReceipt } from '../PurchaseReceipt';
|
|
|
|
const fyo = getTestFyo();
|
|
setupTestFyo(fyo, __filename);
|
|
|
|
const item = 'Pen';
|
|
const location = 'Common';
|
|
const party = 'Someone';
|
|
const testDocs = {
|
|
Item: {
|
|
[item]: getItem(item, 100),
|
|
},
|
|
Location: {
|
|
[location]: { name: location },
|
|
},
|
|
Party: { [party]: { name: party, Role: 'Both' } },
|
|
} as Record<string, Record<string, { name: string; [key: string]: RawValue }>>;
|
|
|
|
test('insert test docs', async (t) => {
|
|
for (const schemaName in testDocs) {
|
|
for (const name in testDocs[schemaName]) {
|
|
await fyo.doc.getNewDoc(schemaName, testDocs[schemaName][name]).sync();
|
|
}
|
|
}
|
|
|
|
t.ok(await fyo.db.exists(ModelNameEnum.Party, party), 'party created');
|
|
t.ok(
|
|
await fyo.db.exists(ModelNameEnum.Location, location),
|
|
'location created'
|
|
);
|
|
t.ok(await fyo.db.exists(ModelNameEnum.Item, item), 'item created');
|
|
});
|
|
|
|
test('inventory settings', async (t) => {
|
|
const doc = (await fyo.doc.getDoc(
|
|
ModelNameEnum.InventorySettings
|
|
)) as InventorySettings;
|
|
|
|
t.ok(doc.stockInHand, 'stock in hand set');
|
|
t.ok(doc.stockReceivedButNotBilled, 'stock rec. but not billed set');
|
|
});
|
|
|
|
test('PurchaseReceipt, create inward stock movement', async (t) => {
|
|
const date = new Date('2022-01-01');
|
|
const rate = (testDocs['Item'][item].rate as number) ?? 0;
|
|
const quantity = 10;
|
|
const doc = await getStockTransfer(
|
|
ModelNameEnum.PurchaseReceipt,
|
|
party,
|
|
date,
|
|
[
|
|
{
|
|
item,
|
|
location,
|
|
quantity,
|
|
rate,
|
|
},
|
|
],
|
|
fyo
|
|
);
|
|
|
|
await doc.sync();
|
|
const grandTotal = quantity * rate;
|
|
t.equal(doc.grandTotal?.float, quantity * rate);
|
|
|
|
await doc.submit();
|
|
|
|
t.equal(
|
|
(await fyo.db.getAllRaw(ModelNameEnum.PurchaseReceipt)).length,
|
|
1,
|
|
'purchase receipt created'
|
|
);
|
|
t.equal(
|
|
(await getSLEs(doc.name!, doc.schemaName, fyo)).length,
|
|
1,
|
|
'sle created'
|
|
);
|
|
t.equal(
|
|
await fyo.db.getStockQuantity(item, location),
|
|
quantity,
|
|
'stock purchased'
|
|
);
|
|
t.ok(doc.name?.startsWith('PREC-'));
|
|
|
|
const ales = await getALEs(doc.name!, doc.schemaName, fyo);
|
|
for (const ale of ales) {
|
|
t.equal(ale.party, party, 'party matches');
|
|
if (ale.account === 'Stock Received But Not Billed') {
|
|
t.equal(parseFloat(ale.debit), 0);
|
|
t.equal(parseFloat(ale.credit), grandTotal);
|
|
} else {
|
|
t.equal(parseFloat(ale.credit), 0);
|
|
t.equal(parseFloat(ale.debit), grandTotal);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('Shipment, create outward stock movement', async (t) => {
|
|
const date = new Date('2022-01-02');
|
|
const rate = (testDocs['Item'][item].rate as number) ?? 0;
|
|
const quantity = 5;
|
|
const doc = await getStockTransfer(
|
|
ModelNameEnum.Shipment,
|
|
party,
|
|
date,
|
|
[
|
|
{
|
|
item,
|
|
location,
|
|
quantity,
|
|
rate,
|
|
},
|
|
],
|
|
fyo
|
|
);
|
|
|
|
await doc.sync();
|
|
const grandTotal = quantity * rate;
|
|
t.equal(doc.grandTotal?.float, grandTotal);
|
|
|
|
await doc.submit();
|
|
|
|
t.equal(
|
|
(await fyo.db.getAllRaw(ModelNameEnum.Shipment)).length,
|
|
1,
|
|
'shipment created'
|
|
);
|
|
t.equal(
|
|
(await getSLEs(doc.name!, doc.schemaName, fyo)).length,
|
|
1,
|
|
'sle created'
|
|
);
|
|
t.equal(
|
|
await fyo.db.getStockQuantity(item, location),
|
|
10 - quantity,
|
|
'stock purchased'
|
|
);
|
|
t.ok(doc.name?.startsWith('SHPM-'));
|
|
|
|
const ales = await getALEs(doc.name!, doc.schemaName, fyo);
|
|
for (const ale of ales) {
|
|
t.equal(ale.party, party, 'party matches');
|
|
if (ale.account === 'Cost of Goods Sold') {
|
|
t.equal(parseFloat(ale.debit), grandTotal);
|
|
t.equal(parseFloat(ale.credit), 0);
|
|
} else {
|
|
t.equal(parseFloat(ale.debit), 0);
|
|
t.equal(parseFloat(ale.credit), grandTotal);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('Shipment, invalid', async (t) => {
|
|
const date = new Date('2022-01-03');
|
|
const rate = (testDocs['Item'][item].rate as number) ?? 0;
|
|
const quantity = 10;
|
|
const doc = await getStockTransfer(
|
|
ModelNameEnum.Shipment,
|
|
party,
|
|
date,
|
|
[
|
|
{
|
|
item,
|
|
location,
|
|
quantity,
|
|
rate,
|
|
},
|
|
],
|
|
fyo
|
|
);
|
|
|
|
await doc.sync();
|
|
const grandTotal = quantity * rate;
|
|
|
|
t.equal(await fyo.db.getStockQuantity(item, location), 5, 'stock unchanged');
|
|
t.equal(doc.grandTotal?.float, grandTotal);
|
|
await assertThrows(async () => await doc.submit());
|
|
|
|
t.equal(
|
|
(await getSLEs(doc.name!, doc.schemaName, fyo)).length,
|
|
0,
|
|
'sles not created'
|
|
);
|
|
t.equal(
|
|
(await getALEs(doc.name!, doc.schemaName, fyo)).length,
|
|
0,
|
|
'ales not created'
|
|
);
|
|
});
|
|
|
|
test('Stock Transfer, invalid cancellation', async (t) => {
|
|
const { name } =
|
|
(
|
|
(await fyo.db.getAllRaw(ModelNameEnum.PurchaseReceipt)) as {
|
|
name: string;
|
|
}[]
|
|
)[0] ?? {};
|
|
|
|
t.ok(name?.startsWith('PREC-'));
|
|
const doc = await fyo.doc.getDoc(ModelNameEnum.PurchaseReceipt, name);
|
|
await assertThrows(async () => await doc.cancel());
|
|
t.equal(await fyo.db.getStockQuantity(item, location), 5, 'stock unchanged');
|
|
t.equal(
|
|
(await getSLEs(name, doc.schemaName, fyo)).length,
|
|
1,
|
|
'sle unchanged'
|
|
);
|
|
const ales = await getALEs(name, doc.schemaName, fyo);
|
|
t.ok(ales.every((i) => !i.reverted) && ales.length === 2, 'ale unchanged');
|
|
});
|
|
|
|
test('Shipment, cancel and delete', async (t) => {
|
|
const { name } =
|
|
(
|
|
(await fyo.db.getAllRaw(ModelNameEnum.Shipment, { order: 'asc' })) as {
|
|
name: string;
|
|
}[]
|
|
)[0] ?? {};
|
|
|
|
t.ok(name?.startsWith('SHPM-'), 'number series matches');
|
|
const doc = await fyo.doc.getDoc(ModelNameEnum.Shipment, name);
|
|
t.ok(doc.isSubmitted, `doc ${name} is submitted`);
|
|
await assertDoesNotThrow(async () => await doc.cancel());
|
|
t.ok(doc.isCancelled), `doc is cancelled`;
|
|
|
|
t.equal(await fyo.db.getStockQuantity(item, location), 10, 'stock changed');
|
|
t.equal((await getSLEs(name, doc.schemaName, fyo)).length, 0, 'sle deleted');
|
|
const ales = await getALEs(name, doc.schemaName, fyo);
|
|
t.ok(ales.every((i) => !!i.reverted) && ales.length === 4, 'ale reverted');
|
|
|
|
await doc.delete();
|
|
t.equal((await getALEs(name, doc.schemaName, fyo)).length, 0, 'ales deleted');
|
|
t.equal(
|
|
(
|
|
await fyo.db.getAllRaw(ModelNameEnum.Shipment, {
|
|
filters: { name: name },
|
|
})
|
|
).length,
|
|
0,
|
|
'doc deleted'
|
|
);
|
|
});
|
|
|
|
test('Purchase Receipt, cancel and delete', async (t) => {
|
|
const { name } =
|
|
(
|
|
(await fyo.db.getAllRaw(ModelNameEnum.PurchaseReceipt, {
|
|
order: 'asc',
|
|
})) as {
|
|
name: string;
|
|
}[]
|
|
)[0] ?? {};
|
|
|
|
t.ok(name?.startsWith('PREC-'), 'number series matches');
|
|
const doc = await fyo.doc.getDoc(ModelNameEnum.PurchaseReceipt, name);
|
|
t.ok(doc.isSubmitted, `doc ${name} is submitted`);
|
|
await assertDoesNotThrow(async () => await doc.cancel());
|
|
t.ok(doc.isCancelled), `doc is cancelled`;
|
|
|
|
t.equal(await fyo.db.getStockQuantity(item, location), null, 'stock changed');
|
|
t.equal((await getSLEs(name, doc.schemaName, fyo)).length, 0, 'sle deleted');
|
|
const ales = await getALEs(name, doc.schemaName, fyo);
|
|
t.ok(ales.every((i) => !!i.reverted) && ales.length === 4, 'ale reverted');
|
|
|
|
await doc.delete();
|
|
t.equal((await getALEs(name, doc.schemaName, fyo)).length, 0, 'ales deleted');
|
|
t.equal(
|
|
(
|
|
await fyo.db.getAllRaw(ModelNameEnum.Shipment, {
|
|
filters: { name: name },
|
|
})
|
|
).length,
|
|
0,
|
|
'doc deleted'
|
|
);
|
|
});
|
|
|
|
test('Purchase Invoice then Purchase Receipt', async (t) => {
|
|
const rate = testDocs.Item[item].rate as number;
|
|
const quantity = 3;
|
|
const pinv = fyo.doc.getNewDoc(ModelNameEnum.PurchaseInvoice) as Invoice;
|
|
|
|
const date = new Date('2022-01-04');
|
|
await pinv.set({
|
|
date,
|
|
party,
|
|
account: 'Creditors',
|
|
});
|
|
await pinv.append('items', { item, quantity, rate });
|
|
await pinv.sync();
|
|
await pinv.submit();
|
|
|
|
t.equal(pinv.name, 'PINV-1001', 'PINV name matches');
|
|
t.equal(pinv.stockNotTransferred, quantity, 'stock not transferred');
|
|
const prec = await pinv.getStockTransfer();
|
|
if (prec === null) {
|
|
return t.ok(false, 'prec was null');
|
|
}
|
|
|
|
prec.date = new Date('2022-01-05');
|
|
t.equal(
|
|
ModelNameEnum.PurchaseReceipt,
|
|
prec.schemaName,
|
|
'stock transfer is a PREC'
|
|
);
|
|
t.equal(prec.backReference, pinv.name, 'back reference is set');
|
|
t.equal(prec.items?.[0].quantity, quantity, 'PREC transfers quantity');
|
|
|
|
await assertDoesNotThrow(async () => await prec.sync());
|
|
await assertDoesNotThrow(async () => await prec.submit());
|
|
|
|
t.equal(prec.name, 'PREC-1002', 'PREC name matches');
|
|
t.equal(pinv.stockNotTransferred, 0, 'stock has been transferred');
|
|
t.equal(pinv.items?.[0].stockNotTransferred, 0, 'stock has been transferred');
|
|
});
|
|
|
|
test('Back Ref Purchase Receipt cancel', async (t) => {
|
|
const prec = (await fyo.doc.getDoc(
|
|
ModelNameEnum.PurchaseReceipt,
|
|
'PREC-1002'
|
|
)) as StockTransfer;
|
|
|
|
t.equal(prec.backReference, 'PINV-1001', 'back reference matches');
|
|
await assertDoesNotThrow(async () => {
|
|
await prec.cancel();
|
|
});
|
|
|
|
const pinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.PurchaseInvoice,
|
|
'PINV-1001'
|
|
)) as Invoice;
|
|
|
|
t.equal(pinv.stockNotTransferred, 3, 'pinv stock untransferred');
|
|
t.equal(
|
|
pinv.items?.[0].stockNotTransferred,
|
|
3,
|
|
'pinv item stock untransferred'
|
|
);
|
|
});
|
|
|
|
test('Cancel Purchase Invoice after Purchase Receipt is created', async (t) => {
|
|
const pinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.PurchaseInvoice,
|
|
'PINV-1001'
|
|
)) as Invoice;
|
|
|
|
const prec = await pinv.getStockTransfer();
|
|
if (prec === null) {
|
|
return t.ok(false, 'prec was null');
|
|
}
|
|
|
|
prec.date = new Date('2022-01-05');
|
|
await prec.sync();
|
|
await prec.submit();
|
|
|
|
t.equal(prec.name, 'PREC-1003', 'PREC name matches');
|
|
t.equal(prec.backReference, 'PINV-1001', 'PREC backref matches');
|
|
|
|
await assertThrows(async () => {
|
|
await pinv.cancel();
|
|
}, 'cancel prevented cause of PREC');
|
|
|
|
const ales = await fyo.db.getAllRaw(ModelNameEnum.AccountingLedgerEntry, {
|
|
fields: ['name', 'reverted'],
|
|
filters: { referenceName: pinv.name!, reverted: true },
|
|
});
|
|
|
|
t.equal(ales.length, 0);
|
|
});
|
|
|
|
test('Sales Invoice then partial Shipment', async (t) => {
|
|
const rate = testDocs.Item[item].rate as number;
|
|
const sinv = fyo.doc.getNewDoc(ModelNameEnum.SalesInvoice) as Invoice;
|
|
|
|
await sinv.set({
|
|
party,
|
|
date: new Date('2022-01-06'),
|
|
account: 'Debtors',
|
|
});
|
|
await sinv.append('items', { item, quantity: 3, rate });
|
|
await sinv.sync();
|
|
await sinv.submit();
|
|
|
|
t.equal(sinv.name, 'SINV-1001', 'SINV name matches');
|
|
t.equal(sinv.stockNotTransferred, 3, 'stock not transferred');
|
|
|
|
const shpm = await sinv.getStockTransfer();
|
|
if (shpm === null) {
|
|
return t.ok(false, 'shpm was null');
|
|
}
|
|
|
|
shpm.date = new Date('2022-01-07');
|
|
await shpm.items?.[0].set('quantity', 1);
|
|
|
|
await assertDoesNotThrow(async () => await shpm.sync());
|
|
await assertDoesNotThrow(async () => await shpm.submit());
|
|
|
|
t.equal(ModelNameEnum.Shipment, shpm.schemaName, 'stock transfer is a SHPM');
|
|
t.equal(shpm.backReference, sinv.name, 'back reference is set');
|
|
t.equal(shpm.items?.[0].quantity, 1, 'shpm transfers quantity 1');
|
|
|
|
t.equal(shpm.name, 'SHPM-1003', 'SHPM name matches');
|
|
t.equal(sinv.stockNotTransferred, 2, 'stock qty 2 has not been transferred');
|
|
t.equal(
|
|
sinv.items?.[0].stockNotTransferred,
|
|
2,
|
|
'item stock qty 2 has not been transferred'
|
|
);
|
|
});
|
|
|
|
test('Sales Invoice then another Shipment', async (t) => {
|
|
const sinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.SalesInvoice,
|
|
'SINV-1001'
|
|
)) as Invoice;
|
|
|
|
const shpm = await sinv.getStockTransfer();
|
|
if (shpm === null) {
|
|
return t.ok(false, 'shpm was null');
|
|
}
|
|
|
|
await assertDoesNotThrow(async () => await shpm.sync());
|
|
await assertDoesNotThrow(async () => await shpm.submit());
|
|
|
|
t.equal(shpm.name, 'SHPM-1004', 'SHPM name matches');
|
|
t.equal(shpm.items?.[0].quantity, 2, 'shpm transfers quantity 2');
|
|
t.equal(sinv.stockNotTransferred, 0, 'stock has been transferred');
|
|
t.equal(
|
|
sinv.items?.[0].stockNotTransferred,
|
|
0,
|
|
'item stock has been transferred'
|
|
);
|
|
t.equal(await sinv.getStockTransfer(), null, 'no more stock transfers');
|
|
});
|
|
|
|
test('Cancel Sales Invoice after Shipment is created', async (t) => {
|
|
const sinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.SalesInvoice,
|
|
'SINV-1001'
|
|
)) as Invoice;
|
|
await assertThrows(
|
|
async () => await sinv.cancel(),
|
|
'cancel prevent cause of SHPM'
|
|
);
|
|
|
|
const ales = await fyo.db.getAllRaw(ModelNameEnum.AccountingLedgerEntry, {
|
|
fields: ['name', 'reverted'],
|
|
filters: { referenceName: sinv.name!, reverted: true },
|
|
});
|
|
|
|
t.equal(ales.length, 0);
|
|
});
|
|
|
|
test('Cancel partial Shipment', async (t) => {
|
|
let shpm = (await fyo.doc.getDoc(
|
|
ModelNameEnum.Shipment,
|
|
'SHPM-1003'
|
|
)) as StockTransfer;
|
|
|
|
t.equal(shpm.backReference, 'SINV-1001', 'SHPM 1 back ref is set');
|
|
t.equal(shpm.items?.[0].quantity, 1, 'SHPM transfers qty 1');
|
|
|
|
await assertDoesNotThrow(async () => await shpm.cancel());
|
|
t.ok(shpm.isCancelled, 'SHPM cancelled');
|
|
|
|
const sinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.SalesInvoice,
|
|
'SINV-1001'
|
|
)) as Invoice;
|
|
t.equal(sinv.stockNotTransferred, 1, 'stock qty 1 untransferred');
|
|
|
|
shpm = (await fyo.doc.getDoc(
|
|
ModelNameEnum.Shipment,
|
|
'SHPM-1004'
|
|
)) as StockTransfer;
|
|
|
|
t.equal(shpm.backReference, 'SINV-1001', 'SHPM 2 back ref is set');
|
|
t.equal(shpm.items?.[0].quantity, 2, 'SHPM transfers qty 2');
|
|
|
|
await assertDoesNotThrow(async () => await shpm.cancel());
|
|
t.ok(shpm.isCancelled, 'SHPM cancelled');
|
|
|
|
t.equal(sinv.stockNotTransferred, 3, 'all stock untransferred');
|
|
});
|
|
|
|
test('Duplicate Shipment, backref unset', async (t) => {
|
|
const shpm = (await fyo.doc.getDoc(
|
|
ModelNameEnum.Shipment,
|
|
'SHPM-1003'
|
|
)) as StockTransfer;
|
|
|
|
t.ok(shpm.backReference, 'SHPM back ref is set');
|
|
|
|
const doc = shpm.duplicate();
|
|
t.notOk(doc.backReference, 'Duplicate SHPM back ref is not set');
|
|
});
|
|
|
|
test('Cancel and Delete Sales Invoice with cancelled Shipments', async (t) => {
|
|
const sinv = (await fyo.doc.getDoc(
|
|
ModelNameEnum.SalesInvoice,
|
|
'SINV-1001'
|
|
)) as Invoice;
|
|
|
|
await assertDoesNotThrow(async () => await sinv.cancel());
|
|
t.ok(sinv.isCancelled, 'sinv cancelled');
|
|
|
|
const transfers = (await fyo.db.getAllRaw(ModelNameEnum.Shipment, {
|
|
fields: ['name'],
|
|
filters: { backReference: 'SINV-1001' },
|
|
})) as { name: string }[];
|
|
|
|
await assertDoesNotThrow(async () => await sinv.delete());
|
|
t.notOk(
|
|
await fyo.db.exists(ModelNameEnum.SalesInvoice, 'SINV-1001'),
|
|
'SINV-1001 deleted'
|
|
);
|
|
|
|
for (const { name } of transfers) {
|
|
t.notOk(
|
|
await fyo.db.exists(ModelNameEnum.Shipment, 'SINV-1001'),
|
|
`linked Shipment ${name} deleted`
|
|
);
|
|
}
|
|
});
|
|
|
|
test('Create Shipment from manually set Back Ref', async (t) => {
|
|
const rate = (testDocs['Item'][item].rate as number) ?? 0;
|
|
const totalQuantity = 10;
|
|
const prec = await getStockTransfer(
|
|
ModelNameEnum.PurchaseReceipt,
|
|
party,
|
|
new Date('2022-01-08'),
|
|
[
|
|
{
|
|
item,
|
|
location,
|
|
quantity: totalQuantity,
|
|
rate,
|
|
},
|
|
],
|
|
fyo
|
|
);
|
|
await (await prec.sync()).submit();
|
|
|
|
const sinv = fyo.doc.getNewDoc(ModelNameEnum.SalesInvoice) as Invoice;
|
|
const quantity = 5;
|
|
await sinv.set({
|
|
party,
|
|
date: new Date('2022-01-09'),
|
|
account: 'Debtors',
|
|
});
|
|
await sinv.append('items', { item, quantity, rate });
|
|
await (await sinv.sync()).submit();
|
|
|
|
t.equal(sinv.stockNotTransferred, quantity, "stock hasn't been transferred");
|
|
|
|
const shpm = fyo.doc.getNewDoc(ModelNameEnum.Shipment) as Shipment;
|
|
await shpm.set('backReference', sinv.name);
|
|
await shpm.set('date', new Date('2022-01-10'));
|
|
shpm.items?.[0].set('location', location);
|
|
|
|
t.equal(shpm.party, sinv.party, 'party set');
|
|
|
|
await (await shpm.sync()).submit();
|
|
t.equal(
|
|
await fyo.db.getStockQuantity(item, location),
|
|
totalQuantity - quantity,
|
|
'quantity shipped'
|
|
);
|
|
|
|
t.equal(sinv.stockNotTransferred, 0, 'stock has been transferred');
|
|
});
|
|
|
|
test('Create Shipment then create return against it', async (t) => {
|
|
const rate = testDocs.Item[item].rate as number;
|
|
const shpm = fyo.doc.getNewDoc(ModelNameEnum.Shipment) as Shipment;
|
|
|
|
await shpm.set({
|
|
party,
|
|
date: new Date('2023-05-18'),
|
|
items: [{ item, quantity: 3, rate }],
|
|
});
|
|
await shpm.sync();
|
|
await shpm.submit();
|
|
|
|
t.equal(shpm.name, 'SHPM-1006', 'Shipment created');
|
|
|
|
const shpmReturn = (await shpm.getReturnDoc()) as Shipment;
|
|
await shpmReturn.sync();
|
|
await shpmReturn.submit();
|
|
|
|
t.equal(shpmReturn.name, 'SHPM-1007', 'Shipment return created');
|
|
|
|
t.ok(
|
|
shpmReturn.grandTotal?.isNegative(),
|
|
'Shipment return doc has negative grand total '
|
|
);
|
|
|
|
const returnShpmAles = await fyo.db.getAllRaw(
|
|
ModelNameEnum.AccountingLedgerEntry,
|
|
{
|
|
fields: ['name', 'account', 'credit', 'debit'],
|
|
filters: { referenceName: shpmReturn.name! },
|
|
}
|
|
);
|
|
|
|
for (const ale of returnShpmAles) {
|
|
if (ale.account === 'Stock In Hand') {
|
|
t.equal(
|
|
fyo.pesa(ale.debit as string).float,
|
|
shpmReturn.grandTotal?.float,
|
|
`Shipment return amount debited from ${ale.account}`
|
|
);
|
|
}
|
|
|
|
if (ale.account === 'Cost of Goods Sold') {
|
|
t.equal(
|
|
fyo.pesa(ale.credit as string).float,
|
|
shpmReturn.grandTotal?.float,
|
|
`Shipment return amount credited to ${ale.account}`
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('Create Shipment return of batched item', async (t) => {
|
|
const item = {
|
|
name: 'Jacket-B',
|
|
rate: fyo.pesa(100),
|
|
trackItem: true,
|
|
hasBatch: true,
|
|
};
|
|
|
|
const newItemDoc = fyo.doc.getNewDoc(ModelNameEnum.Item, item);
|
|
newItemDoc.sync();
|
|
newItemDoc.submit();
|
|
|
|
t.ok(
|
|
fyo.db.exists(ModelNameEnum.Item, item.name),
|
|
`item ${item.name} created`
|
|
);
|
|
|
|
const batches = [
|
|
{
|
|
name: 'JKT-A-2',
|
|
expiryDate: new Date('2023-07-27T18:30:00.435Z'),
|
|
manufactureDate: new Date('2023-07-27T18:30:00.435Z'),
|
|
},
|
|
{
|
|
name: 'JKT-B-2',
|
|
expiryDate: new Date('2023-07-27T18:30:00.435Z'),
|
|
manufactureDate: new Date('2023-07-27T18:30:00.435Z'),
|
|
},
|
|
];
|
|
|
|
for (const batch of batches) {
|
|
const batchDoc = fyo.doc.getNewDoc(ModelNameEnum.Batch, batch);
|
|
batchDoc.sync();
|
|
t.ok(
|
|
fyo.db.exists(ModelNameEnum.Batch, batch.name),
|
|
`Batch ${batch.name} created`
|
|
);
|
|
}
|
|
|
|
const smovDoc = fyo.doc.getNewDoc(ModelNameEnum.StockMovement, {
|
|
date: new Date('2023-07-27T18:30:00.435Z'),
|
|
movementType: MovementTypeEnum.MaterialReceipt,
|
|
});
|
|
|
|
await smovDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 4,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
batch: batches[0].name,
|
|
});
|
|
|
|
await smovDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 8,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
batch: batches[1].name,
|
|
});
|
|
|
|
await smovDoc.sync();
|
|
await smovDoc.submit();
|
|
|
|
t.equal(smovDoc.name, 'SMOV-1001', 'stock movement created');
|
|
|
|
const shipmentDoc = fyo.doc.getNewDoc(ModelNameEnum.Shipment) as Shipment;
|
|
await shipmentDoc.set({
|
|
party,
|
|
date: new Date(),
|
|
items: [
|
|
{ item: item.name, quantity: 4, rate: item.rate, batch: batches[0].name },
|
|
{ item: item.name, quantity: 8, rate: item.rate, batch: batches[1].name },
|
|
],
|
|
});
|
|
|
|
await shipmentDoc.sync();
|
|
await shipmentDoc.submit();
|
|
|
|
t.equal(shipmentDoc.name, 'SHPM-1008', `Shipment created`);
|
|
|
|
const shpmReturnDoc = fyo.doc.getNewDoc(ModelNameEnum.Shipment) as Shipment;
|
|
|
|
await shpmReturnDoc.set({
|
|
date: new Date(),
|
|
party,
|
|
returnAgainst: shipmentDoc.name,
|
|
items: [
|
|
{ item: item.name, quantity: 2, rate: item.rate, batch: batches[0].name },
|
|
{ item: item.name, quantity: 4, rate: item.rate, batch: batches[1].name },
|
|
],
|
|
});
|
|
await shpmReturnDoc.sync();
|
|
await shpmReturnDoc.submit();
|
|
|
|
t.equal(shpmReturnDoc.name, 'SHPM-1009', 'Shipment return is created');
|
|
|
|
const secondReturnDoc = (await shipmentDoc.getReturnDoc()) as Shipment;
|
|
|
|
for (const item of secondReturnDoc.items!) {
|
|
if (item.batch == batches[0].name) {
|
|
const docItemQty = shipmentDoc.items![0].quantity as number;
|
|
const retItemQty = shpmReturnDoc.items![0].quantity as number;
|
|
const balanceQty = retItemQty - docItemQty;
|
|
|
|
t.equal(
|
|
item.quantity,
|
|
balanceQty,
|
|
`Batch ${item.batch} has ${balanceQty} qty left to return`
|
|
);
|
|
}
|
|
|
|
if (item.batch == batches[1].name) {
|
|
const docItemQty = shipmentDoc.items![1].quantity as number;
|
|
const retItemQty = shpmReturnDoc.items![1].quantity as number;
|
|
const balanceQty = retItemQty - docItemQty;
|
|
|
|
t.equal(
|
|
item.quantity,
|
|
balanceQty,
|
|
`Batch ${item.batch} has ${balanceQty} qty left to return`
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('Create Purchase Reciept then create return against it', async (t) => {
|
|
const rate = testDocs.Item[item].rate as number;
|
|
const prec = fyo.doc.getNewDoc(
|
|
ModelNameEnum.PurchaseReceipt
|
|
) as PurchaseReceipt;
|
|
|
|
await prec.set({
|
|
party,
|
|
date: new Date('2023-05-18'),
|
|
items: [{ item, quantity: 3, rate }],
|
|
});
|
|
await prec.sync();
|
|
await prec.submit();
|
|
|
|
t.equal(prec.name, 'PREC-1005', 'Purchase Receipt created');
|
|
|
|
const precReturn = (await prec.getReturnDoc()) as PurchaseReceipt;
|
|
await precReturn.sync();
|
|
await precReturn.submit();
|
|
|
|
t.equal(precReturn.name, 'PREC-1006', 'Purchase Receipt return created');
|
|
t.ok(
|
|
precReturn.grandTotal?.isNegative(),
|
|
'Purchase Receipt return doc has negative grand total '
|
|
);
|
|
|
|
const returnPrecAles = await fyo.db.getAllRaw(
|
|
ModelNameEnum.AccountingLedgerEntry,
|
|
{
|
|
fields: ['name', 'account', 'credit', 'debit'],
|
|
filters: { referenceName: precReturn.name! },
|
|
}
|
|
);
|
|
|
|
for (const ale of returnPrecAles) {
|
|
if (ale.account === 'Stock In Hand') {
|
|
t.equal(
|
|
fyo.pesa(ale.credit as string).float,
|
|
precReturn.grandTotal?.float,
|
|
`Purchase Receipt return amount credited to ${ale.account}`
|
|
);
|
|
}
|
|
|
|
if (ale.account === 'Stock Received But Not Billed') {
|
|
t.equal(
|
|
fyo.pesa(ale.debit as string).float,
|
|
precReturn.grandTotal?.float,
|
|
`Purchase Receipt return amount debited from ${ale.account}`
|
|
);
|
|
}
|
|
}
|
|
});
|
|
|
|
test('Create Purchase Reciept return of serialized item', async (t) => {
|
|
const item = {
|
|
name: 'Jacket-S',
|
|
rate: fyo.pesa(100),
|
|
trackItem: true,
|
|
hasSerialNumber: true,
|
|
};
|
|
const serialNumbers = {
|
|
row1: ['JKT-S-A-001', 'JKT-S-A-002'],
|
|
row2: ['JKT-S-B-001', 'JKT-S-B-002'],
|
|
};
|
|
|
|
const newItemDoc = fyo.doc.getNewDoc(ModelNameEnum.Item, item);
|
|
newItemDoc.sync();
|
|
newItemDoc.submit();
|
|
|
|
t.ok(
|
|
fyo.db.exists(ModelNameEnum.Item, item.name),
|
|
`item ${item.name} created`
|
|
);
|
|
|
|
const precDoc = fyo.doc.getNewDoc(ModelNameEnum.PurchaseReceipt, {
|
|
date: new Date('2023-07-27T18:30:00.435Z'),
|
|
party,
|
|
}) as PurchaseReceipt;
|
|
|
|
await precDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 2,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
serialNumber: serialNumbers.row1.join('\n'),
|
|
});
|
|
|
|
await precDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 2,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
serialNumber: serialNumbers.row2.join('\n'),
|
|
});
|
|
|
|
await precDoc.sync();
|
|
await precDoc.submit();
|
|
|
|
t.equal(
|
|
precDoc.name,
|
|
'PREC-1007',
|
|
'purchase reciept having serial number created'
|
|
);
|
|
|
|
const returnPrecDoc = fyo.doc.getNewDoc(ModelNameEnum.PurchaseReceipt, {
|
|
date: new Date('2023-07-27T18:30:00.435Z'),
|
|
party,
|
|
returnAgainst: precDoc.name,
|
|
}) as PurchaseReceipt;
|
|
|
|
await returnPrecDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 1,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
serialNumber: serialNumbers.row1[0],
|
|
});
|
|
|
|
await returnPrecDoc.append('items', {
|
|
item: item.name,
|
|
quantity: 1,
|
|
rate: item.rate,
|
|
toLocation: 'Stores',
|
|
serialNumber: serialNumbers.row2[0],
|
|
});
|
|
|
|
await returnPrecDoc.sync();
|
|
await returnPrecDoc.submit();
|
|
|
|
t.equal(
|
|
returnPrecDoc.name,
|
|
'PREC-1008',
|
|
'purchase reciept return having serial number created'
|
|
);
|
|
|
|
const secondPrecReturnDoc = (await precDoc.getReturnDoc()) as PurchaseReceipt;
|
|
const returnBalSerialNumbers = secondPrecReturnDoc.items
|
|
?.map((item) => item.serialNumber?.split('\n'))
|
|
.flat();
|
|
|
|
t.ok(returnBalSerialNumbers?.length === 2);
|
|
|
|
const returnedSerialNumbers = [
|
|
returnPrecDoc.items![0].serialNumber,
|
|
returnPrecDoc.items![1].serialNumber,
|
|
];
|
|
|
|
for (const serialNumber of returnedSerialNumbers) {
|
|
const status = await fyo.getValue(
|
|
ModelNameEnum.SerialNumber,
|
|
serialNumber as string,
|
|
'status'
|
|
);
|
|
|
|
t.equal(
|
|
status,
|
|
'Delivered',
|
|
'serial number status updated to Delivered after return'
|
|
);
|
|
}
|
|
|
|
for (const serialNumber of returnBalSerialNumbers!) {
|
|
const status = await fyo.getValue(
|
|
ModelNameEnum.SerialNumber,
|
|
serialNumber as string,
|
|
'status'
|
|
);
|
|
|
|
t.equal(
|
|
status,
|
|
'Active',
|
|
'serial number status is Active for unreturned item'
|
|
);
|
|
}
|
|
});
|
|
|
|
closeTestFyo(fyo, __filename);
|