2018-12-17 15:13:33 +05:30
|
|
|
import unittest
|
2021-09-02 16:44:59 +05:30
|
|
|
|
2018-12-17 15:13:33 +05:30
|
|
|
import frappe
|
2022-12-18 12:39:54 +05:30
|
|
|
from frappe.tests.utils import FrappeTestCase, change_settings
|
|
|
|
from frappe.utils import add_days, flt, getdate, today
|
2021-09-02 16:44:59 +05:30
|
|
|
|
2022-12-18 12:39:54 +05:30
|
|
|
from erpnext import get_default_cost_center
|
2018-12-17 15:13:33 +05:30
|
|
|
from erpnext.accounts.doctype.payment_entry.payment_entry import get_payment_entry
|
|
|
|
from erpnext.accounts.doctype.sales_invoice.test_sales_invoice import create_sales_invoice
|
|
|
|
from erpnext.accounts.report.accounts_receivable.accounts_receivable import execute
|
2023-08-24 20:46:33 +05:30
|
|
|
from erpnext.accounts.test.accounts_mixin import AccountsTestMixin
|
2022-08-24 13:58:11 +05:30
|
|
|
from erpnext.selling.doctype.sales_order.test_sales_order import make_sales_order
|
2021-09-02 16:44:59 +05:30
|
|
|
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
class TestAccountsReceivable(AccountsTestMixin, FrappeTestCase):
|
2022-08-24 13:58:11 +05:30
|
|
|
def setUp(self):
|
2023-08-24 20:46:33 +05:30
|
|
|
self.create_company()
|
|
|
|
self.create_customer()
|
|
|
|
self.create_item()
|
|
|
|
self.create_usd_receivable_account()
|
|
|
|
self.clear_old_entries()
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2022-08-24 13:58:11 +05:30
|
|
|
def tearDown(self):
|
|
|
|
frappe.db.rollback()
|
|
|
|
|
2022-12-18 12:39:54 +05:30
|
|
|
def create_usd_account(self):
|
|
|
|
name = "Debtors USD"
|
|
|
|
exists = frappe.db.get_list(
|
|
|
|
"Account", filters={"company": "_Test Company 2", "account_name": "Debtors USD"}
|
|
|
|
)
|
|
|
|
if exists:
|
|
|
|
self.debtors_usd = exists[0].name
|
|
|
|
else:
|
|
|
|
debtors = frappe.get_doc(
|
|
|
|
"Account",
|
|
|
|
frappe.db.get_list(
|
|
|
|
"Account", filters={"company": "_Test Company 2", "account_name": "Debtors"}
|
|
|
|
)[0].name,
|
|
|
|
)
|
|
|
|
|
|
|
|
debtors_usd = frappe.new_doc("Account")
|
|
|
|
debtors_usd.company = debtors.company
|
|
|
|
debtors_usd.account_name = "Debtors USD"
|
|
|
|
debtors_usd.account_currency = "USD"
|
|
|
|
debtors_usd.parent_account = debtors.parent_account
|
|
|
|
debtors_usd.account_type = debtors.account_type
|
|
|
|
self.debtors_usd = debtors_usd.save().name
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
def create_sales_invoice(self, no_payment_schedule=False, do_not_submit=False):
|
|
|
|
frappe.set_user("Administrator")
|
|
|
|
si = create_sales_invoice(
|
|
|
|
item=self.item,
|
|
|
|
company=self.company,
|
|
|
|
customer=self.customer,
|
|
|
|
debit_to=self.debit_to,
|
|
|
|
posting_date=today(),
|
|
|
|
parent_cost_center=self.cost_center,
|
|
|
|
cost_center=self.cost_center,
|
|
|
|
rate=100,
|
|
|
|
price_list_rate=100,
|
|
|
|
do_not_save=1,
|
|
|
|
)
|
|
|
|
if not no_payment_schedule:
|
|
|
|
si.append(
|
|
|
|
"payment_schedule",
|
|
|
|
dict(due_date=getdate(add_days(today(), 30)), invoice_portion=30.00, payment_amount=30),
|
|
|
|
)
|
|
|
|
si.append(
|
|
|
|
"payment_schedule",
|
|
|
|
dict(due_date=getdate(add_days(today(), 60)), invoice_portion=50.00, payment_amount=50),
|
|
|
|
)
|
|
|
|
si.append(
|
|
|
|
"payment_schedule",
|
|
|
|
dict(due_date=getdate(add_days(today(), 90)), invoice_portion=20.00, payment_amount=20),
|
|
|
|
)
|
|
|
|
si = si.save()
|
|
|
|
if not do_not_submit:
|
|
|
|
si = si.submit()
|
|
|
|
return si
|
|
|
|
|
|
|
|
def create_payment_entry(self, docname):
|
|
|
|
pe = get_payment_entry("Sales Invoice", docname, bank_account=self.cash, party_amount=40)
|
|
|
|
pe.paid_from = self.debit_to
|
|
|
|
pe.insert()
|
|
|
|
pe.submit()
|
|
|
|
|
|
|
|
def create_credit_note(self, docname):
|
|
|
|
credit_note = create_sales_invoice(
|
|
|
|
company=self.company,
|
|
|
|
customer=self.customer,
|
|
|
|
item=self.item,
|
|
|
|
qty=-1,
|
|
|
|
debit_to=self.debit_to,
|
|
|
|
cost_center=self.cost_center,
|
|
|
|
is_return=1,
|
|
|
|
return_against=docname,
|
|
|
|
)
|
|
|
|
|
|
|
|
return credit_note
|
|
|
|
|
2022-08-24 13:58:11 +05:30
|
|
|
def test_accounts_receivable(self):
|
2018-12-17 15:13:33 +05:30
|
|
|
filters = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"company": self.company,
|
2019-09-03 16:07:46 +05:30
|
|
|
"based_on_payment_terms": 1,
|
|
|
|
"report_date": today(),
|
|
|
|
"range1": 30,
|
|
|
|
"range2": 60,
|
|
|
|
"range3": 90,
|
|
|
|
"range4": 120,
|
2018-12-17 15:13:33 +05:30
|
|
|
}
|
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
# check invoice grand total and invoiced column's value for 3 payment terms
|
2023-08-24 20:46:33 +05:30
|
|
|
si = self.create_sales_invoice()
|
|
|
|
name = si.name
|
|
|
|
|
2018-12-17 15:13:33 +05:30
|
|
|
report = execute(filters)
|
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
expected_data = [[100, 30], [100, 50], [100, 20]]
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
for i in range(3):
|
|
|
|
row = report[1][i - 1]
|
|
|
|
self.assertEqual(expected_data[i - 1], [row.invoice_grand_total, row.invoiced])
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
# check invoice grand total, invoiced, paid and outstanding column's value after payment
|
2023-08-24 20:46:33 +05:30
|
|
|
self.create_payment_entry(si.name)
|
2018-12-17 15:13:33 +05:30
|
|
|
report = execute(filters)
|
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
expected_data_after_payment = [[100, 50, 10, 40], [100, 20, 0, 20]]
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
for i in range(2):
|
|
|
|
row = report[1][i - 1]
|
|
|
|
self.assertEqual(
|
|
|
|
expected_data_after_payment[i - 1],
|
|
|
|
[row.invoice_grand_total, row.invoiced, row.paid, row.outstanding],
|
|
|
|
)
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
# check invoice grand total, invoiced, paid and outstanding column's value after credit note
|
2023-08-24 20:46:33 +05:30
|
|
|
self.create_credit_note(si.name)
|
2018-12-17 15:13:33 +05:30
|
|
|
report = execute(filters)
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
expected_data_after_credit_note = [100, 0, 0, 40, -40, self.debit_to]
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2019-09-03 16:07:46 +05:30
|
|
|
row = report[1][0]
|
|
|
|
self.assertEqual(
|
|
|
|
expected_data_after_credit_note,
|
2022-03-30 15:00:16 +05:30
|
|
|
[
|
|
|
|
row.invoice_grand_total,
|
|
|
|
row.invoiced,
|
|
|
|
row.paid,
|
|
|
|
row.credit_note,
|
|
|
|
row.outstanding,
|
|
|
|
row.party_account,
|
|
|
|
],
|
2019-09-03 16:07:46 +05:30
|
|
|
)
|
2022-03-28 18:52:46 +05:30
|
|
|
|
2022-08-24 13:58:11 +05:30
|
|
|
def test_payment_againt_po_in_receivable_report(self):
|
|
|
|
"""
|
|
|
|
Payments made against Purchase Order will show up as outstanding amount
|
|
|
|
"""
|
|
|
|
|
|
|
|
so = make_sales_order(
|
2023-08-24 20:46:33 +05:30
|
|
|
company=self.company,
|
|
|
|
customer=self.customer,
|
|
|
|
warehouse=self.warehouse,
|
|
|
|
debit_to=self.debit_to,
|
|
|
|
income_account=self.income_account,
|
|
|
|
expense_account=self.expense_account,
|
|
|
|
cost_center=self.cost_center,
|
2022-08-24 13:58:11 +05:30
|
|
|
)
|
|
|
|
|
|
|
|
pe = get_payment_entry(so.doctype, so.name)
|
|
|
|
pe = pe.save().submit()
|
|
|
|
|
|
|
|
filters = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"company": self.company,
|
2022-08-24 13:58:11 +05:30
|
|
|
"based_on_payment_terms": 0,
|
|
|
|
"report_date": today(),
|
|
|
|
"range1": 30,
|
|
|
|
"range2": 60,
|
|
|
|
"range3": 90,
|
|
|
|
"range4": 120,
|
|
|
|
}
|
|
|
|
|
|
|
|
report = execute(filters)
|
|
|
|
|
|
|
|
expected_data_after_payment = [0, 1000, 0, -1000]
|
|
|
|
|
|
|
|
row = report[1][0]
|
|
|
|
self.assertEqual(
|
|
|
|
expected_data_after_payment,
|
|
|
|
[
|
|
|
|
row.invoiced,
|
|
|
|
row.paid,
|
|
|
|
row.credit_note,
|
|
|
|
row.outstanding,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
2022-12-18 12:39:54 +05:30
|
|
|
@change_settings(
|
2023-08-24 20:46:33 +05:30
|
|
|
"Accounts Settings",
|
|
|
|
{"allow_multi_currency_invoices_against_single_party_account": 1, "allow_stale": 0},
|
2022-12-18 12:39:54 +05:30
|
|
|
)
|
|
|
|
def test_exchange_revaluation_for_party(self):
|
|
|
|
"""
|
2023-08-24 20:46:33 +05:30
|
|
|
Exchange Revaluation for party on Receivable/Payable should be included
|
2022-12-18 12:39:54 +05:30
|
|
|
"""
|
|
|
|
|
|
|
|
# Using Exchange Gain/Loss account for unrealized as well.
|
2023-08-24 20:46:33 +05:30
|
|
|
company_doc = frappe.get_doc("Company", self.company)
|
2022-12-18 12:39:54 +05:30
|
|
|
company_doc.unrealized_exchange_gain_loss_account = company_doc.exchange_gain_loss_account
|
|
|
|
company_doc.save()
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
si = self.create_sales_invoice(no_payment_schedule=True, do_not_submit=True)
|
2022-12-18 12:39:54 +05:30
|
|
|
si.currency = "USD"
|
2023-08-24 20:46:33 +05:30
|
|
|
si.conversion_rate = 80
|
2022-12-18 12:39:54 +05:30
|
|
|
si.debit_to = self.debtors_usd
|
|
|
|
si = si.save().submit()
|
|
|
|
|
|
|
|
# Exchange Revaluation
|
|
|
|
err = frappe.new_doc("Exchange Rate Revaluation")
|
2023-08-24 20:46:33 +05:30
|
|
|
err.company = self.company
|
2022-12-18 12:39:54 +05:30
|
|
|
err.posting_date = today()
|
|
|
|
accounts = err.get_accounts_data()
|
|
|
|
err.extend("accounts", accounts)
|
2023-08-24 20:46:33 +05:30
|
|
|
err.accounts[0].new_exchange_rate = 85
|
2022-12-18 12:39:54 +05:30
|
|
|
row = err.accounts[0]
|
|
|
|
row.new_balance_in_base_currency = flt(
|
|
|
|
row.new_exchange_rate * flt(row.balance_in_account_currency)
|
|
|
|
)
|
|
|
|
row.gain_loss = row.new_balance_in_base_currency - flt(row.balance_in_base_currency)
|
|
|
|
err.set_total_gain_loss()
|
|
|
|
err = err.save().submit()
|
2018-12-17 15:13:33 +05:30
|
|
|
|
2022-12-18 12:39:54 +05:30
|
|
|
# Submit JV for ERR
|
2023-01-02 14:33:14 +05:30
|
|
|
err_journals = err.make_jv_entries()
|
|
|
|
je = frappe.get_doc("Journal Entry", err_journals.get("revaluation_jv"))
|
|
|
|
je = je.submit()
|
2022-12-18 12:39:54 +05:30
|
|
|
|
|
|
|
filters = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"company": self.company,
|
2022-12-18 12:39:54 +05:30
|
|
|
"report_date": today(),
|
|
|
|
"range1": 30,
|
|
|
|
"range2": 60,
|
|
|
|
"range3": 90,
|
|
|
|
"range4": 120,
|
|
|
|
}
|
|
|
|
report = execute(filters)
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
expected_data_for_err = [0, -500, 0, 500]
|
2023-01-02 14:33:14 +05:30
|
|
|
row = [x for x in report[1] if x.voucher_type == je.doctype and x.voucher_no == je.name][0]
|
2022-12-18 12:39:54 +05:30
|
|
|
self.assertEqual(
|
|
|
|
expected_data_for_err,
|
|
|
|
[
|
|
|
|
row.invoiced,
|
|
|
|
row.paid,
|
|
|
|
row.credit_note,
|
|
|
|
row.outstanding,
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
2023-06-12 17:35:13 +05:30
|
|
|
def test_payment_against_credit_note(self):
|
|
|
|
"""
|
|
|
|
Payment against credit/debit note should be considered against the parent invoice
|
|
|
|
"""
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
si1 = self.create_sales_invoice()
|
2023-06-12 17:35:13 +05:30
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
pe = get_payment_entry(si1.doctype, si1.name, bank_account=self.cash)
|
|
|
|
pe.paid_from = self.debit_to
|
2023-06-12 17:35:13 +05:30
|
|
|
pe.insert()
|
|
|
|
pe.submit()
|
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
cr_note = self.create_credit_note(si1.name)
|
2023-06-12 17:35:13 +05:30
|
|
|
|
2023-08-24 20:46:33 +05:30
|
|
|
si2 = self.create_sales_invoice()
|
2023-06-12 17:35:13 +05:30
|
|
|
|
|
|
|
# manually link cr_note with si2 using journal entry
|
|
|
|
je = frappe.new_doc("Journal Entry")
|
2023-08-24 20:46:33 +05:30
|
|
|
je.company = self.company
|
2023-06-12 17:35:13 +05:30
|
|
|
je.voucher_type = "Credit Note"
|
|
|
|
je.posting_date = today()
|
|
|
|
|
|
|
|
debit_entry = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"account": self.debit_to,
|
2023-06-12 17:35:13 +05:30
|
|
|
"party_type": "Customer",
|
2023-08-24 20:46:33 +05:30
|
|
|
"party": self.customer,
|
2023-06-12 17:35:13 +05:30
|
|
|
"debit": 100,
|
|
|
|
"debit_in_account_currency": 100,
|
|
|
|
"reference_type": cr_note.doctype,
|
|
|
|
"reference_name": cr_note.name,
|
2023-08-24 20:46:33 +05:30
|
|
|
"cost_center": self.cost_center,
|
2023-06-12 17:35:13 +05:30
|
|
|
}
|
|
|
|
credit_entry = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"account": self.debit_to,
|
2023-06-12 17:35:13 +05:30
|
|
|
"party_type": "Customer",
|
2023-08-24 20:46:33 +05:30
|
|
|
"party": self.customer,
|
2023-06-12 17:35:13 +05:30
|
|
|
"credit": 100,
|
|
|
|
"credit_in_account_currency": 100,
|
|
|
|
"reference_type": si2.doctype,
|
|
|
|
"reference_name": si2.name,
|
2023-08-24 20:46:33 +05:30
|
|
|
"cost_center": self.cost_center,
|
2023-06-12 17:35:13 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
je.append("accounts", debit_entry)
|
|
|
|
je.append("accounts", credit_entry)
|
|
|
|
je = je.save().submit()
|
|
|
|
|
|
|
|
filters = {
|
2023-08-24 20:46:33 +05:30
|
|
|
"company": self.company,
|
2023-06-12 17:35:13 +05:30
|
|
|
"report_date": today(),
|
|
|
|
"range1": 30,
|
|
|
|
"range2": 60,
|
|
|
|
"range3": 90,
|
|
|
|
"range4": 120,
|
|
|
|
}
|
|
|
|
report = execute(filters)
|
|
|
|
self.assertEqual(report[1], [])
|