Aditya Hase | f3c22f3 | 2019-01-22 18:22:20 +0530 | [diff] [blame] | 1 | from __future__ import unicode_literals |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 2 | import frappe, re |
| 3 | from frappe import _ |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 4 | from frappe.utils import cstr, flt, date_diff, getdate |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 5 | from erpnext.regional.india import states, state_numbers |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 6 | from erpnext.controllers.taxes_and_totals import get_itemised_tax, get_itemised_taxable_amount |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 7 | from erpnext.controllers.accounts_controller import get_taxes_and_charges |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 8 | from erpnext.hr.utils import get_salary_assignment |
| 9 | from erpnext.hr.doctype.salary_structure.salary_structure import make_salary_slip |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 10 | |
| 11 | def validate_gstin_for_india(doc, method): |
FinByz Tech Pvt. Ltd | 237a871 | 2019-01-22 20:49:06 +0530 | [diff] [blame] | 12 | if not hasattr(doc, 'gstin') or not doc.gstin: |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 13 | return |
| 14 | |
Sagar Vora | d75095b | 2019-01-23 14:40:01 +0530 | [diff] [blame] | 15 | doc.gstin = doc.gstin.upper().strip() |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 16 | if not doc.gstin or doc.gstin == 'NA': |
| 17 | return |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 18 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 19 | if len(doc.gstin) != 15: |
| 20 | frappe.throw(_("Invalid GSTIN! A GSTIN must have 15 characters.")) |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 21 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 22 | p = re.compile("^[0-9]{2}[A-Z]{4}[0-9A-Z]{1}[0-9]{4}[A-Z]{1}[1-9A-Z]{1}[1-9A-Z]{1}[0-9A-Z]{1}$") |
| 23 | if not p.match(doc.gstin): |
| 24 | frappe.throw(_("Invalid GSTIN! The input you've entered doesn't match the format of GSTIN.")) |
Rushabh Mehta | 7231f29 | 2017-07-13 15:00:56 +0530 | [diff] [blame] | 25 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 26 | validate_gstin_check_digit(doc.gstin) |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 27 | |
| 28 | if not doc.gst_state: |
Sagar Vora | f99e013 | 2019-01-10 11:57:24 +0530 | [diff] [blame] | 29 | if not doc.state: |
| 30 | return |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 31 | state = doc.state.lower() |
| 32 | states_lowercase = {s.lower():s for s in states} |
| 33 | if state in states_lowercase: |
| 34 | doc.gst_state = states_lowercase[state] |
| 35 | else: |
| 36 | return |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 37 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 38 | doc.gst_state_number = state_numbers[doc.gst_state] |
| 39 | if doc.gst_state_number != doc.gstin[:2]: |
| 40 | frappe.throw(_("Invalid GSTIN! First 2 digits of GSTIN should match with State number {0}.") |
| 41 | .format(doc.gst_state_number)) |
| 42 | |
| 43 | def validate_gstin_check_digit(gstin): |
| 44 | ''' Function to validate the check digit of the GSTIN.''' |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 45 | factor = 1 |
| 46 | total = 0 |
| 47 | code_point_chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 48 | mod = len(code_point_chars) |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 49 | input_chars = gstin[:-1] |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 50 | for char in input_chars: |
| 51 | digit = factor * code_point_chars.find(char) |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 52 | digit = (digit // mod) + (digit % mod) |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 53 | total += digit |
| 54 | factor = 2 if factor == 1 else 1 |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 55 | if gstin[-1] != code_point_chars[((mod - (total % mod)) % mod)]: |
| 56 | frappe.throw(_("Invalid GSTIN! The check digit validation has failed. " + |
| 57 | "Please ensure you've typed the GSTIN correctly.")) |
Rushabh Mehta | 7231f29 | 2017-07-13 15:00:56 +0530 | [diff] [blame] | 58 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 59 | def get_itemised_tax_breakup_header(item_doctype, tax_accounts): |
| 60 | if frappe.get_meta(item_doctype).has_field('gst_hsn_code'): |
| 61 | return [_("HSN/SAC"), _("Taxable Amount")] + tax_accounts |
| 62 | else: |
| 63 | return [_("Item"), _("Taxable Amount")] + tax_accounts |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 64 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 65 | def get_itemised_tax_breakup_data(doc): |
| 66 | itemised_tax = get_itemised_tax(doc.taxes) |
| 67 | |
| 68 | itemised_taxable_amount = get_itemised_taxable_amount(doc.items) |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 69 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 70 | if not frappe.get_meta(doc.doctype + " Item").has_field('gst_hsn_code'): |
| 71 | return itemised_tax, itemised_taxable_amount |
| 72 | |
| 73 | item_hsn_map = frappe._dict() |
| 74 | for d in doc.items: |
| 75 | item_hsn_map.setdefault(d.item_code or d.item_name, d.get("gst_hsn_code")) |
| 76 | |
| 77 | hsn_tax = {} |
| 78 | for item, taxes in itemised_tax.items(): |
| 79 | hsn_code = item_hsn_map.get(item) |
| 80 | hsn_tax.setdefault(hsn_code, frappe._dict()) |
| 81 | for tax_account, tax_detail in taxes.items(): |
| 82 | hsn_tax[hsn_code].setdefault(tax_account, {"tax_rate": 0, "tax_amount": 0}) |
| 83 | hsn_tax[hsn_code][tax_account]["tax_rate"] = tax_detail.get("tax_rate") |
| 84 | hsn_tax[hsn_code][tax_account]["tax_amount"] += tax_detail.get("tax_amount") |
| 85 | |
| 86 | # set taxable amount |
| 87 | hsn_taxable_amount = frappe._dict() |
| 88 | for item, taxable_amount in itemised_taxable_amount.items(): |
| 89 | hsn_code = item_hsn_map.get(item) |
| 90 | hsn_taxable_amount.setdefault(hsn_code, 0) |
| 91 | hsn_taxable_amount[hsn_code] += itemised_taxable_amount.get(item) |
| 92 | |
| 93 | return hsn_tax, hsn_taxable_amount |
| 94 | |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 95 | def set_place_of_supply(doc, method=None): |
| 96 | doc.place_of_supply = get_place_of_supply(doc, doc.doctype) |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 97 | |
Rushabh Mehta | 7231f29 | 2017-07-13 15:00:56 +0530 | [diff] [blame] | 98 | # don't remove this function it is used in tests |
| 99 | def test_method(): |
| 100 | '''test function''' |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 101 | return 'overridden' |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 102 | |
| 103 | def get_place_of_supply(out, doctype): |
| 104 | if not frappe.get_meta('Address').has_field('gst_state'): return |
| 105 | |
| 106 | if doctype in ("Sales Invoice", "Delivery Note"): |
| 107 | address_name = out.shipping_address_name or out.customer_address |
| 108 | elif doctype == "Purchase Invoice": |
| 109 | address_name = out.shipping_address or out.supplier_address |
| 110 | |
| 111 | if address_name: |
| 112 | address = frappe.db.get_value("Address", address_name, ["gst_state", "gst_state_number"], as_dict=1) |
Rohit Waghchaure | b6a735e | 2018-10-11 10:40:34 +0530 | [diff] [blame] | 113 | if address and address.gst_state and address.gst_state_number: |
Nabin Hait | 2390da6 | 2018-08-30 16:16:35 +0530 | [diff] [blame] | 114 | return cstr(address.gst_state_number) + "-" + cstr(address.gst_state) |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 115 | |
| 116 | def get_regional_address_details(out, doctype, company): |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 117 | out.place_of_supply = get_place_of_supply(out, doctype) |
| 118 | |
| 119 | if not out.place_of_supply: return |
| 120 | |
| 121 | if doctype in ("Sales Invoice", "Delivery Note"): |
| 122 | master_doctype = "Sales Taxes and Charges Template" |
Shreya | 2ad8172 | 2018-06-05 16:49:29 +0530 | [diff] [blame] | 123 | if not out.company_gstin: |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 124 | return |
Shreya | 2ad8172 | 2018-06-05 16:49:29 +0530 | [diff] [blame] | 125 | elif doctype == "Purchase Invoice": |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 126 | master_doctype = "Purchase Taxes and Charges Template" |
Shreya | 2ad8172 | 2018-06-05 16:49:29 +0530 | [diff] [blame] | 127 | if not out.supplier_gstin: |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 128 | return |
| 129 | |
Rohit Waghchaure | de82ad1 | 2018-06-07 13:20:57 +0530 | [diff] [blame] | 130 | if ((doctype in ("Sales Invoice", "Delivery Note") and out.company_gstin |
| 131 | and out.company_gstin[:2] != out.place_of_supply[:2]) or (doctype == "Purchase Invoice" |
| 132 | and out.supplier_gstin and out.supplier_gstin[:2] != out.place_of_supply[:2])): |
Shreya Shah | 4fa600a | 2018-06-05 11:27:53 +0530 | [diff] [blame] | 133 | default_tax = frappe.db.get_value(master_doctype, {"company": company, "is_inter_state":1, "disabled":0}) |
| 134 | else: |
| 135 | default_tax = frappe.db.get_value(master_doctype, {"company": company, "disabled":0, "is_default": 1}) |
| 136 | |
| 137 | if not default_tax: |
| 138 | return |
| 139 | out["taxes_and_charges"] = default_tax |
| 140 | out.taxes = get_taxes_and_charges(master_doctype, default_tax) |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 141 | |
| 142 | def calculate_annual_eligible_hra_exemption(doc): |
Rushabh Mehta | 708e47a | 2018-08-08 16:37:31 +0530 | [diff] [blame] | 143 | basic_component = frappe.get_cached_value('Company', doc.company, "basic_component") |
| 144 | hra_component = frappe.get_cached_value('Company', doc.company, "hra_component") |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 145 | annual_exemption, monthly_exemption, hra_amount = 0, 0, 0 |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 146 | if hra_component and basic_component: |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 147 | assignment = get_salary_assignment(doc.employee, getdate()) |
| 148 | if assignment and frappe.db.exists("Salary Detail", { |
| 149 | "parent": assignment.salary_structure, |
| 150 | "salary_component": hra_component, "parentfield": "earnings"}): |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 151 | basic_amount, hra_amount = get_component_amt_from_salary_slip(doc.employee, |
| 152 | assignment.salary_structure, basic_component, hra_component) |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 153 | if hra_amount: |
| 154 | if doc.monthly_house_rent: |
| 155 | annual_exemption = calculate_hra_exemption(assignment.salary_structure, |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 156 | basic_amount, hra_amount, doc.monthly_house_rent, |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 157 | doc.rented_in_metro_city) |
| 158 | if annual_exemption > 0: |
| 159 | monthly_exemption = annual_exemption / 12 |
| 160 | else: |
| 161 | annual_exemption = 0 |
| 162 | return {"hra_amount": hra_amount, "annual_exemption": annual_exemption, "monthly_exemption": monthly_exemption} |
| 163 | |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 164 | def get_component_amt_from_salary_slip(employee, salary_structure, basic_component, hra_component): |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 165 | salary_slip = make_salary_slip(salary_structure, employee=employee) |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 166 | basic_amt, hra_amt = 0, 0 |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 167 | for earning in salary_slip.earnings: |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 168 | if earning.salary_component == basic_component: |
| 169 | basic_amt = earning.amount |
| 170 | elif earning.salary_component == hra_component: |
| 171 | hra_amt = earning.amount |
| 172 | if basic_amt and hra_amt: |
| 173 | return basic_amt, hra_amt |
Ranjith Kurungadam | 14e94f8 | 2018-07-16 16:12:46 +0530 | [diff] [blame] | 174 | return basic_amt, hra_amt |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 175 | |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 176 | def calculate_hra_exemption(salary_structure, basic, monthly_hra, monthly_house_rent, rented_in_metro_city): |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 177 | # TODO make this configurable |
| 178 | exemptions = [] |
| 179 | frequency = frappe.get_value("Salary Structure", salary_structure, "payroll_frequency") |
| 180 | # case 1: The actual amount allotted by the employer as the HRA. |
| 181 | exemptions.append(get_annual_component_pay(frequency, monthly_hra)) |
| 182 | actual_annual_rent = monthly_house_rent * 12 |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 183 | annual_basic = get_annual_component_pay(frequency, basic) |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 184 | # case 2: Actual rent paid less 10% of the basic salary. |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 185 | exemptions.append(flt(actual_annual_rent) - flt(annual_basic * 0.1)) |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 186 | # case 3: 50% of the basic salary, if the employee is staying in a metro city (40% for a non-metro city). |
Ranjith Kurungadam | b1a756c | 2018-07-01 16:42:38 +0530 | [diff] [blame] | 187 | exemptions.append(annual_basic * 0.5 if rented_in_metro_city else annual_basic * 0.4) |
Ranjith Kurungadam | a8e047a | 2018-06-14 17:56:16 +0530 | [diff] [blame] | 188 | # return minimum of 3 cases |
| 189 | return min(exemptions) |
| 190 | |
| 191 | def get_annual_component_pay(frequency, amount): |
| 192 | if frequency == "Daily": |
| 193 | return amount * 365 |
| 194 | elif frequency == "Weekly": |
| 195 | return amount * 52 |
| 196 | elif frequency == "Fortnightly": |
| 197 | return amount * 26 |
| 198 | elif frequency == "Monthly": |
| 199 | return amount * 12 |
| 200 | elif frequency == "Bimonthly": |
| 201 | return amount * 6 |
| 202 | |
| 203 | def validate_house_rent_dates(doc): |
| 204 | if not doc.rented_to_date or not doc.rented_from_date: |
| 205 | frappe.throw(_("House rented dates required for exemption calculation")) |
| 206 | if date_diff(doc.rented_to_date, doc.rented_from_date) < 14: |
| 207 | frappe.throw(_("House rented dates should be atleast 15 days apart")) |
| 208 | proofs = frappe.db.sql("""select name from `tabEmployee Tax Exemption Proof Submission` |
| 209 | where docstatus=1 and employee='{0}' and payroll_period='{1}' and |
| 210 | (rented_from_date between '{2}' and '{3}' or rented_to_date between |
| 211 | '{2}' and '{3}')""".format(doc.employee, doc.payroll_period, |
| 212 | doc.rented_from_date, doc.rented_to_date)) |
| 213 | if proofs: |
| 214 | frappe.throw(_("House rent paid days overlap with {0}").format(proofs[0][0])) |
| 215 | |
| 216 | def calculate_hra_exemption_for_period(doc): |
| 217 | monthly_rent, eligible_hra = 0, 0 |
| 218 | if doc.house_rent_payment_amount: |
| 219 | validate_house_rent_dates(doc) |
| 220 | # TODO receive rented months or validate dates are start and end of months? |
| 221 | # Calc monthly rent, round to nearest .5 |
| 222 | factor = flt(date_diff(doc.rented_to_date, doc.rented_from_date) + 1)/30 |
| 223 | factor = round(factor * 2)/2 |
| 224 | monthly_rent = doc.house_rent_payment_amount / factor |
| 225 | # update field used by calculate_annual_eligible_hra_exemption |
| 226 | doc.monthly_house_rent = monthly_rent |
| 227 | exemptions = calculate_annual_eligible_hra_exemption(doc) |
| 228 | |
| 229 | if exemptions["monthly_exemption"]: |
| 230 | # calc total exemption amount |
| 231 | eligible_hra = exemptions["monthly_exemption"] * factor |
Ranjith Kurungadam | 4f9744a | 2018-06-20 11:10:56 +0530 | [diff] [blame] | 232 | exemptions["monthly_house_rent"] = monthly_rent |
| 233 | exemptions["total_eligible_hra_exemption"] = eligible_hra |
| 234 | return exemptions |