|
|
@ -4,7 +4,7 @@
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
import frappe, erpnext
|
|
|
|
import frappe, erpnext
|
|
|
|
from frappe import _, scrub
|
|
|
|
from frappe import _, scrub
|
|
|
|
from frappe.utils import getdate, nowdate, flt, cint
|
|
|
|
from frappe.utils import getdate, nowdate, flt, cint, formatdate, cstr
|
|
|
|
|
|
|
|
|
|
|
|
class ReceivablePayableReport(object):
|
|
|
|
class ReceivablePayableReport(object):
|
|
|
|
def __init__(self, filters=None):
|
|
|
|
def __init__(self, filters=None):
|
|
|
@ -57,6 +57,21 @@ class ReceivablePayableReport(object):
|
|
|
|
|
|
|
|
|
|
|
|
credit_or_debit_note = "Credit Note" if args.get("party_type") == "Customer" else "Debit Note"
|
|
|
|
credit_or_debit_note = "Credit Note" if args.get("party_type") == "Customer" else "Debit Note"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if self.filters.based_on_payment_terms:
|
|
|
|
|
|
|
|
columns.append({
|
|
|
|
|
|
|
|
"label": "Payment Term",
|
|
|
|
|
|
|
|
"fieldname": "payment_term",
|
|
|
|
|
|
|
|
"fieldtype": "Data",
|
|
|
|
|
|
|
|
"width": 120
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
columns.append({
|
|
|
|
|
|
|
|
"label": "Invoice Grand Total",
|
|
|
|
|
|
|
|
"fieldname": "invoice_grand_total",
|
|
|
|
|
|
|
|
"fieldtype": "Currency",
|
|
|
|
|
|
|
|
"options": "currency",
|
|
|
|
|
|
|
|
"width": 120
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
for label in ("Invoiced Amount", "Paid Amount", credit_or_debit_note, "Outstanding Amount"):
|
|
|
|
for label in ("Invoiced Amount", "Paid Amount", credit_or_debit_note, "Outstanding Amount"):
|
|
|
|
columns.append({
|
|
|
|
columns.append({
|
|
|
|
"label": label,
|
|
|
|
"label": label,
|
|
|
@ -97,12 +112,6 @@ class ReceivablePayableReport(object):
|
|
|
|
"options": "Currency",
|
|
|
|
"options": "Currency",
|
|
|
|
"width": 100
|
|
|
|
"width": 100
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
|
|
|
|
"fieldname": "pdc/lc_date",
|
|
|
|
|
|
|
|
"label": _("PDC/LC Date"),
|
|
|
|
|
|
|
|
"fieldtype": "Date",
|
|
|
|
|
|
|
|
"width": 110
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
"fieldname": "pdc/lc_ref",
|
|
|
|
"fieldname": "pdc/lc_ref",
|
|
|
|
"label": _("PDC/LC Ref"),
|
|
|
|
"label": _("PDC/LC Ref"),
|
|
|
@ -113,14 +122,14 @@ class ReceivablePayableReport(object):
|
|
|
|
"fieldname": "pdc/lc_amount",
|
|
|
|
"fieldname": "pdc/lc_amount",
|
|
|
|
"label": _("PDC/LC Amount"),
|
|
|
|
"label": _("PDC/LC Amount"),
|
|
|
|
"fieldtype": "Currency",
|
|
|
|
"fieldtype": "Currency",
|
|
|
|
"options": "Currency",
|
|
|
|
"options": "currency",
|
|
|
|
"width": 130
|
|
|
|
"width": 130
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
{
|
|
|
|
"fieldname": "remaining_balance",
|
|
|
|
"fieldname": "remaining_balance",
|
|
|
|
"label": _("Remaining Balance"),
|
|
|
|
"label": _("Remaining Balance"),
|
|
|
|
"fieldtype": "Currency",
|
|
|
|
"fieldtype": "Currency",
|
|
|
|
"options": "Currency",
|
|
|
|
"options": "currency",
|
|
|
|
"width": 130
|
|
|
|
"width": 130
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
|
|
|
|
|
|
|
@ -151,32 +160,122 @@ class ReceivablePayableReport(object):
|
|
|
|
|
|
|
|
|
|
|
|
def get_data(self, party_naming_by, args):
|
|
|
|
def get_data(self, party_naming_by, args):
|
|
|
|
from erpnext.accounts.utils import get_currency_precision
|
|
|
|
from erpnext.accounts.utils import get_currency_precision
|
|
|
|
currency_precision = get_currency_precision() or 2
|
|
|
|
self.currency_precision = get_currency_precision() or 2
|
|
|
|
dr_or_cr = "debit" if args.get("party_type") == "Customer" else "credit"
|
|
|
|
self.dr_or_cr = "debit" if args.get("party_type") == "Customer" else "credit"
|
|
|
|
|
|
|
|
|
|
|
|
future_vouchers = self.get_entries_after(self.filters.report_date, args.get("party_type"))
|
|
|
|
future_vouchers = self.get_entries_after(self.filters.report_date, args.get("party_type"))
|
|
|
|
|
|
|
|
|
|
|
|
if not self.filters.get("company"):
|
|
|
|
if not self.filters.get("company"):
|
|
|
|
self.filters["company"] = frappe.db.get_single_value('Global Defaults', 'default_company')
|
|
|
|
self.filters["company"] = frappe.db.get_single_value('Global Defaults', 'default_company')
|
|
|
|
|
|
|
|
|
|
|
|
company_currency = frappe.get_cached_value('Company', self.filters.get("company"), "default_currency")
|
|
|
|
self.company_currency = frappe.get_cached_value('Company', self.filters.get("company"), "default_currency")
|
|
|
|
|
|
|
|
|
|
|
|
return_entries = self.get_return_entries(args.get("party_type"))
|
|
|
|
return_entries = self.get_return_entries(args.get("party_type"))
|
|
|
|
|
|
|
|
|
|
|
|
data = []
|
|
|
|
data = []
|
|
|
|
pdc_details = get_pdc_details(args.get("party_type"), self.filters.report_date)
|
|
|
|
self.pdc_details = get_pdc_details(args.get("party_type"), self.filters.report_date)
|
|
|
|
gl_entries_data = self.get_entries_till(self.filters.report_date, args.get("party_type"))
|
|
|
|
gl_entries_data = self.get_entries_till(self.filters.report_date, args.get("party_type"))
|
|
|
|
|
|
|
|
|
|
|
|
if gl_entries_data:
|
|
|
|
if gl_entries_data:
|
|
|
|
voucher_nos = [d.voucher_no for d in gl_entries_data] or []
|
|
|
|
voucher_nos = [d.voucher_no for d in gl_entries_data] or []
|
|
|
|
dn_details = get_dn_details(args.get("party_type"), voucher_nos)
|
|
|
|
dn_details = get_dn_details(args.get("party_type"), voucher_nos)
|
|
|
|
voucher_details = get_voucher_details(args.get("party_type"), voucher_nos, dn_details)
|
|
|
|
self.voucher_details = get_voucher_details(args.get("party_type"), voucher_nos, dn_details)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if self.filters.based_on_payment_terms:
|
|
|
|
|
|
|
|
self.payment_term_map = self.get_payment_term_detail(voucher_nos)
|
|
|
|
|
|
|
|
|
|
|
|
for gle in gl_entries_data:
|
|
|
|
for gle in gl_entries_data:
|
|
|
|
if self.is_receivable_or_payable(gle, dr_or_cr, future_vouchers):
|
|
|
|
if self.is_receivable_or_payable(gle, self.dr_or_cr, future_vouchers):
|
|
|
|
outstanding_amount, credit_note_amount = self.get_outstanding_amount(gle,
|
|
|
|
outstanding_amount, credit_note_amount, payment_amount = self.get_outstanding_amount(
|
|
|
|
self.filters.report_date, dr_or_cr, return_entries, currency_precision)
|
|
|
|
gle,self.filters.report_date, self.dr_or_cr, return_entries)
|
|
|
|
if abs(outstanding_amount) > 0.1/10**currency_precision:
|
|
|
|
|
|
|
|
|
|
|
|
temp_outstanding_amt = outstanding_amount
|
|
|
|
|
|
|
|
temp_credit_note_amt = credit_note_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if abs(outstanding_amount) > 0.1/10**self.currency_precision:
|
|
|
|
|
|
|
|
if self.filters.based_on_payment_terms and self.payment_term_map.get(gle.voucher_no):
|
|
|
|
|
|
|
|
for d in self.payment_term_map.get(gle.voucher_no):
|
|
|
|
|
|
|
|
# Allocate payment amount based on payment terms(FIFO order)
|
|
|
|
|
|
|
|
payment_amount, d.payment_amount = self.allocate_based_on_fifo(payment_amount, d.payment_term_amount)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
term_outstanding_amount = d.payment_term_amount - d.payment_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Allocate credit note based on payment terms(FIFO order)
|
|
|
|
|
|
|
|
credit_note_amount, d.credit_note_amount = self.allocate_based_on_fifo(credit_note_amount, term_outstanding_amount)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
term_outstanding_amount -= d.credit_note_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
row_outstanding = term_outstanding_amount
|
|
|
|
|
|
|
|
# Allocate PDC based on payment terms(FIFO order)
|
|
|
|
|
|
|
|
d.pdc_details, d.pdc_amount = self.allocate_pdc_amount_in_fifo(gle, row_outstanding)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if term_outstanding_amount > 0:
|
|
|
|
|
|
|
|
row = self.prepare_row(party_naming_by, args, gle, term_outstanding_amount,
|
|
|
|
|
|
|
|
d.credit_note_amount, d.due_date, d.payment_amount , d.payment_term_amount,
|
|
|
|
|
|
|
|
d.description, d.pdc_amount, d.pdc_details)
|
|
|
|
|
|
|
|
data.append(row)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if credit_note_amount:
|
|
|
|
|
|
|
|
row = self.prepare_row_without_payment_terms(party_naming_by, args, gle, temp_outstanding_amt,
|
|
|
|
|
|
|
|
temp_credit_note_amt)
|
|
|
|
|
|
|
|
data.append(row)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
row = self.prepare_row_without_payment_terms(party_naming_by, args, gle, outstanding_amount,
|
|
|
|
|
|
|
|
credit_note_amount)
|
|
|
|
|
|
|
|
data.append(row)
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def allocate_pdc_amount_in_fifo(self, gle, row_outstanding):
|
|
|
|
|
|
|
|
pdc_list = self.pdc_details.get((gle.voucher_no, gle.party), [])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pdc_details = []
|
|
|
|
|
|
|
|
pdc_amount = 0
|
|
|
|
|
|
|
|
for pdc in pdc_list:
|
|
|
|
|
|
|
|
if row_outstanding <= pdc.pdc_amount:
|
|
|
|
|
|
|
|
pdc_amount += row_outstanding
|
|
|
|
|
|
|
|
pdc.pdc_amount -= row_outstanding
|
|
|
|
|
|
|
|
if row_outstanding and pdc.pdc_ref and pdc.pdc_date:
|
|
|
|
|
|
|
|
pdc_details.append(cstr(pdc.pdc_ref) + "/" + formatdate(pdc.pdc_date))
|
|
|
|
|
|
|
|
row_outstanding = 0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
pdc_amount = pdc.pdc_amount
|
|
|
|
|
|
|
|
if pdc.pdc_amount and pdc.pdc_ref and pdc.pdc_date:
|
|
|
|
|
|
|
|
pdc_details.append(cstr(pdc.pdc_ref) + "/" + formatdate(pdc.pdc_date))
|
|
|
|
|
|
|
|
pdc.pdc_amount = 0
|
|
|
|
|
|
|
|
row_outstanding -= pdc_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return pdc_details, pdc_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def prepare_row_without_payment_terms(self, party_naming_by, args, gle, outstanding_amount, credit_note_amount):
|
|
|
|
|
|
|
|
pdc_list = self.pdc_details.get((gle.voucher_no, gle.party), [])
|
|
|
|
|
|
|
|
pdc_amount = 0
|
|
|
|
|
|
|
|
pdc_details = []
|
|
|
|
|
|
|
|
for d in pdc_list:
|
|
|
|
|
|
|
|
pdc_amount += flt(d.pdc_amount)
|
|
|
|
|
|
|
|
if pdc_amount and d.pdc_ref and d.pdc_date:
|
|
|
|
|
|
|
|
pdc_details.append(cstr(d.pdc_ref) + "/" + formatdate(d.pdc_date))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
row = self.prepare_row(party_naming_by, args, gle, outstanding_amount,
|
|
|
|
|
|
|
|
credit_note_amount, pdc_amount=pdc_amount, pdc_details=pdc_details)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return row
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def allocate_based_on_fifo(self, total_amount, row_amount):
|
|
|
|
|
|
|
|
allocated_amount = 0
|
|
|
|
|
|
|
|
if row_amount <= total_amount:
|
|
|
|
|
|
|
|
allocated_amount = row_amount
|
|
|
|
|
|
|
|
total_amount -= row_amount
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
allocated_amount = total_amount
|
|
|
|
|
|
|
|
total_amount = 0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return total_amount, allocated_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def prepare_row(self, party_naming_by, args, gle, outstanding_amount, credit_note_amount,
|
|
|
|
|
|
|
|
due_date=None, paid_amt=None, payment_term_amount=None, payment_term=None, pdc_amount=None, pdc_details=None):
|
|
|
|
row = [gle.posting_date, gle.party]
|
|
|
|
row = [gle.posting_date, gle.party]
|
|
|
|
|
|
|
|
|
|
|
|
# customer / supplier name
|
|
|
|
# customer / supplier name
|
|
|
@ -184,20 +283,28 @@ class ReceivablePayableReport(object):
|
|
|
|
row += [self.get_party_name(gle.party_type, gle.party)]
|
|
|
|
row += [self.get_party_name(gle.party_type, gle.party)]
|
|
|
|
|
|
|
|
|
|
|
|
# get due date
|
|
|
|
# get due date
|
|
|
|
due_date = voucher_details.get(gle.voucher_no, {}).get("due_date", "")
|
|
|
|
if not due_date:
|
|
|
|
bill_date = voucher_details.get(gle.voucher_no, {}).get("bill_date", "")
|
|
|
|
due_date = self.voucher_details.get(gle.voucher_no, {}).get("due_date", "")
|
|
|
|
|
|
|
|
bill_date = self.voucher_details.get(gle.voucher_no, {}).get("bill_date", "")
|
|
|
|
|
|
|
|
|
|
|
|
row += [gle.voucher_type, gle.voucher_no, due_date]
|
|
|
|
row += [gle.voucher_type, gle.voucher_no, due_date]
|
|
|
|
|
|
|
|
|
|
|
|
# get supplier bill details
|
|
|
|
# get supplier bill details
|
|
|
|
if args.get("party_type") == "Supplier":
|
|
|
|
if args.get("party_type") == "Supplier":
|
|
|
|
row += [
|
|
|
|
row += [
|
|
|
|
voucher_details.get(gle.voucher_no, {}).get("bill_no", ""),
|
|
|
|
self.voucher_details.get(gle.voucher_no, {}).get("bill_no", ""),
|
|
|
|
voucher_details.get(gle.voucher_no, {}).get("bill_date", "")
|
|
|
|
self.voucher_details.get(gle.voucher_no, {}).get("bill_date", "")
|
|
|
|
]
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
# invoiced and paid amounts
|
|
|
|
# invoiced and paid amounts
|
|
|
|
invoiced_amount = gle.get(dr_or_cr) if (gle.get(dr_or_cr) > 0) else 0
|
|
|
|
invoiced_amount = gle.get(self.dr_or_cr) if (gle.get(self.dr_or_cr) > 0) else 0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if self.filters.based_on_payment_terms:
|
|
|
|
|
|
|
|
row+=[payment_term, invoiced_amount]
|
|
|
|
|
|
|
|
if payment_term_amount:
|
|
|
|
|
|
|
|
invoiced_amount = payment_term_amount
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if not payment_term_amount:
|
|
|
|
paid_amt = invoiced_amount - outstanding_amount - credit_note_amount
|
|
|
|
paid_amt = invoiced_amount - outstanding_amount - credit_note_amount
|
|
|
|
row += [invoiced_amount, paid_amt, credit_note_amount, outstanding_amount]
|
|
|
|
row += [invoiced_amount, paid_amt, credit_note_amount, outstanding_amount]
|
|
|
|
|
|
|
|
|
|
|
@ -226,32 +333,29 @@ class ReceivablePayableReport(object):
|
|
|
|
if self.filters.get(scrub(args.get("party_type"))):
|
|
|
|
if self.filters.get(scrub(args.get("party_type"))):
|
|
|
|
row.append(gle.account_currency)
|
|
|
|
row.append(gle.account_currency)
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
row.append(company_currency)
|
|
|
|
row.append(self.company_currency)
|
|
|
|
|
|
|
|
|
|
|
|
pdc = pdc_details.get((gle.voucher_no, gle.party), {})
|
|
|
|
remaining_balance = outstanding_amount - flt(pdc_amount)
|
|
|
|
|
|
|
|
pdc_details = ", ".join(pdc_details)
|
|
|
|
remaining_balance = outstanding_amount - flt(pdc.get("pdc_amount"))
|
|
|
|
row += [pdc_details, pdc_amount, remaining_balance]
|
|
|
|
row += [pdc.get("pdc_date"), pdc.get("pdc_ref"),
|
|
|
|
|
|
|
|
flt(pdc.get("pdc_amount")), remaining_balance]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if args.get('party_type') == 'Customer':
|
|
|
|
if args.get('party_type') == 'Customer':
|
|
|
|
# customer LPO
|
|
|
|
# customer LPO
|
|
|
|
row += [voucher_details.get(gle.voucher_no, {}).get("po_no")]
|
|
|
|
row += [self.voucher_details.get(gle.voucher_no, {}).get("po_no")]
|
|
|
|
|
|
|
|
|
|
|
|
# Delivery Note
|
|
|
|
# Delivery Note
|
|
|
|
row += [voucher_details.get(gle.voucher_no, {}).get("delivery_note")]
|
|
|
|
row += [self.voucher_details.get(gle.voucher_no, {}).get("delivery_note")]
|
|
|
|
|
|
|
|
|
|
|
|
# customer territory / supplier group
|
|
|
|
# customer territory / supplier group
|
|
|
|
if args.get("party_type") == "Customer":
|
|
|
|
if args.get("party_type") == "Customer":
|
|
|
|
row += [self.get_territory(gle.party), self.get_customer_group(gle.party),
|
|
|
|
row += [self.get_territory(gle.party), self.get_customer_group(gle.party),
|
|
|
|
voucher_details.get(gle.voucher_no, {}).get("sales_person")]
|
|
|
|
self.voucher_details.get(gle.voucher_no, {}).get("sales_person")]
|
|
|
|
if args.get("party_type") == "Supplier":
|
|
|
|
if args.get("party_type") == "Supplier":
|
|
|
|
row += [self.get_supplier_group(gle.party)]
|
|
|
|
row += [self.get_supplier_group(gle.party)]
|
|
|
|
|
|
|
|
|
|
|
|
row.append(gle.remarks)
|
|
|
|
row.append(gle.remarks)
|
|
|
|
data.append(row)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return data
|
|
|
|
return row
|
|
|
|
|
|
|
|
|
|
|
|
def get_entries_after(self, report_date, party_type):
|
|
|
|
def get_entries_after(self, report_date, party_type):
|
|
|
|
# returns a distinct list
|
|
|
|
# returns a distinct list
|
|
|
@ -280,25 +384,25 @@ class ReceivablePayableReport(object):
|
|
|
|
doctype = "Sales Invoice" if party_type=="Customer" else "Purchase Invoice"
|
|
|
|
doctype = "Sales Invoice" if party_type=="Customer" else "Purchase Invoice"
|
|
|
|
return [d.name for d in frappe.get_all(doctype, filters={"is_return": 1, "docstatus": 1})]
|
|
|
|
return [d.name for d in frappe.get_all(doctype, filters={"is_return": 1, "docstatus": 1})]
|
|
|
|
|
|
|
|
|
|
|
|
def get_outstanding_amount(self, gle, report_date, dr_or_cr, return_entries, currency_precision):
|
|
|
|
def get_outstanding_amount(self, gle, report_date, dr_or_cr, return_entries):
|
|
|
|
payment_amount, credit_note_amount = 0.0, 0.0
|
|
|
|
payment_amount, credit_note_amount = 0.0, 0.0
|
|
|
|
reverse_dr_or_cr = "credit" if dr_or_cr=="debit" else "debit"
|
|
|
|
reverse_dr_or_cr = "credit" if dr_or_cr=="debit" else "debit"
|
|
|
|
|
|
|
|
|
|
|
|
for e in self.get_gl_entries_for(gle.party, gle.party_type, gle.voucher_type, gle.voucher_no):
|
|
|
|
for e in self.get_gl_entries_for(gle.party, gle.party_type, gle.voucher_type, gle.voucher_no):
|
|
|
|
if getdate(e.posting_date) <= report_date and e.name!=gle.name:
|
|
|
|
if getdate(e.posting_date) <= report_date and e.name!=gle.name:
|
|
|
|
amount = flt(e.get(reverse_dr_or_cr), currency_precision) - flt(e.get(dr_or_cr), currency_precision)
|
|
|
|
amount = flt(e.get(reverse_dr_or_cr), self.currency_precision) - flt(e.get(dr_or_cr), self.currency_precision)
|
|
|
|
if e.voucher_no not in return_entries:
|
|
|
|
if e.voucher_no not in return_entries:
|
|
|
|
payment_amount += amount
|
|
|
|
payment_amount += amount
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
credit_note_amount += amount
|
|
|
|
credit_note_amount += amount
|
|
|
|
|
|
|
|
|
|
|
|
outstanding_amount = (flt((flt(gle.get(dr_or_cr), currency_precision)
|
|
|
|
outstanding_amount = (flt((flt(gle.get(dr_or_cr), self.currency_precision)
|
|
|
|
- flt(gle.get(reverse_dr_or_cr), currency_precision)
|
|
|
|
- flt(gle.get(reverse_dr_or_cr), self.currency_precision)
|
|
|
|
- payment_amount - credit_note_amount), currency_precision))
|
|
|
|
- payment_amount - credit_note_amount), self.currency_precision))
|
|
|
|
|
|
|
|
|
|
|
|
credit_note_amount = flt(credit_note_amount, currency_precision)
|
|
|
|
credit_note_amount = flt(credit_note_amount, self.currency_precision)
|
|
|
|
|
|
|
|
|
|
|
|
return outstanding_amount, credit_note_amount
|
|
|
|
return outstanding_amount, credit_note_amount, payment_amount
|
|
|
|
|
|
|
|
|
|
|
|
def get_party_name(self, party_type, party_name):
|
|
|
|
def get_party_name(self, party_type, party_name):
|
|
|
|
return self.get_party_map(party_type).get(party_name, {}).get("customer_name" if party_type == "Customer" else "supplier_name") or ""
|
|
|
|
return self.get_party_map(party_type).get(party_name, {}).get("customer_name" if party_type == "Customer" else "supplier_name") or ""
|
|
|
@ -432,6 +536,31 @@ class ReceivablePayableReport(object):
|
|
|
|
.get(against_voucher_type, {})\
|
|
|
|
.get(against_voucher_type, {})\
|
|
|
|
.get(against_voucher, [])
|
|
|
|
.get(against_voucher, [])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_payment_term_detail(self, voucher_nos):
|
|
|
|
|
|
|
|
payment_term_map = frappe._dict()
|
|
|
|
|
|
|
|
payment_terms_details = frappe.db.sql(""" select si.name,
|
|
|
|
|
|
|
|
party_account_currency, currency, si.conversion_rate,
|
|
|
|
|
|
|
|
ps.due_date, ps.payment_amount, ps.description
|
|
|
|
|
|
|
|
from `tabSales Invoice` si, `tabPayment Schedule` ps
|
|
|
|
|
|
|
|
where si.name = ps.parent and
|
|
|
|
|
|
|
|
si.docstatus = 1 and si.company = '%s' and
|
|
|
|
|
|
|
|
si.name in (%s) order by ps.due_date
|
|
|
|
|
|
|
|
""" % (frappe.db.escape(self.filters.company), ','.join(['%s'] *len(voucher_nos))),
|
|
|
|
|
|
|
|
(tuple(voucher_nos)), as_dict = 1)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for d in payment_terms_details:
|
|
|
|
|
|
|
|
if self.filters.get("customer") and d.currency == d.party_account_currency:
|
|
|
|
|
|
|
|
payment_term_amount = d.payment_amount
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
payment_term_amount = flt(flt(d.payment_amount) * flt(d.conversion_rate), self.currency_precision)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
payment_term_map.setdefault(d.name, []).append(frappe._dict({
|
|
|
|
|
|
|
|
"due_date": d.due_date,
|
|
|
|
|
|
|
|
"payment_term_amount": payment_term_amount,
|
|
|
|
|
|
|
|
"description": d.description
|
|
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
return payment_term_map
|
|
|
|
|
|
|
|
|
|
|
|
def get_chart_data(self, columns, data):
|
|
|
|
def get_chart_data(self, columns, data):
|
|
|
|
ageing_columns = columns[self.ageing_col_idx_start : self.ageing_col_idx_start+4]
|
|
|
|
ageing_columns = columns[self.ageing_col_idx_start : self.ageing_col_idx_start+4]
|
|
|
|
|
|
|
|
|
|
|
@ -479,12 +608,11 @@ def get_ageing_data(first_range, second_range, third_range, age_as_on, entry_dat
|
|
|
|
|
|
|
|
|
|
|
|
def get_pdc_details(party_type, report_date):
|
|
|
|
def get_pdc_details(party_type, report_date):
|
|
|
|
pdc_details = frappe._dict()
|
|
|
|
pdc_details = frappe._dict()
|
|
|
|
|
|
|
|
pdc_via_pe = frappe.db.sql("""
|
|
|
|
for pdc in frappe.db.sql("""
|
|
|
|
|
|
|
|
select
|
|
|
|
select
|
|
|
|
pref.reference_name as invoice_no, pent.party, pent.party_type,
|
|
|
|
pref.reference_name as invoice_no, pent.party, pent.party_type,
|
|
|
|
max(pent.posting_date) as pdc_date, sum(ifnull(pref.allocated_amount,0)) as pdc_amount,
|
|
|
|
pent.posting_date as pdc_date, ifnull(pref.allocated_amount,0) as pdc_amount,
|
|
|
|
GROUP_CONCAT(pent.reference_no SEPARATOR ', ') as pdc_ref
|
|
|
|
pent.reference_no as pdc_ref
|
|
|
|
from
|
|
|
|
from
|
|
|
|
`tabPayment Entry` as pent inner join `tabPayment Entry Reference` as pref
|
|
|
|
`tabPayment Entry` as pent inner join `tabPayment Entry Reference` as pref
|
|
|
|
on
|
|
|
|
on
|
|
|
@ -492,19 +620,22 @@ def get_pdc_details(party_type, report_date):
|
|
|
|
where
|
|
|
|
where
|
|
|
|
pent.docstatus < 2 and pent.posting_date > %s
|
|
|
|
pent.docstatus < 2 and pent.posting_date > %s
|
|
|
|
and pent.party_type = %s
|
|
|
|
and pent.party_type = %s
|
|
|
|
group by pent.party, pref.reference_name""", (report_date, party_type), as_dict=1):
|
|
|
|
""", (report_date, party_type), as_dict=1)
|
|
|
|
pdc_details.setdefault((pdc.invoice_no, pdc.party), pdc)
|
|
|
|
|
|
|
|
|
|
|
|
for pdc in pdc_via_pe:
|
|
|
|
|
|
|
|
pdc_details.setdefault((pdc.invoice_no, pdc.party), []).append(pdc)
|
|
|
|
|
|
|
|
|
|
|
|
if scrub(party_type):
|
|
|
|
if scrub(party_type):
|
|
|
|
amount_field = ("jea.debit_in_account_currency"
|
|
|
|
amount_field = ("jea.debit_in_account_currency"
|
|
|
|
if party_type == 'Supplier' else "jea.credit_in_account_currency")
|
|
|
|
if party_type == 'Supplier' else "jea.credit_in_account_currency")
|
|
|
|
else:
|
|
|
|
else:
|
|
|
|
amount_field = "jea.debit + jea.credit"
|
|
|
|
amount_field = "jea.debit + jea.credit"
|
|
|
|
|
|
|
|
|
|
|
|
for pdc in frappe.db.sql("""
|
|
|
|
pdc_via_je = frappe.db.sql("""
|
|
|
|
select
|
|
|
|
select
|
|
|
|
jea.reference_name as invoice_no, jea.party, jea.party_type,
|
|
|
|
jea.reference_name as invoice_no, jea.party, jea.party_type,
|
|
|
|
max(je.posting_date) as pdc_date, sum(ifnull({0},0)) as pdc_amount,
|
|
|
|
je.posting_date as pdc_date, ifnull({0},0) as pdc_amount,
|
|
|
|
GROUP_CONCAT(je.cheque_no SEPARATOR ', ') as pdc_ref
|
|
|
|
je.cheque_no as pdc_ref
|
|
|
|
from
|
|
|
|
from
|
|
|
|
`tabJournal Entry` as je inner join `tabJournal Entry Account` as jea
|
|
|
|
`tabJournal Entry` as je inner join `tabJournal Entry Account` as jea
|
|
|
|
on
|
|
|
|
on
|
|
|
@ -512,16 +643,10 @@ def get_pdc_details(party_type, report_date):
|
|
|
|
where
|
|
|
|
where
|
|
|
|
je.docstatus < 2 and je.posting_date > %s
|
|
|
|
je.docstatus < 2 and je.posting_date > %s
|
|
|
|
and jea.party_type = %s
|
|
|
|
and jea.party_type = %s
|
|
|
|
group by jea.party, jea.reference_name""".format(amount_field), (report_date, party_type), as_dict=1):
|
|
|
|
""".format(amount_field), (report_date, party_type), as_dict=1)
|
|
|
|
if (pdc.invoice_no, pdc.party) in pdc_details:
|
|
|
|
|
|
|
|
key = (pdc.invoice_no, pdc.party)
|
|
|
|
for pdc in pdc_via_je:
|
|
|
|
pdc_details[key]["pdc_amount"] += pdc.pdc_amount
|
|
|
|
pdc_details.setdefault((pdc.invoice_no, pdc.party), []).append(pdc)
|
|
|
|
if pdc.pdc_ref:
|
|
|
|
|
|
|
|
pdc_details[key]["pdc_ref"] += ", " + pdc.pdc_ref
|
|
|
|
|
|
|
|
if pdc.pdc_date:
|
|
|
|
|
|
|
|
pdc_details[key]["pdc_date"] = max(pdc_details[key]["pdc_date"], pdc.pdc_date)
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
pdc_details.setdefault((pdc.invoice_no, pdc.party), pdc)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return pdc_details
|
|
|
|
return pdc_details
|
|
|
|
|
|
|
|
|
|
|
|