2021-11-04 16:01:26 +05:30
|
|
|
import frappe from 'frappejs';
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2021-11-04 16:01:26 +05:30
|
|
|
export default class AccountsReceivablePayable {
|
2019-08-01 17:22:58 +05:30
|
|
|
async run(reportType, { date }) {
|
|
|
|
const rows = await getReceivablePayable({
|
|
|
|
reportType,
|
|
|
|
date
|
|
|
|
});
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
return { rows };
|
|
|
|
}
|
|
|
|
};
|
2018-04-29 14:37:59 +05:30
|
|
|
|
|
|
|
async function getReceivablePayable({ reportType = 'Receivable', date }) {
|
2019-08-01 17:22:58 +05:30
|
|
|
let entries = [];
|
|
|
|
const debitOrCredit = reportType === 'Receivable' ? 'debit' : 'credit';
|
|
|
|
const referenceType =
|
|
|
|
reportType === 'Receivable' ? 'SalesInvoice' : 'PurchaseInvoice';
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
entries = await getLedgerEntries();
|
|
|
|
const vouchers = await getVouchers();
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
const futureEntries = getFutureEntries();
|
|
|
|
const returnEntries = getReturnEntries();
|
|
|
|
const pdc = getPDC();
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
const validEntries = getValidEntries();
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
let data = [];
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
for (let entry of validEntries) {
|
|
|
|
const { outStandingAmount, creditNoteAmount } = getOutstandingAmount(entry);
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
if (outStandingAmount > 0.1 / 10) {
|
|
|
|
const row = {
|
|
|
|
date: entry.date,
|
|
|
|
party: entry.party
|
|
|
|
};
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
// due date / bill date
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
row.voucherType = entry.referenceType;
|
|
|
|
row.voucherNo = entry.referenceName;
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
// bill details
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
const invoicedAmount = entry[debitOrCredit] || 0;
|
|
|
|
const paidAmount = invoicedAmount - outStandingAmount - creditNoteAmount;
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
Object.assign(row, {
|
|
|
|
invoicedAmount,
|
|
|
|
paidAmount,
|
|
|
|
outStandingAmount,
|
|
|
|
creditNoteAmount
|
|
|
|
});
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
// ageing
|
2018-04-29 14:37:59 +05:30
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
data.push(row);
|
2018-04-29 14:37:59 +05:30
|
|
|
}
|
2019-08-01 17:22:58 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return data;
|
|
|
|
|
|
|
|
// helpers
|
|
|
|
|
|
|
|
async function getVouchers() {
|
|
|
|
return await frappe.db.getAll({
|
|
|
|
doctype: referenceType,
|
|
|
|
fields: ['name', 'date'],
|
|
|
|
filters: {
|
|
|
|
submitted: 1
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function getValidEntries() {
|
|
|
|
return entries.filter(entry => {
|
|
|
|
return (
|
|
|
|
entry.date <= date &&
|
|
|
|
entry.referenceType === referenceType &&
|
|
|
|
entry[debitOrCredit] > 0
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function getOutstandingAmount(entry) {
|
|
|
|
let paymentAmount = 0.0,
|
|
|
|
creditNoteAmount = 0.0;
|
|
|
|
let reverseDebitOrCredit = debitOrCredit === 'debit' ? 'credit' : 'debit';
|
|
|
|
|
|
|
|
for (let e of getEntriesFor(
|
|
|
|
entry.party,
|
|
|
|
entry.referenceType,
|
|
|
|
entry.referenceName
|
|
|
|
)) {
|
|
|
|
if (e.date <= date) {
|
|
|
|
const amount = e[reverseDebitOrCredit] - e[debitOrCredit];
|
|
|
|
|
|
|
|
if (!Object.keys(returnEntries).includes(e.referenceName)) {
|
|
|
|
paymentAmount += amount;
|
|
|
|
} else {
|
|
|
|
creditNoteAmount += amount;
|
2018-04-29 14:37:59 +05:30
|
|
|
}
|
2019-08-01 17:22:58 +05:30
|
|
|
}
|
2018-04-29 14:37:59 +05:30
|
|
|
}
|
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
return {
|
|
|
|
outStandingAmount:
|
|
|
|
entry[debitOrCredit] -
|
|
|
|
entry[reverseDebitOrCredit] -
|
|
|
|
paymentAmount -
|
|
|
|
creditNoteAmount,
|
|
|
|
creditNoteAmount
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function getEntriesFor(party, againstVoucherType, againstVoucher) {
|
|
|
|
// TODO
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
function getFutureEntries() {
|
|
|
|
return entries.filter(entry => entry.date > date);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getReturnEntries() {
|
|
|
|
// TODO
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPDC() {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getLedgerEntries() {
|
|
|
|
if (entries.length) {
|
|
|
|
return entries;
|
2018-04-29 14:37:59 +05:30
|
|
|
}
|
|
|
|
|
2019-08-01 17:22:58 +05:30
|
|
|
const partyType = reportType === 'Receivable' ? 'customer' : 'supplier';
|
|
|
|
const partyList = (await frappe.db.getAll({
|
|
|
|
doctype: 'Party',
|
|
|
|
filters: {
|
|
|
|
[partyType]: 1
|
|
|
|
}
|
|
|
|
})).map(d => d.name);
|
|
|
|
|
|
|
|
return await frappe.db.getAll({
|
|
|
|
doctype: 'AccountingLedgerEntry',
|
|
|
|
fields: [
|
|
|
|
'name',
|
|
|
|
'date',
|
|
|
|
'account',
|
|
|
|
'party',
|
|
|
|
'referenceType',
|
|
|
|
'referenceName',
|
|
|
|
'sum(debit) as debit',
|
|
|
|
'sum(credit) as credit'
|
|
|
|
],
|
|
|
|
filters: {
|
|
|
|
party: ['in', partyList]
|
|
|
|
},
|
|
|
|
groupBy: ['referenceType', 'referenceName', 'party'],
|
|
|
|
orderBy: 'date'
|
|
|
|
});
|
|
|
|
}
|
2018-04-29 14:37:59 +05:30
|
|
|
}
|