# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors # License: GNU General Public License v3. See license.txt from collections import OrderedDict import frappe from frappe import _, _dict from frappe.utils import cstr, getdate from erpnext import get_company_currency, get_default_company from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import ( get_accounting_dimensions, get_dimension_with_children, ) from erpnext.accounts.report.financial_statements import get_cost_centers_with_children from erpnext.accounts.report.utils import convert_to_presentation_currency, get_currency from erpnext.accounts.utils import get_account_currency # to cache translations TRANSLATIONS = frappe._dict() def execute(filters=None): if not filters: return [], [] account_details = {} if filters and filters.get("print_in_account_currency") and not filters.get("account"): frappe.throw(_("Select an account to print in account currency")) for acc in frappe.db.sql("""select name, is_group from tabAccount""", as_dict=1): account_details.setdefault(acc.name, acc) if filters.get("party"): filters.party = frappe.parse_json(filters.get("party")) validate_filters(filters, account_details) validate_party(filters) filters = set_account_currency(filters) columns = get_columns(filters) update_translations() res = get_result(filters, account_details) return columns, res def update_translations(): TRANSLATIONS.update( dict(OPENING=_("Opening"), TOTAL=_("Total"), CLOSING_TOTAL=_("Closing (Opening + Total)")) ) def validate_filters(filters, account_details): if not filters.get("company"): frappe.throw(_("{0} is mandatory").format(_("Company"))) if not filters.get("from_date") and not filters.get("to_date"): frappe.throw( _("{0} and {1} are mandatory").format(frappe.bold(_("From Date")), frappe.bold(_("To Date"))) ) if filters.get("account"): filters.account = frappe.parse_json(filters.get("account")) for account in filters.account: if not account_details.get(account): frappe.throw(_("Account {0} does not exists").format(account)) if filters.get("account") and filters.get("group_by") == "Group by Account": filters.account = frappe.parse_json(filters.get("account")) for account in filters.account: if account_details[account].is_group == 0: frappe.throw(_("Can not filter based on Child Account, if grouped by Account")) if filters.get("voucher_no") and filters.get("group_by") in ["Group by Voucher"]: frappe.throw(_("Can not filter based on Voucher No, if grouped by Voucher")) if filters.from_date > filters.to_date: frappe.throw(_("From Date must be before To Date")) if filters.get("project"): filters.project = frappe.parse_json(filters.get("project")) if filters.get("cost_center"): filters.cost_center = frappe.parse_json(filters.get("cost_center")) def validate_party(filters): party_type, party = filters.get("party_type"), filters.get("party") if party and party_type: for d in party: if not frappe.db.exists(party_type, d): frappe.throw(_("Invalid {0}: {1}").format(party_type, d)) def set_account_currency(filters): if filters.get("account") or (filters.get("party") and len(filters.party) == 1): filters["company_currency"] = frappe.get_cached_value( "Company", filters.company, "default_currency" ) account_currency = None if filters.get("account"): if len(filters.get("account")) == 1: account_currency = get_account_currency(filters.account[0]) else: currency = get_account_currency(filters.account[0]) is_same_account_currency = True for account in filters.get("account"): if get_account_currency(account) != currency: is_same_account_currency = False break if is_same_account_currency: account_currency = currency elif filters.get("party"): gle_currency = frappe.db.get_value( "GL Entry", {"party_type": filters.party_type, "party": filters.party[0], "company": filters.company}, "account_currency", ) if gle_currency: account_currency = gle_currency else: account_currency = ( None if filters.party_type in ["Employee", "Shareholder", "Member"] else frappe.db.get_value(filters.party_type, filters.party[0], "default_currency") ) filters["account_currency"] = account_currency or filters.company_currency if filters.account_currency != filters.company_currency and not filters.presentation_currency: filters.presentation_currency = filters.account_currency return filters def get_result(filters, account_details): accounting_dimensions = [] if filters.get("include_dimensions"): accounting_dimensions = get_accounting_dimensions() gl_entries = get_gl_entries(filters, accounting_dimensions) data = get_data_with_opening_closing(filters, account_details, accounting_dimensions, gl_entries) result = get_result_as_list(data, filters) return result def get_gl_entries(filters, accounting_dimensions): currency_map = get_currency(filters) select_fields = """, debit, credit, debit_in_account_currency, credit_in_account_currency """ order_by_statement = "order by posting_date, account, creation" if filters.get("include_dimensions"): order_by_statement = "order by posting_date, creation" if filters.get("group_by") == "Group by Voucher": order_by_statement = "order by posting_date, voucher_type, voucher_no" if filters.get("group_by") == "Group by Account": order_by_statement = "order by account, posting_date, creation" if filters.get("include_default_book_entries"): filters["company_fb"] = frappe.db.get_value( "Company", filters.get("company"), "default_finance_book" ) dimension_fields = "" if accounting_dimensions: dimension_fields = ", ".join(accounting_dimensions) + "," gl_entries = frappe.db.sql( """ select name as gl_entry, posting_date, account, party_type, party, voucher_type, voucher_no, {dimension_fields} cost_center, project, against_voucher_type, against_voucher, account_currency, remarks, against, is_opening, creation {select_fields} from `tabGL Entry` where company=%(company)s {conditions} {order_by_statement} """.format( dimension_fields=dimension_fields, select_fields=select_fields, conditions=get_conditions(filters), order_by_statement=order_by_statement, ), filters, as_dict=1, ) if filters.get("presentation_currency"): return convert_to_presentation_currency(gl_entries, currency_map, filters.get("company")) else: return gl_entries def get_conditions(filters): conditions = [] if filters.get("account"): filters.account = get_accounts_with_children(filters.account) conditions.append("account in %(account)s") if filters.get("cost_center"): filters.cost_center = get_cost_centers_with_children(filters.cost_center) conditions.append("cost_center in %(cost_center)s") if filters.get("voucher_no"): conditions.append("voucher_no=%(voucher_no)s") if filters.get("group_by") == "Group by Party" and not filters.get("party_type"): conditions.append("party_type in ('Customer', 'Supplier')") if filters.get("party_type"): conditions.append("party_type=%(party_type)s") if filters.get("party"): conditions.append("party in %(party)s") if not ( filters.get("account") or filters.get("party") or filters.get("group_by") in ["Group by Account", "Group by Party"] ): conditions.append("(posting_date >=%(from_date)s or is_opening = 'Yes')") conditions.append("(posting_date <=%(to_date)s)") if filters.get("project"): conditions.append("project in %(project)s") if filters.get("finance_book"): if filters.get("include_default_book_entries"): conditions.append( "(finance_book in (%(finance_book)s, %(company_fb)s, '') OR finance_book IS NULL)" ) else: conditions.append("finance_book in (%(finance_book)s)") if not filters.get("show_cancelled_entries"): conditions.append("is_cancelled = 0") from frappe.desk.reportview import build_match_conditions match_conditions = build_match_conditions("GL Entry") if match_conditions: conditions.append(match_conditions) if filters.get("include_dimensions"): accounting_dimensions = get_accounting_dimensions(as_list=False) if accounting_dimensions: for dimension in accounting_dimensions: if not dimension.disabled: if filters.get(dimension.fieldname): if frappe.get_cached_value("DocType", dimension.document_type, "is_tree"): filters[dimension.fieldname] = get_dimension_with_children( dimension.document_type, filters.get(dimension.fieldname) ) conditions.append("{0} in %({0})s".format(dimension.fieldname)) else: conditions.append("{0} in %({0})s".format(dimension.fieldname)) return "and {}".format(" and ".join(conditions)) if conditions else "" def get_accounts_with_children(accounts): if not isinstance(accounts, list): accounts = [d.strip() for d in accounts.strip().split(",") if d] all_accounts = [] for d in accounts: if frappe.db.exists("Account", d): lft, rgt = frappe.db.get_value("Account", d, ["lft", "rgt"]) children = frappe.get_all("Account", filters={"lft": [">=", lft], "rgt": ["<=", rgt]}) all_accounts += [c.name for c in children] else: frappe.throw(_("Account: {0} does not exist").format(d)) return list(set(all_accounts)) def get_data_with_opening_closing(filters, account_details, accounting_dimensions, gl_entries): data = [] gle_map = initialize_gle_map(gl_entries, filters) totals, entries = get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map) # Opening for filtered account data.append(totals.opening) if filters.get("group_by") != "Group by Voucher (Consolidated)": for acc, acc_dict in gle_map.items(): # acc if acc_dict.entries: # opening data.append({}) if filters.get("group_by") != "Group by Voucher": data.append(acc_dict.totals.opening) data += acc_dict.entries # totals data.append(acc_dict.totals.total) # closing if filters.get("group_by") != "Group by Voucher": data.append(acc_dict.totals.closing) data.append({}) else: data += entries # totals data.append(totals.total) # closing data.append(totals.closing) return data def get_totals_dict(): def _get_debit_credit_dict(label): return _dict( account="'{0}'".format(label), debit=0.0, credit=0.0, debit_in_account_currency=0.0, credit_in_account_currency=0.0, ) return _dict( opening=_get_debit_credit_dict(TRANSLATIONS.OPENING), total=_get_debit_credit_dict(TRANSLATIONS.TOTAL), closing=_get_debit_credit_dict(TRANSLATIONS.CLOSING_TOTAL), ) def group_by_field(group_by): if group_by == "Group by Party": return "party" elif group_by in ["Group by Voucher (Consolidated)", "Group by Account"]: return "account" else: return "voucher_no" def initialize_gle_map(gl_entries, filters): gle_map = OrderedDict() group_by = group_by_field(filters.get("group_by")) for gle in gl_entries: gle_map.setdefault(gle.get(group_by), _dict(totals=get_totals_dict(), entries=[])) return gle_map def get_accountwise_gle(filters, accounting_dimensions, gl_entries, gle_map): totals = get_totals_dict() entries = [] consolidated_gle = OrderedDict() group_by = group_by_field(filters.get("group_by")) group_by_voucher_consolidated = filters.get("group_by") == "Group by Voucher (Consolidated)" if filters.get("show_net_values_in_party_account"): account_type_map = get_account_type_map(filters.get("company")) def update_value_in_dict(data, key, gle): data[key].debit += gle.debit data[key].credit += gle.credit data[key].debit_in_account_currency += gle.debit_in_account_currency data[key].credit_in_account_currency += gle.credit_in_account_currency if filters.get("show_net_values_in_party_account") and account_type_map.get( data[key].account ) in ("Receivable", "Payable"): net_value = data[key].debit - data[key].credit net_value_in_account_currency = ( data[key].debit_in_account_currency - data[key].credit_in_account_currency ) if net_value < 0: dr_or_cr = "credit" rev_dr_or_cr = "debit" else: dr_or_cr = "debit" rev_dr_or_cr = "credit" data[key][dr_or_cr] = abs(net_value) data[key][dr_or_cr + "_in_account_currency"] = abs(net_value_in_account_currency) data[key][rev_dr_or_cr] = 0 data[key][rev_dr_or_cr + "_in_account_currency"] = 0 if data[key].against_voucher and gle.against_voucher: data[key].against_voucher += ", " + gle.against_voucher from_date, to_date = getdate(filters.from_date), getdate(filters.to_date) show_opening_entries = filters.get("show_opening_entries") for gle in gl_entries: group_by_value = gle.get(group_by) if gle.posting_date < from_date or (cstr(gle.is_opening) == "Yes" and not show_opening_entries): if not group_by_voucher_consolidated: update_value_in_dict(gle_map[group_by_value].totals, "opening", gle) update_value_in_dict(gle_map[group_by_value].totals, "closing", gle) update_value_in_dict(totals, "opening", gle) update_value_in_dict(totals, "closing", gle) elif gle.posting_date <= to_date or (cstr(gle.is_opening) == "Yes" and show_opening_entries): if not group_by_voucher_consolidated: update_value_in_dict(gle_map[group_by_value].totals, "total", gle) update_value_in_dict(gle_map[group_by_value].totals, "closing", gle) update_value_in_dict(totals, "total", gle) update_value_in_dict(totals, "closing", gle) gle_map[group_by_value].entries.append(gle) elif group_by_voucher_consolidated: keylist = [ gle.get("voucher_type"), gle.get("voucher_no"), gle.get("account"), gle.get("party_type"), gle.get("party"), ] if filters.get("include_dimensions"): for dim in accounting_dimensions: keylist.append(gle.get(dim)) keylist.append(gle.get("cost_center")) key = tuple(keylist) if key not in consolidated_gle: consolidated_gle.setdefault(key, gle) else: update_value_in_dict(consolidated_gle, key, gle) for key, value in consolidated_gle.items(): update_value_in_dict(totals, "total", value) update_value_in_dict(totals, "closing", value) entries.append(value) return totals, entries def get_account_type_map(company): account_type_map = frappe._dict( frappe.get_all( "Account", fields=["name", "account_type"], filters={"company": company}, as_list=1 ) ) return account_type_map def get_result_as_list(data, filters): balance, balance_in_account_currency = 0, 0 inv_details = get_supplier_invoice_details() for d in data: if not d.get("posting_date"): balance, balance_in_account_currency = 0, 0 balance = get_balance(d, balance, "debit", "credit") d["balance"] = balance d["account_currency"] = filters.account_currency d["bill_no"] = inv_details.get(d.get("against_voucher"), "") return data def get_supplier_invoice_details(): inv_details = {} for d in frappe.db.sql( """ select name, bill_no from `tabPurchase Invoice` where docstatus = 1 and bill_no is not null and bill_no != '' """, as_dict=1, ): inv_details[d.name] = d.bill_no return inv_details def get_balance(row, balance, debit_field, credit_field): balance += row.get(debit_field, 0) - row.get(credit_field, 0) return balance def get_columns(filters): if filters.get("presentation_currency"): currency = filters["presentation_currency"] else: if filters.get("company"): currency = get_company_currency(filters["company"]) else: company = get_default_company() currency = get_company_currency(company) columns = [ { "label": _("GL Entry"), "fieldname": "gl_entry", "fieldtype": "Link", "options": "GL Entry", "hidden": 1, }, {"label": _("Posting Date"), "fieldname": "posting_date", "fieldtype": "Date", "width": 90}, { "label": _("Account"), "fieldname": "account", "fieldtype": "Link", "options": "Account", "width": 180, }, { "label": _("Debit ({0})").format(currency), "fieldname": "debit", "fieldtype": "Float", "width": 100, }, { "label": _("Credit ({0})").format(currency), "fieldname": "credit", "fieldtype": "Float", "width": 100, }, { "label": _("Balance ({0})").format(currency), "fieldname": "balance", "fieldtype": "Float", "width": 130, }, {"label": _("Voucher Type"), "fieldname": "voucher_type", "width": 120}, { "label": _("Voucher No"), "fieldname": "voucher_no", "fieldtype": "Dynamic Link", "options": "voucher_type", "width": 180, }, {"label": _("Against Account"), "fieldname": "against", "width": 120}, {"label": _("Party Type"), "fieldname": "party_type", "width": 100}, {"label": _("Party"), "fieldname": "party", "width": 100}, {"label": _("Project"), "options": "Project", "fieldname": "project", "width": 100}, ] if filters.get("include_dimensions"): for dim in get_accounting_dimensions(as_list=False): columns.append( {"label": _(dim.label), "options": dim.label, "fieldname": dim.fieldname, "width": 100} ) columns.append( {"label": _("Cost Center"), "options": "Cost Center", "fieldname": "cost_center", "width": 100} ) columns.extend( [ {"label": _("Against Voucher Type"), "fieldname": "against_voucher_type", "width": 100}, { "label": _("Against Voucher"), "fieldname": "against_voucher", "fieldtype": "Dynamic Link", "options": "against_voucher_type", "width": 100, }, {"label": _("Supplier Invoice No"), "fieldname": "bill_no", "fieldtype": "Data", "width": 100}, {"label": _("Remarks"), "fieldname": "remarks", "width": 400}, ] ) return columns