2
0
mirror of https://github.com/frappe/books.git synced 2025-01-03 15:17:30 +00:00
books/reports/AccountReport.ts

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

655 lines
16 KiB
TypeScript
Raw Normal View History

import { Fyo, t } from 'fyo';
2022-05-16 10:10:35 +00:00
import { cloneDeep } from 'lodash';
import { DateTime } from 'luxon';
import { AccountRootType } from 'models/baseModels/Account/types';
import { isCredit } from 'models/helpers';
import { ModelNameEnum } from 'models/types';
import { LedgerReport } from 'reports/LedgerReport';
import {
Account,
AccountList,
AccountListNode,
AccountNameValueMapMap,
AccountTree,
AccountTreeNode,
BasedOn,
ColumnField,
DateRange,
GroupedMap,
LedgerEntry,
Periodicity,
ReportCell,
ReportData,
ReportRow,
Tree,
TreeNode,
ValueMap,
} from 'reports/types';
import { Field } from 'schemas/types';
import { getMapFromList } from 'utils';
import { QueryFilter } from 'utils/db/types';
2022-05-16 13:01:58 +00:00
export const ACC_NAME_WIDTH = 2;
2022-05-16 16:10:15 +00:00
export const ACC_BAL_WIDTH = 1.25;
2022-05-16 10:10:35 +00:00
export abstract class AccountReport extends LedgerReport {
toDate?: string;
count = 3;
2022-05-16 10:10:35 +00:00
fromYear?: number;
toYear?: number;
consolidateColumns = false;
hideGroupAmounts = false;
2022-05-16 10:10:35 +00:00
periodicity: Periodicity = 'Monthly';
basedOn: BasedOn = 'Until Date';
_rawData: LedgerEntry[] = [];
_dateRanges?: DateRange[];
accountMap?: Record<string, Account>;
async setDefaultFilters(): Promise<void> {
if (this.basedOn === 'Until Date' && !this.toDate) {
this.toDate = DateTime.now().plus({ days: 1 }).toISODate();
2022-05-16 10:10:35 +00:00
}
if (this.basedOn === 'Fiscal Year' && !this.toYear) {
this.fromYear = DateTime.now().year;
this.toYear = this.fromYear + 1;
}
await this._setDateRanges();
}
async _setDateRanges() {
this._dateRanges = await this._getDateRanges();
}
getRootNodes(
2022-05-16 10:10:35 +00:00
rootType: AccountRootType,
accountTree: AccountTree
): AccountTreeNode[] | undefined {
2022-05-16 10:10:35 +00:00
const rootNodeList = Object.values(accountTree);
return rootNodeList.filter((n) => n.rootType === rootType);
2022-05-16 10:10:35 +00:00
}
getEmptyRow(): ReportRow {
const columns = this.getColumns();
return {
isEmpty: true,
cells: columns.map(
(c) =>
({
value: '',
rawValue: '',
width: c.width,
align: 'left',
} as ReportCell)
),
};
}
getTotalNode(rootNodes: AccountTreeNode[], name: string): AccountListNode {
const accountTree: Tree = {};
for (const rootNode of rootNodes) {
accountTree[rootNode.name] = rootNode;
}
2022-05-16 10:10:35 +00:00
const leafNodes = getListOfLeafNodes(accountTree) as AccountTreeNode[];
const totalMap = leafNodes.reduce((acc, node) => {
for (const key of this._dateRanges!) {
2022-05-16 13:01:58 +00:00
const bal = acc.get(key)?.balance ?? 0;
const val = node.valueMap?.get(key)?.balance ?? 0;
acc.set(key, { balance: bal + val });
2022-05-16 10:10:35 +00:00
}
return acc;
}, new Map() as ValueMap);
return { name, valueMap: totalMap, level: 0 } as AccountListNode;
}
getReportRowsFromAccountList(accountList: AccountList): ReportData {
return accountList.map((al) => {
return this.getRowFromAccountListNode(al);
});
}
getRowFromAccountListNode(al: AccountListNode) {
const nameCell = {
value: al.name,
rawValue: al.name,
align: 'left',
width: ACC_NAME_WIDTH,
bold: !al.level,
indent: al.level ?? 0,
} as ReportCell;
const balanceCells = this._dateRanges!.map((k) => {
2022-05-16 13:01:58 +00:00
const rawValue = al.valueMap?.get(k)?.balance ?? 0;
2022-05-16 10:10:35 +00:00
let value = this.fyo.format(rawValue, 'Currency');
2022-05-16 13:01:58 +00:00
if (this.hideGroupAmounts && al.isGroup) {
2022-05-16 10:10:35 +00:00
value = '';
}
return {
rawValue,
value,
align: 'right',
width: ACC_BAL_WIDTH,
} as ReportCell;
});
return {
cells: [nameCell, balanceCells].flat(),
level: al.level,
isGroup: !!al.isGroup,
folded: false,
foldedBelow: false,
} as ReportRow;
}
async _getGroupedByDateRanges(
map: GroupedMap
): Promise<AccountNameValueMapMap> {
const accountValueMap: AccountNameValueMapMap = new Map();
if (!this.accountMap) {
await this._setAndReturnAccountMap();
}
2022-05-16 10:10:35 +00:00
for (const account of map.keys()) {
const valueMap: ValueMap = new Map();
/**
* Set Balance for every DateRange key
*/
for (const entry of map.get(account)!) {
const key = this._getRangeMapKey(entry);
if (key === null) {
continue;
}
if (!this.accountMap?.[entry.account]) {
await this._setAndReturnAccountMap(true);
}
const totalBalance = valueMap.get(key)?.balance ?? 0;
2022-05-16 10:10:35 +00:00
const balance = (entry.debit ?? 0) - (entry.credit ?? 0);
const rootType = this.accountMap![entry.account]?.rootType;
2022-05-16 10:10:35 +00:00
if (isCredit(rootType)) {
valueMap.set(key, { balance: totalBalance - balance });
2022-05-16 10:10:35 +00:00
} else {
valueMap.set(key, { balance: totalBalance + balance });
2022-05-16 10:10:35 +00:00
}
}
accountValueMap.set(account, valueMap);
}
return accountValueMap;
}
async _getAccountTree(rangeGroupedMap: AccountNameValueMapMap) {
const accountTree = cloneDeep(
await this._setAndReturnAccountMap()
) as AccountTree;
2022-05-16 10:10:35 +00:00
setPruneFlagOnAccountTreeNodes(accountTree);
setValueMapOnAccountTreeNodes(accountTree, rangeGroupedMap);
setChildrenOnAccountTreeNodes(accountTree);
deleteNonRootAccountTreeNodes(accountTree);
pruneAccountTree(accountTree);
return accountTree;
}
async _setAndReturnAccountMap(force = false) {
if (this.accountMap && !force) {
2022-05-16 10:10:35 +00:00
return this.accountMap;
}
const accountList: Account[] = (
await this.fyo.db.getAllRaw('Account', {
fields: ['name', 'rootType', 'isGroup', 'parentAccount'],
})
).map((rv) => ({
name: rv.name as string,
rootType: rv.rootType as AccountRootType,
isGroup: Boolean(rv.isGroup),
parentAccount: rv.parentAccount as string | null,
}));
this.accountMap = getMapFromList(accountList, 'name');
return this.accountMap;
}
_getRangeMapKey(entry: LedgerEntry): DateRange | null {
const entryDate = DateTime.fromISO(
entry.date!.toISOString().split('T')[0]
).toMillis();
for (const dr of this._dateRanges!) {
const toDate = dr.toDate.toMillis();
const fromDate = dr.fromDate.toMillis();
2023-01-05 06:16:52 +00:00
if (entryDate >= fromDate && entryDate < toDate) {
2022-05-16 10:10:35 +00:00
return dr;
}
}
return null;
}
// Fix arythmetic on dates when adding or substracting months. If the
// reference date was the last day in month, ensure that the resulting date is
// also the last day.
_fixMonthsJump(refDate: DateTime, date: DateTime): DateTime {
if (refDate.day == refDate.daysInMonth && date.day != date.daysInMonth) {
return date.set({ day: date.daysInMonth });
} else {
return date;
}
}
2022-05-16 10:10:35 +00:00
async _getDateRanges(): Promise<DateRange[]> {
const endpoints = await this._getFromAndToDates();
const fromDate = DateTime.fromISO(endpoints.fromDate);
const toDate = DateTime.fromISO(endpoints.toDate);
if (this.consolidateColumns) {
return [
{
toDate,
fromDate,
},
];
}
const months: number = monthsMap[this.periodicity];
const dateRanges: DateRange[] = [
{
toDate,
fromDate: this._fixMonthsJump(toDate, toDate.minus({ months })),
},
2022-05-16 10:10:35 +00:00
];
let count = this.count ?? 1;
if (this.basedOn === 'Fiscal Year') {
count = Math.ceil(((this.toYear! - this.fromYear!) * 12) / months);
}
for (let i = 1; i < count; i++) {
const lastRange = dateRanges.at(-1)!;
dateRanges.push({
toDate: lastRange.fromDate,
fromDate: this._fixMonthsJump(
toDate,
lastRange.fromDate.minus({ months })
),
2022-05-16 10:10:35 +00:00
});
}
return dateRanges.sort((b, a) => b.toDate.toMillis() - a.toDate.toMillis());
}
async _getFromAndToDates() {
let toDate: string;
let fromDate: string;
if (this.basedOn === 'Until Date') {
2023-01-05 06:16:52 +00:00
toDate = DateTime.fromISO(this.toDate!).plus({ days: 1 }).toISODate();
2022-05-16 10:10:35 +00:00
const months = monthsMap[this.periodicity] * Math.max(this.count ?? 1, 1);
2023-01-05 06:16:52 +00:00
fromDate = DateTime.fromISO(this.toDate!).minus({ months }).toISODate();
2022-05-16 10:10:35 +00:00
} else {
const fy = await getFiscalEndpoints(
this.toYear!,
this.fromYear!,
this.fyo
);
toDate = DateTime.fromISO(fy.toDate).plus({ days: 1 }).toISODate();
2022-05-16 10:10:35 +00:00
fromDate = fy.fromDate;
}
return { fromDate, toDate };
}
async _getQueryFilters(): Promise<QueryFilter> {
const filters: QueryFilter = {};
const { fromDate, toDate } = await this._getFromAndToDates();
const dateFilter: string[] = [];
2023-01-05 06:16:52 +00:00
dateFilter.push('<', toDate);
2022-05-16 10:10:35 +00:00
dateFilter.push('>=', fromDate);
filters.date = dateFilter;
filters.reverted = false;
return filters;
}
getFilters(): Field[] {
const periodNameMap: Record<Periodicity, string> = {
Monthly: t`Months`,
Quarterly: t`Quarters`,
'Half Yearly': t`Half Years`,
Yearly: t`Years`,
};
const filters = [
{
fieldtype: 'Select',
options: [
{ label: t`Fiscal Year`, value: 'Fiscal Year' },
{ label: t`Until Date`, value: 'Until Date' },
],
label: t`Based On`,
fieldname: 'basedOn',
},
{
fieldtype: 'Select',
options: [
{ label: t`Monthly`, value: 'Monthly' },
{ label: t`Quarterly`, value: 'Quarterly' },
{ label: t`Half Yearly`, value: 'Half Yearly' },
{ label: t`Yearly`, value: 'Yearly' },
],
label: t`Periodicity`,
fieldname: 'periodicity',
},
,
] as Field[];
let dateFilters = [
{
fieldtype: 'Int',
2023-01-05 06:16:52 +00:00
fieldname: 'fromYear',
placeholder: t`From Year`,
label: t`From Year`,
2022-05-16 10:10:35 +00:00
minvalue: 2000,
required: true,
},
{
fieldtype: 'Int',
2023-01-05 06:16:52 +00:00
fieldname: 'toYear',
placeholder: t`To Year`,
label: t`To Year`,
2022-05-16 10:10:35 +00:00
minvalue: 2000,
required: true,
},
] as Field[];
if (this.basedOn === 'Until Date') {
dateFilters = [
{
fieldtype: 'Date',
fieldname: 'toDate',
placeholder: t`To Date`,
label: t`To Date`,
required: true,
},
{
fieldtype: 'Int',
fieldname: 'count',
minvalue: 1,
placeholder: t`Number of ${periodNameMap[this.periodicity]}`,
label: t`Number of ${periodNameMap[this.periodicity]}`,
required: true,
},
] as Field[];
}
return [
filters,
dateFilters,
{
fieldtype: 'Check',
label: t`Consolidate Columns`,
fieldname: 'consolidateColumns',
} as Field,
{
fieldtype: 'Check',
2022-05-16 13:01:58 +00:00
label: t`Hide Group Amounts`,
fieldname: 'hideGroupAmounts',
2022-05-16 10:10:35 +00:00
} as Field,
].flat();
}
getColumns(): ColumnField[] {
const columns = [
{
label: t`Account`,
fieldtype: 'Link',
fieldname: 'account',
align: 'left',
width: ACC_NAME_WIDTH,
},
] as ColumnField[];
const dateColumns = this._dateRanges!.sort(
(a, b) => b.toDate.toMillis() - a.toDate.toMillis()
2023-01-05 06:16:52 +00:00
).map((d) => {
const toDate = d.toDate.minus({ days: 1 });
const label = this.fyo.format(toDate.toJSDate(), 'Date');
return {
label,
fieldtype: 'Data',
fieldname: 'toDate',
align: 'right',
width: ACC_BAL_WIDTH,
} as ColumnField;
});
2022-05-16 10:10:35 +00:00
return [columns, dateColumns].flat();
}
metaFilters: string[] = ['basedOn'];
}
export async function getFiscalEndpoints(
toYear: number,
fromYear: number,
fyo: Fyo
) {
2022-05-16 10:10:35 +00:00
const fys = (await fyo.getValue(
ModelNameEnum.AccountingSettings,
'fiscalYearStart'
)) as Date;
const fye = (await fyo.getValue(
ModelNameEnum.AccountingSettings,
'fiscalYearEnd'
)) as Date;
/**
* Get the month and the day, and
* prepend with the passed year.
*/
const fromDate = [
fromYear,
(fys.getMonth() + 1).toString().padStart(2, '0'),
fys.getDate().toString().padStart(2, '0'),
].join('-');
const toDate = [
toYear,
(fye.getMonth() + 1).toString().padStart(2, '0'),
fye.getDate().toString().padStart(2, '0'),
].join('-');
2022-05-16 10:10:35 +00:00
return { fromDate, toDate };
}
const monthsMap: Record<Periodicity, number> = {
Monthly: 1,
Quarterly: 3,
'Half Yearly': 6,
Yearly: 12,
};
function setPruneFlagOnAccountTreeNodes(accountTree: AccountTree) {
for (const account of Object.values(accountTree)) {
account.prune = true;
}
}
function setValueMapOnAccountTreeNodes(
accountTree: AccountTree,
rangeGroupedMap: AccountNameValueMapMap
) {
for (const name of rangeGroupedMap.keys()) {
if (!accountTree[name]) {
continue;
}
2022-05-16 10:10:35 +00:00
const valueMap = rangeGroupedMap.get(name)!;
accountTree[name].valueMap = valueMap;
accountTree[name].prune = false;
/**
* Set the update the parent account values recursively
* also prevent pruning of the parent accounts.
*/
let parentAccountName: string | null = accountTree[name].parentAccount;
while (parentAccountName !== null) {
parentAccountName = updateParentAccountWithChildValues(
accountTree,
parentAccountName,
valueMap
);
}
}
}
function updateParentAccountWithChildValues(
accountTree: AccountTree,
parentAccountName: string,
valueMap: ValueMap
): string {
const parentAccount = accountTree[parentAccountName];
parentAccount.prune = false;
parentAccount.valueMap ??= new Map();
for (const key of valueMap.keys()) {
const value = parentAccount.valueMap.get(key);
2022-05-16 13:01:58 +00:00
const childValue = valueMap.get(key);
const map: Record<string, number> = {};
for (const key of Object.keys(childValue!)) {
map[key] = (value?.[key] ?? 0) + (childValue?.[key] ?? 0);
}
parentAccount.valueMap.set(key, map);
2022-05-16 10:10:35 +00:00
}
return parentAccount.parentAccount!;
}
function setChildrenOnAccountTreeNodes(accountTree: AccountTree) {
const parentNodes: Set<string> = new Set();
for (const name of Object.keys(accountTree)) {
const ac = accountTree[name];
if (!ac.parentAccount) {
continue;
}
accountTree[ac.parentAccount].children ??= [];
accountTree[ac.parentAccount].children!.push(ac);
2022-05-16 10:10:35 +00:00
parentNodes.add(ac.parentAccount);
}
}
function deleteNonRootAccountTreeNodes(accountTree: AccountTree) {
for (const name of Object.keys(accountTree)) {
const ac = accountTree[name];
if (!ac.parentAccount) {
continue;
}
delete accountTree[name];
}
}
function pruneAccountTree(accountTree: AccountTree) {
for (const root of Object.keys(accountTree)) {
if (accountTree[root].prune) {
delete accountTree[root];
}
}
for (const root of Object.keys(accountTree)) {
accountTree[root].children = getPrunedChildren(
accountTree[root].children ?? []
);
2022-05-16 10:10:35 +00:00
}
}
function getPrunedChildren(children: AccountTreeNode[]): AccountTreeNode[] {
return children.filter((child) => {
if (child.children?.length) {
2022-05-16 10:10:35 +00:00
child.children = getPrunedChildren(child.children);
}
return !child.prune;
});
}
export function convertAccountRootNodesToAccountList(
rootNodes: AccountTreeNode[]
2022-05-16 10:10:35 +00:00
): AccountList {
if (!rootNodes || rootNodes.length == 0) {
2022-05-16 10:10:35 +00:00
return [];
}
const accountList: AccountList = [];
for (const rootNode of rootNodes) {
pushToAccountList(rootNode, accountList, 0);
}
2022-05-16 10:10:35 +00:00
return accountList;
}
function pushToAccountList(
accountTreeNode: AccountTreeNode,
accountList: AccountList,
level: number
) {
accountList.push({
name: accountTreeNode.name,
rootType: accountTreeNode.rootType,
isGroup: accountTreeNode.isGroup,
parentAccount: accountTreeNode.parentAccount,
valueMap: accountTreeNode.valueMap,
level,
});
const children = accountTreeNode.children ?? [];
const childLevel = level + 1;
for (const childNode of children) {
pushToAccountList(childNode, accountList, childLevel);
}
}
function getListOfLeafNodes(tree: Tree): TreeNode[] {
const nonGroupChildren: TreeNode[] = [];
for (const node of Object.values(tree)) {
if (!node) {
continue;
}
const groupChildren = node.children ?? [];
while (groupChildren.length) {
const child = groupChildren.shift()!;
if (!child?.children?.length) {
nonGroupChildren.push(child);
continue;
}
groupChildren.unshift(...(child.children ?? []));
}
}
return nonGroupChildren;
}