mirror of
https://github.com/frappe/books.git
synced 2025-01-11 18:38:47 +00:00
1757f8f266
- Tree structured reports - Profit and Loss
291 lines
6.8 KiB
JavaScript
291 lines
6.8 KiB
JavaScript
const frappe = require('frappejs');
|
|
const { DateTime } = require('luxon');
|
|
const { unique } = require('frappejs/utils');
|
|
|
|
async function getData({
|
|
rootType,
|
|
balanceMustBe = 'Debit',
|
|
fromDate,
|
|
toDate,
|
|
periodicity = 'Monthly',
|
|
accumulateValues = false
|
|
}) {
|
|
let accounts = await getAccounts(rootType);
|
|
let fiscalYear = await getFiscalYear();
|
|
let ledgerEntries = await getLedgerEntries(fromDate, toDate, accounts);
|
|
let periodList = getPeriodList(fromDate, toDate, periodicity, fiscalYear);
|
|
|
|
for (let account of accounts) {
|
|
const entries = ledgerEntries.filter(
|
|
entry => entry.account === account.name
|
|
);
|
|
|
|
for (let entry of entries) {
|
|
let periodKey = getPeriodKey(entry.date, periodicity);
|
|
|
|
if (!account[periodKey]) {
|
|
account[periodKey] = 0.0;
|
|
}
|
|
|
|
const multiplier = balanceMustBe === 'Debit' ? 1 : -1;
|
|
const value = (entry.debit - entry.credit) * multiplier;
|
|
account[periodKey] += value;
|
|
}
|
|
}
|
|
|
|
if (accumulateValues) {
|
|
periodList.forEach((periodKey, i) => {
|
|
if (i === 0) return;
|
|
const previousPeriodKey = periodList[i - 1];
|
|
|
|
for (let account of accounts) {
|
|
if (!account[periodKey]) {
|
|
account[periodKey] = 0.0;
|
|
}
|
|
account[periodKey] += account[previousPeriodKey] || 0.0;
|
|
}
|
|
});
|
|
}
|
|
|
|
// calculate totalRow
|
|
let totalRow = {
|
|
account: `Total ${rootType} (${balanceMustBe})`
|
|
};
|
|
|
|
periodList.forEach(periodKey => {
|
|
if (!totalRow[periodKey]) {
|
|
totalRow[periodKey] = 0.0;
|
|
}
|
|
|
|
for (let account of accounts) {
|
|
totalRow[periodKey] += account[periodKey] || 0.0;
|
|
}
|
|
});
|
|
|
|
return { accounts, totalRow, periodList };
|
|
}
|
|
|
|
async function getTrialBalance({ rootType, fromDate, toDate }) {
|
|
let accounts = await getAccounts(rootType);
|
|
let ledgerEntries = await getLedgerEntries(null, toDate, accounts);
|
|
|
|
for (let account of accounts) {
|
|
const accountEntries = ledgerEntries.filter(
|
|
entry => entry.account === account.name
|
|
);
|
|
// opening
|
|
const beforePeriodEntries = accountEntries.filter(
|
|
entry => entry.date < fromDate
|
|
);
|
|
account.opening = beforePeriodEntries.reduce((acc, entry) => {
|
|
return acc + (entry.debit - entry.credit);
|
|
}, 0);
|
|
|
|
if (account.opening >= 0) {
|
|
account.openingDebit = account.opening;
|
|
} else {
|
|
account.openingCredit = -account.opening;
|
|
}
|
|
|
|
// debit / credit
|
|
const periodEntries = accountEntries.filter(
|
|
entry => entry.date >= fromDate && entry.date < toDate
|
|
);
|
|
account.debit = periodEntries.reduce((acc, entry) => acc + entry.debit, 0);
|
|
account.credit = periodEntries.reduce(
|
|
(acc, entry) => acc + entry.credit,
|
|
0
|
|
);
|
|
|
|
// closing
|
|
account.closing = account.opening + account.debit - account.credit;
|
|
|
|
if (account.closing >= 0) {
|
|
account.closingDebit = account.closing;
|
|
} else {
|
|
account.closingCredit = -account.closing;
|
|
}
|
|
|
|
if (account.debit != 0 || account.credit != 0) {
|
|
setParentEntry(account, account.parentAccount);
|
|
}
|
|
}
|
|
|
|
function setParentEntry(leafAccount, parentName) {
|
|
for (let acc of accounts) {
|
|
if (acc.name === parentName) {
|
|
acc.debit += leafAccount.debit;
|
|
acc.credit += leafAccount.credit;
|
|
acc.closing = acc.opening + acc.debit - acc.credit;
|
|
if (acc.closing >= 0) {
|
|
acc.closingDebit = acc.closing;
|
|
} else {
|
|
acc.closingCredit = -acc.closing;
|
|
}
|
|
if (acc.parentAccount) {
|
|
setParentEntry(leafAccount, acc.parentAccount);
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return accounts;
|
|
}
|
|
|
|
function getPeriodList(fromDate, toDate, periodicity, fiscalYear) {
|
|
if (!fromDate) {
|
|
fromDate = fiscalYear.start;
|
|
}
|
|
|
|
let monthsToAdd = {
|
|
Monthly: 1,
|
|
Quarterly: 3,
|
|
'Half Yearly': 6,
|
|
Yearly: 12
|
|
}[periodicity];
|
|
|
|
let startDate = DateTime.fromISO(fromDate).startOf('month');
|
|
let endDate = DateTime.fromISO(toDate).endOf('month');
|
|
let curDate = startDate;
|
|
let out = [];
|
|
|
|
while (curDate <= endDate) {
|
|
out.push(getPeriodKey(curDate, periodicity));
|
|
curDate = curDate.plus({ months: monthsToAdd });
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
function getPeriodKey(date, periodicity) {
|
|
let key;
|
|
let dateObj = DateTime.fromISO(date);
|
|
let year = dateObj.year;
|
|
let quarter = dateObj.quarter;
|
|
let month = dateObj.month;
|
|
|
|
let getKey = {
|
|
Monthly: () => `${dateObj.monthShort} ${year}`,
|
|
Quarterly: () => {
|
|
return {
|
|
1: `Jan ${year} - Mar ${year}`,
|
|
2: `Apr ${year} - Jun ${year}`,
|
|
3: `Jun ${year} - Sep ${year}`,
|
|
4: `Oct ${year} - Dec ${year}`
|
|
}[quarter];
|
|
},
|
|
'Half Yearly': () => {
|
|
return {
|
|
1: `Apr ${year} - Sep ${year}`,
|
|
2: `Oct ${year} - Mar ${year}`
|
|
}[[2, 3].includes(quarter) ? 1 : 2];
|
|
},
|
|
Yearly: () => {
|
|
if (month > 3) {
|
|
return `${year} - ${year + 1}`;
|
|
}
|
|
return `${year - 1} - ${year}`;
|
|
}
|
|
}[periodicity];
|
|
|
|
return getKey();
|
|
}
|
|
|
|
function setIndentLevel(accounts, parentAccount, level) {
|
|
if (!parentAccount) {
|
|
// root
|
|
parentAccount = null;
|
|
level = 0;
|
|
}
|
|
|
|
accounts.forEach(account => {
|
|
if (
|
|
account.parentAccount === parentAccount &&
|
|
account.indent === undefined
|
|
) {
|
|
account.indent = level;
|
|
setIndentLevel(accounts, account.name, level + 1);
|
|
}
|
|
});
|
|
|
|
return accounts;
|
|
}
|
|
|
|
function sortAccounts(accounts) {
|
|
let out = [];
|
|
let pushed = {};
|
|
|
|
pushToOut(null);
|
|
|
|
function pushToOut(parentAccount) {
|
|
accounts.forEach(account => {
|
|
if (account.parentAccount === parentAccount && !pushed[account.name]) {
|
|
out.push(account);
|
|
pushed[account.name] = 1;
|
|
|
|
pushToOut(account.name);
|
|
}
|
|
});
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
async function getLedgerEntries(fromDate, toDate, accounts) {
|
|
const dateFilter = () => {
|
|
const before = ['<=', toDate];
|
|
const after = ['>=', fromDate];
|
|
if (fromDate) {
|
|
return [...after, ...before];
|
|
}
|
|
return before;
|
|
};
|
|
|
|
const ledgerEntries = await frappe.db.getAll({
|
|
doctype: 'AccountingLedgerEntry',
|
|
fields: ['account', 'debit', 'credit', 'date'],
|
|
filters: {
|
|
account: ['in', accounts.map(d => d.name)],
|
|
date: dateFilter()
|
|
}
|
|
});
|
|
|
|
return ledgerEntries;
|
|
}
|
|
|
|
async function getAccounts(rootType) {
|
|
let accounts = await frappe.db.getAll({
|
|
doctype: 'Account',
|
|
fields: ['name', 'parentAccount', 'isGroup'],
|
|
filters: {
|
|
rootType
|
|
}
|
|
});
|
|
|
|
accounts = setIndentLevel(accounts);
|
|
accounts = sortAccounts(accounts);
|
|
|
|
accounts.forEach(account => {
|
|
account.account = account.name;
|
|
});
|
|
|
|
return accounts;
|
|
}
|
|
|
|
async function getFiscalYear() {
|
|
let { fiscalYearStart, fiscalYearEnd } = await frappe.getSingle(
|
|
'AccountingSettings'
|
|
);
|
|
return {
|
|
start: fiscalYearStart,
|
|
end: fiscalYearEnd
|
|
};
|
|
}
|
|
|
|
module.exports = {
|
|
getData,
|
|
getTrialBalance
|
|
};
|