Merge pull request #36488 from frappe/mergify/bp/version-14-hotfix/pr-36333

fix: AP report does not show expense claim payables (#36333)
This commit is contained in:
Deepesh Garg
2023-08-04 17:52:52 +05:30
committed by GitHub
5 changed files with 175 additions and 120 deletions

View File

@@ -14,6 +14,7 @@ from frappe.contacts.doctype.address.address import (
from frappe.contacts.doctype.contact.contact import get_contact_details from frappe.contacts.doctype.contact.contact import get_contact_details
from frappe.core.doctype.user_permission.user_permission import get_permitted_documents from frappe.core.doctype.user_permission.user_permission import get_permitted_documents
from frappe.model.utils import get_fetch_values from frappe.model.utils import get_fetch_values
from frappe.query_builder.functions import Date, Sum
from frappe.utils import ( from frappe.utils import (
add_days, add_days,
add_months, add_months,
@@ -883,32 +884,35 @@ def get_party_shipping_address(doctype: str, name: str) -> Optional[str]:
def get_partywise_advanced_payment_amount( def get_partywise_advanced_payment_amount(
party_type, posting_date=None, future_payment=0, company=None, party=None party_type, posting_date=None, future_payment=0, company=None, party=None, account_type=None
): ):
cond = "1=1" gle = frappe.qb.DocType("GL Entry")
query = (
frappe.qb.from_(gle)
.select(gle.party)
.where(
(gle.party_type.isin(party_type)) & (gle.against_voucher.isnull()) & (gle.is_cancelled == 0)
)
.groupby(gle.party)
)
if account_type == "Receivable":
query = query.select(Sum(gle.credit).as_("amount"))
else:
query = query.select(Sum(gle.debit).as_("amount"))
if posting_date: if posting_date:
if future_payment: if future_payment:
cond = "(posting_date <= '{0}' OR DATE(creation) <= '{0}')" "".format(posting_date) query = query.where((gle.posting_date <= posting_date) | (Date(gle.creation) <= posting_date))
else: else:
cond = "posting_date <= '{0}'".format(posting_date) query = query.where(gle.posting_date <= posting_date)
if company: if company:
cond += "and company = {0}".format(frappe.db.escape(company)) query = query.where(gle.company == company)
if party: if party:
cond += "and party = {0}".format(frappe.db.escape(party)) query = query.where(gle.party == party)
data = frappe.db.sql( data = query.run(as_dict=True)
""" SELECT party, sum({0}) as amount
FROM `tabGL Entry`
WHERE
party_type = %s and against_voucher is null
and is_cancelled = 0
and {1} GROUP BY party""".format(
("credit") if party_type == "Customer" else "debit", cond
),
party_type,
)
if data: if data:
return frappe._dict(data) return frappe._dict(data)

View File

@@ -7,7 +7,7 @@ from erpnext.accounts.report.accounts_receivable.accounts_receivable import Rece
def execute(filters=None): def execute(filters=None):
args = { args = {
"party_type": "Supplier", "account_type": "Payable",
"naming_by": ["Buying Settings", "supp_master_name"], "naming_by": ["Buying Settings", "supp_master_name"],
} }
return ReceivablePayableReport(filters).run(args) return ReceivablePayableReport(filters).run(args)

View File

@@ -9,7 +9,7 @@ from erpnext.accounts.report.accounts_receivable_summary.accounts_receivable_sum
def execute(filters=None): def execute(filters=None):
args = { args = {
"party_type": "Supplier", "account_type": "Payable",
"naming_by": ["Buying Settings", "supp_master_name"], "naming_by": ["Buying Settings", "supp_master_name"],
} }
return AccountsReceivableSummary(filters).run(args) return AccountsReceivableSummary(filters).run(args)

View File

@@ -7,7 +7,7 @@ from collections import OrderedDict
import frappe import frappe
from frappe import _, qb, scrub from frappe import _, qb, scrub
from frappe.query_builder import Criterion from frappe.query_builder import Criterion
from frappe.query_builder.functions import Date from frappe.query_builder.functions import Date, Sum
from frappe.utils import cint, cstr, flt, getdate, nowdate from frappe.utils import cint, cstr, flt, getdate, nowdate
from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import ( from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import (
@@ -34,7 +34,7 @@ from erpnext.accounts.utils import get_currency_precision
def execute(filters=None): def execute(filters=None):
args = { args = {
"party_type": "Customer", "account_type": "Receivable",
"naming_by": ["Selling Settings", "cust_master_name"], "naming_by": ["Selling Settings", "cust_master_name"],
} }
return ReceivablePayableReport(filters).run(args) return ReceivablePayableReport(filters).run(args)
@@ -70,8 +70,11 @@ class ReceivablePayableReport(object):
"Company", self.filters.get("company"), "default_currency" "Company", self.filters.get("company"), "default_currency"
) )
self.currency_precision = get_currency_precision() or 2 self.currency_precision = get_currency_precision() or 2
self.dr_or_cr = "debit" if self.filters.party_type == "Customer" else "credit" self.dr_or_cr = "debit" if self.filters.account_type == "Receivable" else "credit"
self.party_type = self.filters.party_type self.account_type = self.filters.account_type
self.party_type = frappe.db.get_all(
"Party Type", {"account_type": self.account_type}, pluck="name"
)
self.party_details = {} self.party_details = {}
self.invoices = set() self.invoices = set()
self.skip_total_row = 0 self.skip_total_row = 0
@@ -197,6 +200,7 @@ class ReceivablePayableReport(object):
# no invoice, this is an invoice / stand-alone payment / credit note # no invoice, this is an invoice / stand-alone payment / credit note
row = self.voucher_balance.get((ple.voucher_type, ple.voucher_no, ple.party)) row = self.voucher_balance.get((ple.voucher_type, ple.voucher_no, ple.party))
row.party_type = ple.party_type
return row return row
def update_voucher_balance(self, ple): def update_voucher_balance(self, ple):
@@ -207,8 +211,9 @@ class ReceivablePayableReport(object):
return return
# amount in "Party Currency", if its supplied. If not, amount in company currency # amount in "Party Currency", if its supplied. If not, amount in company currency
if self.filters.get(scrub(self.party_type)): for party_type in self.party_type:
amount = ple.amount_in_account_currency if self.filters.get(scrub(party_type)):
amount = ple.amount_in_account_currency
else: else:
amount = ple.amount amount = ple.amount
amount_in_account_currency = ple.amount_in_account_currency amount_in_account_currency = ple.amount_in_account_currency
@@ -362,7 +367,7 @@ class ReceivablePayableReport(object):
def get_invoice_details(self): def get_invoice_details(self):
self.invoice_details = frappe._dict() self.invoice_details = frappe._dict()
if self.party_type == "Customer": if self.account_type == "Receivable":
si_list = frappe.db.sql( si_list = frappe.db.sql(
""" """
select name, due_date, po_no select name, due_date, po_no
@@ -390,7 +395,7 @@ class ReceivablePayableReport(object):
d.sales_person d.sales_person
) )
if self.party_type == "Supplier": if self.account_type == "Payable":
for pi in frappe.db.sql( for pi in frappe.db.sql(
""" """
select name, due_date, bill_no, bill_date select name, due_date, bill_no, bill_date
@@ -421,8 +426,10 @@ class ReceivablePayableReport(object):
# customer / supplier name # customer / supplier name
party_details = self.get_party_details(row.party) or {} party_details = self.get_party_details(row.party) or {}
row.update(party_details) row.update(party_details)
if self.filters.get(scrub(self.filters.party_type)): for party_type in self.party_type:
row.currency = row.account_currency if self.filters.get(scrub(party_type)):
row.currency = row.account_currency
break
else: else:
row.currency = self.company_currency row.currency = self.company_currency
@@ -532,65 +539,67 @@ class ReceivablePayableReport(object):
self.future_payments.setdefault((d.invoice_no, d.party), []).append(d) self.future_payments.setdefault((d.invoice_no, d.party), []).append(d)
def get_future_payments_from_payment_entry(self): def get_future_payments_from_payment_entry(self):
return frappe.db.sql( pe = frappe.qb.DocType("Payment Entry")
""" pe_ref = frappe.qb.DocType("Payment Entry Reference")
select return (
ref.reference_name as invoice_no, frappe.qb.from_(pe)
payment_entry.party, .inner_join(pe_ref)
payment_entry.party_type, .on(pe_ref.parent == pe.name)
payment_entry.posting_date as future_date, .select(
ref.allocated_amount as future_amount, (pe_ref.reference_name).as_("invoice_no"),
payment_entry.reference_no as future_ref pe.party,
from pe.party_type,
`tabPayment Entry` as payment_entry inner join `tabPayment Entry Reference` as ref (pe.posting_date).as_("future_date"),
on (pe_ref.allocated_amount).as_("future_amount"),
(ref.parent = payment_entry.name) (pe.reference_no).as_("future_ref"),
where )
payment_entry.docstatus < 2 .where(
and payment_entry.posting_date > %s (pe.docstatus < 2)
and payment_entry.party_type = %s & (pe.posting_date > self.filters.report_date)
""", & (pe.party_type.isin(self.party_type))
(self.filters.report_date, self.party_type), )
as_dict=1, ).run(as_dict=True)
)
def get_future_payments_from_journal_entry(self): def get_future_payments_from_journal_entry(self):
if self.filters.get("party"): je = frappe.qb.DocType("Journal Entry")
amount_field = ( jea = frappe.qb.DocType("Journal Entry Account")
"jea.debit_in_account_currency - jea.credit_in_account_currency" query = (
if self.party_type == "Supplier" frappe.qb.from_(je)
else "jea.credit_in_account_currency - jea.debit_in_account_currency" .inner_join(jea)
) .on(jea.parent == je.name)
else: .select(
amount_field = "jea.debit - " if self.party_type == "Supplier" else "jea.credit" jea.reference_name.as_("invoice_no"),
return frappe.db.sql(
"""
select
jea.reference_name as invoice_no,
jea.party, jea.party,
jea.party_type, jea.party_type,
je.posting_date as future_date, je.posting_date.as_("future_date"),
sum('{0}') as future_amount, je.cheque_no.as_("future_ref"),
je.cheque_no as future_ref )
from .where(
`tabJournal Entry` as je inner join `tabJournal Entry Account` as jea (je.docstatus < 2)
on & (je.posting_date > self.filters.report_date)
(jea.parent = je.name) & (jea.party_type.isin(self.party_type))
where & (jea.reference_name.isnotnull())
je.docstatus < 2 & (jea.reference_name != "")
and je.posting_date > %s )
and jea.party_type = %s
and jea.reference_name is not null and jea.reference_name != ''
group by je.name, jea.reference_name
having future_amount > 0
""".format(
amount_field
),
(self.filters.report_date, self.party_type),
as_dict=1,
) )
if self.filters.get("party"):
if self.account_type == "Payable":
query = query.select(
Sum(jea.debit_in_account_currency - jea.credit_in_account_currency).as_("future_amount")
)
else:
query = query.select(
Sum(jea.credit_in_account_currency - jea.debit_in_account_currency).as_("future_amount")
)
else:
query = query.select(
Sum(jea.debit if self.account_type == "Payable" else jea.credit).as_("future_amount")
)
query = query.having(qb.Field("future_amount") > 0)
return query.run(as_dict=True)
def allocate_future_payments(self, row): def allocate_future_payments(self, row):
# future payments are captured in additional columns # future payments are captured in additional columns
# this method allocates pending future payments against a voucher to # this method allocates pending future payments against a voucher to
@@ -619,13 +628,17 @@ class ReceivablePayableReport(object):
row.future_ref = ", ".join(row.future_ref) row.future_ref = ", ".join(row.future_ref)
def get_return_entries(self): def get_return_entries(self):
doctype = "Sales Invoice" if self.party_type == "Customer" else "Purchase Invoice" doctype = "Sales Invoice" if self.account_type == "Receivable" else "Purchase Invoice"
filters = {"is_return": 1, "docstatus": 1, "company": self.filters.company} filters = {"is_return": 1, "docstatus": 1, "company": self.filters.company}
party_field = scrub(self.filters.party_type) or_filters = {}
if self.filters.get(party_field): for party_type in self.party_type:
filters.update({party_field: self.filters.get(party_field)}) party_field = scrub(party_type)
if self.filters.get(party_field):
or_filters.update({party_field: self.filters.get(party_field)})
self.return_entries = frappe._dict( self.return_entries = frappe._dict(
frappe.get_all(doctype, filters, ["name", "return_against"], as_list=1) frappe.get_all(
doctype, filters=filters, or_filters=or_filters, fields=["name", "return_against"], as_list=1
)
) )
def set_ageing(self, row): def set_ageing(self, row):
@@ -716,6 +729,7 @@ class ReceivablePayableReport(object):
) )
.where(ple.delinked == 0) .where(ple.delinked == 0)
.where(Criterion.all(self.qb_selection_filter)) .where(Criterion.all(self.qb_selection_filter))
.where(Criterion.any(self.or_filters))
) )
if self.filters.get("group_by_party"): if self.filters.get("group_by_party"):
@@ -746,16 +760,18 @@ class ReceivablePayableReport(object):
def prepare_conditions(self): def prepare_conditions(self):
self.qb_selection_filter = [] self.qb_selection_filter = []
party_type_field = scrub(self.party_type) self.or_filters = []
self.qb_selection_filter.append(self.ple.party_type == self.party_type) for party_type in self.party_type:
party_type_field = scrub(party_type)
self.or_filters.append(self.ple.party_type == party_type)
self.add_common_filters(party_type_field=party_type_field) self.add_common_filters(party_type_field=party_type_field)
if party_type_field == "customer": if party_type_field == "customer":
self.add_customer_filters() self.add_customer_filters()
elif party_type_field == "supplier": elif party_type_field == "supplier":
self.add_supplier_filters() self.add_supplier_filters()
if self.filters.cost_center: if self.filters.cost_center:
self.get_cost_center_conditions() self.get_cost_center_conditions()
@@ -784,11 +800,10 @@ class ReceivablePayableReport(object):
self.qb_selection_filter.append(self.ple.account == self.filters.party_account) self.qb_selection_filter.append(self.ple.account == self.filters.party_account)
else: else:
# get GL with "receivable" or "payable" account_type # get GL with "receivable" or "payable" account_type
account_type = "Receivable" if self.party_type == "Customer" else "Payable"
accounts = [ accounts = [
d.name d.name
for d in frappe.get_all( for d in frappe.get_all(
"Account", filters={"account_type": account_type, "company": self.filters.company} "Account", filters={"account_type": self.account_type, "company": self.filters.company}
) )
] ]
@@ -878,7 +893,7 @@ class ReceivablePayableReport(object):
def get_party_details(self, party): def get_party_details(self, party):
if not party in self.party_details: if not party in self.party_details:
if self.party_type == "Customer": if self.account_type == "Receivable":
fields = ["customer_name", "territory", "customer_group", "customer_primary_contact"] fields = ["customer_name", "territory", "customer_group", "customer_primary_contact"]
if self.filters.get("sales_partner"): if self.filters.get("sales_partner"):
@@ -901,14 +916,20 @@ class ReceivablePayableReport(object):
self.columns = [] self.columns = []
self.add_column("Posting Date", fieldtype="Date") self.add_column("Posting Date", fieldtype="Date")
self.add_column( self.add_column(
label=_(self.party_type), label="Party Type",
fieldname="party_type",
fieldtype="Data",
width=100,
)
self.add_column(
label="Party",
fieldname="party", fieldname="party",
fieldtype="Link", fieldtype="Dynamic Link",
options=self.party_type, options="party_type",
width=180, width=180,
) )
self.add_column( self.add_column(
label="Receivable Account" if self.party_type == "Customer" else "Payable Account", label=self.account_type + " Account",
fieldname="party_account", fieldname="party_account",
fieldtype="Link", fieldtype="Link",
options="Account", options="Account",
@@ -916,13 +937,19 @@ class ReceivablePayableReport(object):
) )
if self.party_naming_by == "Naming Series": if self.party_naming_by == "Naming Series":
if self.account_type == "Payable":
label = "Supplier Name"
fieldname = "supplier_name"
else:
label = "Customer Name"
fieldname = "customer_name"
self.add_column( self.add_column(
_("{0} Name").format(self.party_type), label=label,
fieldname=scrub(self.party_type) + "_name", fieldname=fieldname,
fieldtype="Data", fieldtype="Data",
) )
if self.party_type == "Customer": if self.account_type == "Receivable":
self.add_column( self.add_column(
_("Customer Contact"), _("Customer Contact"),
fieldname="customer_primary_contact", fieldname="customer_primary_contact",
@@ -942,7 +969,7 @@ class ReceivablePayableReport(object):
self.add_column(label="Due Date", fieldtype="Date") self.add_column(label="Due Date", fieldtype="Date")
if self.party_type == "Supplier": if self.account_type == "Payable":
self.add_column(label=_("Bill No"), fieldname="bill_no", fieldtype="Data") self.add_column(label=_("Bill No"), fieldname="bill_no", fieldtype="Data")
self.add_column(label=_("Bill Date"), fieldname="bill_date", fieldtype="Date") self.add_column(label=_("Bill Date"), fieldname="bill_date", fieldtype="Date")
@@ -952,7 +979,7 @@ class ReceivablePayableReport(object):
self.add_column(_("Invoiced Amount"), fieldname="invoiced") self.add_column(_("Invoiced Amount"), fieldname="invoiced")
self.add_column(_("Paid Amount"), fieldname="paid") self.add_column(_("Paid Amount"), fieldname="paid")
if self.party_type == "Customer": if self.account_type == "Receivable":
self.add_column(_("Credit Note"), fieldname="credit_note") self.add_column(_("Credit Note"), fieldname="credit_note")
else: else:
# note: fieldname is still `credit_note` # note: fieldname is still `credit_note`
@@ -970,7 +997,7 @@ class ReceivablePayableReport(object):
self.add_column(label=_("Future Payment Amount"), fieldname="future_amount") self.add_column(label=_("Future Payment Amount"), fieldname="future_amount")
self.add_column(label=_("Remaining Balance"), fieldname="remaining_balance") self.add_column(label=_("Remaining Balance"), fieldname="remaining_balance")
if self.filters.party_type == "Customer": if self.filters.account_type == "Receivable":
self.add_column(label=_("Customer LPO"), fieldname="po_no", fieldtype="Data") self.add_column(label=_("Customer LPO"), fieldname="po_no", fieldtype="Data")
# comma separated list of linked delivery notes # comma separated list of linked delivery notes
@@ -991,7 +1018,7 @@ class ReceivablePayableReport(object):
if self.filters.sales_partner: if self.filters.sales_partner:
self.add_column(label=_("Sales Partner"), fieldname="default_sales_partner", fieldtype="Data") self.add_column(label=_("Sales Partner"), fieldname="default_sales_partner", fieldtype="Data")
if self.filters.party_type == "Supplier": if self.filters.account_type == "Payable":
self.add_column( self.add_column(
label=_("Supplier Group"), label=_("Supplier Group"),
fieldname="supplier_group", fieldname="supplier_group",

View File

@@ -12,7 +12,7 @@ from erpnext.accounts.report.accounts_receivable.accounts_receivable import Rece
def execute(filters=None): def execute(filters=None):
args = { args = {
"party_type": "Customer", "account_type": "Receivable",
"naming_by": ["Selling Settings", "cust_master_name"], "naming_by": ["Selling Settings", "cust_master_name"],
} }
@@ -21,7 +21,10 @@ def execute(filters=None):
class AccountsReceivableSummary(ReceivablePayableReport): class AccountsReceivableSummary(ReceivablePayableReport):
def run(self, args): def run(self, args):
self.party_type = args.get("party_type") self.account_type = args.get("account_type")
self.party_type = frappe.db.get_all(
"Party Type", {"account_type": self.account_type}, pluck="name"
)
self.party_naming_by = frappe.db.get_value( self.party_naming_by = frappe.db.get_value(
args.get("naming_by")[0], None, args.get("naming_by")[1] args.get("naming_by")[0], None, args.get("naming_by")[1]
) )
@@ -35,13 +38,19 @@ class AccountsReceivableSummary(ReceivablePayableReport):
self.get_party_total(args) self.get_party_total(args)
party = None
for party_type in self.party_type:
if self.filters.get(scrub(party_type)):
party = self.filters.get(scrub(party_type))
party_advance_amount = ( party_advance_amount = (
get_partywise_advanced_payment_amount( get_partywise_advanced_payment_amount(
self.party_type, self.party_type,
self.filters.report_date, self.filters.report_date,
self.filters.show_future_payments, self.filters.show_future_payments,
self.filters.company, self.filters.company,
party=self.filters.get(scrub(self.party_type)), party=party,
account_type=self.account_type,
) )
or {} or {}
) )
@@ -57,9 +66,13 @@ class AccountsReceivableSummary(ReceivablePayableReport):
row.party = party row.party = party
if self.party_naming_by == "Naming Series": if self.party_naming_by == "Naming Series":
row.party_name = frappe.get_cached_value( if self.account_type == "Payable":
self.party_type, party, scrub(self.party_type) + "_name" doctype = "Supplier"
) fieldname = "supplier_name"
else:
doctype = "Customer"
fieldname = "customer_name"
row.party_name = frappe.get_cached_value(doctype, party, fieldname)
row.update(party_dict) row.update(party_dict)
@@ -93,6 +106,7 @@ class AccountsReceivableSummary(ReceivablePayableReport):
# set territory, customer_group, sales person etc # set territory, customer_group, sales person etc
self.set_party_details(d) self.set_party_details(d)
self.party_total[d.party].update({"party_type": d.party_type})
def init_party_total(self, row): def init_party_total(self, row):
self.party_total.setdefault( self.party_total.setdefault(
@@ -131,17 +145,27 @@ class AccountsReceivableSummary(ReceivablePayableReport):
def get_columns(self): def get_columns(self):
self.columns = [] self.columns = []
self.add_column( self.add_column(
label=_(self.party_type), label="Party Type",
fieldname="party_type",
fieldtype="Data",
width=100,
)
self.add_column(
label="Party",
fieldname="party", fieldname="party",
fieldtype="Link", fieldtype="Dynamic Link",
options=self.party_type, options="party_type",
width=180, width=180,
) )
if self.party_naming_by == "Naming Series": if self.party_naming_by == "Naming Series":
self.add_column(_("{0} Name").format(self.party_type), fieldname="party_name", fieldtype="Data") self.add_column(
label="Supplier Name" if self.account_type == "Payable" else "Customer Name",
fieldname="party_name",
fieldtype="Data",
)
credit_debit_label = "Credit Note" if self.party_type == "Customer" else "Debit Note" credit_debit_label = "Credit Note" if self.account_type == "Receivable" else "Debit Note"
self.add_column(_("Advance Amount"), fieldname="advance") self.add_column(_("Advance Amount"), fieldname="advance")
self.add_column(_("Invoiced Amount"), fieldname="invoiced") self.add_column(_("Invoiced Amount"), fieldname="invoiced")
@@ -159,7 +183,7 @@ class AccountsReceivableSummary(ReceivablePayableReport):
self.add_column(label=_("Future Payment Amount"), fieldname="future_amount") self.add_column(label=_("Future Payment Amount"), fieldname="future_amount")
self.add_column(label=_("Remaining Balance"), fieldname="remaining_balance") self.add_column(label=_("Remaining Balance"), fieldname="remaining_balance")
if self.party_type == "Customer": if self.account_type == "Receivable":
self.add_column( self.add_column(
label=_("Territory"), fieldname="territory", fieldtype="Link", options="Territory" label=_("Territory"), fieldname="territory", fieldtype="Link", options="Territory"
) )