fix: tcs amount calculation
diff --git a/erpnext/accounts/doctype/sales_invoice/sales_invoice.py b/erpnext/accounts/doctype/sales_invoice/sales_invoice.py
index 0be63a8..a106af7 100644
--- a/erpnext/accounts/doctype/sales_invoice/sales_invoice.py
+++ b/erpnext/accounts/doctype/sales_invoice/sales_invoice.py
@@ -161,16 +161,18 @@
return
accounts = []
+ tax_withholding_account = tax_withholding_details.get("account_head")
+
for d in self.taxes:
- if d.account_head == tax_withholding_details.get("account_head"):
+ if d.account_head == tax_withholding_account:
d.update(tax_withholding_details)
accounts.append(d.account_head)
- if not accounts or tax_withholding_details.get("account_head") not in accounts:
+ if not accounts or tax_withholding_account not in accounts:
self.append("taxes", tax_withholding_details)
to_remove = [d for d in self.taxes
- if not d.tax_amount and d.account_head == tax_withholding_details.get("account_head")]
+ if not d.tax_amount and d.charge_type == "Actual" and d.account_head == tax_withholding_account]
for d in to_remove:
self.remove(d)
diff --git a/erpnext/accounts/doctype/tax_withholding_category/tax_withholding_category.py b/erpnext/accounts/doctype/tax_withholding_category/tax_withholding_category.py
index f2c4973..4cbca6c 100644
--- a/erpnext/accounts/doctype/tax_withholding_category/tax_withholding_category.py
+++ b/erpnext/accounts/doctype/tax_withholding_category/tax_withholding_category.py
@@ -12,22 +12,22 @@
class TaxWithholdingCategory(Document):
pass
-def get_party_details(ref_doc):
+def get_party_details(inv):
party_type, party = '', ''
- if ref_doc.doctype == 'Sales Invoice':
+ if inv.doctype == 'Sales Invoice':
party_type = 'Customer'
- party = ref_doc.customer
+ party = inv.customer
else:
party_type = 'Supplier'
- party = ref_doc.supplier
+ party = inv.supplier
return party_type, party
-def get_party_tax_withholding_details(ref_doc, tax_withholding_category=None):
+def get_party_tax_withholding_details(inv, tax_withholding_category=None):
pan_no = ''
parties = []
- party_type, party = get_party_details(ref_doc)
+ party_type, party = get_party_details(inv)
if not tax_withholding_category:
tax_withholding_category, pan_no = frappe.db.get_value(party_type, party, ['tax_withholding_category', 'pan'])
@@ -46,24 +46,28 @@
if not parties:
parties.append(party)
- fiscal_year = get_fiscal_year(ref_doc.posting_date, company=ref_doc.company)
- tax_details = get_tax_withholding_details(tax_withholding_category, fiscal_year[0], ref_doc.company)
+ fiscal_year = get_fiscal_year(inv.posting_date, company=inv.company)
+ tax_details = get_tax_withholding_details(tax_withholding_category, fiscal_year[0], inv.company)
if not tax_details:
frappe.throw(_('Please set associated account in Tax Withholding Category {0} against Company {1}')
- .format(tax_withholding_category, ref_doc.company))
+ .format(tax_withholding_category, inv.company))
if party_type == 'Customer' and not tax_details.cumulative_threshold:
+ # TCS is only chargeable on sum of invoiced value
frappe.throw(_('Tax Withholding Category {} against Company {} for Customer {} should have Cumulative Threshold value.')
- .format(tax_withholding_category, ref_doc.company, party))
+ .format(tax_withholding_category, inv.company, party))
- tax_amount = get_tax_amount(
+ tax_amount, tax_deducted = get_tax_amount(
party_type, parties,
- ref_doc, tax_details,
+ inv, tax_details,
fiscal_year, pan_no
)
- tax_row = get_tax_row(tax_details, tax_amount)
+ if party_type == 'Supplier':
+ tax_row = get_tax_row_for_tds(tax_details, tax_amount)
+ else:
+ tax_row = get_tax_row_for_tcs(inv, tax_details, tax_amount, tax_deducted)
return tax_row
@@ -90,14 +94,44 @@
frappe.throw(_("No Tax Withholding data found for the current Fiscal Year."))
-def get_tax_row(tax_details, tax_amount):
+def get_tax_row_for_tcs(inv, tax_details, tax_amount, tax_deducted):
+ row = {
+ "category": "Total",
+ "charge_type": "Actual",
+ "tax_amount": tax_amount,
+ "description": tax_details.description,
+ "account_head": tax_details.account_head
+ }
+
+ if tax_deducted:
+ # TCS already deducted on previous invoices
+ # So, TCS will be calculated by 'Previous Row Total'
+
+ taxes_excluding_tcs = [d for d in inv.taxes if d.account_head != tax_details.account_head]
+ if taxes_excluding_tcs:
+ # chargeable amount is the total amount after other charges are applied
+ row.update({
+ "charge_type": "On Previous Row Total",
+ "row_id": len(taxes_excluding_tcs),
+ "rate": tax_details.rate
+ })
+ else:
+ # if only TCS is to be charged, then net total is chargeable amount
+ row.update({
+ "charge_type": "On Net Total",
+ "rate": tax_details.rate
+ })
+
+ return row
+
+def get_tax_row_for_tds(tax_details, tax_amount):
return {
"category": "Total",
- "add_deduct_tax": "Deduct",
"charge_type": "Actual",
- "account_head": tax_details.account_head,
+ "tax_amount": tax_amount,
+ "add_deduct_tax": "Deduct",
"description": tax_details.description,
- "tax_amount": tax_amount
+ "account_head": tax_details.account_head
}
def get_lower_deduction_certificate(fiscal_year, pan_no):
@@ -105,57 +139,46 @@
if ldc_name:
return frappe.get_doc('Lower Deduction Certificate', ldc_name)
-def get_tax_amount(party_type, parties, ref_doc, tax_details, fiscal_year_details, pan_no=None):
+def get_tax_amount(party_type, parties, inv, tax_details, fiscal_year_details, pan_no=None):
fiscal_year = fiscal_year_details[0]
- vouchers = get_invoice_vouchers(parties, fiscal_year, ref_doc.company, party_type=party_type)
- advance_vouchers = get_advance_vouchers(parties, fiscal_year, ref_doc.company, party_type=party_type)
+ vouchers = get_invoice_vouchers(parties, fiscal_year, inv.company, party_type=party_type)
+ advance_vouchers = get_advance_vouchers(parties, fiscal_year, inv.company, party_type=party_type)
taxable_vouchers = vouchers + advance_vouchers
tax_deducted = 0
if taxable_vouchers:
- # check if tds / tcs is already charged on taxable vouchers
- filters = {
- 'is_cancelled': 0,
- 'credit': ['>', 0],
- 'fiscal_year': fiscal_year,
- 'account': tax_details.account_head,
- 'voucher_no': ['in', taxable_vouchers],
- }
- field = "sum(credit)"
-
- tax_deducted = frappe.db.get_value('GL Entry', filters, field) or 0.0
+ tax_deducted = get_deducted_tax(taxable_vouchers, fiscal_year, tax_details)
tax_amount = 0
- posting_date = ref_doc.posting_date
+ posting_date = inv.posting_date
if party_type == 'Supplier':
ldc = get_lower_deduction_certificate(fiscal_year, pan_no)
if tax_deducted:
- net_total = ref_doc.net_total
+ net_total = inv.net_total
if ldc:
tax_amount = get_tds_amount_from_ldc(ldc, parties, fiscal_year, pan_no, tax_details, posting_date, net_total)
else:
tax_amount = net_total * tax_details.rate / 100 if net_total > 0 else 0
else:
tax_amount = get_tds_amount(
- ldc, parties, ref_doc, tax_details,
+ ldc, parties, inv, tax_details,
fiscal_year_details, vouchers
)
elif party_type == 'Customer':
if tax_deducted:
- grand_total = get_invoice_total_without_tcs(ref_doc, tax_details)
- # if already tcs is charged, then (net total + gst amount) of invoice is chargeable
- tax_amount = grand_total * tax_details.rate / 100 if grand_total > 0 else 0
+ # if already TCS is charged, then amount will be calculated based on 'Previous Row Total'
+ tax_amount = 0
else:
- # if no tcs has been charged in FY,
+ # if no TCS has been charged in FY,
# then chargeable value is "prev invoices + advances" value which cross the threshold
tax_amount = get_tcs_amount(
- parties, ref_doc, tax_details,
+ parties, inv, tax_details,
fiscal_year_details, vouchers, advance_vouchers
)
- return tax_amount
+ return tax_amount, tax_deducted
def get_invoice_vouchers(parties, fiscal_year, company, party_type='Supplier'):
dr_or_cr = 'credit' if party_type == 'Supplier' else 'debit'
@@ -194,7 +217,20 @@
return frappe.get_all('GL Entry', filters=filters, distinct=1, pluck='voucher_no') or [""]
-def get_tds_amount(ldc, parties, ref_doc, tax_details, fiscal_year_details, vouchers):
+def get_deducted_tax(taxable_vouchers, fiscal_year, tax_details):
+ # check if TDS / TCS account is already charged on taxable vouchers
+ filters = {
+ 'is_cancelled': 0,
+ 'credit': ['>', 0],
+ 'fiscal_year': fiscal_year,
+ 'account': tax_details.account_head,
+ 'voucher_no': ['in', taxable_vouchers],
+ }
+ field = "sum(credit)"
+
+ return frappe.db.get_value('GL Entry', filters, field) or 0.0
+
+def get_tds_amount(ldc, parties, inv, tax_details, fiscal_year_details, vouchers):
tds_amount = 0
supp_credit_amt = frappe.db.get_value('Purchase Invoice', {
@@ -207,9 +243,9 @@
}, 'sum(credit_in_account_currency)') or 0.0
supp_credit_amt += supp_jv_credit_amt
- supp_credit_amt += ref_doc.net_total
+ supp_credit_amt += inv.net_total
- debit_note_amount = get_debit_note_amount(parties, fiscal_year_details, ref_doc.company)
+ debit_note_amount = get_debit_note_amount(parties, fiscal_year_details, inv.company)
supp_credit_amt -= debit_note_amount
threshold = tax_details.get('threshold', 0)
@@ -218,7 +254,7 @@
if ((threshold and supp_credit_amt >= threshold) or (cumulative_threshold and supp_credit_amt >= cumulative_threshold)):
if ldc and is_valid_certificate(
ldc.valid_from, ldc.valid_upto,
- ref_doc.posting_date, tax_deducted,
+ inv.posting_date, tax_deducted,
net_total, ldc.certificate_limit
):
tds_amount = get_ltds_amount(supp_credit_amt, 0, ldc.certificate_limit, ldc.rate, tax_details)
@@ -227,7 +263,7 @@
return tds_amount
-def get_tcs_amount(parties, ref_doc, tax_details, fiscal_year_details, vouchers, adv_vouchers):
+def get_tcs_amount(parties, inv, tax_details, fiscal_year_details, vouchers, adv_vouchers):
tcs_amount = 0
fiscal_year, _, _ = fiscal_year_details
@@ -235,7 +271,7 @@
invoiced_amt = frappe.db.get_value('GL Entry', {
'is_cancelled': 0,
'party': ['in', parties],
- 'company': ref_doc.company,
+ 'company': inv.company,
'voucher_no': ['in', vouchers],
}, 'sum(debit)') or 0.0
@@ -243,7 +279,7 @@
advance_amt = frappe.db.get_value('GL Entry', {
'is_cancelled': 0,
'party': ['in', parties],
- 'company': ref_doc.company,
+ 'company': inv.company,
'voucher_no': ['in', adv_vouchers],
}, 'sum(credit)') or 0.0
@@ -253,13 +289,13 @@
'credit': ['>', 0],
'party': ['in', parties],
'fiscal_year': fiscal_year,
- 'company': ref_doc.company,
+ 'company': inv.company,
'voucher_type': 'Sales Invoice',
}, 'sum(credit)') or 0.0
cumulative_threshold = tax_details.get('cumulative_threshold', 0)
- current_invoice_total = get_invoice_total_without_tcs(ref_doc, tax_details)
+ current_invoice_total = get_invoice_total_without_tcs(inv, tax_details)
total_invoiced_amt = current_invoice_total + invoiced_amt + advance_amt - credit_note_amt
if ((cumulative_threshold and total_invoiced_amt >= cumulative_threshold)):
@@ -268,11 +304,11 @@
return tcs_amount
-def get_invoice_total_without_tcs(ref_doc, tax_details):
- tcs_tax_row = [d for d in ref_doc.taxes if d.account_head == tax_details.account_head]
+def get_invoice_total_without_tcs(inv, tax_details):
+ tcs_tax_row = [d for d in inv.taxes if d.account_head == tax_details.account_head]
tcs_tax_row_amount = tcs_tax_row[0].base_tax_amount if tcs_tax_row else 0
- return ref_doc.grand_total - tcs_tax_row_amount
+ return inv.grand_total - tcs_tax_row_amount
def get_tds_amount_from_ldc(ldc, parties, fiscal_year, pan_no, tax_details, posting_date, net_total):
tds_amount = 0
diff --git a/erpnext/accounts/doctype/tax_withholding_category/test_tax_withholding_category.py b/erpnext/accounts/doctype/tax_withholding_category/test_tax_withholding_category.py
index 1d8fa45..9ce8e3f 100644
--- a/erpnext/accounts/doctype/tax_withholding_category/test_tax_withholding_category.py
+++ b/erpnext/accounts/doctype/tax_withholding_category/test_tax_withholding_category.py
@@ -18,6 +18,9 @@
create_records()
create_tax_with_holding_category()
+ def tearDown(self):
+ cancel_invoices()
+
def test_cumulative_threshold_tds(self):
frappe.db.set_value("Supplier", "Test TDS Supplier", "tax_withholding_category", "Cumulative Threshold TDS")
invoices = []
@@ -161,6 +164,23 @@
for d in invoices:
d.cancel()
+def cancel_invoices():
+ purchase_invoices = frappe.get_all("Purchase Invoice", {
+ 'supplier': ['in', ['Test TDS Supplier', 'Test TDS Supplier1', 'Test TDS Supplier2']],
+ 'docstatus': 1
+ }, pluck="name")
+
+ sales_invoices = frappe.get_all("Sales Invoice", {
+ 'customer': 'Test TCS Customer',
+ 'docstatus': 1
+ }, pluck="name")
+
+ for d in purchase_invoices:
+ frappe.get_doc('Purchase Invoice', d).cancel()
+
+ for d in sales_invoices:
+ frappe.get_doc('Sales Invoice', d).cancel()
+
def create_purchase_invoice(**args):
# return sales invoice doc object
item = frappe.db.get_value('Item', {'item_name': 'TDS Item'}, "name")