Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 1 | import frappe, re |
| 2 | from frappe import _ |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 3 | from frappe.utils import cstr |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 4 | from erpnext.regional.india import states, state_numbers |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 5 | from erpnext.controllers.taxes_and_totals import get_itemised_tax, get_itemised_taxable_amount |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 6 | |
| 7 | def validate_gstin_for_india(doc, method): |
| 8 | if not hasattr(doc, 'gstin'): |
| 9 | return |
| 10 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 11 | doc.gstin = doc.gstin.upper().strip() |
| 12 | if not doc.gstin or doc.gstin == 'NA': |
| 13 | return |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 14 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 15 | if len(doc.gstin) != 15: |
| 16 | frappe.throw(_("Invalid GSTIN! A GSTIN must have 15 characters.")) |
Rushabh Mehta | b3c8f44 | 2017-06-21 17:22:38 +0530 | [diff] [blame] | 17 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 18 | 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}$") |
| 19 | if not p.match(doc.gstin): |
| 20 | 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] | 21 | |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 22 | validate_gstin_check_digit(doc.gstin) |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 23 | |
Sagar Vora | f99e013 | 2019-01-10 11:57:24 +0530 | [diff] [blame] | 24 | if not doc.gst_state: |
| 25 | if not doc.state: |
| 26 | return |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 27 | state = doc.state.lower() |
| 28 | states_lowercase = {s.lower():s for s in states} |
| 29 | if state in states_lowercase: |
| 30 | doc.gst_state = states_lowercase[state] |
| 31 | else: |
| 32 | return |
| 33 | |
| 34 | doc.gst_state_number = state_numbers[doc.gst_state] |
| 35 | if doc.gst_state_number != doc.gstin[:2]: |
| 36 | frappe.throw(_("Invalid GSTIN! First 2 digits of GSTIN should match with State number {0}.") |
| 37 | .format(doc.gst_state_number)) |
| 38 | |
| 39 | def validate_gstin_check_digit(gstin): |
| 40 | ''' Function to validate the check digit of the GSTIN.''' |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 41 | factor = 1 |
| 42 | total = 0 |
| 43 | code_point_chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 44 | mod = len(code_point_chars) |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 45 | input_chars = gstin[:-1] |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 46 | for char in input_chars: |
| 47 | digit = factor * code_point_chars.find(char) |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 48 | digit = (digit // mod) + (digit % mod) |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 49 | total += digit |
| 50 | factor = 2 if factor == 1 else 1 |
Sagar Vora | 07cf4e8 | 2019-01-10 11:07:51 +0530 | [diff] [blame] | 51 | if gstin[-1] != code_point_chars[((mod - (total % mod)) % mod)]: |
| 52 | frappe.throw(_("Invalid GSTIN! The check digit validation has failed. " + |
| 53 | "Please ensure you've typed the GSTIN correctly.")) |
karthikeyan5 | 2825b92 | 2019-01-09 19:15:10 +0530 | [diff] [blame] | 54 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 55 | def get_itemised_tax_breakup_header(item_doctype, tax_accounts): |
| 56 | if frappe.get_meta(item_doctype).has_field('gst_hsn_code'): |
| 57 | return [_("HSN/SAC"), _("Taxable Amount")] + tax_accounts |
| 58 | else: |
| 59 | return [_("Item"), _("Taxable Amount")] + tax_accounts |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 60 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 61 | def get_itemised_tax_breakup_data(doc): |
| 62 | itemised_tax = get_itemised_tax(doc.taxes) |
| 63 | |
| 64 | itemised_taxable_amount = get_itemised_taxable_amount(doc.items) |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 65 | |
Nabin Hait | b962fc1 | 2017-07-17 18:02:31 +0530 | [diff] [blame] | 66 | if not frappe.get_meta(doc.doctype + " Item").has_field('gst_hsn_code'): |
| 67 | return itemised_tax, itemised_taxable_amount |
| 68 | |
| 69 | item_hsn_map = frappe._dict() |
| 70 | for d in doc.items: |
| 71 | item_hsn_map.setdefault(d.item_code or d.item_name, d.get("gst_hsn_code")) |
| 72 | |
| 73 | hsn_tax = {} |
| 74 | for item, taxes in itemised_tax.items(): |
| 75 | hsn_code = item_hsn_map.get(item) |
| 76 | hsn_tax.setdefault(hsn_code, frappe._dict()) |
| 77 | for tax_account, tax_detail in taxes.items(): |
| 78 | hsn_tax[hsn_code].setdefault(tax_account, {"tax_rate": 0, "tax_amount": 0}) |
| 79 | hsn_tax[hsn_code][tax_account]["tax_rate"] = tax_detail.get("tax_rate") |
| 80 | hsn_tax[hsn_code][tax_account]["tax_amount"] += tax_detail.get("tax_amount") |
| 81 | |
| 82 | # set taxable amount |
| 83 | hsn_taxable_amount = frappe._dict() |
| 84 | for item, taxable_amount in itemised_taxable_amount.items(): |
| 85 | hsn_code = item_hsn_map.get(item) |
| 86 | hsn_taxable_amount.setdefault(hsn_code, 0) |
| 87 | hsn_taxable_amount[hsn_code] += itemised_taxable_amount.get(item) |
| 88 | |
| 89 | return hsn_tax, hsn_taxable_amount |
| 90 | |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 91 | def set_place_of_supply(doc, method): |
| 92 | if not frappe.get_meta('Address').has_field('gst_state'): return |
| 93 | |
Vishal Dhayagude | 2505c74 | 2018-04-04 11:05:21 +0530 | [diff] [blame] | 94 | if doc.doctype in ("Sales Invoice", "Delivery Note"): |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 95 | address_name = doc.shipping_address_name or doc.customer_address |
| 96 | elif doc.doctype == "Purchase Invoice": |
| 97 | address_name = doc.shipping_address or doc.supplier_address |
| 98 | |
| 99 | if address_name: |
| 100 | address = frappe.db.get_value("Address", address_name, ["gst_state", "gst_state_number"], as_dict=1) |
| 101 | doc.place_of_supply = cstr(address.gst_state_number) + "-" + cstr(address.gst_state) |
| 102 | |
Rushabh Mehta | 7231f29 | 2017-07-13 15:00:56 +0530 | [diff] [blame] | 103 | # don't remove this function it is used in tests |
| 104 | def test_method(): |
| 105 | '''test function''' |
Nabin Hait | b95ecd7 | 2018-02-16 13:19:04 +0530 | [diff] [blame] | 106 | return 'overridden' |