diff --git a/india_compliance/gst_india/api_classes/base.py b/india_compliance/gst_india/api_classes/base.py
index a986497083..b61a2539b8 100644
--- a/india_compliance/gst_india/api_classes/base.py
+++ b/india_compliance/gst_india/api_classes/base.py
@@ -154,6 +154,10 @@ def _make_request(
)
response_json = self.process_response(response_json)
+
+ if response_json.get("error_type") == "invalid_public_key":
+ return self._make_request(method, endpoint, params, headers, json)
+
return response_json.get("result", response_json)
except Exception as e:
diff --git a/india_compliance/gst_india/api_classes/public.py b/india_compliance/gst_india/api_classes/public.py
index f45a289a50..3653d7ef22 100644
--- a/india_compliance/gst_india/api_classes/public.py
+++ b/india_compliance/gst_india/api_classes/public.py
@@ -28,3 +28,8 @@ def get_gstin_info(self, gstin):
)
return response
+
+ def get_returns_info(self, gstin, fy):
+ return self.get(
+ "returns", params={"action": "RETTRACK", "gstin": gstin, "fy": fy}
+ )
diff --git a/india_compliance/gst_india/api_classes/returns.py b/india_compliance/gst_india/api_classes/returns.py
index 0b74c67b52..f2bdedcd45 100644
--- a/india_compliance/gst_india/api_classes/returns.py
+++ b/india_compliance/gst_india/api_classes/returns.py
@@ -22,8 +22,12 @@
class PublicCertificate(BaseAPI):
BASE_PATH = "static"
- def get_gstn_public_certificate(self) -> str:
+ def get_gstn_public_certificate(self, error_message=None) -> str:
response = self.get(endpoint="gstn_g2b_prod_public")
+
+ if response.certificate == self.settings.gstn_public_certificate:
+ frappe.throw(error_message or _("Public Certificate is already up to date"))
+
self.settings.db_set("gstn_public_certificate", response.certificate)
return response.certificate
@@ -225,13 +229,16 @@ class ReturnsAPI(ReturnsAuthenticate):
"RET2B1023": "not_generated",
"RET2B1016": "no_docs_found",
"RT-3BAS1009": "no_docs_found",
+ "RET11417": "no_docs_found", # GSTR-1 Exports
"RET2B1018": "requested_before_cutoff_date",
"RTN_24": "queued",
+ "AUTH158": "invalid_otp", # Invalid OTP
"AUTH4033": "invalid_otp", # Invalid Session
# "AUTH4034": "invalid_otp", # Invalid OTP
"AUTH4038": "authorization_failed", # Session Expired
"RET11402": "authorization_failed", # API Authorization Failed for 2A
"RET2B1010": "authorization_failed", # API Authorization Failed for 2B
+ "TEC4002": "invalid_public_key",
}
def setup(self, company_gstin):
@@ -336,6 +343,14 @@ def handle_error_response(self, response):
title=_("API Request Failed"),
)
+ # Handle invalid public key
+ if response.error_type == "invalid_public_key":
+ PublicCertificate().get_gstn_public_certificate(
+ error_message=_(
+ "Looks like Public Key of GSTN used for encryption is Invalid"
+ )
+ )
+
def is_ignored_error(self, response):
error_code = response.get("error", {}).get("error_cd")
@@ -396,3 +411,25 @@ def get_data(self, action, return_period, otp=None):
endpoint="returns/gstr2a",
otp=otp,
)
+
+
+class GSTR1API(ReturnsAPI):
+ API_NAME = "GSTR-1"
+
+ def get_gstr_1_data(self, action, return_period, otp=None):
+ return self.get(
+ action,
+ return_period,
+ params={"ret_period": return_period},
+ endpoint="returns/gstr1",
+ otp=otp,
+ )
+
+ def get_einvoice_data(self, section, return_period, otp=None):
+ return self.get(
+ "EINV",
+ return_period,
+ params={"ret_period": return_period, "sec": section},
+ endpoint="returns/einvoice",
+ otp=otp,
+ )
diff --git a/india_compliance/gst_india/doctype/gst_settings/gst_settings.js b/india_compliance/gst_india/doctype/gst_settings/gst_settings.js
index 668b6a2f67..f8b59aa925 100644
--- a/india_compliance/gst_india/doctype/gst_settings/gst_settings.js
+++ b/india_compliance/gst_india/doctype/gst_settings/gst_settings.js
@@ -51,35 +51,22 @@ function filter_accounts(frm, account_field) {
function show_ic_api_promo(frm) {
if (!frm.doc.__onload?.can_show_promo) return;
+ const alert_message = `
+ Looking for API Features?
+
+ Get started with the India Compliance API!
+ `;
- const alert = $(`
-
- `).prependTo(frm.layout.wrapper);
-
- alert.on("closed.bs.alert", () => {
- frappe.xcall(
- "india_compliance.gst_india.doctype.gst_settings.gst_settings.disable_api_promo"
- );
- });
+ india_compliance.show_dismissable_alert(
+ frm.layout.wrapper,
+ alert_message,
+ "primary",
+ () => {
+ frappe.xcall(
+ "india_compliance.gst_india.doctype.gst_settings.gst_settings.disable_api_promo"
+ );
+ }
+ );
}
function show_update_gst_category_button(frm) {
diff --git a/india_compliance/gst_india/doctype/gst_settings/gst_settings.json b/india_compliance/gst_india/doctype/gst_settings/gst_settings.json
index ebb3162082..aee01737c6 100644
--- a/india_compliance/gst_india/doctype/gst_settings/gst_settings.json
+++ b/india_compliance/gst_india/doctype/gst_settings/gst_settings.json
@@ -40,6 +40,12 @@
"column_break_17",
"e_invoice_applicable_from",
"e_invoice_applicable_companies",
+ "gstr_1_section_break",
+ "compare_gstr_1_data",
+ "filing_frequency",
+ "column_break_cxmn",
+ "restrict_changes_after_gstr_1",
+ "role_allowed_to_modify",
"other_apis_section",
"autofill_party_info",
"archive_party_info_days",
@@ -560,12 +566,50 @@
"fieldtype": "Check",
"hidden": 1,
"label": "Is Retry e-Invoice/e-Waybill Pending"
+ },
+ {
+ "fieldname": "gstr_1_section_break",
+ "fieldtype": "Section Break",
+ "label": "GSTR-1"
+ },
+ {
+ "fieldname": "column_break_cxmn",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "filing_frequency",
+ "fieldtype": "Select",
+ "label": "Filing Frequency",
+ "options": "Monthly\nQuarterly"
+ },
+ {
+ "depends_on": "eval: doc.restrict_changes_after_gstr_1",
+ "fieldname": "role_allowed_to_modify",
+ "fieldtype": "Link",
+ "label": "Role Allowed to Modify Transactions",
+ "options": "Role"
+ },
+ {
+ "default": "0",
+ "depends_on": "eval: india_compliance.is_api_enabled(doc)",
+ "description": "Prevent any modifications to transactions once they have been reported in GSTR-1",
+ "fieldname": "restrict_changes_after_gstr_1",
+ "fieldtype": "Check",
+ "label": "Restrict Changes to Transactions After Filing"
+ },
+ {
+ "default": "0",
+ "depends_on": "eval: india_compliance.is_api_enabled(doc)",
+ "description": "Use APIs to compare records with GST Portal Data Before and After Filing",
+ "fieldname": "compare_gstr_1_data",
+ "fieldtype": "Check",
+ "label": "Compare Data with GST Portal"
}
],
"index_web_pages_for_search": 1,
"issingle": 1,
"links": [],
- "modified": "2024-03-21 19:36:00.878806",
+ "modified": "2024-06-09 17:27:54.720233",
"modified_by": "Administrator",
"module": "GST India",
"name": "GST Settings",
diff --git a/india_compliance/gst_india/doctype/gst_settings/gst_settings.py b/india_compliance/gst_india/doctype/gst_settings/gst_settings.py
index eabdb13014..a25bf48648 100644
--- a/india_compliance/gst_india/doctype/gst_settings/gst_settings.py
+++ b/india_compliance/gst_india/doctype/gst_settings/gst_settings.py
@@ -5,7 +5,7 @@
from frappe import _
from frappe.model.document import Document
from frappe.query_builder.functions import IfNull
-from frappe.utils import getdate
+from frappe.utils import add_to_date, getdate
from india_compliance.gst_india.constants import GST_ACCOUNT_FIELDS, GST_PARTY_TYPES
from india_compliance.gst_india.constants.custom_fields import (
@@ -13,13 +13,13 @@
E_WAYBILL_FIELDS,
SALES_REVERSE_CHARGE_FIELDS,
)
+from india_compliance.gst_india.doctype.gstin.gstin import get_gstr_1_filed_upto
from india_compliance.gst_india.page.india_compliance_account import (
_disable_api_promo,
post_login,
)
from india_compliance.gst_india.utils import can_enable_api, is_api_enabled
from india_compliance.gst_india.utils.custom_fields import toggle_custom_fields
-from india_compliance.gst_india.utils.e_invoice import get_e_invoice_applicability_date
from india_compliance.gst_india.utils.gstin_info import get_gstin_info
E_INVOICE_START_DATE = "2021-01-01"
@@ -270,6 +270,42 @@ def validate_e_invoice_applicable_companies(self):
company_list.append(row.company)
+ def is_sek_valid(self, gstin, throw=False, threshold=30):
+ for credential in self.credentials:
+ if credential.service == "Returns" and credential.gstin == gstin:
+ break
+
+ else:
+ if throw:
+ frappe.throw(
+ _(
+ "No credential found for the GSTIN {0} in the GST Settings"
+ ).format(gstin)
+ )
+
+ return False
+
+ if credential.session_expiry and credential.session_expiry > add_to_date(
+ None, minutes=threshold * -1
+ ):
+ return True
+
+ def has_valid_credentials(self, gstin, service, throw=False):
+ for credential in self.credentials:
+ if credential.gstin == gstin and credential.service == service:
+ break
+ else:
+ message = _(
+ "No credential found for the GSTIN {0} in the GST Settings"
+ ).format(gstin)
+
+ if throw:
+ frappe.throw(message)
+
+ return False
+
+ return True
+
@frappe.whitelist()
def disable_api_promo():
@@ -345,6 +381,24 @@ def update_e_invoice_status():
update_not_applicable_status(e_invoice_applicability_date, company)
+def get_e_invoice_applicability_date(company, settings=None, throw=True):
+ if not settings:
+ settings = frappe.get_cached_doc("GST Settings")
+
+ e_invoice_applicable_from = settings.e_invoice_applicable_from
+
+ if settings.apply_e_invoice_only_for_selected_companies:
+ for row in settings.e_invoice_applicable_companies:
+ if company == row.company:
+ e_invoice_applicable_from = row.applicable_from
+ break
+
+ else:
+ return
+
+ return e_invoice_applicable_from
+
+
def update_pending_status(e_invoice_applicability_date, company=None):
if not e_invoice_applicability_date:
return
@@ -394,3 +448,52 @@ def update_not_applicable_status(e_invoice_applicability_date=None, company=None
company = query.where(sales_invoice.company == company)
query.run()
+
+
+def restrict_gstr_1_transaction_for(posting_date, company_gstin, gst_settings=None):
+ """
+ Check if the user is allowed to modify transactions before the GSTR-1 filing date
+ Additionally, update the `is_not_latest_gstr1_data` field in the GSTR-1 Log
+ """
+ posting_date = getdate(posting_date)
+
+ if not gst_settings:
+ gst_settings = frappe.get_cached_doc("GST Settings")
+
+ restrict = True
+
+ if not gst_settings.restrict_changes_after_gstr_1:
+ restrict = False
+
+ gstr_1_filed_upto = get_gstr_1_filed_upto(company_gstin)
+
+ if not gstr_1_filed_upto:
+ return False
+
+ if posting_date > getdate(gstr_1_filed_upto):
+ restrict = False
+
+ if (
+ gst_settings.role_allowed_to_modify in frappe.get_roles()
+ or frappe.session.user == "Administrator"
+ ):
+ restrict = False
+
+ if restrict:
+ return gstr_1_filed_upto
+
+ update_is_not_latest_gstr1_data(posting_date, company_gstin)
+
+ return None
+
+
+def update_is_not_latest_gstr1_data(posting_date, company_gstin):
+ period = posting_date.strftime("%m%Y")
+
+ frappe.db.set_value("GSTR-1 Log", f"{period}-{company_gstin}", "is_latest_data", 0)
+
+ frappe.publish_realtime(
+ "is_not_latest_data",
+ message={"filters": {"company_gstin": company_gstin, "period": period}},
+ doctype="GSTR-1 Beta",
+ )
diff --git a/india_compliance/gst_india/doctype/gstin/gstin.json b/india_compliance/gst_india/doctype/gstin/gstin.json
index 91731bbcb4..234b1382c2 100644
--- a/india_compliance/gst_india/doctype/gstin/gstin.json
+++ b/india_compliance/gst_india/doctype/gstin/gstin.json
@@ -12,7 +12,9 @@
"is_blocked",
"column_break_nrjd",
"last_updated_on",
- "cancelled_date"
+ "cancelled_date",
+ "section_break_ttzc",
+ "gstr_1_filed_upto"
],
"fields": [
{
@@ -59,12 +61,22 @@
"in_list_view": 1,
"in_standard_filter": 1,
"label": "Is Blocked"
+ },
+ {
+ "fieldname": "section_break_ttzc",
+ "fieldtype": "Section Break"
+ },
+ {
+ "fieldname": "gstr_1_filed_upto",
+ "fieldtype": "Date",
+ "hidden": 1,
+ "label": "GSTR-1 Filed Upto"
}
],
"in_create": 1,
"index_web_pages_for_search": 1,
"links": [],
- "modified": "2023-08-17 11:32:42.332746",
+ "modified": "2024-05-17 15:38:05.867522",
"modified_by": "Administrator",
"module": "GST India",
"name": "GSTIN",
diff --git a/india_compliance/gst_india/doctype/gstin/gstin.py b/india_compliance/gst_india/doctype/gstin/gstin.py
index 1feb1ac6af..96a5f71607 100644
--- a/india_compliance/gst_india/doctype/gstin/gstin.py
+++ b/india_compliance/gst_india/doctype/gstin/gstin.py
@@ -304,3 +304,10 @@ def get_transporter_id_info(transporter_id):
"status": "Active" if response.transin else "Invalid",
}
)
+
+
+def get_gstr_1_filed_upto(gstin):
+ if not gstin:
+ return
+
+ return frappe.db.get_value("GSTIN", gstin, "gstr_1_filed_upto")
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/__init__.py b/india_compliance/gst_india/doctype/gstr_1_beta/__init__.py
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.css b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.css
new file mode 100644
index 0000000000..70c4050cea
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.css
@@ -0,0 +1,142 @@
+div[data-page-route="GSTR-1 Beta"] {
+ --dt-row-height: 34px;
+}
+
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_html"] .section-body {
+ margin-top: 0;
+}
+
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_html"] .form-tabs-list {
+ margin-top: var(--margin-sm);
+ display: flex;
+ justify-content: space-between;
+ align-items: center;
+ padding-right: var(--padding-lg);
+ position: inherit;
+}
+
+div[data-page-route="GSTR-1 Beta"]
+ [data-fieldname="tabs_html"]
+ .form-tabs-list
+ .custom-button-group {
+ display: flex;
+}
+
+div[data-page-route="GSTR-1 Beta"]
+ [data-fieldname="tabs_html"]
+ .form-tabs-list
+ .inner-group-button,
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_html"] .filter-selector,
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_html"] .view-group {
+ margin-bottom: 8px;
+ margin-left: 8px;
+}
+
+div[data-page-route="GSTR-1 Beta"]
+ [data-fieldname="tabs_html"]
+ .form-tabs-list
+ .custom-button-group
+ .btn {
+ padding: 5px 10px;
+}
+
+div[data-page-route="GSTR-1 Beta"] .tab-title-text {
+ font-size: 1.2rem;
+}
+
+div[data-page-route="GSTR-1 Beta"] .tab-subtitle-text {
+ font-size: 0.8rem;
+}
+
+div[data-page-route="GSTR-1 Beta"] .datatable .dt-scrollable {
+ overflow-y: auto !important;
+}
+
+div[data-page-route="GSTR-1 Beta"] .datatable .dt-row {
+ height: unset;
+}
+
+div[data-page-route="GSTR-1 Beta"] .datatable .dt-row-filter {
+ height: var(--dt-row-height);
+}
+
+div[data-page-route="GSTR-1 Beta"] .datatable .dt-row-filter .dt-cell {
+ max-height: var(--dt-row-height);
+}
+
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_empty_state"],
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_no_data"] {
+ min-height: 320px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ flex-direction: column;
+}
+
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_empty_state"] > img,
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_no_data"] > img {
+ margin-bottom: var(--margin-md);
+ max-height: 100px;
+}
+
+div[data-page-route="GSTR-1 Beta"] .tab-actions {
+ display: flex;
+ flex-direction: row;
+ justify-content: space-between;
+ align-items: center;
+}
+
+div[data-page-route="GSTR-1 Beta"] .frappe-control [data-fieldtype="HTML"] {
+ margin-left: -11px;
+ margin-right: -15px;
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item > .nav-link {
+ padding: 4px 8px;
+ outline: 1px solid var(--dark-border-color);
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item > .nav-link.focus {
+ outline: 1px solid var(--dark-border-color);
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item > .nav-link.active {
+ color: var(--primary);
+ background-color: #ecf5fe;
+ outline: 1px solid var(--dark-border-color);
+ box-shadow: none;
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item:first-child:not(:has(.disabled)):hover {
+ background-color: var(--fg-hover-color);
+ color: var(--text-color);
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item:last-child:not(:has(.disabled)):hover {
+ background-color: var(--fg-hover-color);
+ color: var(--text-color);
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item > .nav-link.active:hover {
+ background-color: var(--fg-hover-color);
+ color: var(--primary);
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item:first-child > .nav-link {
+ border-radius: 0.4rem 0 0 0.4rem;
+}
+
+div[data-page-route="GSTR-1 Beta"] .custom-tabs > .nav-item:last-child > .nav-link {
+ border-radius: 0 0.4rem 0.4rem 0;
+}
+
+div[data-page-route="GSTR-1 Beta"] [data-fieldname="tabs_html"] .reconcile-row {
+ position: absolute;
+ top: 50%;
+ left: 50%;
+ transform: translate(-50%, -50%);
+}
+
+.not-matched {
+ color: red;
+}
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.js b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.js
new file mode 100644
index 0000000000..9283a81097
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.js
@@ -0,0 +1,2187 @@
+// Copyright (c) 2024, Resilient Tech and contributors
+// For license information, please see license.txt
+
+frappe.provide("india_compliance");
+
+const DOCTYPE = "GSTR-1 Beta";
+const GSTR1_Category = {
+ B2B: "B2B, SEZ, DE",
+ EXP: "Exports",
+ B2CL: "B2C (Large)",
+ B2CS: "B2C (Others)",
+ NIL_EXEMPT: "Nil-Rated, Exempted, Non-GST",
+ CDNR: "Credit/Debit Notes (Registered)",
+ CDNUR: "Credit/Debit Notes (Unregistered)",
+ // Other Categories
+ AT: "Advances Received",
+ TXP: "Advances Adjusted",
+ HSN: "HSN Summary",
+ DOC_ISSUE: "Document Issued",
+};
+
+const GSTR1_SubCategory = {
+ B2B_REGULAR: "B2B Regular",
+ B2B_REVERSE_CHARGE: "B2B Reverse Charge",
+ SEZWP: "SEZ With Payment of Tax",
+ SEZWOP: "SEZ Without Payment of Tax",
+ DE: "Deemed Exports",
+ EXPWP: "Export With Payment of Tax",
+ EXPWOP: "Export Without Payment of Tax",
+ B2CL: "B2C (Large)",
+ B2CS: "B2C (Others)",
+ NIL_EXEMPT: "Nil-Rated, Exempted, Non-GST",
+ CDNR: "Credit/Debit Notes (Registered)",
+ CDNUR: "Credit/Debit Notes (Unregistered)",
+
+ AT: "Advances Received",
+ TXP: "Advances Adjusted",
+ HSN: "HSN Summary",
+ DOC_ISSUE: "Document Issued",
+
+ SUPECOM_52: "TCS collected by E-commerce Operator u/s 52",
+ SUPECOM_9_5: "GST Payable on RCM by E-commerce Operator u/s 9(5)",
+};
+
+const INVOICE_TYPE = {
+ [GSTR1_Category.B2B]: [
+ GSTR1_SubCategory.B2B_REGULAR,
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE,
+ GSTR1_SubCategory.SEZWP,
+ GSTR1_SubCategory.SEZWOP,
+ GSTR1_SubCategory.DE,
+ ],
+ [GSTR1_Category.B2CL]: [GSTR1_SubCategory.B2CL],
+ [GSTR1_Category.EXP]: [GSTR1_SubCategory.EXPWP, GSTR1_SubCategory.EXPWOP],
+ [GSTR1_Category.NIL_EXEMPT]: [GSTR1_SubCategory.NIL_EXEMPT],
+ [GSTR1_Category.CDNR]: [GSTR1_SubCategory.CDNR],
+ [GSTR1_Category.CDNUR]: [GSTR1_SubCategory.CDNUR],
+ [GSTR1_Category.AT]: [GSTR1_SubCategory.AT],
+ [GSTR1_Category.TXP]: [GSTR1_SubCategory.TXP],
+ [GSTR1_Category.HSN]: [GSTR1_SubCategory.HSN],
+ [GSTR1_Category.DOC_ISSUE]: [GSTR1_SubCategory.DOC_ISSUE],
+};
+
+const GSTR1_DataField = {
+ TRANSACTION_TYPE: "transaction_type",
+ CUST_GSTIN: "customer_gstin",
+ ECOMMERCE_GSTIN: "ecommerce_gstin",
+ CUST_NAME: "customer_name",
+ DOC_DATE: "document_date",
+ DOC_NUMBER: "document_number",
+ DOC_TYPE: "document_type",
+ DOC_VALUE: "document_value",
+ POS: "place_of_supply",
+ DIFF_PERCENTAGE: "diff_percentage",
+ REVERSE_CHARGE: "reverse_charge",
+ TAXABLE_VALUE: "total_taxable_value",
+ TAX_RATE: "tax_rate",
+ IGST: "total_igst_amount",
+ CGST: "total_cgst_amount",
+ SGST: "total_sgst_amount",
+ CESS: "total_cess_amount",
+ UPLOAD_STATUS: "upload_status",
+
+ SHIPPING_BILL_NUMBER: "shipping_bill_number",
+ SHIPPING_BILL_DATE: "shipping_bill_date",
+ SHIPPING_PORT_CODE: "shipping_port_code",
+
+ EXEMPTED_AMOUNT: "exempted_amount",
+ NIL_RATED_AMOUNT: "nil_rated_amount",
+ NON_GST_AMOUNT: "non_gst_amount",
+
+ HSN_CODE: "hsn_code",
+ DESCRIPTION: "description",
+ UOM: "uom",
+ QUANTITY: "quantity",
+
+ FROM_SR: "from_sr_no",
+ TO_SR: "to_sr_no",
+ TOTAL_COUNT: "total_count",
+ DRAFT_COUNT: "draft_count",
+ CANCELLED_COUNT: "cancelled_count",
+};
+
+frappe.ui.form.on(DOCTYPE, {
+ async setup(frm) {
+ patch_set_indicator(frm);
+ frappe.require("gstr1.bundle.js").then(() => {
+ frm.gstr1 = new GSTR1(frm);
+ frm.trigger("company");
+ });
+
+ frm.filing_frequency = gst_settings.filing_frequency;
+
+ // Set Default Values
+ set_default_company_gstin(frm);
+ set_options_for_year(frm);
+ set_options_for_month_or_quarter(frm);
+
+ frm.__setup_complete = true;
+
+ // Setup Listeners
+ frappe.realtime.on("is_not_latest_data", message => {
+ const { filters } = message;
+
+ const [month_or_quarter, year] =
+ india_compliance.get_month_year_from_period(filters.period);
+
+ if (
+ frm.doc.company_gstin !== filters.company_gstin ||
+ frm.doc.month_or_quarter != month_or_quarter ||
+ frm.doc.year != year
+ )
+ return;
+
+ if (frm.$wrapper.find(".form-message.orange").length) return;
+ frm.set_intro(
+ __(
+ "Books data was updated after the computation of GSTR-1 data. Please generate GSTR-1 again."
+ ),
+ "orange"
+ );
+ });
+
+ frappe.realtime.on("gstr1_generation_failed", message => {
+ const { error, filters } = message;
+ let alert = `GSTR-1 Generation Failed for ${filters.company_gstin} - ${filters.month_or_quarter} - ${filters.year}.
${error}`;
+
+ frappe.msgprint({
+ title: __("GSTR-1 Generation Failed"),
+ message: alert,
+ });
+ });
+
+ frappe.realtime.on("gstr1_data_prepared", message => {
+ const { data, filters } = message;
+
+ if (
+ frm.doc.company_gstin !== filters.company_gstin ||
+ frm.doc.month_or_quarter != filters.month_or_quarter ||
+ frm.doc.year != filters.year
+ )
+ return;
+
+ frappe.after_ajax(() => {
+ frm.doc.__onload = { data };
+ frm.trigger("after_save");
+ });
+ });
+ },
+
+ async company(frm) {
+ render_empty_state(frm);
+
+ if (!frm.doc.company) return;
+ const options = await india_compliance.set_gstin_options(frm);
+
+ frm.set_value("company_gstin", options[0]);
+ },
+
+ company_gstin: render_empty_state,
+
+ month_or_quarter(frm) {
+ render_empty_state(frm);
+ },
+
+ year(frm) {
+ render_empty_state(frm);
+ set_options_for_month_or_quarter(frm);
+ },
+
+ refresh(frm) {
+ // Primary Action
+ frm.disable_save();
+ frm.page.set_primary_action(__("Generate"), () => frm.save());
+ },
+
+ before_save(frm) {
+ frm.doc.__unsaved = true;
+ },
+
+ async after_save(frm) {
+ const data = frm.doc.__onload?.data;
+ if (!frm._otp_requested && data == "otp_requested") {
+ frm._otp_requested = true;
+
+ india_compliance
+ .authenticate_otp(frm.doc.company_gstin)
+ .then(() => frm.save());
+ return;
+ }
+
+ frm._otp_requested = false;
+
+ if (!data?.status) return;
+ frm.gstr1.status = data.status;
+ frm.gstr1.refresh_data(data);
+ },
+});
+
+class GSTR1 {
+ // Render page / Setup Listeners / Setup Data
+ TABS = [
+ {
+ label: __("Books"),
+ name: "books",
+ is_active: true,
+ _TabManager: BooksTab,
+ },
+ {
+ label: __("Unfiled"),
+ name: "unfiled",
+ _TabManager: UnfiledTab,
+ },
+ {
+ label: __("Reconcile"),
+ name: "reconcile",
+ _TabManager: ReconcileTab,
+ },
+ {
+ label: __("Filed"),
+ name: "filed",
+ _TabManager: FiledTab,
+ },
+ ];
+
+ constructor(frm) {
+ this.init(frm);
+ this.render();
+ }
+
+ init(frm) {
+ this.frm = frm;
+ this.data = frm.doc._data;
+ this.filters = [];
+ this.$wrapper = frm.fields_dict.tabs_html.$wrapper;
+ }
+
+ refresh_data(data) {
+ this.render_indicator();
+
+ // clear filters if any and set default view
+ this.active_view = "Summary";
+ this.filter_group.filter_x_button.click();
+
+ if (data) this.data = data;
+
+ // set data for filing return
+ if (!this.data["filed"]) {
+ // deepcopy
+ const filed = JSON.parse(JSON.stringify(this.data["books"]));
+ Object.assign(filed, filed.aggregate_data);
+
+ this.data["filed"] = filed;
+ this.data["filed_summary"] = this.data["books_summary"];
+ }
+
+ // set idx for reconcile rows (for detail view)
+ if (this.data["reconcile"]) {
+ Object.values(this.data["reconcile"]).forEach(category => {
+ category instanceof Array &&
+ category.forEach((row, idx) => {
+ row.idx = idx;
+ });
+ });
+ }
+
+ this.set_output_gst_balances();
+
+ // refresh tabs
+ this.TABS.forEach(_tab => {
+ const tab_name = _tab.name;
+ const tab = this.tabs[`${tab_name}_tab`];
+
+ if (!this.data[tab_name]) {
+ tab.hide();
+ _tab.shown = false;
+ return;
+ }
+
+ tab.show();
+ _tab.shown = true;
+ tab.tabmanager.refresh_data(
+ this.data[tab_name],
+ this.data[`${tab_name}_summary`],
+ this.status
+ );
+ });
+ }
+
+ async refresh_view() {
+ // for change in view (Summary/Detailed)
+ this.viewgroup.set_active_view(this.active_view);
+
+ this.toggle_filter_selector();
+
+ let detailed_view_filters = [];
+ if (this.active_view === "Detailed") {
+ detailed_view_filters = this.filter_group.get_filters();
+ }
+
+ // refresh tabs
+ this.TABS.forEach(tab => {
+ if (!tab.shown) return;
+ this.tabs[`${tab.name}_tab`].tabmanager.refresh_view(
+ this.active_view,
+ this.filter_category,
+ detailed_view_filters
+ );
+ });
+ }
+
+ // RENDER
+
+ render() {
+ this.render_tab_group();
+ this.render_indicator();
+ this.setup_filter_button();
+ this.render_view_groups();
+ this.render_tabs();
+ this.setup_detail_view_listener();
+ }
+
+ render_tab_group() {
+ const tab_fields = this.TABS.reduce(
+ (acc, tab) => [
+ ...acc,
+ {
+ fieldtype: "Tab Break",
+ fieldname: `${tab.name}_tab`,
+ label: __(tab.label),
+ active: tab.is_active ? 1 : 0,
+ depends_on: tab.depends_on,
+ },
+ {
+ fieldtype: "HTML",
+ fieldname: `${tab.name}_html`,
+ },
+ ],
+ []
+ );
+
+ this.tab_group = new frappe.ui.FieldGroup({
+ fields: [
+ {
+ //hack: for the FieldGroup(Layout) to avoid rendering default "Detailed" tab
+ fieldtype: "Section Break",
+ },
+ ...tab_fields,
+ ],
+ body: this.$wrapper,
+ frm: this.frm,
+ });
+ this.tab_group.make();
+
+ // make tabs_dict for easy access
+ this.tabs = Object.fromEntries(
+ this.tab_group.tabs.map(tab => [tab.df.fieldname, tab])
+ );
+
+ // Fix css
+ this.$wrapper.find(".form-tabs-list").append(``);
+
+ // Remove padding around data table
+ this.$wrapper.closest(".form-column").css("padding", "0px");
+ this.$wrapper.closest(".row.form-section").css("padding", "0px");
+ }
+
+ render_view_groups() {
+ this.active_view = "Summary";
+ const wrapper = this.$wrapper.find(".tab-actions").find(".custom-button-group");
+
+ this.viewgroup = new india_compliance.ViewGroup({
+ $wrapper: wrapper,
+ view_names: ["Summary", "Detailed"],
+ active_view: this.active_view,
+ parent: this,
+ callback: this.change_view,
+ });
+
+ this.viewgroup.disable_view(
+ "Detailed",
+ "Click on a category from summary to view details"
+ );
+ }
+
+ render_tabs() {
+ this.TABS.forEach(tab => {
+ const wrapper = this.tab_group.get_field(`${tab.name}_html`).$wrapper;
+ this.tabs[`${tab.name}_tab`].tabmanager = new tab._TabManager(
+ this,
+ wrapper,
+ this.show_filtered_category,
+ this.filter_detailed_view
+ );
+ });
+ }
+
+ render_indicator() {
+ if (!this.status) {
+ this.frm.page.clear_indicator();
+ return;
+ }
+
+ const tab_name = this.status === "Filed" ? "Filed" : "File";
+ const color = this.status === "Filed" ? "green" : "orange";
+
+ this.$wrapper.find(`[data-fieldname="filed_tab"]`).html(tab_name);
+ this.frm.page.set_indicator(this.status, color);
+ this.frm.refresh();
+ }
+
+ // SETUP
+
+ setup_filter_button() {
+ this.filter_group = new india_compliance.FilterGroup({
+ doctype: DOCTYPE,
+ parent: this.$wrapper.find(".tab-actions"),
+ filter_options: {
+ fieldname: "description",
+ filter_fields: this.get_category_filter_fields(),
+ },
+ on_change: () => {
+ if (this.is_category_changed) {
+ this.is_category_changed = false;
+ return;
+ }
+
+ this.refresh_view();
+ },
+ });
+
+ this.toggle_filter_selector();
+ }
+
+ setup_detail_view_listener() {
+ const me = this;
+ this.$wrapper.on("click", ".btn.eye.reconcile-row", function (e) {
+ const row_index = $(this).attr("data-row-index");
+ const data = me.data.reconcile[me.filter_category][row_index];
+
+ const category_columns = me.tabs.filed_tab.tabmanager.category_columns;
+ const field_label_map = category_columns.map(col => [
+ col.fieldname,
+ col.name,
+ ]);
+
+ new DetailViewDialog(data, field_label_map);
+ });
+ }
+
+ // UTILS
+
+ show_filtered_category = category => {
+ category = category.trim();
+
+ if (category != this.filter_category) {
+ this.is_category_changed = true;
+ }
+
+ this.filter_category = category;
+
+ if (this.filter_category) this.active_view = "Detailed";
+ else this.active_view = "Summary";
+
+ this.viewgroup.enable_view("Detailed");
+
+ this.refresh_filter_options();
+ this.refresh_view();
+ };
+
+ refresh_filter_options() {
+ const filter_options = this.filter_group.filter_options;
+ this.filter_fields = this.get_category_filter_fields();
+
+ if (!this.filter_fields.length) return;
+
+ filter_options.fieldname = this.filter_fields[0].fieldname;
+ filter_options.filter_fields = this.filter_fields;
+
+ if (this.is_category_changed) {
+ this.filter_group.filter_x_button.click();
+ }
+ }
+
+ get_category_filter_fields() {
+ let fields = [];
+
+ if (
+ [
+ GSTR1_SubCategory.B2B_REGULAR,
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE,
+ GSTR1_SubCategory.SEZWOP,
+ GSTR1_SubCategory.SEZWP,
+ GSTR1_SubCategory.DE,
+ GSTR1_SubCategory.CDNR,
+ ].includes(this.filter_category)
+ ) {
+ fields = [
+ {
+ label: "Customer GSTIN",
+ fieldname: GSTR1_DataField.CUST_GSTIN,
+ fieldtype: "Data",
+ },
+ {
+ label: "Reverse Charge",
+ fieldname: GSTR1_DataField.REVERSE_CHARGE,
+ fieldtype: "Data",
+ },
+ {
+ label: "Place of Supply",
+ fieldname: GSTR1_DataField.POS,
+ fieldtype: "Data",
+ },
+ ];
+ } else if (
+ [GSTR1_SubCategory.EXPWP, GSTR1_SubCategory.EXPWOP].includes(
+ this.filter_category
+ )
+ ) {
+ fields = [
+ {
+ label: "Port Code",
+ fieldname: GSTR1_DataField.SHIPPING_PORT_CODE,
+ fieldtype: "Data",
+ },
+ ];
+ } else if (
+ [
+ GSTR1_SubCategory.B2CL,
+ GSTR1_SubCategory.B2CS,
+ GSTR1_SubCategory.AT,
+ GSTR1_SubCategory.TXP,
+ GSTR1_SubCategory.CDNUR,
+ ].includes(this.filter_category)
+ ) {
+ fields = [
+ {
+ label: "Place of Supply",
+ fieldname: GSTR1_DataField.POS,
+ fieldtype: "Data",
+ },
+ ];
+ } else if (
+ [GSTR1_SubCategory.NIL_EXEMPT, GSTR1_SubCategory.DOC_ISSUE].includes(
+ this.filter_category
+ )
+ ) {
+ fields = [
+ {
+ label: "Document Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ fieldtype: "Data",
+ },
+ ];
+ } else if (this.filter_category === GSTR1_SubCategory.HSN) {
+ fields = [
+ {
+ label: "HSN Code",
+ fieldname: GSTR1_DataField.HSN_CODE,
+ fieldtype: "Data",
+ },
+ {
+ label: "UOM",
+ fieldname: GSTR1_DataField.UOM,
+ fieldtype: "Data",
+ },
+ ];
+ }
+
+ fields.forEach(field => (field.parent = DOCTYPE));
+ return fields;
+ }
+
+ filter_detailed_view = async (fieldname, value) => {
+ await this.filter_group.push_new_filter([DOCTYPE, fieldname, "=", value]);
+ this.filter_group.apply();
+ };
+
+ show_summary_view = () => {
+ this.viewgroup.set_active_view("Summary");
+ this.change_view("Summary");
+ };
+
+ change_view = target_view => {
+ const current_view = this.active_view;
+
+ if (!this.filter_category && current_view === "Summary") return;
+
+ this.active_view = target_view;
+ this.refresh_view();
+ };
+
+ toggle_filter_selector() {
+ if (this.active_view === "Detailed" && this.filter_fields.length)
+ this.$wrapper.find(".filter-selector").show();
+ else this.$wrapper.find(".filter-selector").hide();
+ }
+
+ async set_output_gst_balances() {
+ //Checks if gst-ledger-difference element is there and removes if already present
+ const gst_liability = await get_net_gst_liability(this.frm);
+
+ if ($(".gst-ledger-difference").length) {
+ $(".gst-ledger-difference").remove();
+ }
+
+ $(function () {
+ $('[data-toggle="tooltip"]').tooltip();
+ });
+
+ const net_transactions = {
+ IGST: gst_liability["total_igst_amount"] || 0,
+ CGST: gst_liability["total_cgst_amount"] || 0,
+ SGST: gst_liability["total_sgst_amount"] || 0,
+ CESS: gst_liability["total_cess_amount"] || 0,
+ };
+
+ const ledger_balance_cards = Object.entries(net_transactions)
+ .map(
+ ([type, net_amount]) => `
+
+
${type} Account
+
+ ${format_currency(net_amount)}
+ `
+ )
+ .join("");
+
+ const gst_liability_html = `
+
+
+ Net Output GST Liability (Credit - Debit)
+
+
+ ${ledger_balance_cards}
+
+
`;
+
+ let element = $('[data-fieldname="tabs_html"]');
+ element.closest(".row.form-section").prepend(gst_liability_html);
+ }
+}
+
+class TabManager {
+ DEFAULT_NO_DATA_MESSAGE = __("No Data");
+ CATEGORY_COLUMNS = {};
+ DEFAULT_SUMMARY = {
+ // description: "",
+ no_of_records: 0,
+ total_taxable_value: 0,
+ total_igst_amount: 0,
+ total_cgst_amount: 0,
+ total_sgst_amount: 0,
+ total_cess_amount: 0,
+ };
+
+ constructor(instance, wrapper, summary_view_callback, detailed_view_callback) {
+ this.DEFAULT_TITLE = "";
+ this.DEFAULT_SUBTITLE = "";
+ this.creation_time_string = "";
+
+ this.instance = instance;
+ this.wrapper = wrapper;
+ this.summary_view_callback = summary_view_callback;
+ this.detailed_view_callback = detailed_view_callback;
+
+ this.reset_data();
+ this.setup_wrapper();
+ this.setup_datatable(wrapper);
+ this.setup_footer(wrapper);
+ }
+
+ reset_data() {
+ this.data = {}; // Raw Data
+ this.filtered_data = {}; // Filtered Data / Detailed View
+ this.summary = {};
+ }
+
+ refresh_data(data, summary_data, status) {
+ this.data = data;
+ this.summary = summary_data;
+ this.status = status;
+ this.remove_tab_custom_buttons();
+ this.setup_actions();
+ this.datatable.refresh(this.summary);
+ this.set_default_title();
+ this.set_creation_time_string();
+ }
+
+ refresh_view(view, category, filters) {
+ if (!category && view === "Detailed") return;
+
+ this.filter_category = category;
+ let subtitle = "";
+
+ if (view === "Detailed") {
+ this.filter_fieldnames = this.instance.filter_fields.map(
+ filter => filter.fieldname
+ );
+
+ const columns_func = this.CATEGORY_COLUMNS[category];
+ if (!columns_func) return;
+
+ this.category_columns = columns_func.call(this);
+ this.setup_datatable(
+ this.wrapper,
+ this.filter_data(this.data[category], filters),
+ this.category_columns
+ );
+ this.set_title(category, null, true);
+ } else if (view === "Summary") {
+ this.setup_datatable(
+ this.wrapper,
+ this.summary,
+ this.get_summary_columns()
+ );
+ subtitle = this.DEFAULT_SUBTITLE;
+ this.set_title(this.DEFAULT_TITLE, subtitle);
+ }
+
+ this.setup_footer(this.wrapper);
+ this.set_creation_time_string();
+ }
+
+ filter_data(data, filters) {
+ if (!data) return [];
+ if (!filters || !filters.length) return data;
+
+ return data.filter(row => {
+ return filters.every(filter =>
+ india_compliance.FILTER_OPERATORS[filter[2]](
+ filter[3] || "",
+ row[filter[1]] || ""
+ )
+ );
+ });
+ }
+
+ // SETUP
+
+ set_title(title, subtitle, with_back_button = false) {
+ if (title) this.wrapper.find(".tab-title-text").text(title);
+ else this.wrapper.find(".tab-title-text").html(" ");
+
+ if (subtitle) this.wrapper.find(".tab-subtitle-text").text(subtitle);
+ else this.wrapper.find(".tab-subtitle-text").html("");
+
+ if (with_back_button) this.wrapper.find(".tab-back-button").show();
+ else this.wrapper.find(".tab-back-button").hide();
+ }
+
+ set_default_title() {
+ this.set_title(this.DEFAULT_TITLE, this.DEFAULT_SUBTITLE);
+ }
+
+ setup_wrapper() {
+ this.wrapper.append(`
+
+
+
+ `);
+
+ this.setup_back_button_listener();
+ }
+
+ setup_back_button_listener() {
+ this.wrapper.find(".tab-back-button").on("click", () => {
+ this.instance.show_summary_view();
+ });
+ }
+
+ setup_datatable(wrapper, data, columns) {
+ const _columns = columns || this.get_summary_columns();
+ const _data = data || [];
+ const treeView = this.instance.active_view === "Summary";
+
+ this.datatable = new india_compliance.DataTableManager({
+ $wrapper: wrapper.find(".data-table"),
+ columns: _columns,
+ data: _data,
+ options: {
+ showTotalRow: true,
+ checkboxColumn: false,
+ treeView: treeView,
+ noDataMessage: this.DEFAULT_NO_DATA_MESSAGE,
+ headerDropdown: [
+ {
+ label: "Collapse All Node",
+ action: () => {
+ this.datatable.datatable.rowmanager.collapseAllNodes();
+ },
+ },
+ {
+ label: "Expand All Node",
+ action: () => {
+ this.datatable.datatable.rowmanager.expandAllNodes();
+ },
+ },
+ ],
+ hooks: {
+ columnTotal: (_, row) => {
+ if (this.instance.active_view !== "Summary") return null;
+
+ if (row.colIndex === 1)
+ return (row.content = "Total Liability");
+
+ const column_field = row.column.fieldname;
+ if (!this.summary) return null;
+
+ const total = this.summary.reduce((acc, row) => {
+ if (row.indent !== 1) return acc;
+ if (
+ row.consider_in_total_taxable_value &&
+ ["no_of_records", "total_taxable_value"].includes(
+ column_field
+ )
+ )
+ acc += row[column_field] || 0;
+ else if (row.consider_in_total_tax)
+ acc += row[column_field] || 0;
+
+ return acc;
+ }, 0);
+
+ return total;
+ },
+ },
+ },
+ no_data_message: __("No data found"),
+ });
+
+ this.setup_datatable_listeners(treeView);
+ }
+
+ setup_datatable_listeners(isSummaryView) {
+ const me = this;
+
+ // Summary View
+ if (isSummaryView) {
+ this.datatable.$datatable.on("click", ".description", async function (e) {
+ e.preventDefault();
+
+ const summary_description = $(this).text();
+ me.summary_view_callback &&
+ me.summary_view_callback(summary_description);
+ });
+ return;
+ }
+
+ // Detailed View
+ this.instance.filter_fields.forEach(field => {
+ this.datatable.$datatable.on("click", `.${field.fieldname}`, function (e) {
+ e.preventDefault();
+
+ const fieldname = field.fieldname;
+ const value = $(this).text();
+ me.detailed_view_callback &&
+ me.detailed_view_callback(fieldname, value);
+ });
+ });
+ }
+
+ setup_footer(wrapper) {
+ const treeView = this.instance.active_view === "Summary";
+ if (!treeView) {
+ $(wrapper).find("[data-action=collapse_all_rows]").hide();
+ $(wrapper).find("[data-action=expand_all_rows]").hide();
+ } else {
+ $(wrapper).find("[data-action=collapse_all_rows]").show();
+ $(wrapper).find("[data-action=expand_all_rows]").hide();
+ }
+
+ this.setup_footer_actions(wrapper);
+ }
+
+ setup_footer_actions(wrapper) {
+ const me = this;
+ ["expand", "collapse"].forEach(action => {
+ $(wrapper).on("click", `.${action}`, function (e) {
+ e.preventDefault();
+ me.datatable.datatable.rowmanager[`${action}AllNodes`]();
+ $(wrapper).find("[data-action=collapse_all_rows]").toggle();
+ $(wrapper).find("[data-action=expand_all_rows]").toggle();
+ });
+ });
+ }
+
+ set_creation_time_string() {
+ const creation_time_string = this.get_creation_time_string();
+ if (!creation_time_string) return;
+
+ if ($(this.wrapper).find(".creation-time").length)
+ $(this.wrapper).find(".creation-time").remove();
+
+ this.wrapper
+ .find(".report-footer")
+ .append(
+ `${creation_time_string}
`
+ );
+ }
+
+ get_creation_time_string() {
+ if (!this.data.creation) return;
+
+ const creation = frappe.utils.to_title_case(
+ frappe.datetime.prettyDate(this.data.creation)
+ );
+
+ return `Created ${creation}`;
+ }
+
+ // UTILS
+
+ add_tab_custom_button(label, action) {
+ let button = this.wrapper.find(
+ `button[data-label="${encodeURIComponent(label)}"]`
+ );
+ if (button.length) return;
+
+ $(`
+
+ `)
+ .appendTo(this.wrapper.find(".custom-button-group"))
+ .on("click", action);
+ }
+
+ remove_tab_custom_buttons() {
+ this.wrapper.find(".custom-button-group").empty();
+ }
+
+ format_summary_table_cell(args) {
+ const isDescriptionCell = args[1]?.id === "description";
+ let value = args[0];
+
+ if (args[1]?._fieldtype === "Currency") value = format_currency(value);
+ else if (args[1]?._fieldtype === "Float") value = format_number(value);
+
+ value =
+ args[2]?.indent == 0
+ ? `${value}`
+ : isDescriptionCell
+ ? `
+ ${value}
+ `
+ : value;
+
+ return value;
+ }
+
+ format_detailed_table_cell(args) {
+ /**
+ * Update fieldname as a class to the cell
+ * and make it clickable.
+ *
+ * This is used to simplify filtering of data
+ */
+ let value = frappe.format(...args);
+
+ if (this.filter_fieldnames.includes(args[1]?.id))
+ value = `
+
+ ${value}
+ `;
+
+ return value;
+ }
+
+ get_icon(value, column, data, icon) {
+ if (!data) return "";
+ return `
+ `;
+ }
+}
+
+class GSTR1_TabManager extends TabManager {
+ // COLUMNS
+ get_summary_columns() {
+ return [
+ {
+ name: "Description",
+ fieldname: "description",
+ width: 300,
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "Total Docs",
+ fieldname: "no_of_records",
+ _fieldtype: "Float",
+ width: 100,
+ align: "center",
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "Taxable Value",
+ fieldname: GSTR1_DataField.TAXABLE_VALUE,
+ _fieldtype: "Float",
+ width: 180,
+
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "IGST",
+ fieldname: GSTR1_DataField.IGST,
+ _fieldtype: "Float",
+ width: 150,
+
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "CGST",
+ fieldname: GSTR1_DataField.CGST,
+ _fieldtype: "Float",
+ width: 150,
+
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "SGST",
+ fieldname: GSTR1_DataField.SGST,
+ _fieldtype: "Float",
+ width: 150,
+
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ {
+ name: "CESS",
+ fieldname: GSTR1_DataField.CESS,
+ _fieldtype: "Float",
+ width: 150,
+
+ _value: (...args) => this.format_summary_table_cell(args),
+ },
+ ];
+ }
+
+ get_invoice_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Invoice Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Invoice Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer GSTIN",
+ fieldname: GSTR1_DataField.CUST_GSTIN,
+ width: 160,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ {
+ name: "Invoice Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 150,
+ },
+ {
+ name: "Reverse Charge",
+ fieldname: GSTR1_DataField.REVERSE_CHARGE,
+ width: 120,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ ...this.get_tax_columns(),
+ {
+ name: "Invoice Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_export_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Invoice Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Invoice Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Invoice Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 150,
+ },
+ {
+ name: "Shipping Bill Number",
+ fieldname: GSTR1_DataField.SHIPPING_BILL_NUMBER,
+ width: 150,
+ },
+ {
+ name: "Shipping Bill Date",
+ fieldname: GSTR1_DataField.SHIPPING_BILL_DATE,
+ width: 120,
+ },
+ {
+ name: "Port Code",
+ fieldname: GSTR1_DataField.SHIPPING_PORT_CODE,
+ width: 100,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ ...this.get_igst_tax_columns(),
+ {
+ name: "Invoice Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_document_columns() {
+ // `Transaction Type` + Invoice Columns with `Document` as title instead of `Invoice`
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Transaction Type",
+ fieldname: GSTR1_DataField.TRANSACTION_TYPE,
+ width: 100,
+ },
+ {
+ name: "Document Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Document Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer GSTIN",
+ fieldname: GSTR1_DataField.CUST_GSTIN,
+ width: 160,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ {
+ name: "Document Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 150,
+ },
+ {
+ name: "Reverse Charge",
+ fieldname: GSTR1_DataField.REVERSE_CHARGE,
+ width: 120,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ ...this.get_tax_columns(),
+ {
+ name: "Document Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_hsn_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "HSN Code",
+ fieldname: GSTR1_DataField.HSN_CODE,
+ width: 150,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ {
+ name: "Description",
+ fieldname: GSTR1_DataField.DESCRIPTION,
+ width: 300,
+ },
+ {
+ name: "UOM",
+ fieldname: GSTR1_DataField.UOM,
+ width: 100,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ {
+ name: "Total Quantity",
+ fieldname: GSTR1_DataField.QUANTITY,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "Tax Rate",
+ fieldname: GSTR1_DataField.TAX_RATE,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "Taxable Value",
+ fieldname: GSTR1_DataField.TAXABLE_VALUE,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "IGST",
+ fieldname: GSTR1_DataField.IGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "CGST",
+ fieldname: GSTR1_DataField.CGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "SGST",
+ fieldname: GSTR1_DataField.SGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "CESS",
+ fieldname: GSTR1_DataField.CESS,
+ fieldtype: "Float",
+ width: 100,
+ },
+ ];
+ }
+
+ get_documents_issued_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Document Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 200,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ {
+ name: "Sr No From",
+ fieldname: GSTR1_DataField.FROM_SR,
+ width: 150,
+ },
+ {
+ name: "Sr No To",
+ fieldname: GSTR1_DataField.TO_SR,
+ width: 150,
+ },
+ {
+ name: "Total Count",
+ fieldname: GSTR1_DataField.TOTAL_COUNT,
+ width: 120,
+ },
+ {
+ name: "Draft Count",
+ fieldname: GSTR1_DataField.DRAFT_COUNT,
+ width: 120,
+ },
+ {
+ name: "Cancelled Count",
+ fieldname: GSTR1_DataField.CANCELLED_COUNT,
+ width: 120,
+ },
+ ];
+ }
+
+ get_advances_received_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ ...this.get_match_columns(),
+ ...this.get_tax_columns(),
+ ];
+ }
+
+ get_advances_adjusted_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ ...this.get_match_columns(),
+ ...this.get_tax_columns(),
+ ];
+ }
+
+ // Common Columns
+
+ get_tax_columns() {
+ return [
+ {
+ name: "Place of Supply",
+ fieldname: GSTR1_DataField.POS,
+ width: 150,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ {
+ name: "Tax Rate",
+ fieldname: GSTR1_DataField.TAX_RATE,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "Taxable Value",
+ fieldname: GSTR1_DataField.TAXABLE_VALUE,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "IGST",
+ fieldname: GSTR1_DataField.IGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "CGST",
+ fieldname: GSTR1_DataField.CGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "SGST",
+ fieldname: GSTR1_DataField.SGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "CESS",
+ fieldname: GSTR1_DataField.CESS,
+ fieldtype: "Float",
+ width: 100,
+ },
+ ];
+ }
+
+ get_igst_tax_columns(with_pos) {
+ const columns = [];
+
+ if (with_pos)
+ columns.push({
+ name: "Place of Supply",
+ fieldname: GSTR1_DataField.POS,
+ width: 150,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ });
+
+ columns.push(
+ {
+ name: "Tax Rate",
+ fieldname: GSTR1_DataField.TAX_RATE,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "Taxable Value",
+ fieldname: GSTR1_DataField.TAXABLE_VALUE,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "IGST",
+ fieldname: GSTR1_DataField.IGST,
+ fieldtype: "Float",
+ width: 100,
+ },
+ {
+ name: "CESS",
+ fieldname: GSTR1_DataField.CESS,
+ fieldtype: "Float",
+ width: 100,
+ }
+ );
+
+ return columns;
+ }
+
+ get_match_columns() {
+ return [];
+ }
+
+ get_detail_view_column() {
+ return [];
+ }
+}
+
+class BooksTab extends GSTR1_TabManager {
+ CATEGORY_COLUMNS = {
+ // [GSTR1_Categories.NIL_EXEMPT]: this.get_document_columns,
+
+ // SUBCATEGORIES
+ [GSTR1_SubCategory.B2B_REGULAR]: this.get_invoice_columns,
+ [GSTR1_SubCategory.B2B_REVERSE_CHARGE]: this.get_invoice_columns,
+ [GSTR1_SubCategory.SEZWP]: this.get_invoice_columns,
+ [GSTR1_SubCategory.SEZWOP]: this.get_invoice_columns,
+ [GSTR1_SubCategory.DE]: this.get_invoice_columns,
+
+ [GSTR1_SubCategory.EXPWP]: this.get_export_columns,
+ [GSTR1_SubCategory.EXPWOP]: this.get_export_columns,
+
+ [GSTR1_SubCategory.B2CL]: this.get_invoice_columns,
+ [GSTR1_SubCategory.B2CS]: this.get_b2cs_columns,
+
+ [GSTR1_SubCategory.NIL_EXEMPT]: this.get_nil_exempt_columns,
+
+ [GSTR1_SubCategory.CDNR]: this.get_document_columns,
+ [GSTR1_SubCategory.CDNUR]: this.get_document_columns,
+
+ [GSTR1_SubCategory.AT]: this.get_advances_received_columns,
+ [GSTR1_SubCategory.TXP]: this.get_advances_adjusted_columns,
+
+ [GSTR1_SubCategory.HSN]: this.get_hsn_columns,
+
+ [GSTR1_SubCategory.DOC_ISSUE]: this.get_documents_issued_columns,
+ };
+
+ DEFAULT_TITLE = "Summary of Books";
+
+ setup_actions() {
+ this.add_tab_custom_button("Download Excel", () =>
+ this.download_books_as_excel()
+ );
+ this.add_tab_custom_button("Recompute", () => this.recompute_books());
+ }
+
+ filter_data(data, filters) {
+ data = super.filter_data(data, filters);
+ return data.filter(row => row.upload_status !== "Missing in Books");
+ }
+
+ // ACTIONS
+
+ download_books_as_excel() {
+ const url =
+ "india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_export.download_books_as_excel";
+
+ open_url_post(`/api/method/${url}`, {
+ company_gstin: this.instance.frm.doc.company_gstin,
+ month_or_quarter: this.instance.frm.doc.month_or_quarter,
+ year: this.instance.frm.doc.year,
+ });
+ }
+
+ recompute_books() {
+ render_empty_state(this.instance.frm);
+ this.instance.frm.call("recompute_books");
+ }
+
+ // COLUMNS
+
+ get_match_columns() {
+ if (this.status === "Filed") return [];
+ return [
+ {
+ name: "Upload Status",
+ fieldname: GSTR1_DataField.UPLOAD_STATUS,
+ width: 150,
+ },
+ ];
+ }
+
+ get_b2cs_columns() {
+ let columns = this.get_document_columns();
+ columns = columns.filter(
+ col =>
+ ![GSTR1_DataField.CUST_GSTIN, GSTR1_DataField.REVERSE_CHARGE].includes(
+ col.fieldname
+ )
+ );
+
+ return columns;
+ }
+
+ get_nil_exempt_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Transaction Type",
+ fieldname: GSTR1_DataField.TRANSACTION_TYPE,
+ width: 100,
+ },
+ {
+ name: "Document Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Document Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer GSTIN",
+ fieldname: GSTR1_DataField.CUST_GSTIN,
+ width: 160,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ {
+ name: "Document Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 150,
+ },
+ ...this.get_match_columns(),
+ {
+ name: "Nil-Rated Supplies",
+ fieldname: GSTR1_DataField.NIL_RATED_AMOUNT,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "Exempted Supplies",
+ fieldname: GSTR1_DataField.EXEMPTED_AMOUNT,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "Non-GST Supplies",
+ fieldname: GSTR1_DataField.NON_GST_AMOUNT,
+ fieldtype: "Float",
+ width: 150,
+ },
+ {
+ name: "Document Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_advances_received_columns() {
+ return [
+ {
+ name: "Advance Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Payment Entry Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Payment Entry",
+ width: 160,
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ ...super.get_advances_received_columns(),
+ ];
+ }
+
+ get_advances_adjusted_columns() {
+ return [
+ {
+ name: "Adjustment Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Adjustment Entry Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ ...super.get_advances_adjusted_columns(),
+ ];
+ }
+}
+
+class FiledTab extends GSTR1_TabManager {
+ CATEGORY_COLUMNS = {
+ [GSTR1_SubCategory.B2B_REGULAR]: this.get_invoice_columns,
+ [GSTR1_SubCategory.B2B_REVERSE_CHARGE]: this.get_invoice_columns,
+ [GSTR1_SubCategory.SEZWP]: this.get_invoice_columns,
+ [GSTR1_SubCategory.SEZWOP]: this.get_invoice_columns,
+ [GSTR1_SubCategory.DE]: this.get_invoice_columns,
+
+ [GSTR1_SubCategory.EXPWP]: this.get_export_columns,
+ [GSTR1_SubCategory.EXPWOP]: this.get_export_columns,
+
+ [GSTR1_SubCategory.B2CL]: this.get_b2cl_columns,
+ [GSTR1_SubCategory.B2CS]: this.get_b2cs_columns,
+
+ [GSTR1_SubCategory.NIL_EXEMPT]: this.get_nil_exempt_columns,
+
+ [GSTR1_SubCategory.CDNR]: this.get_document_columns,
+ [GSTR1_SubCategory.CDNUR]: this.get_cdnur_columns,
+
+ [GSTR1_SubCategory.AT]: this.get_advances_received_columns,
+ [GSTR1_SubCategory.TXP]: this.get_advances_adjusted_columns,
+
+ [GSTR1_SubCategory.HSN]: this.get_hsn_columns,
+ [GSTR1_SubCategory.DOC_ISSUE]: this.get_documents_issued_columns,
+ };
+
+ setup_actions() {
+ this.add_tab_custom_button("Download Excel", () =>
+ this.download_filed_as_excel()
+ );
+
+ if (this.status !== "Filed")
+ this.add_tab_custom_button("Download JSON", () =>
+ this.download_filed_json()
+ );
+
+ if (!is_gstr1_api_enabled()) return;
+
+ if (this.status === "Filed")
+ this.add_tab_custom_button("Sync with GSTN", () =>
+ this.sync_with_gstn("filed")
+ );
+ else {
+ this.add_tab_custom_button("Mark as Filed", () => this.mark_as_filed());
+ }
+ }
+
+ set_default_title() {
+ if (this.status === "Filed") this.DEFAULT_TITLE = "Summary of Filed GSTR-1";
+ else this.DEFAULT_TITLE = "Summary of Draft GSTR-1";
+
+ super.set_default_title();
+ }
+
+ // ACTIONS
+
+ download_filed_as_excel() {
+ const url =
+ "india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_export.download_filed_as_excel";
+
+ open_url_post(`/api/method/${url}`, {
+ company_gstin: this.instance.frm.doc.company_gstin,
+ month_or_quarter: this.instance.frm.doc.month_or_quarter,
+ year: this.instance.frm.doc.year,
+ });
+ }
+
+ sync_with_gstn(sync_for) {
+ render_empty_state(this.instance.frm);
+ this.instance.frm.call("sync_with_gstn", { sync_for });
+ }
+
+ download_filed_json() {
+ const me = this;
+ function get_json_data(dialog) {
+ const { include_uploaded, delete_missing } = dialog
+ ? dialog.get_values()
+ : {
+ include_uploaded: true,
+ delete_missing: false,
+ };
+
+ const doc = me.instance.frm.doc;
+
+ frappe.call({
+ method: "india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_export.download_gstr_1_json",
+ args: {
+ company_gstin: doc.company_gstin,
+ year: doc.year,
+ month_or_quarter: doc.month_or_quarter,
+ include_uploaded,
+ delete_missing,
+ },
+ callback: r => {
+ india_compliance.trigger_file_download(
+ JSON.stringify(r.message.data),
+ r.message.filename
+ );
+ dialog && dialog.hide();
+ },
+ });
+ }
+
+ // without API
+ if (!is_gstr1_api_enabled()) {
+ get_json_data();
+ return;
+ }
+
+ // with API
+ const dialog = new frappe.ui.Dialog({
+ title: __("Download JSON"),
+ fields: [
+ {
+ fieldname: "include_uploaded",
+ label: __("Include Already Uploaded (matching) Invoices"),
+ description: __(
+ `This will include invoices already uploaded (and matching)
+ to GSTN (possibly e-Invoices) and overwrite them in GST Portal.
+ This is not recommended if e-Invoice is applicable to you
+ as it will overwrite the e-Invoice data in GST Portal.`
+ ),
+ fieldtype: "Check",
+ },
+ {
+ fieldname: "delete_missing",
+ label: __(
+ "Delete records that are missing in the Books from GST Portal"
+ ),
+ description: __(
+ "This will delete invoices that are not present in ERP but are present in GST Portal."
+ ),
+ fieldtype: "Check",
+ default: 1,
+ },
+ ],
+ primary_action: () => get_json_data(dialog),
+ });
+
+ dialog.show();
+ }
+
+ mark_as_filed() {
+ render_empty_state(this.instance.frm);
+ this.instance.frm
+ .call("mark_as_filed")
+ .then(() => this.instance.frm.trigger("after_save"));
+ }
+
+ // COLUMNS
+
+ get_b2cl_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Invoice Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Invoice Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ ...this.get_match_columns(),
+ ...this.get_igst_tax_columns(true),
+ {
+ name: "Invoice Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_b2cs_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Invoice Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 100,
+ },
+ ...this.get_tax_columns(),
+ ...this.get_match_columns(),
+ ];
+ }
+
+ get_nil_exempt_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Description",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 200,
+ _value: (...args) => this.format_detailed_table_cell(args),
+ },
+ ...this.get_match_columns(),
+ {
+ name: "Nil-Rated Supplies",
+ fieldname: GSTR1_DataField.NIL_RATED_AMOUNT,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ {
+ name: "Exempted Supplies",
+ fieldname: GSTR1_DataField.EXEMPTED_AMOUNT,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ {
+ name: "Non-GST Supplies",
+ fieldname: GSTR1_DataField.NON_GST_AMOUNT,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ {
+ name: "Total Taxable Value",
+ fieldname: GSTR1_DataField.TAXABLE_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+
+ get_cdnur_columns() {
+ return [
+ ...this.get_detail_view_column(),
+ {
+ name: "Transaction Type",
+ fieldname: GSTR1_DataField.TRANSACTION_TYPE,
+ width: 100,
+ },
+ {
+ name: "Document Date",
+ fieldname: GSTR1_DataField.DOC_DATE,
+ fieldtype: "Date",
+ width: 120,
+ },
+ {
+ name: "Document Number",
+ fieldname: GSTR1_DataField.DOC_NUMBER,
+ fieldtype: "Link",
+ options: "Sales Invoice",
+ width: 160,
+ },
+ {
+ name: "Customer Name",
+ fieldname: GSTR1_DataField.CUST_NAME,
+ width: 200,
+ },
+ {
+ name: "Document Type",
+ fieldname: GSTR1_DataField.DOC_TYPE,
+ width: 150,
+ },
+ ...this.get_match_columns(),
+ ...this.get_igst_tax_columns(true),
+ {
+ name: "Document Value",
+ fieldname: GSTR1_DataField.DOC_VALUE,
+ fieldtype: "Currency",
+ width: 150,
+ },
+ ];
+ }
+}
+
+class UnfiledTab extends FiledTab {
+ setup_actions() {
+ if (!is_gstr1_api_enabled()) return;
+
+ this.add_tab_custom_button("Sync with GSTN", () =>
+ this.sync_with_gstn("unfiled")
+ );
+ }
+
+ set_default_title() {
+ this.DEFAULT_TITLE = "Summary of Invoices Uploaded on GST Portal";
+ TabManager.prototype.set_default_title.call(this);
+ }
+}
+
+class ReconcileTab extends FiledTab {
+ DEFAULT_NO_DATA_MESSAGE = __("No differences found");
+
+ set_default_title() {
+ if (this.instance.data.status === "Filed")
+ this.DEFAULT_TITLE = "Books vs Filed";
+ else this.DEFAULT_TITLE = "Books vs Unfiled";
+
+ this.DEFAULT_SUBTITLE = "Only differences";
+ TabManager.prototype.set_default_title.call(this);
+ }
+
+ setup_actions() {
+ this.add_tab_custom_button("Download Excel", () =>
+ this.download_reconcile_as_excel()
+ );
+ }
+
+ download_reconcile_as_excel() {
+ const url =
+ "india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_export.download_reconcile_as_excel";
+
+ open_url_post(`/api/method/${url}`, {
+ company_gstin: this.instance.frm.doc.company_gstin,
+ month_or_quarter: this.instance.frm.doc.month_or_quarter,
+ year: this.instance.frm.doc.year,
+ });
+ }
+
+ get_creation_time_string() { } // pass
+
+ get_detail_view_column() {
+ return [
+ {
+ fieldname: "detail_view",
+ fieldtype: "html",
+ width: 60,
+ align: "center",
+ _value: (...args) => this.get_icon(...args, "eye"),
+ },
+ ];
+ }
+
+ get_match_columns() {
+ return [
+ {
+ name: "Match Status",
+ fieldname: "match_status",
+ width: 150,
+ },
+ {
+ name: "Differences",
+ fieldname: "differences",
+ width: 150,
+ },
+ ];
+ }
+}
+
+class DetailViewDialog {
+ CURRENCY_FIELD_MAP = {
+ [GSTR1_DataField.TAXABLE_VALUE]: "Taxable Value",
+ [GSTR1_DataField.IGST]: "IGST",
+ [GSTR1_DataField.CGST]: "CGST",
+ [GSTR1_DataField.SGST]: "SGST",
+ [GSTR1_DataField.CESS]: "CESS",
+ [GSTR1_DataField.DOC_VALUE]: "Invoice Value",
+ };
+
+ IGNORED_FIELDS = [
+ GSTR1_DataField.CUST_NAME,
+ GSTR1_DataField.DOC_NUMBER,
+ GSTR1_DataField.DOC_TYPE,
+ "match_status",
+ GSTR1_DataField.DESCRIPTION,
+ ];
+
+ constructor(data, field_label_map) {
+ this.data = data;
+ this.field_label_map = field_label_map;
+ this.show_dialog();
+ }
+
+ show_dialog() {
+ this.init_dialog();
+ this.render_table();
+ this.dialog.show();
+ }
+
+ init_dialog() {
+ this.dialog = new frappe.ui.Dialog({
+ title: "Detail View",
+ fields: [
+ {
+ fieldtype: "HTML",
+ fieldname: "reconcile_data",
+ },
+ ],
+ });
+ }
+
+ render_table() {
+ const detail_table = this.dialog.fields_dict.reconcile_data;
+ const field_label_map = this.field_label_map.filter(
+ field => !this.IGNORED_FIELDS.includes(field[0])
+ );
+
+ detail_table.html(
+ frappe.render_template("gstr_1_detail_comparision", {
+ data: this.data,
+ fieldname_map: field_label_map,
+ currency_map: this.CURRENCY_FIELD_MAP,
+ })
+ );
+ this._set_value_color(detail_table.$wrapper, this.data);
+ }
+
+ _set_value_color(wrapper, data) {
+ if (!Object.keys(data.gov).length || !Object.keys(data.books).length) return;
+
+ let gov_data = data.gov;
+ let books_data = data.books;
+
+ for (const key in gov_data) {
+ if (gov_data[key] === books_data[key] || key === "description") continue;
+
+ wrapper
+ .find(`[data-label='${key}'], [data-label='${key}']`)
+ .addClass("not-matched");
+ }
+ }
+}
+
+// UTILITY FUNCTIONS
+function is_gstr1_api_enabled() {
+ return (
+ india_compliance.is_api_enabled() &&
+ !gst_settings.sandbox_mode &&
+ gst_settings.compare_gstr_1_data
+ );
+}
+
+function patch_set_indicator(frm) {
+ frm.toolbar.set_indicator = function () { };
+}
+
+async function set_default_company_gstin(frm) {
+ frm.set_value("company_gstin", "");
+
+ const company = frm.doc.company;
+ const { message: gstin_list } = await frappe.call(
+ "india_compliance.gst_india.utils.get_gstin_list",
+ { party: company }
+ );
+
+ if (gstin_list && gstin_list.length) {
+ frm.set_value("company_gstin", gstin_list[0]);
+ }
+}
+
+function set_options_for_year(frm) {
+ const today = new Date();
+ const current_year = today.getFullYear();
+ const start_year = 2017;
+ const year_range = current_year - start_year + 1;
+ let options = Array.from({ length: year_range }, (_, index) => start_year + index);
+ options = options.reverse().map(year => year.toString());
+
+ frm.get_field("year").set_data(options);
+ frm.set_value("year", current_year.toString());
+}
+
+function set_options_for_month_or_quarter(frm) {
+ /**
+ * Set options for Month or Quarter based on the year and current date
+ * 1. If the year is current year, then options are till current month
+ * 2. If the year is 2017, then options are from July to December
+ * 3. Else, options are all months or quarters
+ *
+ * @param {Object} frm
+ */
+
+ const today = new Date();
+ const current_year = String(today.getFullYear());
+ const current_month_idx = today.getMonth();
+ let options;
+
+ if (!frm.doc.year) frm.doc.year = current_year;
+
+ if (frm.doc.year === current_year) {
+ // Options for current year till current month
+ if (frm.filing_frequency === "Monthly")
+ options = india_compliance.MONTH.slice(0, current_month_idx + 1);
+ else {
+ let quarter_idx;
+ if (current_month_idx <= 2) quarter_idx = 1;
+ else if (current_month_idx <= 5) quarter_idx = 2;
+ else if (current_month_idx <= 8) quarter_idx = 3;
+ else quarter_idx = 4;
+
+ options = india_compliance.QUARTER.slice(0, quarter_idx);
+ }
+ } else if (frm.doc.year === "2017") {
+ // Options for 2017 from July to December
+ if (frm.filing_frequency === "Monthly")
+ options = india_compliance.MONTH.slice(6);
+ else options = india_compliance.QUARTER.slice(2);
+ } else {
+ if (frm.filing_frequency === "Monthly") options = india_compliance.MONTH;
+ else options = india_compliance.QUARTER;
+ }
+
+ set_field_options("month_or_quarter", options);
+ if (frm.doc.year === current_year)
+ // set second last option as default
+ frm.set_value("month_or_quarter", options[options.length - 2]);
+ // set last option as default
+ else frm.set_value("month_or_quarter", options[options.length - 1]);
+}
+
+function render_empty_state(frm) {
+ if ($(".gst-ledger-difference").length) {
+ $(".gst-ledger-difference").remove();
+ }
+ frm.doc.__onload = null;
+ frm.refresh();
+}
+
+async function get_net_gst_liability(frm) {
+ const response = await frappe.call({
+ method: "india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_beta.get_net_gst_liability",
+ args: {
+ month_or_quarter: frm.doc.month_or_quarter,
+ year: frm.doc.year,
+ company_gstin: frm.doc.company_gstin,
+ company: frm.doc.company,
+ },
+ });
+
+ return response?.message;
+}
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.json b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.json
new file mode 100644
index 0000000000..87f2d16b43
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.json
@@ -0,0 +1,128 @@
+{
+ "actions": [],
+ "beta": 1,
+ "creation": "2024-03-27 17:59:36.078726",
+ "doctype": "DocType",
+ "engine": "InnoDB",
+ "field_order": [
+ "form",
+ "company",
+ "column_break_ejve",
+ "company_gstin",
+ "column_break_ldkv",
+ "year",
+ "column_break_qcor",
+ "month_or_quarter",
+ "data_section",
+ "tabs_html",
+ "tabs_empty_state",
+ "tabs_no_data"
+ ],
+ "fields": [
+ {
+ "fieldname": "form",
+ "fieldtype": "Section Break"
+ },
+ {
+ "fieldname": "company",
+ "fieldtype": "Link",
+ "in_list_view": 1,
+ "label": "Company",
+ "options": "Company",
+ "reqd": 1
+ },
+ {
+ "fieldname": "company_gstin",
+ "fieldtype": "Autocomplete",
+ "label": "Company GSTIN",
+ "reqd": 1
+ },
+ {
+ "fieldname": "column_break_ejve",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "column_break_ldkv",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "column_break_qcor",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "year",
+ "fieldtype": "Autocomplete",
+ "label": "Year",
+ "reqd": 1
+ },
+ {
+ "depends_on": "eval: doc.__onload?.data && Object.keys(doc.__onload.data).length",
+ "fieldname": "tabs_html",
+ "fieldtype": "HTML"
+ },
+ {
+ "depends_on": "eval: !doc.__onload?.data",
+ "fieldname": "tabs_empty_state",
+ "fieldtype": "HTML",
+ "options": "\n\t{{ __(\"Generate to view the data\") }}
"
+ },
+ {
+ "depends_on": "eval: doc.__onload?.data && Object.keys(doc.__onload.data).length === 0",
+ "fieldname": "tabs_no_data",
+ "fieldtype": "HTML",
+ "options": "\n\t{{ __(\"No data available for selected filters\") }}
"
+ },
+ {
+ "fieldname": "data_section",
+ "fieldtype": "Section Break"
+ },
+ {
+ "fieldname": "month_or_quarter",
+ "fieldtype": "Select",
+ "label": "Month/Quarter",
+ "reqd": 1
+ }
+ ],
+ "hide_toolbar": 1,
+ "index_web_pages_for_search": 1,
+ "issingle": 1,
+ "links": [],
+ "modified": "2024-05-27 19:30:01.074149",
+ "modified_by": "Administrator",
+ "module": "GST India",
+ "name": "GSTR-1 Beta",
+ "owner": "Administrator",
+ "permissions": [
+ {
+ "create": 1,
+ "export": 1,
+ "read": 1,
+ "role": "System Manager",
+ "write": 1
+ },
+ {
+ "create": 1,
+ "export": 1,
+ "read": 1,
+ "role": "Accounts Manager",
+ "write": 1
+ },
+ {
+ "create": 1,
+ "export": 1,
+ "read": 1,
+ "role": "Accounts User",
+ "write": 1
+ },
+ {
+ "create": 1,
+ "export": 1,
+ "read": 1,
+ "role": "Auditor",
+ "write": 1
+ }
+ ],
+ "sort_field": "modified",
+ "sort_order": "DESC",
+ "states": []
+}
\ No newline at end of file
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.py b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.py
new file mode 100644
index 0000000000..dde610a83a
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_beta.py
@@ -0,0 +1,245 @@
+# Copyright (c) 2024, Resilient Tech and contributors
+# For license information, please see license.txt
+
+from datetime import datetime
+
+import frappe
+from frappe import _
+from frappe.desk.form.load import run_onload
+from frappe.model.document import Document
+from frappe.query_builder.functions import Date, Sum
+from frappe.utils import get_last_day, getdate
+
+from india_compliance.gst_india.utils import get_gst_accounts_by_type
+from india_compliance.gst_india.utils.gstin_info import get_gstr_1_return_status
+from india_compliance.gst_india.utils.gstr_utils import request_otp
+
+
+class GSTR1Beta(Document):
+
+ def onload(self):
+ data = getattr(self, "data", None)
+ if data is not None:
+ self.set_onload("data", data)
+
+ @frappe.whitelist()
+ def recompute_books(self):
+ self.validate(recompute_books=True)
+
+ @frappe.whitelist()
+ def sync_with_gstn(self, sync_for):
+ self.validate(sync_for=sync_for)
+
+ @frappe.whitelist()
+ def mark_as_filed(self):
+ period = get_period(self.month_or_quarter, self.year)
+ return_status = get_gstr_1_return_status(
+ self.company, self.company_gstin, period
+ )
+
+ if return_status != "Filed":
+ frappe.msgprint(
+ _("GSTR-1 is not yet filed on the GST Portal"), indicator="red"
+ )
+
+ else:
+ frappe.db.set_value(
+ "GSTR-1 Log",
+ f"{period}-{self.company_gstin}",
+ "filing_status",
+ return_status,
+ )
+
+ self.validate()
+ run_onload(self)
+
+ def validate(self, sync_for=None, recompute_books=False):
+ period = get_period(self.month_or_quarter, self.year)
+
+ # get gstr1 log
+ if log_name := frappe.db.exists("GSTR-1 Log", f"{period}-{self.company_gstin}"):
+
+ gstr1_log = frappe.get_doc("GSTR-1 Log", log_name)
+
+ message = None
+ if gstr1_log.status == "In Progress":
+ message = (
+ "GSTR-1 is being prepared. Please wait for the process to complete."
+ )
+
+ elif gstr1_log.status == "Queued":
+ message = (
+ "GSTR-1 download is queued and could take some time. Please wait"
+ " for the process to complete."
+ )
+
+ if message:
+ frappe.msgprint(_(message), title=_("GSTR-1 Generation In Progress"))
+ return
+
+ else:
+ gstr1_log = frappe.new_doc("GSTR-1 Log")
+ gstr1_log.company = self.company
+ gstr1_log.gstin = self.company_gstin
+ gstr1_log.return_period = period
+ gstr1_log.insert()
+
+ settings = frappe.get_cached_doc("GST Settings")
+
+ if sync_for:
+ gstr1_log.remove_json_for(sync_for)
+
+ if recompute_books:
+ gstr1_log.remove_json_for("books")
+
+ # files are already present
+ if gstr1_log.has_all_files(settings):
+ data = gstr1_log.load_data()
+
+ if data:
+ self.data = data
+ self.data["status"] = gstr1_log.filing_status or "Not Filed"
+ gstr1_log.update_status("Generated")
+ return
+
+ # request OTP
+ if gstr1_log.is_sek_needed(settings) and not settings.is_sek_valid(
+ self.company_gstin
+ ):
+ request_otp(self.company_gstin)
+ self.data = "otp_requested"
+ return
+
+ self.gstr1_log = gstr1_log
+
+ # generate gstr1
+ gstr1_log.update_status("In Progress")
+ frappe.enqueue(self.generate_gstr1, queue="short")
+ frappe.msgprint(_("GSTR-1 is being prepared"), alert=True)
+
+ def generate_gstr1(self):
+ """
+ Try to generate GSTR-1 data. Wrapper for generating GSTR-1 data
+ """
+
+ filters = frappe._dict(
+ company=self.company,
+ company_gstin=self.company_gstin,
+ month_or_quarter=self.month_or_quarter,
+ year=self.year,
+ )
+
+ try:
+ self.gstr1_log.generate_gstr1_data(filters, callback=self.on_generate)
+
+ except Exception as e:
+ self.gstr1_log.update_status("Failed", commit=True)
+
+ frappe.publish_realtime(
+ "gstr1_generation_failed",
+ message={"error": str(e), "filters": filters},
+ user=frappe.session.user,
+ doctype=self.doctype,
+ )
+
+ raise e
+
+ def on_generate(self, data, filters):
+ """
+ Once data is generated, update the status and publish the data
+ """
+ self.gstr1_log.db_set({"generation_status": "Generated", "is_latest_data": 1})
+
+ frappe.publish_realtime(
+ "gstr1_data_prepared",
+ message={"data": data, "filters": filters},
+ user=frappe.session.user,
+ doctype=self.doctype,
+ )
+
+
+####### DATA ######################################################################################
+
+
+@frappe.whitelist()
+def get_net_gst_liability(company, company_gstin, month_or_quarter, year):
+ """
+ Returns the net output balance for the given return period as per ledger entries
+ """
+
+ frappe.has_permission("GSTR-1 Beta", throw=True)
+
+ from_date, to_date = get_gstr_1_from_and_to_date(month_or_quarter, year)
+
+ filters = frappe._dict(
+ {
+ "company": company,
+ "company_gstin": company_gstin,
+ "from_date": from_date,
+ "to_date": to_date,
+ }
+ )
+ accounts = get_gst_accounts_by_type(company, "Output")
+
+ gl_entry = frappe.qb.DocType("GL Entry")
+ gst_ledger = frappe._dict(
+ frappe.qb.from_(gl_entry)
+ .select(gl_entry.account, (Sum(gl_entry.credit) - Sum(gl_entry.debit)))
+ .where(gl_entry.account.isin(list(accounts.values())))
+ .where(gl_entry.company == filters.company)
+ .where(Date(gl_entry.posting_date) >= getdate(filters.from_date))
+ .where(Date(gl_entry.posting_date) <= getdate(filters.to_date))
+ .where(gl_entry.company_gstin == filters.company_gstin)
+ .groupby(gl_entry.account)
+ .run()
+ )
+ net_output_balance = {
+ "total_igst_amount": gst_ledger.get(accounts["igst_account"], 0),
+ "total_cgst_amount": gst_ledger.get(accounts["cgst_account"], 0),
+ "total_sgst_amount": gst_ledger.get(accounts["sgst_account"], 0),
+ "total_cess_amount": gst_ledger.get(accounts["cess_account"], 0)
+ + gst_ledger.get(accounts["cess_non_advol_account"], 0),
+ }
+
+ return net_output_balance
+
+
+####### UTILS ######################################################################################
+
+
+def get_period(month_or_quarter: str, year: str) -> str:
+ """
+ Returns the period in the format MMYYYY
+ as accepted by the GST Portal
+ """
+
+ if "-" in month_or_quarter:
+ # Quarterly
+ last_month = month_or_quarter.split("-")[1]
+ month_number = str(getdate(f"{last_month}-{year}").month).zfill(2)
+
+ else:
+ # Monthly
+ month_number = str(datetime.strptime(month_or_quarter, "%B").month).zfill(2)
+
+ return f"{month_number}{year}"
+
+
+def get_gstr_1_from_and_to_date(month_or_quarter: str, year: str) -> tuple:
+ """
+ Returns the from and to date for the given month or quarter and year
+ This is used to filter the data for the given period in Books
+ """
+
+ filing_frequency = frappe.get_cached_value("GST Settings", None, "filing_frequency")
+
+ if filing_frequency == "Quarterly":
+ start_month, end_month = month_or_quarter.split("-")
+ from_date = getdate(f"{year}-{start_month}-01")
+ to_date = get_last_day(f"{year}-{end_month}-01")
+ else:
+ # Monthly (default)
+ from_date = getdate(f"{year}-{month_or_quarter}-01")
+ to_date = get_last_day(from_date)
+
+ return from_date, to_date
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_detail_comparision.html b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_detail_comparision.html
new file mode 100644
index 0000000000..0b62e8dfef
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_detail_comparision.html
@@ -0,0 +1,83 @@
+{% var gov = data.gov;
+var books = data.books%}
+
+ {%if data.customer_name%}
+
+
Customer Name:
+
{{ data.customer_name}}
+
+ {% endif %}
+
+ {%if data.document_number%}
+
+
Invoice Number:
+
{{ frappe.utils.get_form_link("Sales Invoice",data.document_number, true)
+ }}
+
+ {% endif %}
+ {%if data.document_type%}
+
+
Invoice Type:
+
{{ data.document_type }}
+
+ {% endif %}
+ {%if data.match_status%}
+
+
Match Status:
+
{{ data.match_status }}
+
+ {% endif %}
+
+
+
+ |
+ Books |
+ GSTR-1 |
+
+
+
+ {% var gov_keys=Object.keys(data.gov);
+ var books_keys=Object.keys(data.books);
+ var field_keys = gov_keys.length ? gov_keys : books_keys
+ %}
+ {% for fieldmap in fieldname_map %}
+
+ {% var column_title = fieldmap[1];
+ var fieldname = fieldmap[0];
+
+ var is_currency_field = currency_map[fieldname] || false;
+
+ if(is_currency_field){
+ if(data.books[fieldname])
+ var book_value=format_currency(data.books[fieldname]);
+ else
+ var book_value = "-"
+ }
+ else{
+ var book_value=data.books[fieldname] || "-"
+ }
+
+ if(is_currency_field){
+ if(data.gov[fieldname])
+ var gov_value=format_currency(data.gov[fieldname]);
+ else
+ var gov_value = "-"
+ }
+ else{
+ var gov_value=data.gov[fieldname] || "-"
+ }
+
+ var gov_value = is_currency_field ? format_currency(data.gov[fieldname] || 0) :
+ (data.gov[fieldname] || "-");
+
+ %}
+
+
+ {{ column_title }} |
+ {{ book_value }} |
+ {{ gov_value }} |
+
+ {% endfor %}
+
+
+
\ No newline at end of file
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_export.py b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_export.py
new file mode 100644
index 0000000000..e0f098f303
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/gstr_1_export.py
@@ -0,0 +1,2090 @@
+"""
+Export GSTR-1 data to excel or json
+"""
+
+import json
+from datetime import datetime
+from enum import Enum
+
+import frappe
+from frappe import _
+from frappe.utils import getdate
+
+from india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_beta import get_period
+from india_compliance.gst_india.utils.exporter import ExcelExporter
+from india_compliance.gst_india.utils.gstr_1 import (
+ JSON_CATEGORY_EXCEL_CATEGORY_MAPPING,
+ GovExcelField,
+ GovExcelSheetName,
+ GovJsonKey,
+ GSTR1_DataField,
+ GSTR1_ItemField,
+ GSTR1_SubCategory,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_json_map import (
+ convert_to_gov_data_format,
+ get_category_wise_data,
+)
+
+
+class ExcelWidth(Enum):
+ XS = 10
+ SM = 15
+ MD = 20 # Default
+ LG = 25
+ XL = 30
+ XXL = 35
+
+
+CATEGORIES_WITH_ITEMS = {
+ GovJsonKey.B2B.value,
+ GovJsonKey.B2CL.value,
+ GovJsonKey.EXP.value,
+ GovJsonKey.CDNR.value,
+ GovJsonKey.CDNUR.value,
+}
+
+
+class DataProcessor:
+
+ # transform input data to required format
+ FIELD_TRANSFORMATIONS = {}
+
+ def process_data(self, input_data):
+ """
+ Objective:
+
+ 1. Flatten the input data to a list of invoices
+ 2. Format/Transform the data to match the Gov Excel format
+ """
+
+ category_wise_data = get_category_wise_data(input_data)
+ processed_data = {}
+
+ for category, data in category_wise_data.items():
+ if category in CATEGORIES_WITH_ITEMS:
+ data = self.flatten_invoice_items_to_rows(data)
+
+ if self.FIELD_TRANSFORMATIONS:
+ data = [self.apply_transformations(row) for row in data]
+
+ processed_data[category] = data
+
+ return processed_data
+
+ def apply_transformations(self, row):
+ """
+ Apply transformations to row fields
+ """
+ for field, modifier in self.FIELD_TRANSFORMATIONS.items():
+ if field in row:
+ row[field] = modifier(row[field])
+
+ return row
+
+ def flatten_invoice_items_to_rows(self, invoice_list: list | tuple) -> list:
+ """
+ input_data: List of invoices with items
+ output: List of invoices with item values
+
+ Example:
+ input_data = [
+ {
+ "key": "value",
+ "items": [{ "taxable_value": "100" }, { "taxable_value": "200" }]
+ }
+ ]
+
+ output = [
+ {"key": "value", "taxable_value": "100"},
+ {"key": "value", "taxable_value": "200"}
+ ]
+
+ Purpose: Gov Excel format requires each row to have invoice values
+ """
+ return [
+ {**invoice, **item}
+ for invoice in invoice_list
+ for item in invoice[GSTR1_DataField.ITEMS.value]
+ ]
+
+
+class GovExcel(DataProcessor):
+ """
+ Export GSTR-1 data to excel
+
+ Excel generated as per the format of Returns Offline Tool Version V3.1.8
+
+ Returns Offline Tool download link - https://www.gst.gov.in/download/returns
+ """
+
+ AMOUNT_FORMAT = "#,##0.00"
+ DATE_FORMAT = "dd-mmm-yy"
+ PERCENT_FORMAT = "0.00"
+
+ FIELD_TRANSFORMATIONS = {
+ GSTR1_DataField.DIFF_PERCENTAGE.value: lambda value: (
+ value * 100 if value != 0 else None
+ ),
+ GSTR1_DataField.DOC_DATE.value: lambda value: datetime.strptime(
+ value, "%Y-%m-%d"
+ ),
+ GSTR1_DataField.SHIPPING_BILL_DATE.value: lambda value: datetime.strptime(
+ value, "%Y-%m-%d"
+ ),
+ }
+
+ def generate(self, gstin, period):
+ """
+ Build excel file
+ """
+ self.gstin = gstin
+ self.period = period
+ gstr_1_log = frappe.get_doc("GSTR-1 Log", f"{period}-{gstin}")
+
+ self.file_field = "filed" if gstr_1_log.filed else "books"
+ data = gstr_1_log.load_data(self.file_field)[self.file_field]
+ data = self.process_data(data)
+ self.build_excel(data)
+
+ def process_data(self, data):
+ data = data.update(data.pop("aggregate_data", {}))
+ category_wise_data = super().process_data(data)
+
+ for category, category_data in category_wise_data.items():
+ # filter missing in books
+ category_wise_data[category] = [
+ row
+ for row in category_data
+ if row.get("upload_status") != "Missing in Books"
+ ]
+
+ if category not in [
+ GovJsonKey.CDNR.value,
+ GovJsonKey.CDNUR.value,
+ GovJsonKey.TXP.value,
+ ]:
+ continue
+
+ # convert to positive values
+ for doc in category_wise_data.get(category, []):
+ if doc.get(GSTR1_DataField.DOC_TYPE.value) == "D":
+ continue
+
+ doc.update(
+ {
+ key: abs(value)
+ for key, value in doc.items()
+ if isinstance(value, (int, float))
+ }
+ )
+
+ return category_wise_data
+
+ def build_excel(self, data):
+ excel = ExcelExporter()
+ for category, cat_data in data.items():
+ excel.create_sheet(
+ sheet_name=JSON_CATEGORY_EXCEL_CATEGORY_MAPPING.get(category, category),
+ headers=self.get_category_headers(category),
+ data=cat_data,
+ add_totals=False,
+ default_data_format={"height": 15},
+ )
+
+ excel.remove_sheet("Sheet")
+ excel.export(get_file_name("Gov", self.gstin, self.period))
+
+ def get_category_headers(self, category):
+ return getattr(self, f"get_{category.lower()}_headers")()
+
+ def get_b2b_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.CUST_GSTIN.value),
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.CUST_NAME.value),
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_NUMBER.value),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_DATE.value),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.REVERSE_CHARGE.value),
+ "fieldname": GSTR1_DataField.REVERSE_CHARGE.value,
+ "data_format": {"horizontal": "center"},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_TYPE.value),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.ECOMMERCE_GSTIN.value),
+ # Ignore value, just keep the column
+ "fieldname": f"_{GSTR1_DataField.ECOMMERCE_GSTIN.value}",
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_b2cl_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.INVOICE_NUMBER.value),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_DATE.value),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.ECOMMERCE_GSTIN.value),
+ # Ignore value, just keep the column
+ "fieldname": f"_{GSTR1_DataField.ECOMMERCE_GSTIN.value}",
+ },
+ ]
+
+ def get_b2cs_headers(self):
+ return [
+ {
+ "label": _("Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_DataField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.ECOMMERCE_GSTIN.value),
+ # Ignore value, just keep the column
+ "fieldname": f"_{GSTR1_DataField.ECOMMERCE_GSTIN.value}",
+ },
+ ]
+
+ def get_cdnr_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.CUST_GSTIN.value),
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.CUST_NAME.value),
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _(GovExcelField.NOTE_NO.value),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.NOTE_DATE.value),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.NOTE_TYPE.value),
+ "fieldname": GSTR1_DataField.TRANSACTION_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.REVERSE_CHARGE.value),
+ "fieldname": GSTR1_DataField.REVERSE_CHARGE.value,
+ "data_format": {"horizontal": "center"},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Note Supply Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.NOTE_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_cdnur_headers(self):
+ return [
+ {
+ "label": _("UR Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.NOTE_NO.value),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.NOTE_DATE.value),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.NOTE_TYPE.value),
+ "fieldname": GSTR1_DataField.TRANSACTION_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.NOTE_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_exp_headers(self):
+ return [
+ {
+ "label": _("Export Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.INVOICE_NUMBER.value),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_DATE.value),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.INVOICE_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.PORT_CODE.value),
+ "fieldname": GSTR1_DataField.SHIPPING_PORT_CODE.value,
+ },
+ {
+ "label": _(GovExcelField.SHIPPING_BILL_NO.value),
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.SHIPPING_BILL_DATE.value),
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_DATE.value,
+ "data_format": {"number_format": self.DATE_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_at_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {
+ "number_format": self.PERCENT_FORMAT,
+ },
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Gross Advance Received"),
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_DataField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_txpd_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _(GovExcelField.DIFF_PERCENTAGE.value),
+ "fieldname": GSTR1_DataField.DIFF_PERCENTAGE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Gross Advance Adjusted"),
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_DataField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_nil_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.DESCRIPTION.value),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _("Nil Rated Supplies"),
+ "fieldname": GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _("Exempted(other than nil rated/non GST supply)"),
+ "fieldname": GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _("Non-GST Supplies"),
+ "fieldname": GSTR1_DataField.NON_GST_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_hsn_headers(self):
+ return [
+ {
+ "label": _(GovExcelField.HSN_CODE.value),
+ "fieldname": GSTR1_DataField.HSN_CODE.value,
+ },
+ {
+ "label": _(GovExcelField.DESCRIPTION.value),
+ "fieldname": GSTR1_DataField.DESCRIPTION.value,
+ },
+ {
+ "label": _(GovExcelField.UOM.value),
+ "fieldname": GSTR1_DataField.UOM.value,
+ },
+ {
+ "label": _(GovExcelField.QUANTITY.value),
+ "fieldname": GSTR1_DataField.QUANTITY.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TOTAL_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.IGST.value),
+ "fieldname": GSTR1_DataField.IGST.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CGST.value),
+ "fieldname": GSTR1_DataField.CGST.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.SGST.value),
+ "fieldname": GSTR1_DataField.SGST.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _(GovExcelField.CESS.value),
+ "fieldname": GSTR1_DataField.CESS.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_doc_issue_headers(self):
+ return [
+ {
+ "label": _("Nature of Document"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _("Sr. No. From"),
+ "fieldname": GSTR1_DataField.FROM_SR.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Sr. No. To"),
+ "fieldname": GSTR1_DataField.TO_SR.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Total Number"),
+ "fieldname": GSTR1_DataField.TOTAL_COUNT.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Cancelled"),
+ "fieldname": GSTR1_DataField.CANCELLED_COUNT.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ ]
+
+
+class BooksExcel(DataProcessor):
+ AMOUNT_FORMAT = "#,##0.00"
+ DATE_FORMAT = "dd-mmm-yy"
+ PERCENT_FORMAT = "0.00"
+ DEFAULT_DATA_FORMAT = {"height": 15}
+
+ def __init__(self, company_gstin, month_or_quarter, year):
+ self.company_gstin = company_gstin
+ self.month_or_quarter = month_or_quarter
+ self.year = year
+
+ self.period = get_period(month_or_quarter, year)
+ gstr1_log = frappe.get_doc("GSTR-1 Log", f"{self.period}-{company_gstin}")
+
+ self.data = self.process_data(gstr1_log.load_data("books")["books"])
+
+ def process_data(self, data):
+ category_wise_data = super().process_data(data)
+
+ DOC_ITEM_FIELD_MAP = {
+ GSTR1_DataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GSTR1_DataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GSTR1_DataField.CGST.value: GSTR1_ItemField.CGST.value,
+ GSTR1_DataField.SGST.value: GSTR1_ItemField.SGST.value,
+ GSTR1_DataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ for category, category_data in category_wise_data.items():
+ # filter missing in books
+ category_wise_data[category] = [
+ doc
+ for doc in category_data
+ if doc.get("upload_status") != "Missing in Books"
+ ]
+
+ # copy doc value to item fields
+ if category != GovJsonKey.B2CS.value:
+ continue
+
+ for doc in category_wise_data[category]:
+ for doc_field, item_field in DOC_ITEM_FIELD_MAP.items():
+ doc[item_field] = doc.get(doc_field, 0)
+
+ return category_wise_data
+
+ def export_data(self):
+ excel = ExcelExporter()
+ excel.remove_sheet("Sheet")
+
+ excel.create_sheet(
+ sheet_name="invoices",
+ headers=self.get_document_headers(),
+ data=self.get_document_data(),
+ default_data_format=self.DEFAULT_DATA_FORMAT,
+ add_totals=False,
+ )
+
+ self.create_other_sheets(excel)
+ excel.export(get_file_name("Books", self.company_gstin, self.period))
+
+ def create_other_sheets(self, excel: ExcelExporter):
+ for category in ("NIL_EXEMPT", "HSN", "AT", "TXP", "DOC_ISSUE"):
+ data = self.data.get(GovJsonKey[category].value)
+
+ if not data:
+ continue
+
+ excel.create_sheet(
+ sheet_name=GovExcelSheetName[category].value,
+ headers=getattr(self, f"get_{category.lower()}_headers")(),
+ data=data,
+ default_data_format=self.DEFAULT_DATA_FORMAT,
+ add_totals=False,
+ )
+
+ def get_document_data(self):
+ taxable_inv_categories = [
+ GovJsonKey.B2B.value,
+ GovJsonKey.EXP.value,
+ GovJsonKey.B2CL.value,
+ GovJsonKey.CDNR.value,
+ GovJsonKey.CDNUR.value,
+ GovJsonKey.B2CS.value,
+ ]
+
+ category_data = []
+ for key, values in self.data.items():
+ if key not in taxable_inv_categories:
+ continue
+
+ category_data.extend(values)
+
+ return category_data
+
+ def get_document_headers(self):
+ return [
+ {
+ "label": _("Transaction Type"),
+ "fieldname": GSTR1_DataField.TRANSACTION_TYPE.value,
+ },
+ {
+ "label": _("Document Date"),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Document Number"),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Customer GSTIN"),
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Customer Name"),
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _("Document Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ },
+ {
+ "label": _(GovExcelField.SHIPPING_BILL_NO.value),
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _(GovExcelField.SHIPPING_BILL_DATE.value),
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_DATE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.PORT_CODE.value),
+ "fieldname": GSTR1_DataField.SHIPPING_PORT_CODE.value,
+ },
+ {
+ "label": _(GovExcelField.REVERSE_CHARGE.value),
+ "fieldname": GSTR1_DataField.REVERSE_CHARGE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Upload Status"),
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": _("Tax Rate"),
+ "fieldname": GSTR1_ItemField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "fieldname": GSTR1_ItemField.TAXABLE_VALUE.value,
+ "label": _("Taxable Value"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_ItemField.IGST.value,
+ "label": _("IGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_ItemField.CGST.value,
+ "label": _("CGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_ItemField.SGST.value,
+ "label": _("SGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_ItemField.CESS.value,
+ "label": _("CESS"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": _("Document Value"),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ },
+ ]
+
+ def get_at_headers(self):
+ return [
+ {
+ "label": _("Advance Date"),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Payment Entry Number"),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Customer"),
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": "Upload Status",
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ *self.get_amount_headers(),
+ ]
+
+ def get_txp_headers(self):
+ return [
+ {
+ "label": _("Adjustment Date"),
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Adjustment Entry Number"),
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": _("Customer"),
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _(GovExcelField.POS.value),
+ "fieldname": GSTR1_DataField.POS.value,
+ },
+ {
+ "label": "Upload Status",
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ *self.get_amount_headers(),
+ ]
+
+ def get_hsn_headers(self):
+ return [
+ {
+ "label": _("HSN Code"),
+ "fieldname": GSTR1_DataField.HSN_CODE.value,
+ },
+ {
+ "label": _(GovExcelField.DESCRIPTION.value),
+ "fieldname": GSTR1_DataField.DESCRIPTION.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": _("UOM"),
+ "fieldname": GSTR1_DataField.UOM.value,
+ },
+ {
+ "label": _(GovExcelField.TAX_RATE.value),
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "data_format": {"number_format": self.PERCENT_FORMAT},
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": "Upload Status",
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ {
+ "label": _(GovExcelField.QUANTITY.value),
+ "fieldname": GSTR1_DataField.QUANTITY.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _(GovExcelField.TOTAL_VALUE.value),
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ *self.get_amount_headers(),
+ ]
+
+ def get_doc_issue_headers(self):
+ return [
+ {
+ "label": _("Document Type"),
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": "Upload Status",
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ {
+ "label": _("Sr No From"),
+ "fieldname": GSTR1_DataField.FROM_SR.value,
+ },
+ {
+ "label": _("Sr No To"),
+ "fieldname": GSTR1_DataField.TO_SR.value,
+ },
+ {
+ "label": _("Total Count"),
+ "fieldname": GSTR1_DataField.TOTAL_COUNT.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Draft Count"),
+ "fieldname": GSTR1_DataField.DRAFT_COUNT.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": _("Cancelled Count"),
+ "fieldname": GSTR1_DataField.CANCELLED_COUNT.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ ]
+
+ def get_amount_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "label": _("Taxable Value"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.IGST.value,
+ "label": _("IGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.CGST.value,
+ "label": _("CGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.SGST.value,
+ "label": _("SGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.CESS.value,
+ "label": _("CESS"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+ def get_nil_exempt_headers(self):
+ return [
+ {
+ "label": "Transaction Type",
+ "fieldname": GSTR1_DataField.TRANSACTION_TYPE.value,
+ },
+ {
+ "label": "Documenrt Date",
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "label": "Document Number",
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "label": "Customer Name",
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": "Document Type",
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "label": "Upload Status",
+ "fieldname": GSTR1_DataField.UPLOAD_STATUS.value,
+ },
+ {
+ "label": "Nil Rated Supplies",
+ "fieldname": GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": "Exempted Supplies",
+ "fieldname": GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": "Non-GST Supplies",
+ "fieldname": GSTR1_DataField.NON_GST_AMOUNT.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "label": "Document Value",
+ "fieldname": GSTR1_DataField.DOC_VALUE.value,
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+
+
+class ReconcileExcel:
+ AMOUNT_FORMAT = "#,##0.00"
+ DATE_FORMAT = "dd-mmm-yy"
+
+ COLOR_PALLATE = frappe._dict(
+ {
+ "dark_gray": "d9d9d9",
+ "light_gray": "f2f2f2",
+ "dark_pink": "e6b9b8",
+ "light_pink": "f2dcdb",
+ "sky_blue": "c6d9f1",
+ "light_blue": "dce6f2",
+ "green": "d7e4bd",
+ "light_green": "ebf1de",
+ }
+ )
+
+ DEFAULT_HEADER_FORMAT = {"bg_color": COLOR_PALLATE.dark_gray}
+ DEFAULT_DATA_FORMAT = {"bg_color": COLOR_PALLATE.light_gray}
+
+ def __init__(self, company_gstin, month_or_quarter, year):
+ self.company_gstin = company_gstin
+ self.month_or_quarter = month_or_quarter
+ self.year = year
+
+ self.period = get_period(month_or_quarter, year)
+ gstr1_log = frappe.get_doc("GSTR-1 Log", f"{self.period}-{company_gstin}")
+
+ self.summary = gstr1_log.load_data("reconcile_summary")["reconcile_summary"]
+ data = gstr1_log.load_data("reconcile")["reconcile"]
+ self.data = get_category_wise_data(data)
+
+ def export_data(self):
+ excel = ExcelExporter()
+ excel.remove_sheet("Sheet")
+
+ excel.create_sheet(
+ sheet_name="reconcile summary",
+ headers=self.get_reconcile_summary_headers(),
+ data=self.get_reconcile_summary_data(),
+ default_data_format=self.DEFAULT_DATA_FORMAT,
+ default_header_format=self.DEFAULT_HEADER_FORMAT,
+ add_totals=False,
+ )
+
+ for category in (
+ "B2B",
+ "EXP",
+ "B2CL",
+ "B2CS",
+ "NIL_EXEMPT",
+ "CDNR",
+ "CDNUR",
+ "AT",
+ "TXP",
+ "HSN",
+ "DOC_ISSUE",
+ ):
+ self.create_sheet(excel, category)
+
+ excel.export(get_file_name("Reconcile", self.company_gstin, self.period))
+
+ def get_reconcile_summary_headers(self):
+ headers = [
+ {
+ "fieldname": GSTR1_DataField.DESCRIPTION.value,
+ "label": _(GovExcelField.DESCRIPTION.value),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.TAXABLE_VALUE.value,
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.IGST.value,
+ "label": _("IGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.CGST.value,
+ "label": _("CGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.SGST.value,
+ "label": _("SGST"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ {
+ "fieldname": GSTR1_DataField.CESS.value,
+ "label": _("CESS"),
+ "data_format": {"number_format": self.AMOUNT_FORMAT},
+ },
+ ]
+ return headers
+
+ def get_reconcile_summary_data(self):
+ excel_data = []
+ for row in self.summary:
+ if row["indent"] == 1:
+ continue
+ excel_data.append(row)
+
+ return excel_data
+
+ def create_sheet(self, excel: ExcelExporter, category):
+ data = self.get_data(category)
+ if not data:
+ return
+
+ category_key = GovJsonKey[category].value
+ merged_headers = getattr(
+ self,
+ f"get_merge_headers_for_{category_key}",
+ self.get_merge_headers,
+ )()
+
+ excel.create_sheet(
+ sheet_name=GovExcelSheetName[category].value,
+ merged_headers=merged_headers,
+ headers=getattr(self, f"get_{category_key}_headers")(),
+ data=data,
+ default_data_format=self.DEFAULT_DATA_FORMAT,
+ default_header_format=self.DEFAULT_HEADER_FORMAT,
+ add_totals=False,
+ )
+
+ def get_data(self, category):
+ data = self.data.get(GovJsonKey[category].value, [])
+ excel_data = []
+
+ for row in data:
+ row_dict = self.get_row_dict(row)
+ excel_data.append(row_dict)
+
+ return excel_data
+
+ def get_merge_headers(self):
+ return frappe._dict(
+ {
+ "Books": [
+ "books_" + GSTR1_DataField.POS.value,
+ "books_" + GSTR1_DataField.CESS.value,
+ ],
+ "GSTR-1": [
+ "gstr_1_" + GSTR1_DataField.POS.value,
+ "gstr_1_" + GSTR1_DataField.CESS.value,
+ ],
+ }
+ )
+
+ def get_merge_headers_for_exp(self):
+ return self.get_merge_headers_for_b2cs()
+
+ def get_merge_headers_for_b2cs(self):
+ return frappe._dict(
+ {
+ "Books": [
+ "books_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "books_" + GSTR1_DataField.CESS.value,
+ ],
+ "GSTR-1": [
+ "gstr_1_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "gstr_1_" + GSTR1_DataField.CESS.value,
+ ],
+ }
+ )
+
+ def get_merge_headers_for_nil(self):
+ return frappe._dict(
+ {
+ "Books": [
+ "books_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "books_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ ],
+ "GSTR-1": [
+ "gstr_1_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "gstr_1_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ ],
+ }
+ )
+
+ def get_merge_headers_for_doc_issue(self):
+ return frappe._dict(
+ {
+ "Books": [
+ "books_" + GSTR1_DataField.FROM_SR.value,
+ "books_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ ],
+ "GSTR-1": [
+ "gstr_1_" + GSTR1_DataField.FROM_SR.value,
+ "gstr_1_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ ],
+ }
+ )
+
+ def get_merge_headers_for_hsn(self):
+ return frappe._dict(
+ {
+ "Books": [
+ "books_" + GSTR1_DataField.QUANTITY.value,
+ "books_" + GSTR1_DataField.CESS.value,
+ ],
+ "GSTR-1": [
+ "gstr_1_" + GSTR1_DataField.QUANTITY.value,
+ "gstr_1_" + GSTR1_DataField.CESS.value,
+ ],
+ }
+ )
+
+ def get_merge_headers_for_at(self):
+ return self.get_merge_headers_for_b2cs()
+
+ def get_merge_headers_for_txpd(self):
+ return self.get_merge_headers_for_b2cs()
+
+ def get_b2b_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "label": _("Document Date"),
+ "header_format": {
+ "width": ExcelWidth.XS.value,
+ "number_format": self.DATE_FORMAT,
+ },
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "label": _("Document No"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "label": _("Customer GSTIN"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "label": _("Customer Name"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ *self.get_common_compare_columns(),
+ ]
+
+ def get_b2cl_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "label": _("Document Date"),
+ "header_format": {
+ "width": ExcelWidth.XS.value,
+ "number_format": self.DATE_FORMAT,
+ },
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "label": _("Document No"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "label": _("Customer Name"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ {
+ "fieldname": "books_" + GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ "compare_with": "gstr_1_" + GSTR1_DataField.POS.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ },
+ },
+ *self.get_amount_field_columns(for_books=True, only_igst=True),
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ "compare_with": "books_" + GSTR1_DataField.POS.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ },
+ },
+ *self.get_amount_field_columns(for_books=False, only_igst=True),
+ ]
+
+ def get_exp_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "label": _("Document Date"),
+ "header_format": {
+ "width": ExcelWidth.XS.value,
+ "number_format": self.DATE_FORMAT,
+ },
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "label": _("Document No"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "label": _("Customer Name"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_NUMBER.value,
+ "label": _(GovExcelField.SHIPPING_BILL_NO.value),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.SHIPPING_BILL_DATE.value,
+ "label": _(GovExcelField.SHIPPING_BILL_DATE.value),
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.SHIPPING_PORT_CODE.value,
+ "label": _("Shipping Port Code"),
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ *self.get_amount_field_columns(for_books=True, only_igst=True),
+ *self.get_amount_field_columns(for_books=False, only_igst=True),
+ ]
+
+ def get_b2cs_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ },
+ {
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "label": _("Tax Rate"),
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ *self.get_amount_field_columns(for_books=True),
+ *self.get_amount_field_columns(for_books=False),
+ ]
+
+ def get_nil_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ {
+ "fieldname": "books_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "label": _("Nil-Rated Supplies"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.green},
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "label": _("Exempted Supplies"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.green},
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.NON_GST_AMOUNT.value,
+ "label": _("Non-GST Supplies"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.NON_GST_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.green},
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "compare_with": "gstr_1_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.green},
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "label": _("Nil-Rated Supplies"),
+ "compare_with": "books_" + GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.sky_blue},
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "label": _("Exempted Supplies"),
+ "compare_with": "books_" + GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.sky_blue},
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.NON_GST_AMOUNT.value,
+ "label": _("Non-GST Supplies"),
+ "compare_with": "books_" + GSTR1_DataField.NON_GST_AMOUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.sky_blue},
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "compare_with": "books_" + GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {"bg_color": self.COLOR_PALLATE.sky_blue},
+ },
+ ]
+
+ def get_cdnr_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "label": _("Document Date"),
+ "header_format": {
+ "width": ExcelWidth.XS.value,
+ "number_format": self.DATE_FORMAT,
+ },
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "label": _("Document No"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "label": _("Customer GSTIN"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "label": _("Customer Name"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ *self.get_common_compare_columns(),
+ ]
+
+ def get_cdnur_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_DATE.value,
+ "label": _("Document Date"),
+ "header_format": {
+ "width": ExcelWidth.XS.value,
+ "number_format": self.DATE_FORMAT,
+ },
+ },
+ {
+ "fieldname": GSTR1_DataField.DOC_NUMBER.value,
+ "label": _("Document No"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_GSTIN.value,
+ "label": _("Customer GSTIN"),
+ "header_format": {"width": ExcelWidth.SM.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.CUST_NAME.value,
+ "label": _("Customer Name"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ {
+ "fieldname": "books_" + GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ "compare_with": "gstr_1_" + GSTR1_DataField.POS.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ },
+ },
+ *self.get_amount_field_columns(for_books=True, only_igst=True),
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ "compare_with": "books_" + GSTR1_DataField.POS.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ },
+ },
+ *self.get_amount_field_columns(for_books=False, only_igst=True),
+ ]
+
+ def get_doc_issue_headers(self):
+ headers = [
+ {
+ "fieldname": GSTR1_DataField.DOC_TYPE.value,
+ "label": _("Document Type"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "fieldname": "match_status",
+ "label": _("Match Status"),
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.FROM_SR.value,
+ "label": _("SR No From"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.FROM_SR.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ },
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.TO_SR.value,
+ "label": _("SR No To"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.TO_SR.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ },
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.TOTAL_COUNT.value,
+ "label": _("Total Count"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.TOTAL_COUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ {
+ "fieldname": "books_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ "label": _("Cancelled Count"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.FROM_SR.value,
+ "label": _("Sr No From"),
+ "compare_with": "books_" + GSTR1_DataField.FROM_SR.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ },
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.TO_SR.value,
+ "label": _("Sr No To"),
+ "compare_with": "books_" + GSTR1_DataField.TO_SR.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ },
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.TOTAL_COUNT.value,
+ "label": _("Total Count"),
+ "compare_with": "books_" + GSTR1_DataField.TOTAL_COUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ "label": _("Cancelled Count"),
+ "compare_with": "books_" + GSTR1_DataField.CANCELLED_COUNT.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ ]
+
+ return headers
+
+ def get_hsn_headers(self):
+ headers = [
+ {"fieldname": GSTR1_DataField.HSN_CODE.value, "label": _("HSN Code")},
+ {
+ "fieldname": GSTR1_DataField.DESCRIPTION.value,
+ "label": _("Description"),
+ "header_format": {"width": ExcelWidth.XXL.value},
+ },
+ {
+ "fieldname": GSTR1_DataField.UOM.value,
+ "label": _(GovExcelField.UOM.value),
+ },
+ {
+ "fieldname": GSTR1_DataField.TAX_RATE.value,
+ "label": _(GovExcelField.TAX_RATE.value),
+ "header_format": {"width": ExcelWidth.XS.value},
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ {
+ "fieldname": "books_" + GSTR1_DataField.QUANTITY.value,
+ "label": _("Quantity"),
+ "compare_with": "gstr_1_" + GSTR1_DataField.QUANTITY.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.green,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ *self.get_amount_field_columns(for_books=True),
+ {
+ "fieldname": "gstr_1_" + GSTR1_DataField.QUANTITY.value,
+ "label": _("Quantity"),
+ "compare_with": "books_" + GSTR1_DataField.QUANTITY.value,
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.sky_blue,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ *self.get_amount_field_columns(for_books=False),
+ ]
+
+ return headers
+
+ def get_at_headers(self):
+ return [
+ {
+ "fieldname": GSTR1_DataField.POS.value,
+ "label": _("POS"),
+ },
+ {"fieldname": "match_status", "label": _("Match Status")},
+ *self.get_tax_difference_columns(),
+ *self.get_amount_field_columns(for_books=True),
+ *self.get_amount_field_columns(for_books=False),
+ ]
+
+ def get_txpd_headers(self):
+ return self.get_at_headers()
+
+ def get_row_dict(self, row: dict) -> dict:
+ books = row.pop("books", {})
+ gstr_1 = row.pop("gov", {})
+
+ row.update({"books_" + key: value for key, value in books.items()})
+ row.update({"gstr_1_" + key: value for key, value in gstr_1.items()})
+
+ doc_date = row.get(GSTR1_DataField.DOC_DATE.value)
+ row[GSTR1_DataField.DOC_DATE.value] = getdate(doc_date) if doc_date else ""
+
+ self.update_differences(row)
+
+ return row
+
+ def update_differences(self, row_dict):
+ taxable_value_key = GSTR1_DataField.TAXABLE_VALUE.value
+ igst_key = GSTR1_DataField.IGST.value
+ cgst_key = GSTR1_DataField.CGST.value
+ sgst_key = GSTR1_DataField.SGST.value
+ cess_key = GSTR1_DataField.CESS.value
+
+ row_dict["taxable_value_difference"] = (
+ row_dict.get("books_" + taxable_value_key, 0)
+ ) - (row_dict.get("gstr_1_" + taxable_value_key, 0))
+
+ row_dict["tax_difference"] = 0
+ for tax_key in [igst_key, cgst_key, sgst_key, cess_key]:
+ row_dict["tax_difference"] += row_dict.get("books_" + tax_key, 0) - (
+ row_dict.get("gstr_1_" + tax_key, 0)
+ )
+
+ # COMMON COLUMNS
+
+ def get_tax_difference_columns(self):
+ return [
+ {
+ "fieldname": "taxable_value_difference",
+ "label": _("Taxable Value Difference"),
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_pink,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.dark_pink,
+ },
+ },
+ {
+ "fieldname": "tax_difference",
+ "label": _("Tax Difference"),
+ "data_format": {
+ "bg_color": self.COLOR_PALLATE.light_pink,
+ "number_format": self.AMOUNT_FORMAT,
+ },
+ "header_format": {
+ "bg_color": self.COLOR_PALLATE.dark_pink,
+ },
+ },
+ ]
+
+ def get_common_compare_columns(self):
+ return [
+ *self.get_tax_details_columns(for_books=True),
+ *self.get_amount_field_columns(for_books=True),
+ *self.get_tax_details_columns(for_books=False),
+ *self.get_amount_field_columns(for_books=False),
+ ]
+
+ def get_amount_field_columns(self, for_books=True, only_igst=False):
+ if for_books:
+ field_prefix = "books_"
+ compare_with = "gstr_1_"
+ data_format = {
+ "bg_color": self.COLOR_PALLATE.light_green,
+ "number_format": self.AMOUNT_FORMAT,
+ }
+ header_format = {"bg_color": self.COLOR_PALLATE.green}
+
+ else:
+ field_prefix = "gstr_1_"
+ compare_with = "books_"
+ data_format = {
+ "bg_color": self.COLOR_PALLATE.light_blue,
+ "number_format": self.AMOUNT_FORMAT,
+ }
+ header_format = {"bg_color": self.COLOR_PALLATE.sky_blue}
+
+ def get_cgst_sgst_columns():
+ if only_igst:
+ return []
+
+ return [
+ {
+ "fieldname": field_prefix + GSTR1_DataField.CGST.value,
+ "label": _("CGST"),
+ "compare_with": compare_with + GSTR1_DataField.CGST.value,
+ "data_format": data_format,
+ "header_format": header_format,
+ },
+ {
+ "fieldname": field_prefix + GSTR1_DataField.SGST.value,
+ "label": _("SGST"),
+ "compare_with": compare_with + GSTR1_DataField.SGST.value,
+ "data_format": data_format,
+ "header_format": header_format,
+ },
+ ]
+
+ return [
+ {
+ "fieldname": field_prefix + GSTR1_DataField.TAXABLE_VALUE.value,
+ "label": _(GovExcelField.TAXABLE_VALUE.value),
+ "compare_with": compare_with + GSTR1_DataField.TAXABLE_VALUE.value,
+ "data_format": data_format,
+ "header_format": header_format,
+ },
+ {
+ "fieldname": field_prefix + GSTR1_DataField.IGST.value,
+ "label": _("IGST"),
+ "compare_with": compare_with + GSTR1_DataField.IGST.value,
+ "data_format": data_format,
+ "header_format": header_format,
+ },
+ *get_cgst_sgst_columns(),
+ {
+ "fieldname": field_prefix + GSTR1_DataField.CESS.value,
+ "label": _("CESS"),
+ "compare_with": compare_with + GSTR1_DataField.CESS.value,
+ "data_format": data_format,
+ "header_format": header_format,
+ },
+ ]
+
+ def get_tax_details_columns(self, for_books=True):
+ if for_books:
+ field_prefix = "books_"
+ compare_with = "gstr_1_"
+ data_color = self.COLOR_PALLATE.light_green
+ header_color = self.COLOR_PALLATE.green
+
+ else:
+ field_prefix = "gstr_1_"
+ compare_with = "books_"
+ data_color = self.COLOR_PALLATE.light_blue
+ header_color = self.COLOR_PALLATE.sky_blue
+
+ return [
+ {
+ "fieldname": field_prefix + GSTR1_DataField.POS.value,
+ "label": _(GovExcelField.POS.value),
+ "compare_with": compare_with + GSTR1_DataField.POS.value,
+ "data_format": {"bg_color": data_color},
+ "header_format": {"bg_color": header_color},
+ },
+ {
+ "fieldname": field_prefix + GSTR1_DataField.REVERSE_CHARGE.value,
+ "label": _(GovExcelField.REVERSE_CHARGE.value),
+ "compare_with": compare_with + GSTR1_DataField.REVERSE_CHARGE.value,
+ "data_format": {"bg_color": data_color},
+ "header_format": {
+ "bg_color": header_color,
+ "width": ExcelWidth.XS.value,
+ },
+ },
+ ]
+
+
+@frappe.whitelist()
+def download_filed_as_excel(company_gstin, month_or_quarter, year):
+ frappe.has_permission("GSTR-1 Beta", "export", throw=True)
+ GovExcel().generate(company_gstin, get_period(month_or_quarter, year))
+
+
+@frappe.whitelist()
+def download_books_as_excel(company_gstin, month_or_quarter, year):
+ frappe.has_permission("GSTR-1 Beta", "export", throw=True)
+
+ books_excel = BooksExcel(company_gstin, month_or_quarter, year)
+ books_excel.export_data()
+
+
+@frappe.whitelist()
+def download_reconcile_as_excel(company_gstin, month_or_quarter, year):
+ frappe.has_permission("GSTR-1 Beta", "export", throw=True)
+
+ reconcile_excel = ReconcileExcel(company_gstin, month_or_quarter, year)
+ reconcile_excel.export_data()
+
+
+@frappe.whitelist()
+def download_gstr_1_json(
+ company_gstin,
+ year,
+ month_or_quarter,
+ include_uploaded=False,
+ delete_missing=False,
+):
+ frappe.has_permission("GSTR-1 Beta", "export", throw=True)
+
+ if isinstance(include_uploaded, str):
+ include_uploaded = json.loads(include_uploaded)
+
+ if isinstance(delete_missing, str):
+ delete_missing = json.loads(delete_missing)
+
+ period = get_period(month_or_quarter, year)
+ gstr1_log = frappe.get_doc("GSTR-1 Log", f"{period}-{company_gstin}")
+
+ data = gstr1_log.get_json_for("books")
+ data = data.update(data.pop("aggregate_data", {}))
+
+ for subcategory, subcategory_data in data.items():
+ if subcategory in {
+ GSTR1_SubCategory.NIL_EXEMPT.value,
+ GSTR1_SubCategory.HSN.value,
+ }:
+ continue
+
+ discard_invoices = []
+
+ if isinstance(subcategory_data, str):
+ continue
+
+ for key, row in subcategory_data.items():
+ if isinstance(row, list):
+ row = row[0]
+
+ if not row.get("upload_status"):
+ continue
+
+ if row.get("upload_status") == "Uploaded" and not include_uploaded:
+ discard_invoices.append(key)
+ continue
+
+ if row.get("upload_status") == "Missing in Books":
+ if delete_missing:
+ row["flag"] = "D"
+ else:
+ discard_invoices.append(key)
+
+ for key in discard_invoices:
+ subcategory_data.pop(key)
+
+ gstr1_log.normalize_data(data)
+
+ return {
+ "data": {
+ "gstin": company_gstin,
+ "fp": period,
+ **convert_to_gov_data_format(data, company_gstin),
+ },
+ "filename": f"GSTR-1-Gov-{company_gstin}-{period}.json",
+ }
+
+
+def get_file_name(field_name, gstin, period):
+ return f"GSTR-1-{field_name}-{gstin}-{period}"
diff --git a/india_compliance/gst_india/doctype/gstr_1_beta/test_gstr_1_beta.py b/india_compliance/gst_india/doctype/gstr_1_beta/test_gstr_1_beta.py
new file mode 100644
index 0000000000..67158cb746
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_beta/test_gstr_1_beta.py
@@ -0,0 +1,9 @@
+# Copyright (c) 2024, Resilient Tech and Contributors
+# See license.txt
+
+# import frappe
+from frappe.tests.utils import FrappeTestCase
+
+
+class TestGSTR1Beta(FrappeTestCase):
+ pass
diff --git a/india_compliance/gst_india/doctype/gstr_1_log/__init__.py b/india_compliance/gst_india/doctype/gstr_1_log/__init__.py
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.js b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.js
new file mode 100644
index 0000000000..30af20e146
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.js
@@ -0,0 +1,33 @@
+// Copyright (c) 2024, Resilient Tech and contributors
+// For license information, please see license.txt
+
+
+frappe.ui.form.on("GSTR-1 Log", {
+ refresh(frm) {
+ const [month_or_quarter, year] = india_compliance.get_month_year_from_period(frm.doc.return_period);
+
+ frm.add_custom_button(__("View GSTR-1"), () => {
+ frappe.set_route("Form", "GSTR-1 Beta")
+
+ // after form loads
+ new Promise((resolve) => {
+ const interval = setInterval(() => {
+ if (cur_frm.doctype === "GSTR-1 Beta" && cur_frm.__setup_complete) {
+ clearInterval(interval);
+ resolve();
+ }
+ }, 100);
+
+ }).then(async () => {
+ await cur_frm.set_value({
+ "company": frm.doc.company,
+ "company_gstin": frm.doc.gstin,
+ "year": year,
+ "month_or_quarter": month_or_quarter,
+ });
+ cur_frm.save();
+
+ });
+ });
+ },
+});
diff --git a/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.json b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.json
new file mode 100644
index 0000000000..3955a768e5
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.json
@@ -0,0 +1,238 @@
+{
+ "actions": [],
+ "autoname": "format:{return_period}-{gstin}",
+ "creation": "2024-03-11 11:59:06.887429",
+ "description": "Keeps the log of GSTR-1 filed by GSTIN and Period",
+ "doctype": "DocType",
+ "engine": "InnoDB",
+ "field_order": [
+ "filing_summary_section",
+ "filing_status",
+ "return_period",
+ "company",
+ "column_break_sqwh",
+ "filing_date",
+ "acknowledgement_number",
+ "gstin",
+ "generation_status",
+ "section_break_oisv",
+ "unfiled",
+ "filed",
+ "column_break_hxfu",
+ "unfiled_summary",
+ "filed_summary",
+ "computed_data_section",
+ "is_latest_data",
+ "section_break_emlz",
+ "books",
+ "column_break_ehcm",
+ "books_summary",
+ "reconciled_data_section",
+ "reconcile",
+ "column_break_ndup",
+ "reconcile_summary"
+ ],
+ "fields": [
+ {
+ "fieldname": "filing_summary_section",
+ "fieldtype": "Section Break",
+ "label": "Filing Summary"
+ },
+ {
+ "fieldname": "filing_status",
+ "fieldtype": "Data",
+ "label": "Filing Status",
+ "read_only": 1
+ },
+ {
+ "fieldname": "return_period",
+ "fieldtype": "Data",
+ "label": "Return Period",
+ "read_only": 1
+ },
+ {
+ "fieldname": "column_break_sqwh",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "filing_date",
+ "fieldtype": "Date",
+ "label": "Filing Date",
+ "read_only": 1
+ },
+ {
+ "fieldname": "acknowledgement_number",
+ "fieldtype": "Data",
+ "label": "Acknowledgement Number",
+ "read_only": 1
+ },
+ {
+ "fieldname": "section_break_oisv",
+ "fieldtype": "Section Break",
+ "label": "Government Data"
+ },
+ {
+ "fieldname": "gstin",
+ "fieldtype": "Data",
+ "label": "GSTIN",
+ "read_only": 1
+ },
+ {
+ "fieldname": "column_break_hxfu",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "computed_data_section",
+ "fieldtype": "Section Break",
+ "hide_border": 1,
+ "label": "Computed Data"
+ },
+ {
+ "fieldname": "reconciled_data_section",
+ "fieldtype": "Section Break",
+ "label": "Reconciled Data"
+ },
+ {
+ "fieldname": "column_break_ndup",
+ "fieldtype": "Column Break"
+ },
+ {
+ "default": "0",
+ "fieldname": "is_latest_data",
+ "fieldtype": "Check",
+ "label": "Is Latest Data",
+ "read_only": 1
+ },
+ {
+ "fieldname": "generation_status",
+ "fieldtype": "Select",
+ "hidden": 1,
+ "label": "Generation Status",
+ "options": "\nIn Progress\nGenerated\nFailed",
+ "read_only": 1
+ },
+ {
+ "fieldname": "filed",
+ "fieldtype": "Attach",
+ "label": "Filed Data",
+ "read_only": 1
+ },
+ {
+ "fieldname": "filed_summary",
+ "fieldtype": "Attach",
+ "label": "Filed Summary",
+ "read_only": 1
+ },
+ {
+ "fieldname": "books",
+ "fieldtype": "Attach",
+ "label": "Books Data",
+ "read_only": 1
+ },
+ {
+ "fieldname": "books_summary",
+ "fieldtype": "Attach",
+ "label": "Books Summary",
+ "read_only": 1
+ },
+ {
+ "fieldname": "reconcile",
+ "fieldtype": "Attach",
+ "label": "Reconcile GSTR-1",
+ "read_only": 1
+ },
+ {
+ "fieldname": "reconcile_summary",
+ "fieldtype": "Attach",
+ "label": "Reconcile Summary",
+ "read_only": 1
+ },
+ {
+ "fieldname": "unfiled",
+ "fieldtype": "Attach",
+ "label": "Unfiled Invoices",
+ "read_only": 1
+ },
+ {
+ "fieldname": "unfiled_summary",
+ "fieldtype": "Attach",
+ "label": "Unfiled Summary",
+ "read_only": 1
+ },
+ {
+ "fieldname": "section_break_emlz",
+ "fieldtype": "Section Break"
+ },
+ {
+ "fieldname": "column_break_ehcm",
+ "fieldtype": "Column Break"
+ },
+ {
+ "fieldname": "company",
+ "fieldtype": "Link",
+ "label": "Company",
+ "options": "Company",
+ "read_only": 1
+ }
+ ],
+ "in_create": 1,
+ "index_web_pages_for_search": 1,
+ "links": [],
+ "modified": "2024-05-28 19:49:55.513493",
+ "modified_by": "Administrator",
+ "module": "GST India",
+ "name": "GSTR-1 Log",
+ "naming_rule": "Expression",
+ "owner": "Administrator",
+ "permissions": [
+ {
+ "create": 1,
+ "delete": 1,
+ "email": 1,
+ "export": 1,
+ "print": 1,
+ "read": 1,
+ "report": 1,
+ "role": "System Manager",
+ "share": 1,
+ "write": 1
+ },
+ {
+ "create": 1,
+ "delete": 1,
+ "email": 1,
+ "export": 1,
+ "print": 1,
+ "read": 1,
+ "report": 1,
+ "role": "Accounts Manager",
+ "share": 1,
+ "write": 1
+ },
+ {
+ "create": 1,
+ "email": 1,
+ "export": 1,
+ "print": 1,
+ "read": 1,
+ "report": 1,
+ "role": "Accounts User",
+ "share": 1,
+ "write": 1
+ },
+ {
+ "create": 1,
+ "email": 1,
+ "export": 1,
+ "print": 1,
+ "read": 1,
+ "report": 1,
+ "role": "Auditor",
+ "share": 1,
+ "write": 1
+ }
+ ],
+ "sort_field": "modified",
+ "sort_order": "DESC",
+ "states": []
+}
\ No newline at end of file
diff --git a/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.py b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.py
new file mode 100644
index 0000000000..069dfb83f3
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_log/gstr_1_log.py
@@ -0,0 +1,916 @@
+# Copyright (c) 2024, Resilient Tech and contributors
+# For license information, please see license.txt
+import gzip
+import itertools
+from datetime import datetime
+
+import frappe
+from frappe import unscrub
+from frappe.model.document import Document
+from frappe.utils import flt, get_datetime, get_datetime_str, get_last_day, getdate
+
+from india_compliance.gst_india.utils import is_production_api_enabled
+from india_compliance.gst_india.utils.gstr_1 import GSTR1_SubCategory
+from india_compliance.gst_india.utils.gstr_1.__init__ import (
+ CATEGORY_SUB_CATEGORY_MAPPING,
+ SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAX,
+ SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE,
+ GSTR1_DataField,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_download import (
+ download_gstr1_json_data,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_json_map import (
+ GSTR1BooksData,
+ summarize_retsum_data,
+)
+from india_compliance.gst_india.utils.gstr_utils import request_otp
+
+
+class SummarizeGSTR1:
+ AMOUNT_FIELDS = {
+ "total_taxable_value": 0,
+ "total_igst_amount": 0,
+ "total_cgst_amount": 0,
+ "total_sgst_amount": 0,
+ "total_cess_amount": 0,
+ }
+
+ def get_summarized_data(self, data, is_filed=False):
+ """
+ Helper function to summarize data for each sub-category
+ """
+ if is_filed and data.get("summary"):
+ return summarize_retsum_data(data.get("summary"))
+
+ subcategory_summary = self.get_subcategory_summary(data)
+
+ return self.get_overall_summary(subcategory_summary)
+
+ def get_overall_summary(self, subcategory_summary):
+ """
+ Summarize data for each category with subcategories
+
+ Steps:
+ 1. Init Category row
+ 2. Summarize category by adding subcategory rows
+ 3. Remove category row if no records
+ 4. Round Values
+ """
+ cateogory_summary = []
+ for category, sub_categories in CATEGORY_SUB_CATEGORY_MAPPING.items():
+ # Init category row
+ category = category.value
+ summary_row = {
+ "description": category,
+ "no_of_records": 0,
+ "indent": 0,
+ **self.AMOUNT_FIELDS,
+ }
+
+ cateogory_summary.append(summary_row)
+ remove_category_row = True
+
+ for subcategory in sub_categories:
+ # update category row
+ subcategory = subcategory.value
+ if subcategory not in subcategory_summary:
+ continue
+
+ subcategory_row = subcategory_summary[subcategory]
+ summary_row["no_of_records"] += subcategory_row["no_of_records"] or 0
+
+ for key in self.AMOUNT_FIELDS:
+ summary_row[key] += subcategory_row[key]
+
+ # add subcategory row
+ cateogory_summary.append(subcategory_row)
+ remove_category_row = False
+
+ if not summary_row["no_of_records"]:
+ summary_row["no_of_records"] = ""
+
+ if remove_category_row:
+ cateogory_summary.remove(summary_row)
+
+ # Round Values
+ for row in cateogory_summary:
+ for key, value in row.items():
+ if isinstance(value, (int, float)):
+ row[key] = flt(value, 2)
+ return cateogory_summary
+
+ def get_subcategory_summary(self, data):
+ """
+ Summarize invoices for each subcategory
+
+ Steps:
+ 1. Init subcategory row
+ 2. Summarize subcategory by adding invoice rows
+ 3. Update no_of_records / count for each subcategory
+ """
+ subcategory_summary = {}
+
+ for subcategory in GSTR1_SubCategory:
+ subcategory = subcategory.value
+ if subcategory not in data:
+ continue
+
+ summary_row = subcategory_summary.setdefault(
+ subcategory, self.default_subcategory_summary(subcategory)
+ )
+
+ _data = data[subcategory]
+ for row in _data:
+ if row.get("upload_status") == "Missing in Books":
+ continue
+
+ for key in self.AMOUNT_FIELDS:
+ summary_row[key] += row.get(key, 0)
+
+ if doc_num := row.get("document_number"):
+ summary_row["unique_records"].add(doc_num)
+
+ elif subcategory == GSTR1_SubCategory.DOC_ISSUE.value:
+ self.count_doc_issue_summary(summary_row, row)
+
+ elif subcategory == GSTR1_SubCategory.HSN.value:
+ self.count_hsn_summary(summary_row)
+
+ for subcategory in subcategory_summary.keys():
+ summary_row = subcategory_summary[subcategory]
+ count = len(summary_row["unique_records"])
+ if count:
+ summary_row["no_of_records"] = count
+
+ summary_row.pop("unique_records")
+
+ return subcategory_summary
+
+ def default_subcategory_summary(self, subcategory):
+ """
+ Considered in total taxable value:
+ Subcategories for which taxable values and counts are considered in front-end
+
+ Considered in total tax:
+ Subcategories for which tax values are considered in front-end
+
+ Indent:
+ 0: Category
+ 1: Subcategory
+ """
+ return {
+ "description": subcategory,
+ "no_of_records": 0,
+ "indent": 1,
+ "consider_in_total_taxable_value": (
+ False
+ if subcategory in SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE
+ else True
+ ),
+ "consider_in_total_tax": (
+ False
+ if subcategory in SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAX
+ else True
+ ),
+ "unique_records": set(),
+ **self.AMOUNT_FIELDS,
+ }
+
+ @staticmethod
+ def count_doc_issue_summary(summary_row, data_row):
+ summary_row["no_of_records"] += data_row.get(
+ GSTR1_DataField.TOTAL_COUNT.value, 0
+ ) - data_row.get(GSTR1_DataField.CANCELLED_COUNT.value, 0)
+
+ @staticmethod
+ def count_hsn_summary(summary_row):
+ summary_row["no_of_records"] += 1
+
+
+class ReconcileGSTR1:
+ IGNORED_FIELDS = {GSTR1_DataField.TAX_RATE.value, GSTR1_DataField.DOC_VALUE.value}
+ UNREQUIRED_KEYS = {
+ GSTR1_DataField.TRANSACTION_TYPE.value,
+ GSTR1_DataField.DOC_NUMBER.value,
+ GSTR1_DataField.DOC_DATE.value,
+ GSTR1_DataField.CUST_GSTIN.value,
+ GSTR1_DataField.CUST_NAME.value,
+ GSTR1_DataField.REVERSE_CHARGE.value,
+ }
+
+ def get_reconcile_gstr1_data(self, gov_data, books_data):
+ """
+ This function reconciles the data between Books and Gov Data
+
+ Steps:
+ 1. If already reconciled, return the reconciled data
+ 2. Update Upload Status for Books Data (if return is not filed)
+ 3. Reconcile for each subcategory
+ - For each row in Books Data, compare with Gov Data
+ - For each row in Gov Data (if not in Books Data)
+ """
+ if self.is_latest_data and self.reconcile:
+ reconcile_data = self.get_json_for("reconcile")
+
+ if reconcile_data:
+ return reconcile_data
+
+ reconciled_data = {}
+ if self.filing_status == "Filed":
+ update_books_match = False
+ else:
+ update_books_match = True
+
+ for subcategory in GSTR1_SubCategory:
+ subcategory = subcategory.value
+ books_subdata = books_data.get(subcategory) or {}
+ gov_subdata = gov_data.get(subcategory) or {}
+
+ if not books_subdata and not gov_subdata:
+ continue
+
+ is_list = False # Object Type for the subdata_value
+
+ reconcile_subdata = {}
+
+ # Books vs Gov
+ for key, books_value in books_subdata.items():
+ if not reconcile_subdata:
+ is_list = isinstance(books_value, list)
+
+ gov_value = gov_subdata.get(key)
+
+ reconcile_row = self.get_reconciled_row(books_value, gov_value)
+
+ if reconcile_row:
+ reconcile_subdata[key] = reconcile_row
+
+ if not update_books_match:
+ continue
+
+ books_values = books_value if is_list else [books_value]
+
+ # Update each row in Books Data
+ for row in books_values:
+ if row.get("upload_status") == "Missing in Books":
+ continue
+
+ if not gov_value:
+ row["upload_status"] = "Not Uploaded"
+ continue
+
+ if reconcile_row:
+ row["upload_status"] = "Mismatch"
+ else:
+ row["upload_status"] = "Uploaded"
+
+ # In Gov but not in Books
+ for key, gov_value in gov_subdata.items():
+ if key in books_subdata:
+ continue
+
+ if not reconcile_subdata:
+ is_list = isinstance(gov_value, list)
+
+ reconcile_subdata[key] = self.get_reconciled_row(None, gov_value)
+
+ if not update_books_match:
+ continue
+
+ books_empty_row = self.get_empty_row(
+ gov_value[0] if is_list else gov_value
+ )
+ books_empty_row["upload_status"] = "Missing in Books"
+
+ books_subdata[key] = [books_empty_row] if is_list else books_empty_row
+
+ if update_books_match and not books_data.get(subcategory):
+ books_data[subcategory] = books_subdata
+
+ if reconcile_subdata:
+ reconciled_data[subcategory] = reconcile_subdata
+
+ if update_books_match:
+ self.update_json_for("books", books_data)
+
+ self.update_json_for("reconcile", reconciled_data)
+
+ return reconciled_data
+
+ @staticmethod
+ def get_reconciled_row(books_row, gov_row):
+ """
+ Compare books_row with gov_row and return the difference
+
+ Args:
+ books_row (dict|list): Books Row Data
+ gov_row (dict|list): Gov Row Data
+
+ Returns:
+ dict|list: Reconciled Row Data
+
+ Steps:
+ 1. Get Empty Row with all values as 0
+ 2. Prefer Gov Row if available to compute empty row
+ 3. Compute comparable Gov and Books Row
+ 4. Compare the rows
+ 5. Compute match status and differences
+ 6. Return the reconciled row only if there are differences
+ """
+ is_list = isinstance(gov_row if gov_row else books_row, list)
+
+ # Get Empty Row
+ if is_list:
+ reconcile_row = ReconcileGSTR1.get_empty_row(
+ gov_row[0] if gov_row else books_row[0], ReconcileGSTR1.UNREQUIRED_KEYS
+ )
+ gov_row = gov_row[0] if gov_row else {}
+ books_row = (
+ AggregateInvoices.get_aggregate_invoices(books_row) if books_row else {}
+ )
+
+ else:
+ reconcile_row = ReconcileGSTR1.get_empty_row(gov_row or books_row)
+ gov_row = gov_row or {}
+ books_row = books_row or {}
+
+ # Default Status
+ reconcile_row["match_status"] = "Matched"
+ reconcile_row["differences"] = []
+
+ if not gov_row:
+ reconcile_row["match_status"] = "Missing in GSTR-1"
+
+ if not books_row:
+ reconcile_row["match_status"] = "Missing in Books"
+
+ # Compute Differences
+ for key, value in reconcile_row.items():
+ if (
+ isinstance(value, (int, float))
+ and key not in AggregateInvoices.IGNORED_FIELDS
+ ):
+ reconcile_row[key] = flt(
+ (books_row.get(key) or 0) - (gov_row.get(key) or 0), 2
+ )
+ has_different_value = reconcile_row[key] != 0
+
+ elif key in ("customer_gstin", "place_of_supply"):
+ has_different_value = books_row.get(key) != gov_row.get(key)
+
+ else:
+ continue
+
+ if not has_different_value:
+ continue
+
+ if "Missing" not in reconcile_row["match_status"]:
+ reconcile_row["match_status"] = "Mismatch"
+ reconcile_row["differences"].append(unscrub(key))
+
+ reconcile_row["differences"] = ", ".join(reconcile_row["differences"])
+
+ # Return
+ if reconcile_row["match_status"] == "Matched":
+ return
+
+ reconcile_row["books"] = books_row
+ reconcile_row["gov"] = gov_row
+
+ if is_list:
+ return [reconcile_row]
+
+ return reconcile_row
+
+ @staticmethod
+ def get_empty_row(row: dict, unrequired_keys=None):
+ """
+ Row with all values as 0
+ """
+ empty_row = row.copy()
+
+ for key, value in empty_row.items():
+ if key in AggregateInvoices.IGNORED_FIELDS:
+ continue
+
+ if unrequired_keys and key in unrequired_keys:
+ empty_row[key] = None
+ continue
+
+ if isinstance(value, (int, float)):
+ empty_row[key] = 0
+
+ if key == "items":
+ empty_row[key] = [{}]
+
+ return empty_row
+
+
+class AggregateInvoices:
+ IGNORED_FIELDS = {GSTR1_DataField.TAX_RATE.value, GSTR1_DataField.DOC_VALUE.value}
+
+ @staticmethod
+ def get_aggregate_data(data: dict):
+ """
+ Aggregate invoices for each subcategory where required
+ and updates the data
+ """
+ sub_categories_requiring_aggregation = [
+ GSTR1_SubCategory.B2CS,
+ GSTR1_SubCategory.NIL_EXEMPT,
+ GSTR1_SubCategory.AT,
+ GSTR1_SubCategory.TXP,
+ ]
+
+ aggregate_data = {}
+
+ for subcategory in sub_categories_requiring_aggregation:
+ subcategory_data = data.get(subcategory.value)
+
+ if not subcategory_data:
+ continue
+
+ aggregate_data[subcategory.value] = (
+ AggregateInvoices.get_aggregate_subcategory(subcategory_data)
+ )
+
+ return aggregate_data
+
+ @staticmethod
+ def get_aggregate_subcategory(subcategory_data: dict):
+ value_keys = []
+ aggregate_invoices = {}
+
+ for _id, invoices in subcategory_data.items():
+ if not value_keys:
+ value_keys = AggregateInvoices.get_value_keys(invoices[0])
+
+ aggregate_invoices[_id] = [
+ AggregateInvoices.get_aggregate_invoices(invoices, value_keys)
+ ]
+
+ return aggregate_invoices
+
+ @staticmethod
+ def get_aggregate_invoices(invoices: list, value_keys: list = None) -> dict:
+ """
+ There can be multiple rows in books data for a single row in gov data
+ Aggregate all the rows to a single row
+ """
+ if not value_keys:
+ value_keys = AggregateInvoices.get_value_keys(invoices[0])
+
+ aggregated_invoice = invoices[0].copy()
+ aggregated_invoice.update(
+ {
+ key: sum([invoice.get(key, 0) for invoice in invoices])
+ for key in value_keys
+ }
+ )
+
+ return aggregated_invoice
+
+ @staticmethod
+ def get_value_keys(invoice: dict):
+ keys = []
+
+ for key, value in invoice.items():
+ if not isinstance(value, (int, float)):
+ continue
+
+ if key in AggregateInvoices.IGNORED_FIELDS:
+ continue
+
+ keys.append(key)
+
+ return keys
+
+
+class GenerateGSTR1(SummarizeGSTR1, ReconcileGSTR1, AggregateInvoices):
+ def generate_gstr1_data(self, filters, callback=None):
+ """
+ Generate GSTR-1 Data
+
+ Steps:
+ 1. Check if APIs are enabled. If not, generate only books data.
+ 2. Get the return status
+ 3. Get Gov Data
+ 4. Get Books Data
+ 5. Reconcile Data
+ 6. Summarize Data and return
+ """
+ data = {}
+
+ # APIs Disabled
+ if not self.is_gstr1_api_enabled():
+ return self.generate_only_books_data(data, filters, callback)
+
+ # APIs Enabled
+ status = self.get_return_status()
+
+ if status == "Filed":
+ gov_data_field = "filed"
+ else:
+ gov_data_field = "unfiled"
+
+ # Get Data
+ gov_data, is_enqueued = self.get_gov_gstr1_data()
+
+ if error_type := gov_data.get("error_type"):
+ # otp_requested, invalid_otp
+
+ if error_type == "invalid_otp":
+ request_otp(filters.company_gstin)
+
+ data = "otp_requested"
+ return callback and callback(data, filters)
+
+ books_data = self.get_books_gstr1_data(filters)
+
+ if is_enqueued:
+ return
+
+ reconcile_data = self.get_reconcile_gstr1_data(gov_data, books_data)
+
+ if status != "Filed":
+ books_data.update({"aggregate_data": self.get_aggregate_data(books_data)})
+ self.update_json_for("books", books_data)
+
+ # Compile Data
+ data["status"] = status
+
+ data["reconcile"] = self.normalize_data(reconcile_data)
+ data[gov_data_field] = self.normalize_data(gov_data)
+ data["books"] = self.normalize_data(books_data)
+
+ self.summarize_data(data)
+ return callback and callback(data, filters)
+
+ def generate_only_books_data(self, data, filters, callback=None):
+ status = "Not Filed"
+
+ books_data = self.get_books_gstr1_data(filters, aggregate=True)
+
+ data["books"] = self.normalize_data(books_data)
+ data["status"] = status
+
+ self.summarize_data(data)
+ return callback and callback(data, filters)
+
+ # GET DATA
+ def get_gov_gstr1_data(self):
+ if self.filing_status == "Filed":
+ data_field = "filed"
+ else:
+ data_field = "unfiled"
+
+ # data exists
+ if self.get(data_field):
+ mapped_data = self.get_json_for(data_field)
+
+ if mapped_data:
+ return mapped_data, False
+
+ # download data
+ return download_gstr1_json_data(self)
+
+ def get_books_gstr1_data(self, filters, aggregate=False):
+ from india_compliance.gst_india.doctype.gstr_1_beta.gstr_1_beta import (
+ get_gstr_1_from_and_to_date,
+ )
+
+ # Query / Process / Map / Sumarize / Optionally Save & Return
+ data_field = "books"
+
+ # data exists
+ if self.is_latest_data and self.get(data_field):
+ books_data = self.get_json_for(data_field)
+
+ if books_data:
+ return books_data
+
+ from_date, to_date = get_gstr_1_from_and_to_date(
+ filters.month_or_quarter, filters.year
+ )
+
+ _filters = frappe._dict(
+ {
+ "company": filters.company,
+ "company_gstin": filters.company_gstin,
+ "from_date": from_date,
+ "to_date": to_date,
+ }
+ )
+
+ # compute data
+ books_data = GSTR1BooksData(_filters).prepare_mapped_data()
+ if aggregate:
+ books_data.update({"aggregate_data": self.get_aggregate_data(books_data)})
+
+ self.update_json_for(data_field, books_data, reset_reconcile=True)
+
+ return books_data
+
+ # DATA MODIFIERS
+ def summarize_data(self, data):
+ """
+ Summarize data for all fields => reconcile, filed, unfiled, books
+
+ If return status is filed, use summary provided by Govt (usecase: amendments manually updated).
+ Else, summarize the data and save it.
+ """
+ summary_fields = {
+ "reconcile": "reconcile_summary",
+ "filed": "filed_summary",
+ "unfiled": "unfiled_summary",
+ "books": "books_summary",
+ }
+
+ for key, field in summary_fields.items():
+ if not data.get(key):
+ continue
+
+ if data.get(field):
+ continue
+
+ if self.is_latest_data and self.get(field):
+ _data = self.get_json_for(field)
+
+ if _data:
+ data[field] = _data
+ continue
+
+ summary_data = self.get_summarized_data(
+ data[key], self.filing_status == "Filed"
+ )
+
+ self.update_json_for(field, summary_data)
+ data[field] = summary_data
+
+ @staticmethod
+ def normalize_data(data):
+ """
+ Helper function to convert complex objects to simple objects
+ Returns object list of rows for each sub-category
+ """
+ for subcategory, subcategory_data in data.items():
+ if subcategory == "aggregate_data":
+ data[subcategory] = GenerateGSTR1.normalize_data(subcategory_data)
+ continue
+
+ if isinstance(subcategory_data, list | tuple | str):
+ continue
+
+ # get first key and value from subcategory_data
+ first_value = next(iter(subcategory_data.values()), None)
+
+ if isinstance(first_value, list | tuple):
+ # flatten the list of objects
+ data[subcategory] = list(itertools.chain(*subcategory_data.values()))
+
+ else:
+ data[subcategory] = [*subcategory_data.values()]
+
+ return data
+
+
+class GSTR1Log(GenerateGSTR1, Document):
+
+ @property
+ def status(self):
+ return self.generation_status
+
+ def update_status(self, status, commit=False):
+ self.db_set("generation_status", status, commit=commit)
+
+ # FILE UTILITY
+ def load_data(self, file_field=None):
+ data = {}
+
+ if file_field:
+ file_fields = [file_field]
+ else:
+ file_fields = self.get_applicable_file_fields()
+
+ for file_field in file_fields:
+ if json_data := self.get_json_for(file_field):
+ if "summary" not in file_field:
+ json_data = self.normalize_data(json_data)
+
+ data[file_field] = json_data
+
+ return data
+
+ def get_json_for(self, file_field):
+ try:
+ if file := get_file_doc(self.doctype, self.name, file_field):
+ return get_decompressed_data(file.get_content())
+
+ except FileNotFoundError:
+ # say File not restored
+ self.db_set(file_field, None)
+ return
+
+ def update_json_for(
+ self, file_field, json_data, overwrite=True, reset_reconcile=False
+ ):
+ if "summary" not in file_field:
+ json_data["creation"] = get_datetime_str(get_datetime())
+ self.remove_json_for(f"{file_field}_summary")
+
+ # reset reconciled data
+ if reset_reconcile:
+ self.remove_json_for("reconcile")
+
+ # new file
+ if not getattr(self, file_field):
+ content = get_compressed_data(json_data)
+ file_name = frappe.scrub("{0}-{1}.json.gz".format(self.name, file_field))
+ file = frappe.get_doc(
+ {
+ "doctype": "File",
+ "attached_to_doctype": self.doctype,
+ "attached_to_name": self.name,
+ "attached_to_field": file_field,
+ "file_name": file_name,
+ "is_private": 1,
+ "content": content,
+ }
+ ).insert()
+ self.db_set(file_field, file.file_url)
+ return
+
+ # existing file
+ file = get_file_doc(self.doctype, self.name, file_field)
+
+ if overwrite:
+ new_json = json_data
+
+ else:
+ new_json = get_decompressed_data(file.get_content())
+ new_json.update(json_data)
+
+ content = get_compressed_data(new_json)
+
+ file.save_file(content=content, overwrite=True)
+ self.db_set(file_field, file.file_url)
+
+ def remove_json_for(self, file_field):
+ if not self.get(file_field):
+ return
+
+ file = get_file_doc(self.doctype, self.name, file_field)
+ if file:
+ file.delete()
+
+ self.db_set(file_field, None)
+
+ if "summary" not in file_field:
+ self.remove_json_for(f"{file_field}_summary")
+
+ if file_field == "filed":
+ self.remove_json_for("unfiled")
+
+ # GSTR 1 UTILITY
+ def is_gstr1_api_enabled(self, settings=None):
+ if not settings:
+ settings = frappe.get_cached_doc("GST Settings")
+
+ return (
+ is_production_api_enabled(settings)
+ and settings.compare_gstr_1_data
+ and settings.has_valid_credentials(self.gstin, "Returns")
+ )
+
+ def is_sek_needed(self, settings=None):
+ if not self.is_gstr1_api_enabled(settings):
+ return False
+
+ if not self.unfiled or self.filing_status != "Filed":
+ return True
+
+ if not self.filed:
+ return True
+
+ return False
+
+ def has_all_files(self, settings=None):
+ if not self.is_latest_data:
+ return False
+
+ file_fields = self.get_applicable_file_fields(settings)
+ return all(getattr(self, file_field) for file_field in file_fields)
+
+ def get_return_status(self):
+ from india_compliance.gst_india.utils.gstin_info import get_gstr_1_return_status
+
+ status = self.get("filing_status")
+ if not status:
+ status = get_gstr_1_return_status(
+ self.company,
+ self.gstin,
+ self.return_period,
+ )
+ self.filing_status = status
+
+ return status
+
+ def get_applicable_file_fields(self, settings=None):
+ # Books aggregated data stored in filed (as to file)
+ fields = ["books", "books_summary"]
+
+ if self.is_gstr1_api_enabled(settings):
+ fields.extend(["reconcile", "reconcile_summary"])
+
+ if self.filing_status == "Filed":
+ fields.extend(["filed", "filed_summary"])
+ else:
+ fields.extend(["unfiled", "unfiled_summary"])
+
+ return fields
+
+
+def process_gstr_1_returns_info(company, gstin, response):
+ return_info = {}
+
+ # compile gstr-1 returns info
+ for info in response.get("EFiledlist"):
+ if info["rtntype"] == "GSTR1":
+ return_info[f"{info['ret_prd']}-{gstin}"] = info
+
+ # existing logs
+ gstr1_logs = frappe._dict(
+ frappe.get_all(
+ "GSTR-1 Log",
+ filters={"name": ("in", list(return_info.keys()))},
+ fields=["name", "acknowledgement_number"],
+ as_list=1,
+ )
+ )
+
+ # update gstr-1 filed upto
+ gstin_doc = frappe.get_doc("GSTIN", gstin)
+ if not gstin_doc:
+ gstin_doc = frappe.new_doc("GSTIN", gstin=gstin, status="Active")
+
+ def _update_gstr_1_filed_upto(filing_date):
+ if not gstin_doc.gstr_1_filed_upto or filing_date > getdate(
+ gstin_doc.gstr_1_filed_upto
+ ):
+ gstin_doc.gstr_1_filed_upto = filing_date
+ gstin_doc.save()
+
+ # create or update filed logs
+ for key, info in return_info.items():
+ filing_details = {
+ "filing_status": info["status"],
+ "acknowledgement_number": info["arn"],
+ "filing_date": datetime.strptime(info["dof"], "%d-%m-%Y").date(),
+ }
+
+ filed_upto = get_last_day(
+ getdate(f"{info['ret_prd'][2:]}-{info['ret_prd'][0:2]}-01")
+ )
+
+ if key in gstr1_logs:
+ if gstr1_logs[key] != info["arn"]:
+ frappe.db.set_value("GSTR-1 Log", key, filing_details)
+ _update_gstr_1_filed_upto(filed_upto)
+
+ # No updates if status is same
+ continue
+
+ frappe.get_doc(
+ {
+ "doctype": "GSTR-1 Log",
+ "company": company,
+ "gstin": gstin,
+ "return_period": info["ret_prd"],
+ **filing_details,
+ }
+ ).insert()
+ _update_gstr_1_filed_upto(filed_upto)
+
+
+def get_file_doc(doctype, docname, attached_to_field):
+ try:
+ return frappe.get_doc(
+ "File",
+ {
+ "attached_to_doctype": doctype,
+ "attached_to_name": docname,
+ "attached_to_field": attached_to_field,
+ },
+ )
+
+ except frappe.DoesNotExistError:
+ return None
+
+
+def get_compressed_data(json_data):
+ return gzip.compress(frappe.safe_encode(frappe.as_json(json_data)))
+
+
+def get_decompressed_data(content):
+ return frappe.parse_json(frappe.safe_decode(gzip.decompress(content)))
diff --git a/india_compliance/gst_india/doctype/gstr_1_log/test_gstr_1_log.py b/india_compliance/gst_india/doctype/gstr_1_log/test_gstr_1_log.py
new file mode 100644
index 0000000000..e8978437ca
--- /dev/null
+++ b/india_compliance/gst_india/doctype/gstr_1_log/test_gstr_1_log.py
@@ -0,0 +1,9 @@
+# Copyright (c) 2024, Resilient Tech and Contributors
+# See license.txt
+
+# import frappe
+from frappe.tests.utils import FrappeTestCase
+
+
+class TestGSTR1FiledLog(FrappeTestCase):
+ pass
diff --git a/india_compliance/gst_india/doctype/gstr_3b_report/gstr_3b_report.py b/india_compliance/gst_india/doctype/gstr_3b_report/gstr_3b_report.py
index b7c0c7cad5..db9c4e5916 100644
--- a/india_compliance/gst_india/doctype/gstr_3b_report/gstr_3b_report.py
+++ b/india_compliance/gst_india/doctype/gstr_3b_report/gstr_3b_report.py
@@ -123,7 +123,13 @@ def get_itc_reversal_entries(self):
def update_itc_reversal_from_purchase_invoice(self):
ineligible_credit = IneligibleITC(
self.company, self.gst_details.get("gstin"), self.month_no, self.year
- ).get_for_purchase_invoice(group_by="ineligibility_reason")
+ ).get_ineligible_itc_us_17_5_for_purchase(group_by="ineligibility_reason")
+
+ ineligible_credit_due_to_pos = IneligibleITC(
+ self.company, self.gst_details.get("gstin"), self.month_no, self.year
+ ).get_ineligible_itc_due_to_pos_for_purchase(group_by="ineligibility_reason")
+
+ ineligible_credit.extend(ineligible_credit_due_to_pos)
return self.process_ineligible_credit(ineligible_credit)
diff --git a/india_compliance/gst_india/doctype/gstr_import_log/gstr_import_log.py b/india_compliance/gst_india/doctype/gstr_import_log/gstr_import_log.py
index 6aa69e282e..869e5dd68b 100644
--- a/india_compliance/gst_india/doctype/gstr_import_log/gstr_import_log.py
+++ b/india_compliance/gst_india/doctype/gstr_import_log/gstr_import_log.py
@@ -72,7 +72,7 @@ def toggle_scheduled_jobs(stopped):
scheduled_job = frappe.db.get_value(
"Scheduled Job Type",
{
- "method": "india_compliance.gst_india.utils.gstr.download_queued_request",
+ "method": "india_compliance.gst_india.utils.gstr_utils.download_queued_request",
},
)
diff --git a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/__init__.py b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/__init__.py
index 746bd7e8ea..3beaffd8ae 100644
--- a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/__init__.py
+++ b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/__init__.py
@@ -16,10 +16,9 @@
from india_compliance.gst_india.utils import (
get_escaped_name,
get_gst_accounts_by_type,
- get_gstin_list,
get_party_for_gstin,
)
-from india_compliance.gst_india.utils.gstr import IMPORT_CATEGORY, ReturnType
+from india_compliance.gst_india.utils.gstr_2 import IMPORT_CATEGORY, ReturnType
class Fields(Enum):
diff --git a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.js b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.js
index 49275b0ca0..91487550f6 100644
--- a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.js
+++ b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.js
@@ -73,19 +73,19 @@ frappe.ui.form.on("Purchase Reconciliation Tool", {
new india_compliance.quick_info_popover(frm, tooltip_info);
await frappe.require("purchase_reconciliation_tool.bundle.js");
+ frm.trigger("company");
frm.purchase_reconciliation_tool = new PurchaseReconciliationTool(frm);
},
onload(frm) {
if (frm.doc.is_modified) frm.doc.reconciliation_data = null;
- frm.trigger("company");
add_gstr2b_alert(frm);
set_date_range_description(frm);
},
async company(frm) {
if (!frm.doc.company) return;
- const options = await set_gstin_options(frm);
+ const options = await india_compliance.set_gstin_options(frm);
if (!frm.doc.company_gstin) frm.set_value("company_gstin", options[0]);
},
@@ -746,8 +746,8 @@ class PurchaseReconciliationTool {
{
label: "Purchase
Invoice",
fieldname: "purchase_invoice_name",
- fieldtype: "Link",
- doctype: "Purchase Invoice",
+ fieldtype: "Dynamic Link",
+ options: "purchase_doctype",
align: "center",
width: 120,
},
@@ -1740,17 +1740,3 @@ async function create_new_purchase_invoice(row, company, company_gstin) {
frappe.new_doc("Purchase Invoice");
}
-async function set_gstin_options(frm) {
- const { query, params } = india_compliance.get_gstin_query(frm.doc.company);
- const { message } = await frappe.call({
- method: query,
- args: params,
- });
-
- if (!message) return [];
- message.unshift("All");
-
- const gstin_field = frm.get_field("company_gstin");
- gstin_field.set_data(message);
- return message;
-}
diff --git a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.py b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.py
index 84aeada014..1558e70d58 100644
--- a/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.py
+++ b/india_compliance/gst_india/doctype/purchase_reconciliation_tool/purchase_reconciliation_tool.py
@@ -10,6 +10,7 @@
from frappe.utils import add_to_date, cint, now_datetime
from frappe.utils.response import json_handler
+from india_compliance.gst_india.api_classes.returns import ReturnsAPI
from india_compliance.gst_india.constants import ORIGINAL_VS_AMENDED
from india_compliance.gst_india.doctype.purchase_reconciliation_tool import (
BaseUtil,
@@ -24,11 +25,10 @@
is_api_enabled,
)
from india_compliance.gst_india.utils.exporter import ExcelExporter
-from india_compliance.gst_india.utils.gstr import (
+from india_compliance.gst_india.utils.gstr_2 import (
ACTIONS,
IMPORT_CATEGORY,
GSTRCategory,
- ReturnsAPI,
ReturnType,
download_gstr_2a,
download_gstr_2b,
diff --git a/india_compliance/gst_india/overrides/payment_entry.py b/india_compliance/gst_india/overrides/payment_entry.py
index c75b16816c..9c302b9e13 100644
--- a/india_compliance/gst_india/overrides/payment_entry.py
+++ b/india_compliance/gst_india/overrides/payment_entry.py
@@ -8,10 +8,16 @@
from erpnext.controllers.accounts_controller import get_advance_payment_entries
from india_compliance.gst_india.overrides.transaction import get_gst_details
+from india_compliance.gst_india.overrides.transaction import (
+ validate_backdated_transaction as _validate_backdated_transaction,
+)
from india_compliance.gst_india.overrides.transaction import (
validate_transaction as validate_transaction_for_advance_payment,
)
-from india_compliance.gst_india.utils import get_all_gst_accounts
+from india_compliance.gst_india.utils import (
+ get_all_gst_accounts,
+ get_gst_accounts_by_type,
+)
@frappe.whitelist()
@@ -79,6 +85,8 @@ def validate(doc, method=None):
return
if doc.party_type == "Customer":
+ validate_backdated_transaction(doc)
+
# Presume is export with GST if GST accounts are present
doc.is_export_with_gst = 1
validate_transaction_for_advance_payment(doc, method)
@@ -100,6 +108,21 @@ def on_update_after_submit(doc, method=None):
make_gst_revesal_entry_from_advance_payment(doc)
+def before_cancel(doc, method=None):
+ if not doc.taxes:
+ return
+
+ validate_backdated_transaction(doc, action="cancel")
+
+
+def validate_backdated_transaction(doc, action="create"):
+ gst_accounts = get_gst_accounts_by_type(doc.company, "Output").values()
+ for row in doc.taxes:
+ if row.account_head in gst_accounts and row.tax_amount != 0:
+ _validate_backdated_transaction(doc, action=action)
+ break
+
+
@frappe.whitelist()
def update_party_details(party_details, doctype, company):
party_details = frappe.parse_json(party_details)
diff --git a/india_compliance/gst_india/overrides/sales_invoice.py b/india_compliance/gst_india/overrides/sales_invoice.py
index 06f9e4ca19..d414982454 100644
--- a/india_compliance/gst_india/overrides/sales_invoice.py
+++ b/india_compliance/gst_india/overrides/sales_invoice.py
@@ -5,6 +5,7 @@
from india_compliance.gst_india.overrides.payment_entry import get_taxes_summary
from india_compliance.gst_india.overrides.transaction import (
ignore_gst_validations,
+ validate_backdated_transaction,
validate_mandatory_fields,
validate_transaction,
)
@@ -57,6 +58,7 @@ def validate(doc, method=None):
gst_settings = frappe.get_cached_doc("GST Settings")
+ validate_backdated_transaction(doc, gst_settings)
validate_invoice_number(doc)
validate_credit_debit_note(doc)
validate_fields_and_set_status_for_e_invoice(doc, gst_settings)
@@ -174,6 +176,10 @@ def on_submit(doc, method=None):
def before_cancel(doc, method=None):
+ if ignore_gst_validations(doc):
+ return
+
+ validate_backdated_transaction(doc, action="cancel")
validate_fields_and_set_status_for_e_invoice(doc)
payment_references = frappe.get_all(
"Payment Entry Reference",
diff --git a/india_compliance/gst_india/overrides/transaction.py b/india_compliance/gst_india/overrides/transaction.py
index 9854ff1c7f..1a810efd6b 100644
--- a/india_compliance/gst_india/overrides/transaction.py
+++ b/india_compliance/gst_india/overrides/transaction.py
@@ -4,7 +4,7 @@
import frappe
from frappe import _, bold
from frappe.model import delete_doc
-from frappe.utils import cint, flt
+from frappe.utils import cint, flt, format_date
from erpnext.controllers.accounts_controller import get_taxes_and_charges
from erpnext.controllers.taxes_and_totals import (
get_itemised_tax,
@@ -17,6 +17,9 @@
STATE_NUMBERS,
)
from india_compliance.gst_india.constants.custom_fields import E_WAYBILL_INV_FIELDS
+from india_compliance.gst_india.doctype.gst_settings.gst_settings import (
+ restrict_gstr_1_transaction_for,
+)
from india_compliance.gst_india.doctype.gstin.gstin import (
_validate_gst_transporter_id_info,
_validate_gstin_info,
@@ -657,6 +660,18 @@ def get_source_state_code(doc):
return (doc.supplier_gstin or doc.company_gstin)[:2]
+def validate_backdated_transaction(doc, gst_settings=None, action="create"):
+ if gstr_1_filed_upto := restrict_gstr_1_transaction_for(
+ doc.posting_date, doc.company_gstin, gst_settings
+ ):
+ frappe.throw(
+ _(
+ "You are not allowed to {0} {1} as GSTR-1 has been filed upto {2}"
+ ).format(action, doc.doctype, frappe.bold(format_date(gstr_1_filed_upto))),
+ title=_("Restricted Changes"),
+ )
+
+
def validate_hsn_codes(doc):
validate_hsn_code, valid_hsn_length = get_hsn_settings()
diff --git a/india_compliance/gst_india/report/gst_itemised_sales_register/gst_itemised_sales_register.py b/india_compliance/gst_india/report/gst_itemised_sales_register/gst_itemised_sales_register.py
index fa3a8ed4a9..dba3cf0ec2 100644
--- a/india_compliance/gst_india/report/gst_itemised_sales_register/gst_itemised_sales_register.py
+++ b/india_compliance/gst_india/report/gst_itemised_sales_register/gst_itemised_sales_register.py
@@ -39,10 +39,8 @@ def get_additional_table_columns():
def get_additional_conditions(filters):
- additional_conditions = ""
+ additional_conditions = {}
if filters.get("company_gstin"):
- additional_conditions += (
- " AND `tabSales Invoice`.company_gstin = %(company_gstin)s"
- )
+ additional_conditions.update({"company_gstin": filters.get("company_gstin")})
return additional_conditions
diff --git a/india_compliance/gst_india/report/gst_sales_register_beta/gst_sales_register_beta.py b/india_compliance/gst_india/report/gst_sales_register_beta/gst_sales_register_beta.py
index 423cd02ff3..1fc832f0e4 100644
--- a/india_compliance/gst_india/report/gst_sales_register_beta/gst_sales_register_beta.py
+++ b/india_compliance/gst_india/report/gst_sales_register_beta/gst_sales_register_beta.py
@@ -5,7 +5,7 @@
from frappe import _
from frappe.utils import getdate
-from india_compliance.gst_india.utils.gstr.gstr_1 import GSTR1Invoices
+from india_compliance.gst_india.utils.gstr_1.gstr_1_data import GSTR1Invoices
def execute(filters=None):
@@ -217,7 +217,7 @@ def get_columns(filters):
},
{
"label": _("UOM"),
- "fieldname": "uom",
+ "fieldname": "stock_uom",
"fieldtype": "Data",
"width": 100,
},
diff --git a/india_compliance/gst_india/report/gst_sales_register_beta/test_sales_register_beta.py b/india_compliance/gst_india/report/gst_sales_register_beta/test_sales_register_beta.py
index 10c2567f94..b01d9d881a 100644
--- a/india_compliance/gst_india/report/gst_sales_register_beta/test_sales_register_beta.py
+++ b/india_compliance/gst_india/report/gst_sales_register_beta/test_sales_register_beta.py
@@ -33,7 +33,7 @@
"total_amount": -300000.0,
"total_tax_amount": 0.0,
"invoice_category": "Credit/Debit Notes (Unregistered)",
- "invoice_sub_category": "CDNUR",
+ "invoice_sub_category": "Credit/Debit Notes (Unregistered)",
"invoice_type": "EXPWOP",
},
{
@@ -52,7 +52,7 @@
"total_amount": -5000.0,
"total_tax_amount": 0.0,
"invoice_category": "Credit/Debit Notes (Unregistered)",
- "invoice_sub_category": "CDNUR",
+ "invoice_sub_category": "Credit/Debit Notes (Unregistered)",
"invoice_type": "EXPWOP",
},
{
@@ -71,7 +71,7 @@
"total_amount": 500000.0,
"total_tax_amount": 0.0,
"invoice_category": "Exports",
- "invoice_sub_category": "EXPWP",
+ "invoice_sub_category": "Export With Payment of Tax",
"invoice_type": "WPAY",
},
{
@@ -90,7 +90,7 @@
"total_amount": 45000.0,
"total_tax_amount": 0.0,
"invoice_category": "Exports",
- "invoice_sub_category": "EXPWP",
+ "invoice_sub_category": "Export With Payment of Tax",
"invoice_type": "WPAY",
},
{
@@ -109,7 +109,7 @@
"total_amount": 140000.0,
"total_tax_amount": 0.0,
"invoice_category": "Exports",
- "invoice_sub_category": "EXPWOP",
+ "invoice_sub_category": "Export Without Payment of Tax",
"invoice_type": "WOPAY",
},
{
@@ -128,7 +128,7 @@
"total_amount": 5000.0,
"total_tax_amount": 0.0,
"invoice_category": "Exports",
- "invoice_sub_category": "EXPWOP",
+ "invoice_sub_category": "Export Without Payment of Tax",
"invoice_type": "WOPAY",
},
{
@@ -148,7 +148,7 @@
"total_tax_amount": -40500.0,
"invoice_category": "Credit/Debit Notes (Registered)",
"invoice_type": "Regular B2B",
- "invoice_sub_category": "CDNR",
+ "invoice_sub_category": "Credit/Debit Notes (Registered)",
},
{
"item_code": "_Test Nil Rated Item",
@@ -166,8 +166,8 @@
"total_amount": -30000.0,
"total_tax_amount": 0.0,
"invoice_category": "Nil-Rated, Exempted, Non-GST",
- "invoice_type": "Inter-State to registered persons",
- "invoice_sub_category": "Nil-Rated",
+ "invoice_type": "Inter-State supplies to registered persons",
+ "invoice_sub_category": "Nil-Rated, Exempted, Non-GST",
},
{
"item_code": "_Test Service Item",
@@ -204,8 +204,8 @@
"total_amount": 60000.0,
"total_tax_amount": 0.0,
"invoice_category": "Nil-Rated, Exempted, Non-GST",
- "invoice_type": "Inter-State to registered persons",
- "invoice_sub_category": "Nil-Rated",
+ "invoice_type": "Inter-State supplies to registered persons",
+ "invoice_sub_category": "Nil-Rated, Exempted, Non-GST",
},
{
"item_code": "_Test Service Item",
@@ -241,8 +241,8 @@
"total_amount": 5000.0,
"total_tax_amount": 0.0,
"invoice_category": "Nil-Rated, Exempted, Non-GST",
- "invoice_type": "Intra-State to unregistered persons",
- "invoice_sub_category": "Nil-Rated",
+ "invoice_type": "Intra-State supplies to unregistered persons",
+ "invoice_sub_category": "Nil-Rated, Exempted, Non-GST",
},
{
"item_code": "_Test Service Item",
@@ -278,8 +278,8 @@
"total_amount": 5000.0,
"total_tax_amount": 0.0,
"invoice_category": "Nil-Rated, Exempted, Non-GST",
- "invoice_type": "Intra-State to unregistered persons",
- "invoice_sub_category": "Nil-Rated",
+ "invoice_type": "Intra-State supplies to unregistered persons",
+ "invoice_sub_category": "Nil-Rated, Exempted, Non-GST",
},
{
"item_code": "_Test Service Item",
@@ -315,8 +315,8 @@
"total_amount": 5000.0,
"total_tax_amount": 0.0,
"invoice_category": "Nil-Rated, Exempted, Non-GST",
- "invoice_type": "Inter-State to unregistered persons",
- "invoice_sub_category": "Nil-Rated",
+ "invoice_type": "Inter-State supplies to unregistered persons",
+ "invoice_sub_category": "Nil-Rated, Exempted, Non-GST",
},
]
@@ -349,7 +349,7 @@
"total_cess_amount": 0,
},
{
- "description": "SEZ with payment",
+ "description": "SEZ With Payment of Tax",
"indent": 1,
"taxable_value": 0,
"igst_amount": 0,
@@ -358,7 +358,7 @@
"total_cess_amount": 0,
},
{
- "description": "SEZ without payment",
+ "description": "SEZ Without Payment of Tax",
"indent": 1,
"taxable_value": 0,
"igst_amount": 0,
@@ -403,7 +403,7 @@
"total_cess_amount": 0.0,
},
{
- "description": "Exports with payment",
+ "description": "Export With Payment of Tax",
"indent": 1,
"taxable_value": 545000.0,
"igst_amount": 0.0,
@@ -412,7 +412,7 @@
"total_cess_amount": 0.0,
},
{
- "description": "Exports without payment",
+ "description": "Export Without Payment of Tax",
"indent": 1,
"taxable_value": 145000.0,
"igst_amount": 0.0,
@@ -448,7 +448,7 @@
"total_cess_amount": 0.0,
},
{
- "description": "Nil-Rated",
+ "description": "Nil-Rated, Exempted, Non-GST",
"indent": 1,
"taxable_value": 45000.0,
"igst_amount": 0.0,
@@ -456,24 +456,6 @@
"sgst_amount": 0.0,
"total_cess_amount": 0.0,
},
- {
- "description": "Exempted",
- "indent": 1,
- "taxable_value": 0,
- "igst_amount": 0,
- "cgst_amount": 0,
- "sgst_amount": 0,
- "total_cess_amount": 0,
- },
- {
- "description": "Non-GST",
- "indent": 1,
- "taxable_value": 0,
- "igst_amount": 0,
- "cgst_amount": 0,
- "sgst_amount": 0,
- "total_cess_amount": 0,
- },
{
"description": "Credit/Debit Notes (Registered)",
"indent": 0,
diff --git a/india_compliance/gst_india/report/gstr_1/gstr_1.js b/india_compliance/gst_india/report/gstr_1/gstr_1.js
index 136121ff39..e9b8286f10 100644
--- a/india_compliance/gst_india/report/gstr_1/gstr_1.js
+++ b/india_compliance/gst_india/report/gstr_1/gstr_1.js
@@ -89,7 +89,10 @@ frappe.query_reports["GSTR-1"] = {
},
},
],
- onload: create_download_buttons,
+ onload(report) {
+ create_download_buttons(report);
+ show_gstr_1_beta_alert(report);
+ },
};
function create_download_buttons(report) {
@@ -166,3 +169,19 @@ function download_full_report_excel(report) {
filters: JSON.stringify(report.get_values()),
});
}
+
+function show_gstr_1_beta_alert(report) {
+ if (report.page.wrapper.find(".alert").length) return;
+
+ const alert_message = `
+
+ GSTR-1 Beta
+
+ is released with improved features and user experience. Try it out now!
+ `;
+
+ india_compliance.show_dismissable_alert(
+ report.page.wrapper.find(".container.page-body"),
+ alert_message
+ );
+}
diff --git a/india_compliance/gst_india/report/gstr_1/gstr_1.py b/india_compliance/gst_india/report/gstr_1/gstr_1.py
index d4942db0cf..bbf51ea6dd 100644
--- a/india_compliance/gst_india/report/gstr_1/gstr_1.py
+++ b/india_compliance/gst_india/report/gstr_1/gstr_1.py
@@ -1072,23 +1072,21 @@ def __init__(self, filters, gst_accounts):
def get_data(self):
if self.filters.get("type_of_business") == "Advances":
- records = self.get_11A_data()
-
+ records = self.get_11A_query().run(as_dict=True)
elif self.filters.get("type_of_business") == "Adjustment":
- records = self.get_11B_data()
+ records = self.get_11B_query().run(as_dict=True)
return self.process_data(records)
- def get_11A_data(self):
+ def get_11A_query(self):
return (
self.get_query()
.select(self.pe.paid_amount.as_("taxable_value"))
.groupby(self.pe.name)
- .run(as_dict=True)
)
- def get_11B_data(self):
- query = (
+ def get_11B_query(self):
+ return (
self.get_query()
.join(self.pe_ref)
.on(self.pe_ref.name == self.gl_entry.voucher_detail_no)
@@ -1096,8 +1094,6 @@ def get_11B_data(self):
.groupby(self.gl_entry.voucher_detail_no)
)
- return query.run(as_dict=True)
-
def get_query(self):
cr_or_dr = (
"credit" if self.filters.get("type_of_business") == "Advances" else "debit"
diff --git a/india_compliance/gst_india/report/gstr_3b_details/gstr_3b_details.py b/india_compliance/gst_india/report/gstr_3b_details/gstr_3b_details.py
index 22827f6f4f..35a808298d 100644
--- a/india_compliance/gst_india/report/gstr_3b_details/gstr_3b_details.py
+++ b/india_compliance/gst_india/report/gstr_3b_details/gstr_3b_details.py
@@ -270,7 +270,7 @@ def get_itc_from_journal_entry(self):
def get_ineligible_itc_from_purchase(self):
ineligible_itc = IneligibleITC(
self.company, self.company_gstin, self.filters.month, self.filters.year
- ).get_for_purchase_invoice()
+ ).get_ineligible_itc_us_17_5_for_purchase()
return self.process_ineligible_itc(ineligible_itc)
@@ -286,10 +286,6 @@ def process_ineligible_itc(self, ineligible_itc):
return []
for row in ineligible_itc.copy():
- if row.itc_classification == "ITC restricted due to PoS rules":
- ineligible_itc.remove(row)
- continue
-
for key in ["iamt", "camt", "samt", "csamt"]:
row[key] = row[key] * -1
@@ -429,11 +425,15 @@ def __init__(self, company, gstin, month, year) -> None:
self.year = year
self.gst_accounts = get_escaped_gst_accounts(company, "Input")
- def get_for_purchase_invoice(self, group_by="name"):
+ def get_ineligible_itc_us_17_5_for_purchase(self, group_by="name"):
+ """
+ - Ineligible As Per Section 17(5)
+ - ITC restricted due to ineligible items in purchase invoice
+ """
ineligible_transactions = self.get_vouchers_with_gst_expense("Purchase Invoice")
if not ineligible_transactions:
- return
+ return []
pi = frappe.qb.DocType("Purchase Invoice")
@@ -446,9 +446,10 @@ def get_for_purchase_invoice(self, group_by="name"):
pi.name.as_("voucher_no"),
pi.ineligibility_reason.as_("itc_classification"),
)
- .where(IfNull(pi.ineligibility_reason, "") != "")
+ .where(
+ IfNull(pi.ineligibility_reason, "") == "Ineligible As Per Section 17(5)"
+ )
.where(pi.name.isin(ineligible_transactions))
- .where(pi.company_gstin != IfNull(pi.supplier_gstin, ""))
.groupby(pi[group_by])
.run(as_dict=1)
)
@@ -465,15 +466,71 @@ def get_for_purchase_invoice(self, group_by="name"):
Sum(pi.itc_state_tax).as_("samt"),
Sum(pi.itc_cess_amount).as_("csamt"),
)
- .where(IfNull(pi.ineligibility_reason, "") != "")
+ .where(
+ IfNull(pi.ineligibility_reason, "") == "Ineligible As Per Section 17(5)"
+ )
.where(pi.name.isin(ineligible_transactions))
- .where(pi.company_gstin != IfNull(pi.supplier_gstin, ""))
.groupby(pi[group_by])
.run(as_dict=1)
)
return self.get_ineligible_credit(credit_availed, credit_available, group_by)
+ def get_ineligible_itc_due_to_pos_for_purchase(self, group_by="name"):
+ """
+ - ITC restricted due to PoS rules
+ """
+ ineligible_transactions = self.get_vouchers_with_gst_expense("Purchase Invoice")
+
+ if not ineligible_transactions:
+ return []
+
+ pi = frappe.qb.DocType("Purchase Invoice")
+ taxes = frappe.qb.DocType("Purchase Taxes and Charges")
+
+ # utility function
+ def get_tax_case_statement(account, alias):
+ return Sum(
+ Case()
+ .when(
+ taxes.account_head.isin(account),
+ taxes.base_tax_amount_after_discount_amount,
+ )
+ .else_(0)
+ ).as_(alias)
+
+ # Credit availed is not required as it will be always 0 for pos
+
+ ineligible_credit = (
+ frappe.qb.from_(pi)
+ .inner_join(taxes)
+ .on(pi.name == taxes.parent)
+ .select(
+ pi.name.as_("voucher_no"),
+ pi.posting_date,
+ pi.ineligibility_reason.as_("itc_classification"),
+ get_tax_case_statement([self.gst_accounts.igst_account], "iamt"),
+ get_tax_case_statement([self.gst_accounts.cgst_account], "camt"),
+ get_tax_case_statement([self.gst_accounts.sgst_account], "camt"),
+ get_tax_case_statement(
+ [
+ self.gst_accounts.cess_account,
+ self.gst_accounts.cess_non_advol_account,
+ ],
+ "csamt",
+ ),
+ )
+ .where(taxes.account_head.isin(list(self.gst_accounts.values())))
+ .where(
+ IfNull(pi.ineligibility_reason, "") == "ITC restricted due to PoS rules"
+ )
+ .where(pi.name.isin(ineligible_transactions))
+ .groupby(pi[group_by])
+ .run(as_dict=True)
+ )
+
+ return ineligible_credit
+
def get_for_bill_of_entry(self, group_by="name"):
ineligible_transactions = self.get_vouchers_with_gst_expense("Bill of Entry")
diff --git a/india_compliance/gst_india/setup/__init__.py b/india_compliance/gst_india/setup/__init__.py
index ce9ec7e288..a4fa622a86 100644
--- a/india_compliance/gst_india/setup/__init__.py
+++ b/india_compliance/gst_india/setup/__init__.py
@@ -215,6 +215,10 @@ def set_default_gst_settings():
"reconcile_on_friday": 1,
"reconcile_for_b2b": 1,
"reconcile_for_cdnr": 1,
+ # GSTR-1
+ "compare_gstr_1_data": 1,
+ "freeze_transactions": 1,
+ "filing_frequency": "Monthly",
}
if frappe.conf.developer_mode:
diff --git a/india_compliance/gst_india/utils/__init__.py b/india_compliance/gst_india/utils/__init__.py
index 4c3ad08f30..0609e897aa 100644
--- a/india_compliance/gst_india/utils/__init__.py
+++ b/india_compliance/gst_india/utils/__init__.py
@@ -648,6 +648,13 @@ def get_titlecase_version(word, all_caps=False, **kwargs):
return word
+def is_production_api_enabled(settings=None):
+ if not settings:
+ settings = frappe.get_cached_doc("GST Settings")
+
+ return is_api_enabled(settings) and not settings.sandbox_mode
+
+
def is_api_enabled(settings=None):
if not settings:
settings = frappe.get_cached_value(
@@ -662,11 +669,7 @@ def is_api_enabled(settings=None):
def is_autofill_party_info_enabled():
settings = frappe.get_cached_doc("GST Settings")
- return (
- is_api_enabled(settings)
- and settings.autofill_party_info
- and not settings.sandbox_mode
- )
+ return is_production_api_enabled(settings) and settings.autofill_party_info
def can_enable_api(settings):
diff --git a/india_compliance/gst_india/utils/e_invoice.py b/india_compliance/gst_india/utils/e_invoice.py
index c18b52019d..af7e0581ff 100644
--- a/india_compliance/gst_india/utils/e_invoice.py
+++ b/india_compliance/gst_india/utils/e_invoice.py
@@ -25,6 +25,9 @@
CANCEL_REASON_CODES,
ITEM_LIMIT,
)
+from india_compliance.gst_india.doctype.gst_settings.gst_settings import (
+ get_e_invoice_applicability_date,
+)
from india_compliance.gst_india.overrides.transaction import (
_validate_hsn_codes,
validate_mandatory_fields,
@@ -413,24 +416,6 @@ def validate_taxable_item(doc, throw=True):
)
-def get_e_invoice_applicability_date(company, settings=None, throw=True):
- if not settings:
- settings = frappe.get_cached_doc("GST Settings")
-
- e_invoice_applicable_from = settings.e_invoice_applicable_from
-
- if settings.apply_e_invoice_only_for_selected_companies:
- for row in settings.e_invoice_applicable_companies:
- if company == row.company:
- e_invoice_applicable_from = row.applicable_from
- break
-
- else:
- return
-
- return e_invoice_applicable_from
-
-
def validate_if_e_invoice_can_be_cancelled(doc):
if not doc.irn:
frappe.throw(_("IRN not found"), title=_("Error Cancelling e-Invoice"))
diff --git a/india_compliance/gst_india/utils/exporter.py b/india_compliance/gst_india/utils/exporter.py
index 7610c93125..7ffeca8db8 100644
--- a/india_compliance/gst_india/utils/exporter.py
+++ b/india_compliance/gst_india/utils/exporter.py
@@ -62,7 +62,6 @@ class Worksheet:
"bold": False,
"horizontal": "general",
"number_format": "General",
- "width": 20,
"height": 20,
"vertical": "center",
"wrap_text": False,
@@ -225,7 +224,9 @@ def apply_style(self, row, column, style):
if style.bg_color:
cell.fill = PatternFill(fill_type="solid", fgColor=style.bg_color)
- self.ws.column_dimensions[get_column_letter(column)].width = style.width
+ if style.get("width"):
+ self.ws.column_dimensions[get_column_letter(column)].width = style.width
+
self.ws.row_dimensions[row].height = style.height
def apply_conditional_formatting(self, has_totals):
diff --git a/india_compliance/gst_india/utils/gstin_info.py b/india_compliance/gst_india/utils/gstin_info.py
index a46d03c0c4..5c9ffdc4e3 100644
--- a/india_compliance/gst_india/utils/gstin_info.py
+++ b/india_compliance/gst_india/utils/gstin_info.py
@@ -4,9 +4,13 @@
import frappe
from frappe import _
+from frappe.utils import getdate
from india_compliance.gst_india.api_classes.base import BASE_URL
from india_compliance.gst_india.api_classes.public import PublicAPI
+from india_compliance.gst_india.doctype.gstr_1_log.gstr_1_log import (
+ process_gstr_1_returns_info,
+)
from india_compliance.gst_india.utils import titlecase, validate_gstin
GST_CATEGORIES = {
@@ -178,3 +182,56 @@ def _extract_address_lines(address):
# "Non Resident Taxable Person"
# "Government Department ID"
+
+
+####################################################################################################
+#### GSTIN RETURNS INFO ##########################################################################
+####################################################################################################
+
+
+def get_gstr_1_return_status(
+ company, gstin, period, process_info=True, year_increment=0
+):
+ """Returns Returns info for the given period"""
+ fy = get_fy(period, year_increment=year_increment)
+
+ response = PublicAPI().get_returns_info(gstin, fy)
+ if not response:
+ return
+
+ if process_info:
+ frappe.enqueue(
+ process_gstr_1_returns_info,
+ company=company,
+ gstin=gstin,
+ response=response,
+ enqueue_after_commit=True,
+ )
+
+ for info in response.get("EFiledlist"):
+ if info["rtntype"] == "GSTR1" and info["ret_prd"] == period:
+ return info["status"]
+
+ # late filing possibility (limitation: only checks for the next FY: good enough)
+ if not year_increment and get_current_fy() != fy:
+ get_gstr_1_return_status(
+ company, gstin, period, process_info=process_info, year_increment=1
+ )
+
+ return "Not Filed"
+
+
+def get_fy(period, year_increment=0):
+ month, year = period[:2], period[2:]
+ year = str(int(year) + year_increment)
+
+ # For the month of March, it's filed in the next FY
+ if int(month) < 3:
+ return f"{int(year) - 1}-{year[-2:]}"
+ else:
+ return f"{year}-{int(year[-2:]) + 1}"
+
+
+def get_current_fy():
+ period = getdate().strftime("%m%Y")
+ return get_fy(period)
diff --git a/india_compliance/gst_india/utils/gstr_1/__init__.py b/india_compliance/gst_india/utils/gstr_1/__init__.py
new file mode 100644
index 0000000000..378c110eb3
--- /dev/null
+++ b/india_compliance/gst_india/utils/gstr_1/__init__.py
@@ -0,0 +1,324 @@
+from enum import Enum
+
+
+class GSTR1_Category(Enum):
+ """
+ Overview Page of GSTR-1
+ """
+
+ # Invoice Items Bifurcation
+ B2B = "B2B, SEZ, DE"
+ EXP = "Exports"
+ B2CL = "B2C (Large)"
+ B2CS = "B2C (Others)"
+ NIL_EXEMPT = "Nil-Rated, Exempted, Non-GST"
+ CDNR = "Credit/Debit Notes (Registered)"
+ CDNUR = "Credit/Debit Notes (Unregistered)"
+
+ # Other Categories
+ AT = "Advances Received"
+ TXP = "Advances Adjusted"
+ HSN = "HSN Summary"
+ DOC_ISSUE = "Document Issued"
+ SUPECOM = "Supplies made through E-commerce Operators"
+
+
+class GSTR1_SubCategory(Enum):
+ """
+ Summary Page of GSTR-1
+ """
+
+ # Invoice Items Bifurcation
+ B2B_REGULAR = "B2B Regular"
+ B2B_REVERSE_CHARGE = "B2B Reverse Charge"
+ SEZWP = "SEZ With Payment of Tax"
+ SEZWOP = "SEZ Without Payment of Tax"
+ DE = "Deemed Exports"
+ EXPWP = "Export With Payment of Tax"
+ EXPWOP = "Export Without Payment of Tax"
+ B2CL = "B2C (Large)"
+ B2CS = "B2C (Others)"
+ NIL_EXEMPT = "Nil-Rated, Exempted, Non-GST"
+ CDNR = "Credit/Debit Notes (Registered)"
+ CDNUR = "Credit/Debit Notes (Unregistered)"
+
+ # Other Sub-Categories
+ AT = "Advances Received"
+ TXP = "Advances Adjusted"
+ HSN = "HSN Summary"
+ DOC_ISSUE = "Document Issued"
+
+ # E-Commerce
+ SUPECOM_52 = "TCS collected by E-commerce Operator u/s 52"
+ SUPECOM_9_5 = "GST Payable on RCM by E-commerce Operator u/s 9(5)"
+
+
+CATEGORY_SUB_CATEGORY_MAPPING = {
+ GSTR1_Category.B2B: (
+ GSTR1_SubCategory.B2B_REGULAR,
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE,
+ GSTR1_SubCategory.SEZWP,
+ GSTR1_SubCategory.SEZWOP,
+ GSTR1_SubCategory.DE,
+ ),
+ GSTR1_Category.B2CL: (GSTR1_SubCategory.B2CL,),
+ GSTR1_Category.EXP: (GSTR1_SubCategory.EXPWP, GSTR1_SubCategory.EXPWOP),
+ GSTR1_Category.B2CS: (GSTR1_SubCategory.B2CS,),
+ GSTR1_Category.NIL_EXEMPT: (GSTR1_SubCategory.NIL_EXEMPT,),
+ GSTR1_Category.CDNR: (GSTR1_SubCategory.CDNR,),
+ GSTR1_Category.CDNUR: (GSTR1_SubCategory.CDNUR,),
+ GSTR1_Category.AT: (GSTR1_SubCategory.AT,),
+ GSTR1_Category.TXP: (GSTR1_SubCategory.TXP,),
+ GSTR1_Category.DOC_ISSUE: (GSTR1_SubCategory.DOC_ISSUE,),
+ GSTR1_Category.HSN: (GSTR1_SubCategory.HSN,),
+ GSTR1_Category.SUPECOM: (
+ GSTR1_SubCategory.SUPECOM_52,
+ GSTR1_SubCategory.SUPECOM_9_5,
+ ),
+}
+
+
+class GSTR1_DataField(Enum):
+ TRANSACTION_TYPE = "transaction_type"
+ CUST_GSTIN = "customer_gstin"
+ ECOMMERCE_GSTIN = "ecommerce_gstin"
+ CUST_NAME = "customer_name"
+ DOC_DATE = "document_date"
+ DOC_NUMBER = "document_number"
+ DOC_TYPE = "document_type"
+ DOC_VALUE = "document_value"
+ POS = "place_of_supply"
+ DIFF_PERCENTAGE = "diff_percentage"
+ REVERSE_CHARGE = "reverse_charge"
+ TAXABLE_VALUE = "total_taxable_value"
+ ITEMS = "items"
+ IGST = "total_igst_amount"
+ CGST = "total_cgst_amount"
+ SGST = "total_sgst_amount"
+ CESS = "total_cess_amount"
+ TAX_RATE = "tax_rate"
+
+ SHIPPING_BILL_NUMBER = "shipping_bill_number"
+ SHIPPING_BILL_DATE = "shipping_bill_date"
+ SHIPPING_PORT_CODE = "shipping_port_code"
+
+ EXEMPTED_AMOUNT = "exempted_amount"
+ NIL_RATED_AMOUNT = "nil_rated_amount"
+ NON_GST_AMOUNT = "non_gst_amount"
+
+ HSN_CODE = "hsn_code"
+ DESCRIPTION = "description"
+ UOM = "uom"
+ QUANTITY = "quantity"
+
+ FROM_SR = "from_sr_no"
+ TO_SR = "to_sr_no"
+ TOTAL_COUNT = "total_count"
+ DRAFT_COUNT = "draft_count"
+ CANCELLED_COUNT = "cancelled_count"
+ NET_ISSUE = "net_issue"
+ UPLOAD_STATUS = "upload_status"
+
+
+class GSTR1_ItemField(Enum):
+ INDEX = "idx"
+ TAXABLE_VALUE = "taxable_value"
+ IGST = "igst_amount"
+ CGST = "cgst_amount"
+ SGST = "sgst_amount"
+ CESS = "cess_amount"
+ TAX_RATE = "tax_rate"
+ ITEM_DETAILS = "item_details"
+ ADDITIONAL_AMOUNT = "additional_amount"
+
+
+class GovDataField(Enum):
+ CUST_GSTIN = "ctin"
+ ECOMMERCE_GSTIN = "etin"
+ DOC_DATE = "idt"
+ DOC_NUMBER = "inum"
+ DOC_VALUE = "val"
+ POS = "pos"
+ DIFF_PERCENTAGE = "diff_percent"
+ REVERSE_CHARGE = "rchrg"
+ TAXABLE_VALUE = "txval"
+ ITEMS = "itms"
+ IGST = "iamt"
+ CGST = "camt"
+ SGST = "samt"
+ CESS = "csamt"
+ TAX_RATE = "rt"
+ ITEM_DETAILS = "itm_det"
+ SHIPPING_BILL_NUMBER = "sbnum"
+ SHIPPING_BILL_DATE = "sbdt"
+ SHIPPING_PORT_CODE = "sbpcode"
+ SUPPLY_TYPE = "sply_ty"
+ NET_TAXABLE_VALUE = "suppval"
+
+ EXEMPTED_AMOUNT = "expt_amt"
+ NIL_RATED_AMOUNT = "nil_amt"
+ NON_GST_AMOUNT = "ngsup_amt"
+
+ HSN_DATA = "data"
+ HSN_CODE = "hsn_sc"
+ DESCRIPTION = "desc"
+ UOM = "uqc"
+ QUANTITY = "qty"
+ ADVANCE_AMOUNT = "ad_amt"
+
+ INDEX = "num"
+ FROM_SR = "from"
+ TO_SR = "to"
+ TOTAL_COUNT = "totnum"
+ CANCELLED_COUNT = "cancel"
+ DOC_ISSUE_DETAILS = "doc_det"
+ DOC_ISSUE_NUMBER = "doc_num"
+ DOC_ISSUE_LIST = "docs"
+ NET_ISSUE = "net_issue"
+
+ INVOICE_TYPE = "inv_typ"
+ INVOICES = "inv"
+ EXPORT_TYPE = "exp_typ"
+ TYPE = "typ"
+
+ NOTE_TYPE = "ntty"
+ NOTE_NUMBER = "nt_num"
+ NOTE_DATE = "nt_dt"
+ NOTE_DETAILS = "nt"
+
+ SUPECOM_52 = "clttx"
+ SUPECOM_9_5 = "paytx"
+
+ FLAG = "flag"
+
+
+class GovExcelField(Enum):
+ CUST_GSTIN = "GSTIN/UIN of Recipient"
+ CUST_NAME = "Receiver Name"
+ INVOICE_NUMBER = "Invoice Number"
+ INVOICE_DATE = "Invoice date"
+ INVOICE_VALUE = "Invoice Value"
+ POS = "Place Of Supply"
+ REVERSE_CHARGE = "Reverse Charge"
+ DIFF_PERCENTAGE = "Applicable % of Tax Rate"
+ INVOICE_TYPE = "Invoice Type"
+ TAXABLE_VALUE = "Taxable Value"
+ ECOMMERCE_GSTIN = "E-Commerce GSTIN"
+ TAX_RATE = "Rate"
+ IGST = "Integrated Tax Amount"
+ CGST = "Central Tax Amount"
+ SGST = "State/UT Tax Amount"
+ CESS = "Cess Amount"
+
+ NOTE_NO = "Note Number"
+ NOTE_DATE = "Note Date"
+ NOTE_TYPE = "Note Type"
+ NOTE_VALUE = "Note Value"
+
+ PORT_CODE = "Port Code"
+ SHIPPING_BILL_NO = "Shipping Bill Number"
+ SHIPPING_BILL_DATE = "Shipping Bill Date"
+
+ DESCRIPTION = "Description"
+ # NIL_RATED = "Nil Rated Supplies"
+ # EXEMPTED = "Exempted (other than nil rated/non-GST supplies)"
+ # NON_GST = "Non-GST Supplies"
+
+ HSN_CODE = "HSN"
+ UOM = "UQC"
+ QUANTITY = "Total Quantity"
+ TOTAL_VALUE = "Total Value"
+
+
+class GovJsonKey(Enum):
+ """
+ Categories / Keys as per Govt JSON file
+ """
+
+ B2B = "b2b"
+ EXP = "exp"
+ B2CL = "b2cl"
+ B2CS = "b2cs"
+ NIL_EXEMPT = "nil"
+ CDNR = "cdnr"
+ CDNUR = "cdnur"
+ AT = "at"
+ TXP = "txpd"
+ HSN = "hsn"
+ DOC_ISSUE = "doc_issue"
+ SUPECOM = "supeco"
+ RET_SUM = "sec_sum"
+
+
+class GovExcelSheetName(Enum):
+ """
+ Categories / Worksheets as per Gov Excel file
+ """
+
+ B2B = "b2b, sez, de"
+ EXP = "exp"
+ B2CL = "b2cl"
+ B2CS = "b2cs"
+ NIL_EXEMPT = "exemp"
+ CDNR = "cdnr"
+ CDNUR = "cdnur"
+ AT = "at"
+ TXP = "atadj"
+ HSN = "hsn"
+ DOC_ISSUE = "docs"
+
+
+SUB_CATEGORY_GOV_CATEGORY_MAPPING = {
+ GSTR1_SubCategory.B2B_REGULAR: GovJsonKey.B2B,
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE: GovJsonKey.B2B,
+ GSTR1_SubCategory.SEZWP: GovJsonKey.B2B,
+ GSTR1_SubCategory.SEZWOP: GovJsonKey.B2B,
+ GSTR1_SubCategory.DE: GovJsonKey.B2B,
+ GSTR1_SubCategory.B2CL: GovJsonKey.B2CL,
+ GSTR1_SubCategory.EXPWP: GovJsonKey.EXP,
+ GSTR1_SubCategory.EXPWOP: GovJsonKey.EXP,
+ GSTR1_SubCategory.B2CS: GovJsonKey.B2CS,
+ GSTR1_SubCategory.NIL_EXEMPT: GovJsonKey.NIL_EXEMPT,
+ GSTR1_SubCategory.CDNR: GovJsonKey.CDNR,
+ GSTR1_SubCategory.CDNUR: GovJsonKey.CDNUR,
+ GSTR1_SubCategory.AT: GovJsonKey.AT,
+ GSTR1_SubCategory.TXP: GovJsonKey.TXP,
+ GSTR1_SubCategory.DOC_ISSUE: GovJsonKey.DOC_ISSUE,
+ GSTR1_SubCategory.HSN: GovJsonKey.HSN,
+ GSTR1_SubCategory.SUPECOM_52: GovJsonKey.SUPECOM,
+ GSTR1_SubCategory.SUPECOM_9_5: GovJsonKey.SUPECOM,
+}
+
+JSON_CATEGORY_EXCEL_CATEGORY_MAPPING = {
+ GovJsonKey.B2B.value: GovExcelSheetName.B2B.value,
+ GovJsonKey.EXP.value: GovExcelSheetName.EXP.value,
+ GovJsonKey.B2CL.value: GovExcelSheetName.B2CL.value,
+ GovJsonKey.B2CS.value: GovExcelSheetName.B2CS.value,
+ GovJsonKey.NIL_EXEMPT.value: GovExcelSheetName.NIL_EXEMPT.value,
+ GovJsonKey.CDNR.value: GovExcelSheetName.CDNR.value,
+ GovJsonKey.CDNUR.value: GovExcelSheetName.CDNUR.value,
+ GovJsonKey.AT.value: GovExcelSheetName.AT.value,
+ GovJsonKey.TXP.value: GovExcelSheetName.TXP.value,
+ GovJsonKey.HSN.value: GovExcelSheetName.HSN.value,
+ GovJsonKey.DOC_ISSUE.value: GovExcelSheetName.DOC_ISSUE.value,
+}
+
+
+class GSTR1_B2B_InvoiceType(Enum):
+ R = "Regular B2B"
+ SEWP = "SEZ supplies with payment"
+ SEWOP = "SEZ supplies without payment"
+ DE = "Deemed Exp"
+
+
+SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE = [
+ GSTR1_SubCategory.HSN.value,
+ GSTR1_SubCategory.DOC_ISSUE.value,
+ GSTR1_SubCategory.SUPECOM_52.value,
+ GSTR1_SubCategory.SUPECOM_9_5.value,
+]
+
+SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAX = [
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE.value,
+ *SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE,
+]
diff --git a/india_compliance/gst_india/utils/gstr/gstr_1.py b/india_compliance/gst_india/utils/gstr_1/gstr_1_data.py
similarity index 75%
rename from india_compliance/gst_india/utils/gstr/gstr_1.py
rename to india_compliance/gst_india/utils/gstr_1/gstr_1_data.py
index e0eebf75ca..17f7dbd9cc 100644
--- a/india_compliance/gst_india/utils/gstr/gstr_1.py
+++ b/india_compliance/gst_india/utils/gstr_1/gstr_1_data.py
@@ -1,8 +1,6 @@
# Copyright (c) 2024, Resilient Tech and contributors
# For license information, please see license.txt
-from enum import Enum
-
from pypika import Order
import frappe
@@ -10,114 +8,41 @@
from frappe.utils import getdate
from india_compliance.gst_india.utils import get_full_gst_uom
+from india_compliance.gst_india.utils.gstr_1 import (
+ CATEGORY_SUB_CATEGORY_MAPPING,
+ GSTR1_B2B_InvoiceType,
+ GSTR1_Category,
+ GSTR1_SubCategory,
+)
B2C_LIMIT = 2_50_000
-# TODO: Enum for Invoice Type
-
-
-class GSTR1_Categories(Enum):
- """
- Overview Page of GSTR-1
- """
-
- # Invoice Items Bifurcation
- B2B = "B2B, SEZ, DE"
- B2CL = "B2C (Large)"
- EXP = "Exports"
- B2CS = "B2C (Others)"
- NIL_EXEMPT = "Nil-Rated, Exempted, Non-GST"
- CDNR = "Credit/Debit Notes (Registered)"
- CDNUR = "Credit/Debit Notes (Unregistered)"
- # Other Categories
- AT = "Advances Received"
- TXP = "Advances Adjusted"
- DOC_ISSUE = "Document Issued"
- HSN = "HSN Summary"
-
-
-class GSTR1_SubCategories(Enum):
- """
- Summary Page of GSTR-1
- """
-
- # Invoice Items Bifurcation
- B2B_REGULAR = "B2B Regular" # Regular B2B
- B2B_REVERSE_CHARGE = "B2B Reverse Charge" # Regular B2B
- SEZWP = "SEZWP" # SEZ supplies with payment
- SEZWOP = "SEZWOP" # SEZ supplies without payment
- DE = "Deemed Exports" # Deemed Exp
- B2CL = "B2C (Large)" # NA
- EXPWP = "EXPWP" # WPAY
- EXPWOP = "EXPWOP" # WOPAY
- B2CS = "B2C (Others)" # NA
- NIL_RATED = "Nil-Rated" # Inter vs Intra & Regis vs UnRegis
- EXEMPTED = "Exempted" # Inter vs Intra & Regis vs UnRegis
- NON_GST = "Non-GST" # Inter vs Intra & Regis vs UnRegis
- CDNR = "CDNR" # Like B2B
- CDNUR = "CDNUR" # B2CL vs EXPWP vs EXPWOP
- # Other Sub-Categories
- # AT = "Advances Received"
- # TXP = "Advances Adjusted"
- # HSN = "HSN Summary"
- # DOC_ISSUE = "Document Issued"
-
-
-CATEGORY_SUB_CATEGORY_MAPPING = {
- GSTR1_Categories.B2B: (
- GSTR1_SubCategories.B2B_REGULAR,
- GSTR1_SubCategories.B2B_REVERSE_CHARGE,
- GSTR1_SubCategories.SEZWP,
- GSTR1_SubCategories.SEZWOP,
- GSTR1_SubCategories.DE,
- ),
- GSTR1_Categories.B2CL: (GSTR1_SubCategories.B2CL,),
- GSTR1_Categories.EXP: (GSTR1_SubCategories.EXPWP, GSTR1_SubCategories.EXPWOP),
- GSTR1_Categories.B2CS: (GSTR1_SubCategories.B2CS,),
- GSTR1_Categories.NIL_EXEMPT: (
- GSTR1_SubCategories.NIL_RATED,
- GSTR1_SubCategories.EXEMPTED,
- GSTR1_SubCategories.NON_GST,
- ),
- GSTR1_Categories.CDNR: (GSTR1_SubCategories.CDNR,),
- GSTR1_Categories.CDNUR: (GSTR1_SubCategories.CDNUR,),
-}
-
-SUB_CATEGORIES_DESCRIPTION = {
- GSTR1_SubCategories.SEZWP: "SEZ with payment",
- GSTR1_SubCategories.SEZWOP: "SEZ without payment",
- GSTR1_SubCategories.EXPWP: "Exports with payment",
- GSTR1_SubCategories.EXPWOP: "Exports without payment",
- GSTR1_SubCategories.CDNR: "Credit/Debit Notes (Registered)",
- GSTR1_SubCategories.CDNUR: "Credit/Debit Notes (Unregistered)",
-}
-
CATEGORY_CONDITIONS = {
- GSTR1_Categories.B2B.value: {
+ GSTR1_Category.B2B.value: {
"category": "is_b2b_invoice",
"sub_category": "set_for_b2b",
},
- GSTR1_Categories.B2CL.value: {
+ GSTR1_Category.B2CL.value: {
"category": "is_b2cl_invoice",
"sub_category": "set_for_b2cl",
},
- GSTR1_Categories.EXP.value: {
+ GSTR1_Category.EXP.value: {
"category": "is_export_invoice",
"sub_category": "set_for_exports",
},
- GSTR1_Categories.B2CS.value: {
+ GSTR1_Category.B2CS.value: {
"category": "is_b2cs_invoice",
"sub_category": "set_for_b2cs",
},
- GSTR1_Categories.NIL_EXEMPT.value: {
+ GSTR1_Category.NIL_EXEMPT.value: {
"category": "is_nil_rated_exempted_non_gst_invoice",
"sub_category": "set_for_nil_exp_non_gst",
},
- GSTR1_Categories.CDNR.value: {
+ GSTR1_Category.CDNR.value: {
"category": "is_cdnr_invoice",
"sub_category": "set_for_cdnr",
},
- GSTR1_Categories.CDNUR.value: {
+ GSTR1_Category.CDNUR.value: {
"category": "is_cdnur_invoice",
"sub_category": "set_for_cdnur",
},
@@ -145,8 +70,9 @@ def get_base_query(self):
.on(self.si.return_against == returned_si.name)
.select(
IfNull(self.si_item.item_code, self.si_item.item_name).as_("item_code"),
+ self.si_item.qty,
self.si_item.gst_hsn_code,
- self.si_item.uom,
+ self.si_item.stock_uom,
self.si.billing_address_gstin,
self.si.company_gstin,
self.si.customer_name,
@@ -371,20 +297,20 @@ def set_for_b2b(self, invoice):
def set_for_b2cl(self, invoice):
# NO INVOICE VALUE
- invoice.invoice_sub_category = GSTR1_SubCategories.B2CL.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.B2CL.value
def set_for_exports(self, invoice):
if invoice.is_export_with_gst:
- invoice.invoice_sub_category = GSTR1_SubCategories.EXPWP.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.EXPWP.value
invoice.invoice_type = "WPAY"
else:
- invoice.invoice_sub_category = GSTR1_SubCategories.EXPWOP.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.EXPWOP.value
invoice.invoice_type = "WOPAY"
def set_for_b2cs(self, invoice):
# NO INVOICE VALUE
- invoice.invoice_sub_category = GSTR1_SubCategories.B2CS.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.B2CS.value
def set_for_nil_exp_non_gst(self, invoice):
# INVOICE TYPE
@@ -394,24 +320,15 @@ def set_for_nil_exp_non_gst(self, invoice):
gst_registration = "registered" if is_registered else "unregistered"
supply_type = "Inter-State" if is_interstate else "Intra-State"
- invoice.invoice_type = f"{supply_type} to {gst_registration} persons"
-
- # INVOICE SUB CATEGORY
- if self.is_nil_rated(invoice):
- invoice.invoice_sub_category = GSTR1_SubCategories.NIL_RATED.value
-
- elif self.is_exempted(invoice):
- invoice.invoice_sub_category = GSTR1_SubCategories.EXEMPTED.value
-
- elif self.is_non_gst(invoice):
- invoice.invoice_sub_category = GSTR1_SubCategories.NON_GST.value
+ invoice.invoice_type = f"{supply_type} supplies to {gst_registration} persons"
+ invoice.invoice_sub_category = GSTR1_SubCategory.NIL_EXEMPT.value
def set_for_cdnr(self, invoice):
self._set_invoice_type_for_b2b_and_cdnr(invoice)
- invoice.invoice_sub_category = GSTR1_SubCategories.CDNR.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.CDNR.value
def set_for_cdnur(self, invoice):
- invoice.invoice_sub_category = GSTR1_SubCategories.CDNUR.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.CDNUR.value
if self.is_export(invoice):
if invoice.is_export_with_gst:
invoice.invoice_type = "EXPWP"
@@ -425,25 +342,25 @@ def set_for_cdnur(self, invoice):
def _set_invoice_type_for_b2b_and_cdnr(self, invoice):
if invoice.gst_category == "Deemed Export":
- invoice.invoice_type = "Deemed Exp"
- invoice.invoice_sub_category = GSTR1_SubCategories.DE.value
+ invoice.invoice_type = GSTR1_B2B_InvoiceType.DE.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.DE.value
elif invoice.gst_category == "SEZ":
if invoice.is_export_with_gst:
- invoice.invoice_type = "SEZ supplies with payment"
- invoice.invoice_sub_category = GSTR1_SubCategories.SEZWP.value
+ invoice.invoice_type = GSTR1_B2B_InvoiceType.SEWP.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.SEZWP.value
else:
- invoice.invoice_type = "SEZ supplies without payment"
- invoice.invoice_sub_category = GSTR1_SubCategories.SEZWOP.value
+ invoice.invoice_type = GSTR1_B2B_InvoiceType.SEWOP.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.SEZWOP.value
elif invoice.is_reverese_charge:
- invoice.invoice_type = "Regular B2B"
- invoice.invoice_sub_category = GSTR1_SubCategories.B2B_REVERSE_CHARGE.value
+ invoice.invoice_type = GSTR1_B2B_InvoiceType.R.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.B2B_REVERSE_CHARGE.value
else:
- invoice.invoice_type = "Regular B2B"
- invoice.invoice_sub_category = GSTR1_SubCategories.B2B_REGULAR.value
+ invoice.invoice_type = GSTR1_B2B_InvoiceType.R.value
+ invoice.invoice_sub_category = GSTR1_SubCategory.B2B_REGULAR.value
class GSTR1Invoices(GSTR1Query, GSTR1Subcategory):
@@ -460,11 +377,23 @@ def __init__(self, filters=None):
def process_invoices(self, invoices):
settings = frappe.get_cached_doc("GST Settings")
-
+ identified_uom = {}
for invoice in invoices:
self.invoice_conditions = {}
self.assign_categories(invoice)
- invoice["uom"] = get_full_gst_uom(invoice.get("uom"), settings)
+
+ if invoice.gst_hsn_code and invoice.gst_hsn_code.startswith("99"):
+ invoice["stock_uom"] = "OTH-OTHERS"
+ invoice["qty"] = 0
+ continue
+
+ stock_uom = invoice.get("stock_uom", "")
+ if stock_uom in identified_uom:
+ invoice["stock_uom"] = identified_uom[stock_uom]
+ else:
+ gst_uom = get_full_gst_uom(stock_uom, settings)
+ identified_uom[stock_uom] = gst_uom
+ invoice["stock_uom"] = gst_uom
def assign_categories(self, invoice):
@@ -507,7 +436,7 @@ def get_invoices_for_hsn_wise_summary(self):
query.gst_hsn_code,
query.gst_rate,
query.gst_treatment,
- query.uom,
+ query.stock_uom,
)
.orderby(
query.posting_date, query.invoice_no, query.item_code, order=Order.desc
@@ -544,7 +473,18 @@ def get_overview(self):
final_summary = []
sub_category_summary = self.get_sub_category_summary()
+ IGNORED_CATEGORIES = (
+ GSTR1_Category.AT,
+ GSTR1_Category.TXP,
+ GSTR1_Category.DOC_ISSUE,
+ GSTR1_Category.HSN,
+ GSTR1_Category.SUPECOM,
+ )
+
for category, sub_categories in CATEGORY_SUB_CATEGORY_MAPPING.items():
+ if category in IGNORED_CATEGORIES:
+ continue
+
category_summary = {
"description": category.value,
"no_of_records": 0,
@@ -572,9 +512,10 @@ def get_sub_category_summary(self):
summary = {}
- for category in GSTR1_SubCategories:
- summary[category.value] = {
- "description": SUB_CATEGORIES_DESCRIPTION.get(category, category.value),
+ for category in GSTR1_SubCategory:
+ category = category.value
+ summary[category] = {
+ "description": category,
"no_of_records": 0,
"indent": 1,
"unique_records": set(),
@@ -597,27 +538,19 @@ def get_sub_category_summary(self):
return summary
def update_overlaping_invoice_summary(self, sub_category_summary, final_summary):
- nil_exempt_non_gst = (
- GSTR1_SubCategories.NIL_RATED.value,
- GSTR1_SubCategories.EXEMPTED.value,
- GSTR1_SubCategories.NON_GST.value,
- )
+ nil_exempt = GSTR1_SubCategory.NIL_EXEMPT.value
# Get Unique Taxable Invoices
unique_invoices = set()
for category, row in sub_category_summary.items():
- if category in nil_exempt_non_gst:
+ if category == nil_exempt:
continue
unique_invoices.update(row["unique_records"])
# Get Overlaping Invoices
- overlaping_invoices = set()
- for category in nil_exempt_non_gst:
- category_invoices = sub_category_summary[category]["unique_records"]
-
- overlaping_invoices.update(category_invoices.intersection(unique_invoices))
- unique_invoices.update(category_invoices)
+ category_invoices = sub_category_summary[nil_exempt]["unique_records"]
+ overlaping_invoices = category_invoices.intersection(unique_invoices)
# Update Summary
if overlaping_invoices:
diff --git a/india_compliance/gst_india/utils/gstr_1/gstr_1_download.py b/india_compliance/gst_india/utils/gstr_1/gstr_1_download.py
new file mode 100644
index 0000000000..94a4d2102d
--- /dev/null
+++ b/india_compliance/gst_india/utils/gstr_1/gstr_1_download.py
@@ -0,0 +1,123 @@
+import frappe
+from frappe import _
+from frappe.utils import cint
+
+from india_compliance.gst_india.api_classes.returns import GSTR1API
+from india_compliance.gst_india.doctype.gstr_import_log.gstr_import_log import (
+ create_import_log,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_json_map import (
+ convert_to_internal_data_format,
+)
+
+UNFILED_ACTIONS = [
+ "B2B",
+ "B2CL",
+ "B2CS",
+ "CDNR",
+ "CDNUR",
+ "EXP",
+ "NIL",
+ "AT",
+ "TXP",
+ # "SUPECO", # 403 Forbidden TODO: Check when this is active
+ "HSNSUM",
+ "DOCISS",
+]
+
+FILED_ACTIONS = [*UNFILED_ACTIONS, "RETSUM"]
+
+
+def download_gstr1_json_data(gstr1_log):
+ """
+ Download GSTR-1 and Unfiled GSTR1 data from GST Portal
+ """
+ gstin = gstr1_log.gstin
+ return_period = gstr1_log.return_period
+
+ is_queued = False
+ json_data = frappe._dict()
+ api = GSTR1API(gstin)
+
+ if gstr1_log.filing_status == "Filed":
+ return_type = "GSTR1"
+ actions = FILED_ACTIONS
+ data_field = "filed"
+
+ else:
+ return_type = "Unfiled GSTR1"
+ actions = UNFILED_ACTIONS
+ data_field = "unfiled"
+
+ # download data
+ for action in actions:
+ response = api.get_gstr_1_data(action, return_period)
+
+ if response.error_type in ["otp_requested", "invalid_otp"]:
+ return response, None
+
+ if response.error_type == "no_docs_found":
+ continue
+
+ # Queued
+ if response.token:
+ create_import_log(
+ gstin,
+ return_type,
+ return_period,
+ classification=action,
+ request_id=response.token,
+ retry_after_mins=cint(response.est),
+ )
+ is_queued = True
+ continue
+
+ if response.error_type:
+ continue
+
+ json_data.update(response)
+
+ mapped_data = convert_to_internal_data_format(json_data)
+ gstr1_log.update_json_for(data_field, mapped_data, reset_reconcile=True)
+
+ if is_queued:
+ gstr1_log.update_status("Queued")
+
+ frappe.publish_realtime(
+ "gstr1_queued",
+ message={"gstin": gstin, "return_period": return_period},
+ user=frappe.session.user,
+ doctype="GSTR-1 Beta",
+ )
+
+ return mapped_data, is_queued
+
+
+def save_gstr_1(gstin, return_period, json_data, return_type):
+ if return_type == "GSTR1":
+ data_field = "filed"
+
+ elif return_type == "Unfiled GSTR1":
+ data_field = "unfiled"
+
+ if not json_data:
+ frappe.throw(
+ _(
+ "Data received seems to be invalid from the GST Portal. Please try"
+ " again or raise support ticket."
+ ),
+ title=_("Invalid Response Received."),
+ )
+
+ mapped_data = convert_to_internal_data_format(json_data)
+
+ gstr1_log = frappe.get_doc("GSTR-1 Log", f"{return_period}-{gstin}")
+ gstr1_log.update_json_for(data_field, mapped_data, overwrite=False)
+
+
+def save_gstr_1_filed_data(gstin, return_period, json_data):
+ save_gstr_1(gstin, return_period, json_data, "GSTR1")
+
+
+def save_gstr_1_unfiled_data(gstin, return_period, json_data):
+ save_gstr_1(gstin, return_period, json_data, "Unfiled GSTR1")
diff --git a/india_compliance/gst_india/utils/gstr_1/gstr_1_json_map.py b/india_compliance/gst_india/utils/gstr_1/gstr_1_json_map.py
new file mode 100644
index 0000000000..20666dac1e
--- /dev/null
+++ b/india_compliance/gst_india/utils/gstr_1/gstr_1_json_map.py
@@ -0,0 +1,2358 @@
+from datetime import datetime
+
+import frappe
+from frappe.utils import flt
+
+from india_compliance.gst_india.constants import STATE_NUMBERS, UOM_MAP
+from india_compliance.gst_india.report.gstr_1.gstr_1 import (
+ GSTR1DocumentIssuedSummary,
+ GSTR11A11BData,
+)
+from india_compliance.gst_india.utils import get_gst_accounts_by_type
+from india_compliance.gst_india.utils.__init__ import get_party_for_gstin
+from india_compliance.gst_india.utils.gstr_1 import (
+ CATEGORY_SUB_CATEGORY_MAPPING,
+ SUB_CATEGORY_GOV_CATEGORY_MAPPING,
+ SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAX,
+ SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE,
+ GovDataField,
+ GovJsonKey,
+ GSTR1_B2B_InvoiceType,
+ GSTR1_Category,
+ GSTR1_DataField,
+ GSTR1_ItemField,
+ GSTR1_SubCategory,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_data import GSTR1Invoices
+
+############################################################################################################
+### Map Govt JSON to Internal Data Structure ###############################################################
+############################################################################################################
+
+
+class GovDataMapper:
+ """
+ GST Developer API Documentation for Returns - https://developer.gst.gov.in/apiportal/taxpayer/returns
+
+ GSTR-1 JSON format - https://developer.gst.gov.in/pages/apiportal/data/Returns/GSTR1%20-%20Save%20GSTR1%20data/v4.0/GSTR1%20-%20Save%20GSTR1%20data%20attributes.xlsx
+ """
+
+ KEY_MAPPING = {}
+ # default item amounts
+ DEFAULT_ITEM_AMOUNTS = {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 0,
+ GSTR1_ItemField.IGST.value: 0,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+
+ FLOAT_FIELDS = {
+ GovDataField.DOC_VALUE.value,
+ GovDataField.TAXABLE_VALUE.value,
+ GovDataField.DIFF_PERCENTAGE.value,
+ GovDataField.IGST.value,
+ GovDataField.CGST.value,
+ GovDataField.SGST.value,
+ GovDataField.CESS.value,
+ GovDataField.NET_TAXABLE_VALUE.value,
+ GovDataField.EXEMPTED_AMOUNT.value,
+ GovDataField.NIL_RATED_AMOUNT.value,
+ GovDataField.NON_GST_AMOUNT.value,
+ GovDataField.QUANTITY.value,
+ GovDataField.ADVANCE_AMOUNT.value,
+ }
+
+ DISCARD_IF_ZERO_FIELDS = {
+ GovDataField.DIFF_PERCENTAGE.value,
+ }
+
+ def __init__(self):
+ self.set_total_defaults()
+
+ self.value_formatters_for_internal = {}
+ self.value_formatters_for_gov = {}
+ self.gstin_party_map = {}
+ # value formatting constants
+
+ self.STATE_NUMBERS = self.reverse_dict(STATE_NUMBERS)
+
+ def format_data(
+ self, data: dict, default_data: dict = None, for_gov: bool = False
+ ) -> dict:
+ """
+ Objective: Convert Object from one format to another.
+ eg: Govt JSON to Internal Data Structure
+
+ Args:
+ data (dict): Data to be converted
+ default_data (dict, optional): Default Data to be added. Hardcoded values.
+ for_gov (bool, optional): If the data is to be converted to Govt JSON. Defaults to False.
+ else it will be converted to Internal Data Structure.
+
+ Steps:
+ 1. Use key mapping to map the keys from one format to another.
+ 2. Use value formatters to format the values of the keys.
+ 3. Round values
+ """
+ output = {}
+
+ if default_data:
+ output.update(default_data)
+
+ key_mapping = self.KEY_MAPPING.copy()
+
+ if for_gov:
+ key_mapping = self.reverse_dict(key_mapping)
+
+ value_formatters = (
+ self.value_formatters_for_gov
+ if for_gov
+ else self.value_formatters_for_internal
+ )
+
+ for old_key, new_key in key_mapping.items():
+ invoice_data_value = data.get(old_key, "")
+
+ if not for_gov and old_key == "flag":
+ continue
+
+ if new_key in self.DISCARD_IF_ZERO_FIELDS and not invoice_data_value:
+ continue
+
+ if not (invoice_data_value or invoice_data_value == 0):
+ # continue if value is None or empty object
+ continue
+
+ value_formatter = value_formatters.get(old_key)
+
+ if callable(value_formatter):
+ output[new_key] = value_formatter(invoice_data_value, data)
+ else:
+ output[new_key] = invoice_data_value
+
+ if new_key in self.FLOAT_FIELDS:
+ output[new_key] = flt(output[new_key], 2)
+
+ return output
+
+ # common utils
+
+ def update_totals(self, invoice, items):
+ """
+ Update item totals to the invoice row
+ """
+ total_data = self.TOTAL_DEFAULTS.copy()
+
+ for item in items:
+ for field, value in item.items():
+ total_field = f"total_{field}"
+
+ if total_field not in total_data:
+ continue
+
+ invoice[total_field] = invoice.setdefault(total_field, 0) + value
+
+ def set_total_defaults(self):
+ self.TOTAL_DEFAULTS = {
+ f"total_{key}": 0 for key in self.DEFAULT_ITEM_AMOUNTS.keys()
+ }
+
+ def reverse_dict(self, data):
+ return {v: k for k, v in data.items()}
+
+ # common value formatters
+ def map_place_of_supply(self, pos, *args):
+ if pos.isnumeric():
+ return f"{pos}-{self.STATE_NUMBERS.get(pos)}"
+
+ return pos.split("-")[0]
+
+ def format_item_for_internal(self, items, *args):
+ return [
+ {
+ **self.DEFAULT_ITEM_AMOUNTS.copy(),
+ **self.format_data(item.get(GovDataField.ITEM_DETAILS.value, {})),
+ }
+ for item in items
+ ]
+
+ def format_item_for_gov(self, items, *args):
+ return [
+ {
+ GovDataField.INDEX.value: index + 1,
+ GovDataField.ITEM_DETAILS.value: self.format_data(item, for_gov=True),
+ }
+ for index, item in enumerate(items)
+ ]
+
+ def guess_customer_name(self, gstin):
+ if party := self.gstin_party_map.get(gstin):
+ return party
+
+ return self.gstin_party_map.setdefault(
+ gstin, get_party_for_gstin(gstin, "Customer") or "Unknown"
+ )
+
+ def format_date_for_internal(self, date, *args):
+ return datetime.strptime(date, "%d-%m-%Y").strftime("%Y-%m-%d")
+
+ def format_date_for_gov(self, date, *args):
+ return datetime.strptime(date, "%Y-%m-%d").strftime("%d-%m-%Y")
+
+
+class B2B(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ [
+ {
+ 'ctin': '24AANFA2641L1ZF',
+ 'inv': [
+ {
+ 'inum': 'S008400',
+ 'itms': [
+ {'num': 1, 'itm_det': {'txval': 10000,
+ ...
+ }}
+ ]
+ }
+ ...
+ ]
+ }
+ ]
+
+ Internal Data Format:
+ {
+ 'B2B Regular': {'S008400': {
+ 'customer_gstin': '24AANFA2641L1ZF',
+ 'document_number': 'S008400',
+ 'items': [
+ {
+ 'taxable_value': 10000,
+ ...
+ }
+ ],
+ ...
+ }}
+ }
+
+ """
+
+ KEY_MAPPING = {
+ # GovDataFields.CUST_GSTIN.value: DataFields.CUST_GSTIN.value,
+ # GovDataFields.INVOICES.value: "invoices",
+ GovDataField.FLAG.value: "flag",
+ GovDataField.DOC_NUMBER.value: GSTR1_DataField.DOC_NUMBER.value,
+ GovDataField.DOC_DATE.value: GSTR1_DataField.DOC_DATE.value,
+ GovDataField.DOC_VALUE.value: GSTR1_DataField.DOC_VALUE.value,
+ GovDataField.POS.value: GSTR1_DataField.POS.value,
+ GovDataField.REVERSE_CHARGE.value: GSTR1_DataField.REVERSE_CHARGE.value,
+ # GovDataFields.ECOMMERCE_GSTIN.value: GSTR1_DataFields.ECOMMERCE_GSTIN.value,
+ GovDataField.INVOICE_TYPE.value: GSTR1_DataField.DOC_TYPE.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ # GovDataFields.INDEX.value: ItemFields.INDEX.value,
+ GovDataField.ITEM_DETAILS.value: GSTR1_ItemField.ITEM_DETAILS.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GovDataField.CGST.value: GSTR1_ItemField.CGST.value,
+ GovDataField.SGST.value: GSTR1_ItemField.SGST.value,
+ GovDataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ # value formatting constants
+ DOCUMENT_CATEGORIES = {
+ "R": GSTR1_B2B_InvoiceType.R.value,
+ "SEWP": GSTR1_B2B_InvoiceType.SEWP.value,
+ "SEWOP": GSTR1_B2B_InvoiceType.SEWOP.value,
+ "DE": GSTR1_B2B_InvoiceType.DE.value,
+ }
+
+ SUBCATEGORIES = {
+ # "B2B": GSTR1_SubCategories.B2B_REGULAR.value,
+ # "B2B": GSTR1_SubCategories.B2B_REVERSE_CHARGE.value,
+ "SEWP": GSTR1_SubCategory.SEZWP.value,
+ "SEWOP": GSTR1_SubCategory.SEZWOP.value,
+ "DE": GSTR1_SubCategory.DE.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.INVOICE_TYPE.value: self.document_category_mapping,
+ GovDataField.POS.value: self.map_place_of_supply,
+ GovDataField.DOC_DATE.value: self.format_date_for_internal,
+ }
+
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.DOC_TYPE.value: self.document_category_mapping,
+ GSTR1_DataField.POS.value: self.map_place_of_supply,
+ GSTR1_DataField.DOC_DATE.value: self.format_date_for_gov,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ """
+ Objective: Convert Govt JSON to Internal Data Structure
+ Args:
+ input_data (list): Govt JSON Data
+ Returns:
+ dict: Internal Data Structure
+ """
+
+ output = {}
+
+ for customer_data in input_data:
+ customer_gstin = customer_data.get(GovDataField.CUST_GSTIN.value)
+
+ default_invoice_data = {
+ GSTR1_DataField.CUST_GSTIN.value: customer_gstin,
+ GSTR1_DataField.CUST_NAME.value: self.guess_customer_name(
+ customer_gstin
+ ),
+ }
+
+ for invoice in customer_data.get(GovDataField.INVOICES.value):
+ invoice_data = self.format_data(invoice, default_invoice_data)
+ self.update_totals(
+ invoice_data, invoice_data.get(GSTR1_DataField.ITEMS.value)
+ )
+
+ subcategory_data = output.setdefault(
+ self.get_document_subcategory(invoice), {}
+ )
+ subcategory_data[invoice_data[GSTR1_DataField.DOC_NUMBER.value]] = (
+ invoice_data
+ )
+
+ return output
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ """
+ Objective: Convert Internal Data Structure to Govt JSON
+ Args:
+ input_data (dict): Internal Data Structure
+ Returns:
+ list: Govt JSON Data
+ """
+ customer_data = {}
+
+ self.DOCUMENT_CATEGORIES = self.reverse_dict(self.DOCUMENT_CATEGORIES)
+
+ for invoice in input_data:
+ customer = customer_data.setdefault(
+ invoice[GSTR1_DataField.CUST_GSTIN.value],
+ {
+ GovDataField.CUST_GSTIN.value: invoice[
+ GSTR1_DataField.CUST_GSTIN.value
+ ],
+ GovDataField.INVOICES.value: [],
+ },
+ )
+
+ customer[GovDataField.INVOICES.value].append(
+ self.format_data(invoice, for_gov=True)
+ )
+
+ return list(customer_data.values())
+
+ def get_document_subcategory(self, invoice_data):
+ if invoice_data.get(GovDataField.INVOICE_TYPE.value) in self.SUBCATEGORIES:
+ return self.SUBCATEGORIES[invoice_data[GovDataField.INVOICE_TYPE.value]]
+
+ if invoice_data.get(GovDataField.REVERSE_CHARGE.value) == "Y":
+ return GSTR1_SubCategory.B2B_REVERSE_CHARGE.value
+
+ return GSTR1_SubCategory.B2B_REGULAR.value
+
+ # value formatting methods
+
+ def document_category_mapping(self, sub_category, data):
+ return self.DOCUMENT_CATEGORIES.get(sub_category, sub_category)
+
+
+class B2CL(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'pos': '05',
+ 'inv': [
+ {
+ 'inum': '92661',
+ 'itms': [
+ {'num': 1,'itm_det': {'txval': 10000,
+ ...
+ }},
+ ...
+ ]
+ }
+ ...
+ ],
+ ...
+ }
+
+ Internal Data Format:
+
+ {
+ 'B2C (Large)': {
+ '92661': {
+ 'place_of_supply': '05-Uttarakhand',
+ 'document_number': '92661',
+ 'items': [
+ {
+ 'taxable_value': 10000,
+ ...
+ },
+ ...
+ ],
+ 'total_taxable_value': 10000,
+ ...
+ }
+ ...
+ }
+ }
+ """
+
+ DOCUMENT_CATEGORY = "B2C (Large)"
+ SUBCATEGORY = GSTR1_SubCategory.B2CL.value
+ DEFAULT_ITEM_AMOUNTS = {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 0,
+ GSTR1_ItemField.IGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+ KEY_MAPPING = {
+ # GovDataFields.POS.value: DataFields.POS.value,
+ # GovDataFields.INVOICES.value: "invoices",
+ GovDataField.FLAG.value: "flag",
+ GovDataField.DOC_NUMBER.value: GSTR1_DataField.DOC_NUMBER.value,
+ GovDataField.DOC_DATE.value: GSTR1_DataField.DOC_DATE.value,
+ GovDataField.DOC_VALUE.value: GSTR1_DataField.DOC_VALUE.value,
+ # GovDataFields.ECOMMERCE_GSTIN.value: GSTR1_DataFields.ECOMMERCE_GSTIN.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ # GovDataFields.INDEX.value: ItemFields.INDEX.value,
+ GovDataField.ITEM_DETAILS.value: GSTR1_ItemField.ITEM_DETAILS.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GovDataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.DOC_DATE.value: self.format_date_for_internal,
+ }
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.DOC_DATE.value: self.format_date_for_gov,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for pos_data in input_data:
+ pos = self.map_place_of_supply(pos_data.get(GovDataField.POS.value))
+
+ default_invoice_data = {
+ GSTR1_DataField.POS.value: pos,
+ GSTR1_DataField.DOC_TYPE.value: self.DOCUMENT_CATEGORY,
+ }
+
+ for invoice in pos_data.get(GovDataField.INVOICES.value):
+ invoice_level_data = self.format_data(invoice, default_invoice_data)
+ self.update_totals(
+ invoice_level_data,
+ invoice_level_data.get(GSTR1_DataField.ITEMS.value),
+ )
+
+ output[invoice_level_data[GSTR1_DataField.DOC_NUMBER.value]] = (
+ invoice_level_data
+ )
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ pos_data = {}
+
+ for invoice in input_data:
+ pos = pos_data.setdefault(
+ invoice[GSTR1_DataField.POS.value],
+ {
+ GovDataField.POS.value: self.map_place_of_supply(
+ invoice[GSTR1_DataField.POS.value]
+ ),
+ GovDataField.INVOICES.value: [],
+ },
+ )
+
+ pos[GovDataField.INVOICES.value].append(
+ self.format_data(invoice, for_gov=True)
+ )
+
+ return list(pos_data.values())
+
+
+class Exports(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'exp_typ': 'WPAY',
+ 'inv': [
+ {
+ 'inum': '81542',
+ 'val': 995048.36,
+ 'itms': [
+ {
+ 'txval': 10000,
+ ...
+ },
+ ...
+ ],
+ ...
+ },
+ ...
+ ]
+ }
+
+ Internal Data Format:
+ {
+ 'Export With Payment of Tax': {
+ '81542': {
+ 'document_number': '81542',
+ 'document_value': 995048.36,
+ 'items': [
+ {
+ 'taxable_value': 10000,
+ ...
+ },
+ ...
+ ],
+ 'total_taxable_value': 10000,
+ ...
+ },
+ ...
+ }
+ }
+ """
+
+ DEFAULT_ITEM_AMOUNTS = {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 0,
+ GSTR1_ItemField.IGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+ KEY_MAPPING = {
+ # GovDataFields.POS.value: DataFields.POS.value,
+ # GovDataFields.INVOICES.value: "invoices",
+ GovDataField.FLAG.value: "flag",
+ # GovDataFields.EXPORT_TYPE.value: DataFields.DOC_TYPE.value,
+ GovDataField.DOC_NUMBER.value: GSTR1_DataField.DOC_NUMBER.value,
+ GovDataField.DOC_DATE.value: GSTR1_DataField.DOC_DATE.value,
+ GovDataField.DOC_VALUE.value: GSTR1_DataField.DOC_VALUE.value,
+ GovDataField.SHIPPING_PORT_CODE.value: GSTR1_DataField.SHIPPING_PORT_CODE.value,
+ GovDataField.SHIPPING_BILL_NUMBER.value: GSTR1_DataField.SHIPPING_BILL_NUMBER.value,
+ GovDataField.SHIPPING_BILL_DATE.value: GSTR1_DataField.SHIPPING_BILL_DATE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GovDataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ SUBCATEGORIES = {
+ "WPAY": GSTR1_SubCategory.EXPWP.value,
+ "WOPAY": GSTR1_SubCategory.EXPWOP.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.DOC_DATE.value: self.format_date_for_internal,
+ GovDataField.SHIPPING_BILL_DATE.value: self.format_date_for_internal,
+ }
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.DOC_DATE.value: self.format_date_for_gov,
+ GSTR1_DataField.SHIPPING_BILL_DATE.value: self.format_date_for_gov,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for export_category in input_data:
+ document_type = export_category.get(GovDataField.EXPORT_TYPE.value)
+ subcategory_data = output.setdefault(
+ self.SUBCATEGORIES.get(document_type, document_type), {}
+ )
+
+ default_invoice_data = {
+ GSTR1_DataField.DOC_TYPE.value: document_type,
+ }
+
+ for invoice in export_category.get(GovDataField.INVOICES.value):
+ invoice_level_data = self.format_data(invoice, default_invoice_data)
+
+ self.update_totals(
+ invoice_level_data,
+ invoice_level_data.get(GSTR1_DataField.ITEMS.value),
+ )
+ subcategory_data[
+ invoice_level_data[GSTR1_DataField.DOC_NUMBER.value]
+ ] = invoice_level_data
+
+ return output
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ export_category_wise_data = {}
+
+ for invoice in input_data:
+ export_category = export_category_wise_data.setdefault(
+ invoice[GSTR1_DataField.DOC_TYPE.value],
+ {
+ GovDataField.EXPORT_TYPE.value: invoice[
+ GSTR1_DataField.DOC_TYPE.value
+ ],
+ GovDataField.INVOICES.value: [],
+ },
+ )
+
+ export_category[GovDataField.INVOICES.value].append(
+ self.format_data(invoice, for_gov=True)
+ )
+
+ return list(export_category_wise_data.values())
+
+ def format_item_for_internal(self, items, *args):
+ return [
+ {
+ **self.DEFAULT_ITEM_AMOUNTS.copy(),
+ **self.format_data(item),
+ }
+ for item in items
+ ]
+
+ def format_item_for_gov(self, items, *args):
+ return [self.format_data(item, for_gov=True) for item in items]
+
+
+class B2CS(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ [
+ {
+ 'typ': 'OE',
+ 'pos': '05',
+ 'txval': 110,
+ ...
+ },
+ ...
+ ]
+
+ Internal Data Format:
+ {
+ 'B2C (Others)': {
+ '05-Uttarakhand - 5.0': [
+ {
+ 'total_taxable_value': 110,
+ 'document_type': 'OE',
+ 'place_of_supply': '05-Uttarakhand',
+ ...
+ },
+ ...
+ ],
+ ...
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.B2CS.value
+ KEY_MAPPING = {
+ GovDataField.FLAG.value: "flag",
+ # GovDataFields.SUPPLY_TYPE.value: "supply_type",
+ GovDataField.TAXABLE_VALUE.value: GSTR1_DataField.TAXABLE_VALUE.value,
+ GovDataField.TYPE.value: GSTR1_DataField.DOC_TYPE.value,
+ # GovDataFields.ECOMMERCE_GSTIN.value: GSTR1_DataFields.ECOMMERCE_GSTIN.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.POS.value: GSTR1_DataField.POS.value,
+ GovDataField.TAX_RATE.value: GSTR1_DataField.TAX_RATE.value,
+ GovDataField.IGST.value: GSTR1_DataField.IGST.value,
+ GovDataField.CGST.value: GSTR1_DataField.CGST.value,
+ GovDataField.SGST.value: GSTR1_DataField.SGST.value,
+ GovDataField.CESS.value: GSTR1_DataField.CESS.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.POS.value: self.map_place_of_supply,
+ }
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.POS.value: self.map_place_of_supply,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for invoice in input_data:
+ invoice_data = self.format_data(invoice)
+
+ output.setdefault(
+ " - ".join(
+ (
+ invoice_data.get(GSTR1_DataField.POS.value, ""),
+ str(flt(invoice_data.get(GSTR1_DataField.TAX_RATE.value, ""))),
+ # invoice_data.get(GSTR1_DataFields.ECOMMERCE_GSTIN.value, ""),
+ )
+ ),
+ [],
+ ).append(invoice_data)
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ self.company_gstin = kwargs.get("company_gstin", "")
+ return [self.format_data(invoice, for_gov=True) for invoice in input_data]
+
+ def format_data(self, data, default_data=None, for_gov=False):
+ data = super().format_data(data, default_data, for_gov)
+ if not for_gov:
+ return data
+
+ data[GovDataField.SUPPLY_TYPE.value] = (
+ "INTRA"
+ if data[GovDataField.POS.value] == self.company_gstin[:2]
+ else "INTER"
+ )
+ return data
+
+
+class NilRated(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'inv': [
+ {
+ 'sply_ty': 'INTRB2B',
+ 'expt_amt': 123.45,
+ 'nil_amt': 1470.85,
+ 'ngsup_amt': 1258.5
+ }
+ ]
+ }
+
+ Internal Data Format:
+ {
+ 'Nil-Rated, Exempted, Non-GST': {
+ 'Inter-State supplies to registered persons': [
+ {
+ 'document_type': 'Inter-State supplies to registered persons',
+ 'exempted_amount': 123.45,
+ 'nil_rated_amount': 1470.85,
+ 'non_gst_amount': 1258.5,
+ 'total_taxable_value': 2852.8
+ }
+ ]
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.NIL_EXEMPT.value
+ KEY_MAPPING = {
+ GovDataField.SUPPLY_TYPE.value: GSTR1_DataField.DOC_TYPE.value,
+ GovDataField.EXEMPTED_AMOUNT.value: GSTR1_DataField.EXEMPTED_AMOUNT.value,
+ GovDataField.NIL_RATED_AMOUNT.value: GSTR1_DataField.NIL_RATED_AMOUNT.value,
+ GovDataField.NON_GST_AMOUNT.value: GSTR1_DataField.NON_GST_AMOUNT.value,
+ }
+
+ DOCUMENT_CATEGORIES = {
+ "INTRB2B": "Inter-State supplies to registered persons",
+ "INTRB2C": "Inter-State supplies to unregistered persons",
+ "INTRAB2B": "Intra-State supplies to registered persons",
+ "INTRAB2C": "Intra-State supplies to unregistered persons",
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.SUPPLY_TYPE.value: self.document_category_mapping
+ }
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.DOC_TYPE.value: self.document_category_mapping
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for invoice in input_data[GovDataField.INVOICES.value]:
+ invoice_data = self.format_data(invoice)
+
+ if not invoice_data:
+ continue
+
+ output.setdefault(
+ invoice_data.get(GSTR1_DataField.DOC_TYPE.value), []
+ ).append(invoice_data)
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ self.DOCUMENT_CATEGORIES = self.reverse_dict(self.DOCUMENT_CATEGORIES)
+
+ return {
+ GovDataField.INVOICES.value: [
+ self.format_data(invoice, for_gov=True) for invoice in input_data
+ ]
+ }
+
+ def format_data(self, data, default_data=None, for_gov=False):
+ invoice_data = super().format_data(data, default_data, for_gov)
+
+ if for_gov:
+ return invoice_data
+
+ # No need to discard if zero fields
+ amounts = [
+ invoice_data.get(GSTR1_DataField.EXEMPTED_AMOUNT.value, 0),
+ invoice_data.get(GSTR1_DataField.NIL_RATED_AMOUNT.value, 0),
+ invoice_data.get(GSTR1_DataField.NON_GST_AMOUNT.value, 0),
+ ]
+
+ if all(amount == 0 for amount in amounts):
+ return
+
+ invoice_data[GSTR1_DataField.TAXABLE_VALUE.value] = sum(amounts)
+ return invoice_data
+
+ # value formatters
+ def document_category_mapping(self, doc_category, data):
+ return self.DOCUMENT_CATEGORIES.get(doc_category, doc_category)
+
+
+class CDNR(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ [
+ {
+ 'ctin': '24AANFA2641L1ZF',
+ 'nt': [
+ {
+ 'ntty': 'C',
+ 'nt_num': '533515',
+ 'val': 123123,
+ 'itms': [
+ {'num': 1,'itm_det': {'txval': 5225.28,
+ ...
+ }},
+ ...
+ ],
+ ...
+ },
+ ...
+ ]
+ },
+ ...
+ ]
+
+ Internal Data Format:
+ {
+ 'Credit/Debit Notes (Registered)': {
+ '533515': {
+ 'transaction_type': 'Credit Note',
+ 'document_number': '533515',
+ 'items': [
+ {
+ 'taxable_value': -5225.28,
+ ...
+ },
+ ...
+ ],
+ 'total_taxable_value': -10450.56,
+ ...
+ },
+ ...
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.CDNR.value
+ KEY_MAPPING = {
+ # GovDataFields.CUST_GSTIN.value: DataFields.CUST_GSTIN.value,
+ GovDataField.FLAG.value: "flag",
+ # GovDataFields.NOTE_DETAILS.value: "credit_debit_note_details",
+ GovDataField.NOTE_TYPE.value: GSTR1_DataField.TRANSACTION_TYPE.value,
+ GovDataField.NOTE_NUMBER.value: GSTR1_DataField.DOC_NUMBER.value,
+ GovDataField.NOTE_DATE.value: GSTR1_DataField.DOC_DATE.value,
+ GovDataField.POS.value: GSTR1_DataField.POS.value,
+ GovDataField.REVERSE_CHARGE.value: GSTR1_DataField.REVERSE_CHARGE.value,
+ GovDataField.INVOICE_TYPE.value: GSTR1_DataField.DOC_TYPE.value,
+ GovDataField.DOC_VALUE.value: GSTR1_DataField.DOC_VALUE.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ # GovDataFields.INDEX.value: ItemFields.INDEX.value,
+ # GovDataFields.ITEM_DETAILS.value: "item_details",
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GovDataField.SGST.value: GSTR1_ItemField.SGST.value,
+ GovDataField.CGST.value: GSTR1_ItemField.CGST.value,
+ GovDataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ DOCUMENT_CATEGORIES = {
+ "R": "Regular B2B",
+ "SEWP": "SEZ supplies with payment",
+ "SEWOP": "SEZ supplies without payment",
+ "DE": "Deemed Exports",
+ }
+
+ DOCUMENT_TYPES = {
+ "C": "Credit Note",
+ "D": "Debit Note",
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.NOTE_TYPE.value: self.document_type_mapping,
+ GovDataField.POS.value: self.map_place_of_supply,
+ GovDataField.INVOICE_TYPE.value: self.document_category_mapping,
+ GovDataField.DOC_VALUE.value: self.format_doc_value,
+ GovDataField.NOTE_DATE.value: self.format_date_for_internal,
+ }
+
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.TRANSACTION_TYPE.value: self.document_type_mapping,
+ GSTR1_DataField.POS.value: self.map_place_of_supply,
+ GSTR1_DataField.DOC_TYPE.value: self.document_category_mapping,
+ GSTR1_DataField.DOC_VALUE.value: lambda val, *args: abs(val),
+ GSTR1_DataField.DOC_DATE.value: self.format_date_for_gov,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for customer_data in input_data:
+ customer_gstin = customer_data.get(GovDataField.CUST_GSTIN.value)
+
+ for document in customer_data.get(GovDataField.NOTE_DETAILS.value):
+ document_data = self.format_data(
+ document,
+ {
+ GSTR1_DataField.CUST_GSTIN.value: customer_gstin,
+ GSTR1_DataField.CUST_NAME.value: self.guess_customer_name(
+ customer_gstin
+ ),
+ },
+ )
+ self.update_totals(
+ document_data, document_data.get(GSTR1_DataField.ITEMS.value)
+ )
+ output[document_data[GSTR1_DataField.DOC_NUMBER.value]] = document_data
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ customer_data = {}
+
+ self.DOCUMENT_CATEGORIES = self.reverse_dict(self.DOCUMENT_CATEGORIES)
+ self.DOCUMENT_TYPES = self.reverse_dict(self.DOCUMENT_TYPES)
+
+ for document in input_data:
+ customer_gstin = document[GSTR1_DataField.CUST_GSTIN.value]
+ customer = customer_data.setdefault(
+ customer_gstin,
+ {
+ GovDataField.CUST_GSTIN.value: customer_gstin,
+ GovDataField.NOTE_DETAILS.value: [],
+ },
+ )
+ customer[GovDataField.NOTE_DETAILS.value].append(
+ self.format_data(document, for_gov=True)
+ )
+
+ return list(customer_data.values())
+
+ def format_item_for_internal(self, items, *args):
+ formatted_items = super().format_item_for_internal(items, *args)
+
+ data = args[0]
+ if data[GovDataField.NOTE_TYPE.value] == "D":
+ return formatted_items
+
+ # for credit notes amounts -ve
+ for item in formatted_items:
+ item.update(
+ {
+ key: value * -1
+ for key, value in item.items()
+ if key in list(self.DEFAULT_ITEM_AMOUNTS.keys())
+ }
+ )
+
+ return formatted_items
+
+ def format_item_for_gov(self, items, *args):
+ keys = set((self.DEFAULT_ITEM_AMOUNTS.keys()))
+ # for credit notes amounts -ve
+ for item in items:
+ for key, value in item.items():
+ if key in keys:
+ item[key] = abs(value)
+
+ return super().format_item_for_gov(items, *args)
+
+ def document_type_mapping(self, doc_type, data):
+ return self.DOCUMENT_TYPES.get(doc_type, doc_type)
+
+ def document_category_mapping(self, doc_category, data):
+ return self.DOCUMENT_CATEGORIES.get(doc_category, doc_category)
+
+ def format_doc_value(self, value, data):
+ return value * -1 if data[GovDataField.NOTE_TYPE.value] == "C" else value
+
+
+class CDNUR(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ [
+ {
+ 'ntty': 'C',
+ 'nt_num': '533515',
+ 'itms': [
+ {'num': 1,'itm_det': { 'txval': 5225.28,
+ ...
+ }},
+ ...
+ ],
+ ...
+ },
+ ...
+ ]
+
+ Internal Data Format:
+ {
+ 'Credit/Debit Notes (Unregistered)': {
+ '533515': {
+ 'transaction_type': 'Credit Note',
+ 'document_number': '533515',
+ 'items': [
+ {
+ 'taxable_value': -5225.28,
+ ...
+ }
+ ],
+ 'total_taxable_value': -5225.28,
+ ...
+ },
+ ...
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.CDNUR.value
+ DEFAULT_ITEM_AMOUNTS = {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 0,
+ GSTR1_ItemField.IGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+ KEY_MAPPING = {
+ GovDataField.FLAG.value: "flag",
+ GovDataField.TYPE.value: GSTR1_DataField.DOC_TYPE.value,
+ GovDataField.NOTE_TYPE.value: GSTR1_DataField.TRANSACTION_TYPE.value,
+ GovDataField.NOTE_NUMBER.value: GSTR1_DataField.DOC_NUMBER.value,
+ GovDataField.NOTE_DATE.value: GSTR1_DataField.DOC_DATE.value,
+ GovDataField.DOC_VALUE.value: GSTR1_DataField.DOC_VALUE.value,
+ GovDataField.POS.value: GSTR1_DataField.POS.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GovDataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+ DOCUMENT_TYPES = {
+ "C": "Credit Note",
+ "D": "Debit Note",
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.NOTE_TYPE.value: self.document_type_mapping,
+ GovDataField.POS.value: self.map_place_of_supply,
+ GovDataField.DOC_VALUE.value: self.format_doc_value,
+ GovDataField.NOTE_DATE.value: self.format_date_for_internal,
+ }
+
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.TRANSACTION_TYPE.value: self.document_type_mapping,
+ GSTR1_DataField.POS.value: self.map_place_of_supply,
+ GSTR1_DataField.DOC_VALUE.value: lambda x, *args: abs(x),
+ GSTR1_DataField.DOC_DATE.value: self.format_date_for_gov,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for invoice in input_data:
+ invoice_data = self.format_data(invoice)
+ self.update_totals(
+ invoice_data, invoice_data.get(GSTR1_DataField.ITEMS.value)
+ )
+ output[invoice_data[GSTR1_DataField.DOC_NUMBER.value]] = invoice_data
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ self.DOCUMENT_TYPES = self.reverse_dict(self.DOCUMENT_TYPES)
+ return [self.format_data(invoice, for_gov=True) for invoice in input_data]
+
+ def format_item_for_internal(self, items, *args):
+ formatted_items = super().format_item_for_internal(items, *args)
+
+ data = args[0]
+ if data[GovDataField.NOTE_TYPE.value] == "D":
+ return formatted_items
+
+ # for credit notes amounts -ve
+ for item in formatted_items:
+ item.update(
+ {
+ key: value * -1
+ for key, value in item.items()
+ if key in list(self.DEFAULT_ITEM_AMOUNTS.keys())
+ }
+ )
+
+ return formatted_items
+
+ def format_item_for_gov(self, items, *args):
+ keys = set(self.DEFAULT_ITEM_AMOUNTS.keys())
+ # for credit notes amounts -ve
+ for item in items:
+ for key, value in item.items():
+ if key in keys:
+ item[key] = abs(value)
+
+ return super().format_item_for_gov(items, *args)
+
+ # value formatters
+ def document_type_mapping(self, doc_type, data):
+ return self.DOCUMENT_TYPES.get(doc_type, doc_type)
+
+ def format_doc_value(self, value, data):
+ return value * -1 if data[GovDataField.NOTE_TYPE.value] == "C" else value
+
+
+class HSNSUM(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'data': [
+ {
+ 'num': 1,
+ 'hsn_sc': '1010',
+ 'desc': 'Goods Description',
+ 'uqc': 'KGS',
+ 'qty': 2.05,
+ 'txval': 10.23,
+ 'iamt': 14.52,
+ 'csamt': 500,
+ 'rt': 0.1
+ }
+ ]
+ }
+
+ Internal Data Format:
+ {
+ 'HSN Summary': {
+ '1010 - KGS-KILOGRAMS - 0.1': {
+ 'hsn_code': '1010',
+ 'description': 'Goods Description',
+ 'uom': 'KGS-KILOGRAMS',
+ 'quantity': 2.05,
+ 'total_taxable_value': 10.23,
+ 'total_igst_amount': 14.52,
+ 'total_cess_amount': 500,
+ 'tax_rate': 0.1
+ }
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.HSN.value
+ KEY_MAPPING = {
+ # GovDataFields.INDEX.value: ItemFields.INDEX.value,
+ GovDataField.HSN_CODE.value: GSTR1_DataField.HSN_CODE.value,
+ GovDataField.DESCRIPTION.value: GSTR1_DataField.DESCRIPTION.value,
+ GovDataField.UOM.value: GSTR1_DataField.UOM.value,
+ GovDataField.QUANTITY.value: GSTR1_DataField.QUANTITY.value,
+ GovDataField.TAXABLE_VALUE.value: GSTR1_DataField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_DataField.IGST.value,
+ GovDataField.CGST.value: GSTR1_DataField.CGST.value,
+ GovDataField.SGST.value: GSTR1_DataField.SGST.value,
+ GovDataField.CESS.value: GSTR1_DataField.CESS.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+ self.value_formatters_for_internal = {GovDataField.UOM.value: self.map_uom}
+ self.value_formatters_for_gov = {
+ GSTR1_DataField.UOM.value: self.map_uom,
+ GSTR1_DataField.DESCRIPTION.value: lambda x, *args: x[:30],
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for invoice in input_data[GovDataField.HSN_DATA.value]:
+ output[
+ " - ".join(
+ (
+ invoice.get(GovDataField.HSN_CODE.value, ""),
+ self.map_uom(invoice.get(GovDataField.UOM.value, "")),
+ str(flt(invoice.get(GovDataField.TAX_RATE.value))),
+ )
+ )
+ ] = self.format_data(invoice)
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ return {
+ GovDataField.HSN_DATA.value: [
+ self.format_data(
+ invoice, {GovDataField.INDEX.value: index + 1}, for_gov=True
+ )
+ for index, invoice in enumerate(input_data)
+ ]
+ }
+
+ def format_data(self, data, default_data=None, for_gov=False):
+ data = super().format_data(data, default_data, for_gov)
+
+ if for_gov:
+ return data
+
+ data[GSTR1_DataField.DOC_VALUE.value] = sum(
+ (
+ data.get(GSTR1_DataField.TAXABLE_VALUE.value, 0),
+ data.get(GSTR1_DataField.IGST.value, 0),
+ data.get(GSTR1_DataField.CGST.value, 0),
+ data.get(GSTR1_DataField.SGST.value, 0),
+ data.get(GSTR1_DataField.CESS.value, 0),
+ )
+ )
+
+ return data
+
+ def map_uom(self, uom, data=None):
+ uom = uom.upper()
+
+ if "-" in uom:
+ if data and data.get(GSTR1_DataField.HSN_CODE.value, "").startswith("99"):
+ return "NA"
+ else:
+ return uom.split("-")[0]
+
+ if uom in UOM_MAP:
+ return f"{uom}-{UOM_MAP[uom]}"
+
+ return f"OTH-{UOM_MAP.get('OTH')}"
+
+
+class AT(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ [
+ {
+ 'pos': '05',
+ 'itms': [
+ {
+ 'rt': 5,
+ 'ad_amt': 100,
+ ...
+ },
+ ...
+ ],
+ ...
+ },
+ ...
+ ]
+
+ Internal Data Format:
+ {
+ 'Advances Received': {
+ '05-Uttarakhand - 5.0': [
+ {
+ 'place_of_supply': '05-Uttarakhand',
+ 'total_taxable_value': 100,
+ 'tax_rate': 5,
+ ...
+ },
+ ...
+ ],
+ ...
+ }
+ }
+ """
+
+ SUBCATEGORY = GSTR1_SubCategory.AT.value
+ KEY_MAPPING = {
+ GovDataField.FLAG.value: "flag",
+ GovDataField.POS.value: GSTR1_DataField.POS.value,
+ GovDataField.DIFF_PERCENTAGE.value: GSTR1_DataField.DIFF_PERCENTAGE.value,
+ GovDataField.ITEMS.value: GSTR1_DataField.ITEMS.value,
+ GovDataField.TAX_RATE.value: GSTR1_ItemField.TAX_RATE.value,
+ GovDataField.ADVANCE_AMOUNT.value: GSTR1_DataField.TAXABLE_VALUE.value,
+ GovDataField.IGST.value: GSTR1_DataField.IGST.value,
+ GovDataField.CGST.value: GSTR1_DataField.CGST.value,
+ GovDataField.SGST.value: GSTR1_DataField.SGST.value,
+ GovDataField.CESS.value: GSTR1_DataField.CESS.value,
+ }
+ DEFAULT_ITEM_AMOUNTS = {
+ GSTR1_DataField.IGST.value: 0,
+ GSTR1_DataField.CESS.value: 0,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: 0,
+ }
+ MULTIPLIER = 1
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ GovDataField.ITEMS.value: self.format_item_for_internal,
+ GovDataField.POS.value: self.map_place_of_supply,
+ }
+
+ self.value_formatters_for_gov = {
+ # GSTR1_DataField.ITEMS.value: self.format_item_for_gov,
+ GSTR1_DataField.POS.value: self.map_place_of_supply,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for invoice in input_data:
+ invoice_data = self.format_data(invoice)
+ items = invoice_data.pop(GSTR1_DataField.ITEMS.value)
+
+ for item in items:
+ if self.MULTIPLIER != 1:
+ item.update(
+ {
+ key: value * self.MULTIPLIER
+ for key, value in item.items()
+ if key in self.DEFAULT_ITEM_AMOUNTS
+ }
+ )
+
+ item_data = invoice_data.copy()
+ item_data.update(item)
+ output[
+ " - ".join(
+ (
+ invoice_data.get(GSTR1_DataField.POS.value, ""),
+ str(flt(item_data.get(GSTR1_DataField.TAX_RATE.value, ""))),
+ )
+ )
+ ] = [item_data]
+
+ return {self.SUBCATEGORY: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ self.company_gstin = kwargs.get("company_gstin", "")
+ pos_wise_data = {}
+
+ for invoice in input_data:
+ formatted_data = self.format_data(invoice, for_gov=True)
+ rate_wise_taxes = self.get_item_details(formatted_data)
+
+ pos_data = pos_wise_data.setdefault(
+ invoice[GSTR1_DataField.POS.value], formatted_data
+ )
+
+ pos_data.setdefault(GovDataField.ITEMS.value, []).extend(
+ rate_wise_taxes[GovDataField.ITEMS.value]
+ )
+
+ return list(pos_wise_data.values())
+
+ def get_item_details(self, invoice):
+ """
+ Transfer document values to item level (by POS and tax rate)
+ """
+ return {
+ GovDataField.ITEMS.value: [
+ {
+ key: invoice.pop(key)
+ for key in [
+ GovDataField.IGST.value,
+ GovDataField.CESS.value,
+ GovDataField.CGST.value,
+ GovDataField.SGST.value,
+ GovDataField.ADVANCE_AMOUNT.value,
+ GovDataField.TAX_RATE.value,
+ ]
+ }
+ ]
+ }
+
+ def format_data(self, data, default_data=None, for_gov=False):
+ if self.MULTIPLIER != 1 and for_gov:
+ data.update(
+ {
+ key: value * self.MULTIPLIER
+ for key, value in data.items()
+ if key in self.DEFAULT_ITEM_AMOUNTS
+ }
+ )
+
+ data = super().format_data(data, default_data, for_gov)
+
+ if not for_gov:
+ return data
+
+ data[GovDataField.SUPPLY_TYPE.value] = (
+ "INTRA"
+ if data[GovDataField.POS.value] == self.company_gstin[:2]
+ else "INTER"
+ )
+ return data
+
+ def format_item_for_internal(self, items, *args):
+ return [
+ {
+ **self.DEFAULT_ITEM_AMOUNTS.copy(),
+ **self.format_data(item),
+ }
+ for item in items
+ ]
+
+ def format_item_for_gov(self, items, *args):
+ return [self.format_data(item, for_gov=True) for item in items]
+
+
+class TXPD(AT):
+ SUBCATEGORY = GSTR1_SubCategory.TXP.value
+ MULTIPLIER = -1
+
+
+class DOC_ISSUE(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'doc_det': [
+ {
+ 'doc_num': 1,
+ 'docs': [
+ {
+ 'num': 1,
+ 'from': '1',
+ 'to': '10',
+ 'totnum': 10,
+ 'cancel': 0,
+ 'net_issue': 10
+ }
+ ]
+ }
+ ]
+ }
+
+ Internal Data Format:
+ {
+ 'Document Issued': {
+ 'Invoices for outward supply - 1': {
+ 'document_type': 'Invoices for outward supply',
+ 'from_sr_no': '1',
+ 'to_sr_no': '10',
+ 'total_count': 10,
+ 'cancelled_count': 0,
+ 'net_issue': 10
+ }
+ }
+ }
+ """
+
+ KEY_MAPPING = {
+ # GovDataFields.INDEX.value: ItemFields.INDEX.value,
+ GovDataField.FROM_SR.value: GSTR1_DataField.FROM_SR.value,
+ GovDataField.TO_SR.value: GSTR1_DataField.TO_SR.value,
+ GovDataField.TOTAL_COUNT.value: GSTR1_DataField.TOTAL_COUNT.value,
+ GovDataField.CANCELLED_COUNT.value: GSTR1_DataField.CANCELLED_COUNT.value,
+ GovDataField.NET_ISSUE.value: GSTR1_DataField.NET_ISSUE.value,
+ }
+ DOCUMENT_NATURE = {
+ 1: "Invoices for outward supply",
+ 2: "Invoices for inward supply from unregistered person",
+ 3: "Revised Invoice",
+ 4: "Debit Note",
+ 5: "Credit Note",
+ 6: "Receipt voucher",
+ 7: "Payment Voucher",
+ 8: "Refund voucher",
+ 9: "Delivery Challan for job work",
+ 10: "Delivery Challan for supply on approval",
+ 11: "Delivery Challan in case of liquid gas",
+ 12: "Delivery Challan in cases other than by way of supply (excluding at S no. 9 to 11)",
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for document in input_data[GovDataField.DOC_ISSUE_DETAILS.value]:
+ document_nature = self.get_document_nature(
+ document.get(GovDataField.DOC_ISSUE_NUMBER.value, "")
+ )
+ output.update(
+ {
+ " - ".join(
+ (document_nature, doc.get(GovDataField.FROM_SR.value))
+ ): self.format_data(
+ doc, {GSTR1_DataField.DOC_TYPE.value: document_nature}
+ )
+ for doc in document[GovDataField.DOC_ISSUE_LIST.value]
+ }
+ )
+
+ return {GSTR1_SubCategory.DOC_ISSUE.value: output}
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ self.DOCUMENT_NATURE = self.reverse_dict(self.DOCUMENT_NATURE)
+
+ output = {GovDataField.DOC_ISSUE_DETAILS.value: []}
+ doc_nature_wise_data = {}
+
+ for invoice in input_data:
+ doc_nature_wise_data.setdefault(
+ invoice[GSTR1_DataField.DOC_TYPE.value], []
+ ).append(invoice)
+
+ input_data = doc_nature_wise_data
+
+ output = {
+ GovDataField.DOC_ISSUE_DETAILS.value: [
+ {
+ GovDataField.DOC_ISSUE_NUMBER.value: self.get_document_nature(
+ doc_nature
+ ),
+ GovDataField.DOC_ISSUE_LIST.value: [
+ self.format_data(
+ document,
+ {GovDataField.INDEX.value: index + 1},
+ for_gov=True,
+ )
+ for index, document in enumerate(documents)
+ ],
+ }
+ for doc_nature, documents in doc_nature_wise_data.items()
+ ]
+ }
+
+ return output
+
+ def format_data(self, data, additional_data=None, for_gov=False):
+ if not for_gov:
+ return super().format_data(data, additional_data)
+
+ # compute additional data
+ data[GSTR1_DataField.CANCELLED_COUNT.value] += data.get(
+ GSTR1_DataField.DRAFT_COUNT.value, 0
+ )
+ data["net_issue"] = data[GSTR1_DataField.TOTAL_COUNT.value] - data.get(
+ GSTR1_DataField.CANCELLED_COUNT.value, 0
+ )
+
+ return super().format_data(data, additional_data, for_gov)
+
+ def get_document_nature(self, doc_nature, *args):
+ return self.DOCUMENT_NATURE.get(doc_nature, doc_nature)
+
+
+class SUPECOM(GovDataMapper):
+ """
+ GST API Version - v4.0
+
+ Government Data Format:
+ {
+ 'clttx': [
+ {
+ 'etin': '20ALYPD6528PQC5',
+ 'suppval': 10000,
+ 'igst': 1000,
+ 'cgst': 0,
+ 'sgst': 0,
+ 'cess': 0
+ }
+ ]
+ }
+
+ Internal Data Format:
+ {
+ 'TCS collected by E-commerce Operator u/s 52': {
+ '20ALYPD6528PQC5': {
+ 'document_type': 'TCS collected by E-commerce Operator u/s 52',
+ 'ecommerce_gstin': '20ALYPD6528PQC5',
+ 'total_taxable_value': 10000,
+ 'igst_amount': 1000,
+ 'cgst_amount': 0,
+ 'sgst_amount': 0,
+ 'cess_amount': 0
+ }
+ }
+ }
+ """
+
+ KEY_MAPPING = {
+ GovDataField.ECOMMERCE_GSTIN.value: GSTR1_DataField.ECOMMERCE_GSTIN.value,
+ GovDataField.NET_TAXABLE_VALUE.value: GSTR1_DataField.TAXABLE_VALUE.value,
+ "igst": GSTR1_ItemField.IGST.value,
+ "cgst": GSTR1_ItemField.CGST.value,
+ "sgst": GSTR1_ItemField.SGST.value,
+ "cess": GSTR1_ItemField.CESS.value,
+ GovDataField.FLAG.value: "flag",
+ }
+ DOCUMENT_CATEGORIES = {
+ GovDataField.SUPECOM_52.value: GSTR1_SubCategory.SUPECOM_52.value,
+ GovDataField.SUPECOM_9_5.value: GSTR1_SubCategory.SUPECOM_9_5.value,
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for section, invoices in input_data.items():
+ document_type = self.DOCUMENT_CATEGORIES.get(section, section)
+ output[document_type] = {
+ invoice.get(GovDataField.ECOMMERCE_GSTIN.value, ""): self.format_data(
+ invoice, {GSTR1_DataField.DOC_TYPE.value: document_type}
+ )
+ for invoice in invoices
+ }
+
+ return output
+
+ def convert_to_gov_data_format(self, input_data, **kwargs):
+ output = {}
+ self.DOCUMENT_CATEGORIES = self.reverse_dict(self.DOCUMENT_CATEGORIES)
+
+ for invoice in input_data:
+ section = invoice[GSTR1_DataField.DOC_TYPE.value]
+ output.setdefault(
+ self.DOCUMENT_CATEGORIES.get(section, section), []
+ ).append(self.format_data(invoice, for_gov=True))
+ return output
+
+
+class RETSUM(GovDataMapper):
+ """
+ Convert GSTR-1 Summary as returned by the API to the internal format
+
+ Usecase: Compute amendment liability for GSTR-1 Summary
+
+ Exceptions:
+ - Only supports latest summary format v4.0 and above
+ """
+
+ KEY_MAPPING = {
+ "sec_nm": GSTR1_DataField.DESCRIPTION.value,
+ "typ": GSTR1_DataField.DESCRIPTION.value,
+ "ttl_rec": "no_of_records",
+ "ttl_val": "total_document_value",
+ "ttl_igst": GSTR1_DataField.IGST.value,
+ "ttl_cgst": GSTR1_DataField.CGST.value,
+ "ttl_sgst": GSTR1_DataField.SGST.value,
+ "ttl_cess": GSTR1_DataField.CESS.value,
+ "ttl_tax": GSTR1_DataField.TAXABLE_VALUE.value,
+ "act_val": "actual_document_value",
+ "act_igst": "actual_igst",
+ "act_sgst": "actual_sgst",
+ "act_cgst": "actual_cgst",
+ "act_cess": "actual_cess",
+ "act_tax": "actual_taxable_value",
+ "ttl_expt_amt": f"total_{GSTR1_DataField.EXEMPTED_AMOUNT.value}",
+ "ttl_ngsup_amt": f"total_{GSTR1_DataField.NON_GST_AMOUNT.value}",
+ "ttl_nilsup_amt": f"total_{GSTR1_DataField.NIL_RATED_AMOUNT.value}",
+ "ttl_doc_issued": GSTR1_DataField.TOTAL_COUNT.value,
+ "ttl_doc_cancelled": GSTR1_DataField.CANCELLED_COUNT.value,
+ }
+
+ SECTION_NAMES = {
+ "AT": GSTR1_Category.AT.value,
+ "B2B_4A": GSTR1_SubCategory.B2B_REGULAR.value,
+ "B2B_4B": GSTR1_SubCategory.B2B_REVERSE_CHARGE.value,
+ "B2B_6C": GSTR1_SubCategory.DE.value,
+ "B2B_SEZWOP": GSTR1_SubCategory.SEZWOP.value,
+ "B2B_SEZWP": GSTR1_SubCategory.SEZWP.value,
+ "B2B": GSTR1_Category.B2B.value,
+ "B2CL": GSTR1_Category.B2CL.value,
+ "B2CS": GSTR1_Category.B2CS.value,
+ "TXPD": GSTR1_Category.TXP.value,
+ "EXP": GSTR1_Category.EXP.value,
+ "CDNR": GSTR1_Category.CDNR.value,
+ "CDNUR": GSTR1_Category.CDNUR.value,
+ "SUPECOM": GSTR1_Category.SUPECOM.value,
+ "ECOM": "ECOM",
+ "ECOM_REG": "ECOM_REG",
+ "ECOM_DE": "ECOM_DE",
+ "ECOM_SEZWOP": "ECOM_SEZWOP",
+ "ECOM_SEZWP": "ECOM_SEZWP",
+ "ECOM_UNREG": "ECOM_UNREG",
+ "ATA": f"{GSTR1_Category.AT.value} (Amended)",
+ "B2BA_4A": f"{GSTR1_SubCategory.B2B_REGULAR.value} (Amended)",
+ "B2BA_4B": f"{GSTR1_SubCategory.B2B_REVERSE_CHARGE.value} (Amended)",
+ "B2BA_6C": f"{GSTR1_SubCategory.DE.value} (Amended)",
+ "B2BA_SEZWOP": f"{GSTR1_SubCategory.SEZWOP.value} (Amended)",
+ "B2BA_SEZWP": f"{GSTR1_SubCategory.SEZWP.value} (Amended)",
+ "B2BA": f"{GSTR1_Category.B2B.value} (Amended)",
+ "B2CLA": f"{GSTR1_Category.B2CL.value} (Amended)",
+ "B2CSA": f"{GSTR1_Category.B2CS.value} (Amended)",
+ "TXPDA": f"{GSTR1_Category.TXP.value} (Amended)",
+ "EXPA": f"{GSTR1_Category.EXP.value} (Amended)",
+ "CDNRA": f"{GSTR1_Category.CDNR.value} (Amended)",
+ "CDNURA": f"{GSTR1_Category.CDNUR.value} (Amended)",
+ "SUPECOMA": f"{GSTR1_Category.SUPECOM.value} (Amended)",
+ "ECOMA": "ECOMA",
+ "ECOMA_REG": "ECOMA_REG",
+ "ECOMA_DE": "ECOMA_DE",
+ "ECOMA_SEZWOP": "ECOMA_SEZWOP",
+ "ECOMA_SEZWP": "ECOMA_SEZWP",
+ "ECOMA_UNREG": "ECOMA_UNREG",
+ "HSN": GSTR1_Category.HSN.value,
+ "NIL": GSTR1_Category.NIL_EXEMPT.value,
+ "DOC_ISSUE": GSTR1_Category.DOC_ISSUE.value,
+ "TTL_LIAB": "Total Liability",
+ }
+
+ SECTIONS_WITH_SUBSECTIONS = {
+ "SUPECOM": {
+ "SUPECOM_14A": GSTR1_SubCategory.SUPECOM_52.value,
+ "SUPECOM_14B": GSTR1_SubCategory.SUPECOM_9_5.value,
+ },
+ "SUPECOMA": {
+ "SUPECOMA_14A": f"{GSTR1_SubCategory.SUPECOM_52.value} (Amended)",
+ "SUPECOMA_14B": f"{GSTR1_SubCategory.SUPECOM_9_5.value} (Amended)",
+ },
+ "EXP": {
+ "EXPWP": GSTR1_SubCategory.EXPWP.value,
+ "EXPWOP": GSTR1_SubCategory.EXPWOP.value,
+ },
+ "EXPA": {
+ "EXPWP": f"{GSTR1_SubCategory.EXPWP.value} (Amended)",
+ "EXPWOP": f"{GSTR1_SubCategory.EXPWOP.value} (Amended)",
+ },
+ }
+
+ def __init__(self):
+ super().__init__()
+
+ self.value_formatters_for_internal = {
+ "sec_nm": self.map_document_types,
+ "typ": self.map_document_types,
+ }
+
+ def convert_to_internal_data_format(self, input_data):
+ output = {}
+
+ for section_data in input_data:
+ section = section_data.get("sec_nm")
+ output[self.SECTION_NAMES.get(section, section)] = self.format_data(
+ section_data
+ )
+
+ if section not in self.SECTIONS_WITH_SUBSECTIONS:
+ continue
+
+ # Unsupported Legacy Summary API. Fallback to self-calculated summary.
+ sub_sections = section_data.get("sub_sections", {})
+ if not sub_sections:
+ return {}
+
+ for subsection_data in sub_sections:
+ formatted_data = self.format_subsection_data(section, subsection_data)
+ output[formatted_data[GSTR1_DataField.DESCRIPTION.value]] = (
+ formatted_data
+ )
+
+ return {"summary": output}
+
+ def format_subsection_data(self, section, subsection_data):
+ subsection = subsection_data.get("typ") or subsection_data.get("sec_nm")
+ formatted_data = self.format_data(subsection_data)
+
+ formatted_data[GSTR1_DataField.DESCRIPTION.value] = (
+ self.SECTIONS_WITH_SUBSECTIONS[section].get(subsection, subsection)
+ )
+ return formatted_data
+
+ def map_document_types(self, doc_type, *args):
+ return self.SECTION_NAMES.get(doc_type, doc_type)
+
+
+CLASS_MAP = {
+ GovJsonKey.B2B.value: B2B,
+ GovJsonKey.B2CL.value: B2CL,
+ GovJsonKey.EXP.value: Exports,
+ GovJsonKey.B2CS.value: B2CS,
+ GovJsonKey.NIL_EXEMPT.value: NilRated,
+ GovJsonKey.CDNR.value: CDNR,
+ GovJsonKey.CDNUR.value: CDNUR,
+ GovJsonKey.HSN.value: HSNSUM,
+ GovJsonKey.DOC_ISSUE.value: DOC_ISSUE,
+ GovJsonKey.AT.value: AT,
+ GovJsonKey.TXP.value: TXPD,
+ GovJsonKey.SUPECOM.value: SUPECOM,
+ GovJsonKey.RET_SUM.value: RETSUM,
+}
+
+
+def convert_to_internal_data_format(gov_data):
+ """
+ Converts Gov data format to internal data format for all categories
+ """
+ output = {}
+
+ for category, mapper_class in CLASS_MAP.items():
+ if not gov_data.get(category):
+ continue
+
+ output.update(
+ mapper_class().convert_to_internal_data_format(gov_data.get(category))
+ )
+
+ return output
+
+
+def get_category_wise_data(
+ subcategory_wise_data: dict,
+ mapping: dict = SUB_CATEGORY_GOV_CATEGORY_MAPPING,
+) -> dict:
+ """
+ returns category wise data from subcategory wise data
+
+ Args:
+ subcategory_wise_data (dict): subcategory wise data
+ mapping (dict): subcategory to category mapping
+ with_subcategory (bool): include subcategory level data
+
+ Returns:
+ dict: category wise data
+
+ Example (with_subcategory=True):
+ {
+ "B2B, SEZ, DE": {
+ "B2B": data,
+ ...
+ }
+ ...
+ }
+
+ Example (with_subcategory=False):
+ {
+ "B2B, SEZ, DE": data,
+ ...
+ }
+ """
+ category_wise_data = {}
+ for subcategory, category in mapping.items():
+ if not subcategory_wise_data.get(subcategory.value):
+ continue
+
+ category_wise_data.setdefault(category.value, []).extend(
+ subcategory_wise_data.get(subcategory.value, [])
+ )
+
+ return category_wise_data
+
+
+def convert_to_gov_data_format(internal_data: dict, company_gstin: str) -> dict:
+ """
+ converts internal data format to Gov data format for all categories
+ """
+
+ category_wise_data = get_category_wise_data(internal_data)
+
+ output = {}
+ for category, mapper_class in CLASS_MAP.items():
+ if not category_wise_data.get(category):
+ continue
+
+ output[category] = mapper_class().convert_to_gov_data_format(
+ category_wise_data.get(category), company_gstin=company_gstin
+ )
+
+ return output
+
+
+def summarize_retsum_data(input_data):
+ if not input_data:
+ return []
+
+ summarized_data = []
+ total_values_keys = [
+ "total_igst_amount",
+ "total_cgst_amount",
+ "total_sgst_amount",
+ "total_cess_amount",
+ "total_taxable_value",
+ ]
+ amended_data = {key: 0 for key in total_values_keys}
+
+ input_data = {row.get("description"): row for row in input_data}
+
+ def _sum(row):
+ return flt(sum([row.get(key, 0) for key in total_values_keys]), 2)
+
+ for category, sub_categories in CATEGORY_SUB_CATEGORY_MAPPING.items():
+ category = category.value
+ if category not in input_data:
+ continue
+
+ # compute total liability and total amended data
+ amended_category_data = input_data.get(f"{category} (Amended)", {})
+ for key in total_values_keys:
+ amended_data[key] += amended_category_data.get(key, 0)
+
+ # add category data
+ if _sum(input_data[category]) == 0:
+ continue
+
+ summarized_data.append({**input_data.get(category), "indent": 0})
+
+ # add subcategory data
+ for sub_category in sub_categories:
+ sub_category = sub_category.value
+ if sub_category not in input_data:
+ continue
+
+ if _sum(input_data[sub_category]) == 0:
+ continue
+
+ summarized_data.append(
+ {
+ **input_data.get(sub_category),
+ "indent": 1,
+ "consider_in_total_taxable_value": (
+ False
+ if sub_category
+ in SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAXABLE_VALUE
+ else True
+ ),
+ "consider_in_total_tax": (
+ False
+ if sub_category in SUBCATEGORIES_NOT_CONSIDERED_IN_TOTAL_TAX
+ else True
+ ),
+ }
+ )
+
+ # add total amendment liability
+ if _sum(amended_data) != 0:
+ summarized_data.extend(
+ [
+ {
+ "description": "Net Liability from Amendments",
+ **amended_data,
+ "indent": 0,
+ "consider_in_total_taxable_value": True,
+ "consider_in_total_tax": True,
+ "no_of_records": 0,
+ }
+ ]
+ )
+
+ return summarized_data
+
+
+####################################################################################################
+### Map Books Data to Internal Data Structure ######################################################
+####################################################################################################
+
+
+class BooksDataMapper:
+ def get_transaction_type(self, invoice):
+ if invoice.is_debit_note:
+ return "Debit Note"
+ elif invoice.is_return:
+ return "Credit Note"
+ else:
+ return "Invoice"
+
+ def process_data_for_invoice_no_key(self, invoice, prepared_data):
+ invoice_sub_category = invoice.invoice_sub_category
+ invoice_no = invoice.invoice_no
+
+ mapped_dict = prepared_data.setdefault(invoice_sub_category, {}).setdefault(
+ invoice_no,
+ {
+ GSTR1_DataField.TRANSACTION_TYPE.value: self.get_transaction_type(
+ invoice
+ ),
+ GSTR1_DataField.CUST_GSTIN.value: invoice.billing_address_gstin,
+ GSTR1_DataField.CUST_NAME.value: invoice.customer_name,
+ GSTR1_DataField.DOC_DATE.value: invoice.posting_date,
+ GSTR1_DataField.DOC_NUMBER.value: invoice.invoice_no,
+ GSTR1_DataField.DOC_VALUE.value: invoice.invoice_total,
+ GSTR1_DataField.POS.value: invoice.place_of_supply,
+ GSTR1_DataField.REVERSE_CHARGE.value: (
+ "Y" if invoice.is_reverse_charge else "N"
+ ),
+ GSTR1_DataField.DOC_TYPE.value: invoice.invoice_type,
+ GSTR1_DataField.TAXABLE_VALUE.value: 0,
+ GSTR1_DataField.IGST.value: 0,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 0,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0,
+ "items": [],
+ },
+ )
+
+ items = mapped_dict["items"]
+
+ for item in items:
+ if item[GSTR1_ItemField.TAX_RATE.value] == invoice.gst_rate:
+ item[GSTR1_ItemField.TAXABLE_VALUE.value] += invoice.taxable_value
+ item[GSTR1_ItemField.IGST.value] += invoice.igst_amount
+ item[GSTR1_ItemField.CGST.value] += invoice.cgst_amount
+ item[GSTR1_ItemField.SGST.value] += invoice.sgst_amount
+ item[GSTR1_ItemField.CESS.value] += invoice.total_cess_amount
+ self.update_totals(mapped_dict, invoice)
+ return
+
+ items.append(
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: invoice.taxable_value,
+ GSTR1_ItemField.IGST.value: invoice.igst_amount,
+ GSTR1_ItemField.CGST.value: invoice.cgst_amount,
+ GSTR1_ItemField.SGST.value: invoice.sgst_amount,
+ GSTR1_ItemField.CESS.value: invoice.total_cess_amount,
+ GSTR1_ItemField.TAX_RATE.value: invoice.gst_rate,
+ }
+ )
+
+ self.update_totals(mapped_dict, invoice)
+
+ def process_data_for_nil_exempt(self, invoice, prepared_data):
+ key = invoice.invoice_category
+ invoices_by_type = prepared_data.setdefault(key, {}).setdefault(
+ invoice.invoice_type, []
+ )
+
+ for mapped_dict in invoices_by_type:
+ if mapped_dict[GSTR1_DataField.DOC_NUMBER.value] == invoice.invoice_no:
+ break
+
+ else:
+ mapped_dict = {
+ GSTR1_DataField.TRANSACTION_TYPE.value: self.get_transaction_type(
+ invoice
+ ),
+ GSTR1_DataField.CUST_GSTIN.value: invoice.billing_address_gstin,
+ GSTR1_DataField.CUST_NAME.value: invoice.customer_name,
+ GSTR1_DataField.DOC_NUMBER.value: invoice.invoice_no,
+ GSTR1_DataField.DOC_DATE.value: invoice.posting_date,
+ GSTR1_DataField.DOC_VALUE.value: invoice.invoice_total,
+ GSTR1_DataField.POS.value: invoice.place_of_supply,
+ GSTR1_DataField.REVERSE_CHARGE.value: (
+ "Y" if invoice.is_reverse_charge else "N"
+ ),
+ GSTR1_DataField.DOC_TYPE.value: invoice.invoice_type,
+ GSTR1_DataField.TAXABLE_VALUE.value: 0,
+ GSTR1_DataField.NIL_RATED_AMOUNT.value: 0,
+ GSTR1_DataField.EXEMPTED_AMOUNT.value: 0,
+ GSTR1_DataField.NON_GST_AMOUNT.value: 0,
+ }
+ invoices_by_type.append(mapped_dict)
+
+ mapped_dict[GSTR1_DataField.TAXABLE_VALUE.value] += invoice.taxable_value
+
+ if invoice.gst_treatment == "Nil-Rated":
+ mapped_dict[GSTR1_DataField.NIL_RATED_AMOUNT.value] += invoice.taxable_value
+ elif invoice.gst_treatment == "Exempted":
+ mapped_dict[GSTR1_DataField.EXEMPTED_AMOUNT.value] += invoice.taxable_value
+ elif invoice.gst_treatment == "Non-GST":
+ mapped_dict[GSTR1_DataField.NON_GST_AMOUNT.value] += invoice.taxable_value
+
+ def process_data_for_b2cs(self, invoice, prepared_data):
+ key = f"{invoice.place_of_supply} - {flt(invoice.gst_rate)}"
+ mapped_dict = prepared_data.setdefault("B2C (Others)", {}).setdefault(key, [])
+
+ for row in mapped_dict:
+ if row[GSTR1_DataField.DOC_NUMBER.value] == invoice.invoice_no:
+ self.update_totals(row, invoice)
+ return
+
+ mapped_dict.append(
+ {
+ GSTR1_DataField.DOC_DATE.value: invoice.posting_date,
+ GSTR1_DataField.DOC_NUMBER.value: invoice.invoice_no,
+ GSTR1_DataField.DOC_VALUE.value: invoice.invoice_total,
+ GSTR1_DataField.CUST_NAME.value: invoice.customer_name,
+ # currently other value is not supported in GSTR-1
+ GSTR1_DataField.DOC_TYPE.value: "OE",
+ GSTR1_DataField.TRANSACTION_TYPE.value: self.get_transaction_type(
+ invoice
+ ),
+ GSTR1_DataField.POS.value: invoice.place_of_supply,
+ GSTR1_DataField.TAX_RATE.value: invoice.gst_rate,
+ GSTR1_DataField.ECOMMERCE_GSTIN.value: invoice.ecommerce_gstin,
+ **self.get_invoice_values(invoice),
+ }
+ )
+
+ def process_data_for_hsn_summary(self, invoice, prepared_data):
+ key = f"{invoice.gst_hsn_code} - {invoice.stock_uom} - {flt(invoice.gst_rate)}"
+
+ if key not in prepared_data:
+ mapped_dict = prepared_data.setdefault(
+ key,
+ {
+ GSTR1_DataField.HSN_CODE.value: invoice.gst_hsn_code,
+ GSTR1_DataField.DESCRIPTION.value: frappe.db.get_value(
+ "GST HSN Code", invoice.gst_hsn_code, "description"
+ ),
+ GSTR1_DataField.UOM.value: invoice.stock_uom,
+ GSTR1_DataField.QUANTITY.value: 0,
+ GSTR1_DataField.TAX_RATE.value: invoice.gst_rate,
+ GSTR1_DataField.TAXABLE_VALUE.value: 0,
+ GSTR1_DataField.IGST.value: 0,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 0,
+ },
+ )
+
+ else:
+ mapped_dict = prepared_data[key]
+
+ self.update_totals(mapped_dict, invoice, for_qty=True)
+
+ mapped_dict[GSTR1_DataField.DOC_VALUE.value] = sum(
+ (
+ mapped_dict.get(GSTR1_DataField.TAXABLE_VALUE.value, 0),
+ mapped_dict.get(GSTR1_DataField.IGST.value, 0),
+ mapped_dict.get(GSTR1_DataField.CGST.value, 0),
+ mapped_dict.get(GSTR1_DataField.SGST.value, 0),
+ mapped_dict.get(GSTR1_DataField.CESS.value, 0),
+ )
+ )
+
+ def process_data_for_document_issued_summary(self, row, prepared_data):
+ key = f"{row['nature_of_document']} - {row['from_serial_no']}"
+ prepared_data.setdefault(
+ key,
+ {
+ GSTR1_DataField.DOC_TYPE.value: row["nature_of_document"],
+ GSTR1_DataField.FROM_SR.value: row["from_serial_no"],
+ GSTR1_DataField.TO_SR.value: row["to_serial_no"],
+ GSTR1_DataField.TOTAL_COUNT.value: row["total_issued"],
+ GSTR1_DataField.DRAFT_COUNT.value: row["total_draft"],
+ GSTR1_DataField.CANCELLED_COUNT.value: row["cancelled"],
+ GSTR1_DataField.NET_ISSUE.value: row["total_submitted"],
+ },
+ )
+
+ def process_data_for_advances_received_or_adjusted(
+ self, row, prepared_data, multiplier=1
+ ):
+ advances = {}
+ tax_rate = round(((row["tax_amount"] / row["taxable_value"]) * 100))
+ key = f"{row['place_of_supply']} - {flt(tax_rate)}"
+
+ mapped_dict = prepared_data.setdefault(key, [])
+
+ advances[GSTR1_DataField.CUST_NAME.value] = row["party"]
+ advances[GSTR1_DataField.DOC_NUMBER.value] = row["name"]
+ advances[GSTR1_DataField.DOC_DATE.value] = row["posting_date"]
+ advances[GSTR1_DataField.POS.value] = row["place_of_supply"]
+ advances[GSTR1_DataField.TAXABLE_VALUE.value] = (
+ row["taxable_value"] * multiplier
+ )
+ advances[GSTR1_DataField.TAX_RATE.value] = tax_rate
+ advances[GSTR1_DataField.CESS.value] = row["cess_amount"] * multiplier
+
+ if row.get("reference_name"):
+ advances["against_voucher"] = row["reference_name"]
+
+ if row["place_of_supply"][0:2] == row["company_gstin"][0:2]:
+ advances[GSTR1_DataField.CGST.value] = row["tax_amount"] / 2 * multiplier
+ advances[GSTR1_DataField.SGST.value] = row["tax_amount"] / 2 * multiplier
+ advances[GSTR1_DataField.IGST.value] = 0
+
+ else:
+ advances[GSTR1_DataField.IGST.value] = row["tax_amount"] * multiplier
+ advances[GSTR1_DataField.CGST.value] = 0
+ advances[GSTR1_DataField.SGST.value] = 0
+
+ mapped_dict.append(advances)
+
+ # utils
+
+ def update_totals(self, mapped_dict, invoice, for_qty=False):
+ data_invoice_amount_map = {
+ GSTR1_DataField.TAXABLE_VALUE.value: GSTR1_ItemField.TAXABLE_VALUE.value,
+ GSTR1_DataField.IGST.value: GSTR1_ItemField.IGST.value,
+ GSTR1_DataField.CGST.value: GSTR1_ItemField.CGST.value,
+ GSTR1_DataField.SGST.value: GSTR1_ItemField.SGST.value,
+ GSTR1_DataField.CESS.value: GSTR1_ItemField.CESS.value,
+ }
+
+ if for_qty:
+ data_invoice_amount_map[GSTR1_DataField.QUANTITY.value] = "qty"
+
+ for key, field in data_invoice_amount_map.items():
+ mapped_dict[key] += invoice.get(field, 0)
+
+ def get_invoice_values(self, invoice):
+ return {
+ GSTR1_DataField.TAXABLE_VALUE.value: invoice.taxable_value,
+ GSTR1_DataField.IGST.value: invoice.igst_amount,
+ GSTR1_DataField.CGST.value: invoice.cgst_amount,
+ GSTR1_DataField.SGST.value: invoice.sgst_amount,
+ GSTR1_DataField.CESS.value: invoice.total_cess_amount,
+ }
+
+
+class GSTR1BooksData(BooksDataMapper):
+ def __init__(self, filters):
+ self.filters = filters
+
+ def prepare_mapped_data(self):
+ prepared_data = {}
+
+ _class = GSTR1Invoices(self.filters)
+ data = _class.get_invoices_for_item_wise_summary()
+ _class.process_invoices(data)
+
+ for invoice in data:
+ if invoice["invoice_category"] in (
+ GSTR1_Category.B2B.value,
+ GSTR1_Category.EXP.value,
+ GSTR1_Category.B2CL.value,
+ GSTR1_Category.CDNR.value,
+ GSTR1_Category.CDNUR.value,
+ ):
+ self.process_data_for_invoice_no_key(invoice, prepared_data)
+ elif invoice["invoice_category"] == GSTR1_Category.NIL_EXEMPT.value:
+ self.process_data_for_nil_exempt(invoice, prepared_data)
+ elif invoice["invoice_category"] == GSTR1_Category.B2CS.value:
+ self.process_data_for_b2cs(invoice, prepared_data)
+
+ other_categories = {
+ GSTR1_Category.AT.value: self.prepare_advances_recevied_data(),
+ GSTR1_Category.TXP.value: self.prepare_advances_adjusted_data(),
+ GSTR1_Category.HSN.value: self.prepare_hsn_data(data),
+ GSTR1_Category.DOC_ISSUE.value: self.prepare_document_issued_data(),
+ }
+
+ for category, data in other_categories.items():
+ if data:
+ prepared_data[category] = data
+
+ return prepared_data
+
+ def prepare_document_issued_data(self):
+ doc_issued_data = {}
+ data = GSTR1DocumentIssuedSummary(self.filters).get_data()
+
+ for row in data:
+ self.process_data_for_document_issued_summary(row, doc_issued_data)
+
+ return doc_issued_data
+
+ def prepare_hsn_data(self, data):
+ hsn_summary_data = {}
+
+ for row in data:
+ self.process_data_for_hsn_summary(row, hsn_summary_data)
+
+ return hsn_summary_data
+
+ def prepare_advances_recevied_data(self):
+ return self.prepare_advances_received_or_adjusted_data("Advances")
+
+ def prepare_advances_adjusted_data(self):
+ return self.prepare_advances_received_or_adjusted_data("Adjustment")
+
+ def prepare_advances_received_or_adjusted_data(self, type_of_business):
+ advances_data = {}
+ self.filters.type_of_business = type_of_business
+ gst_accounts = get_gst_accounts_by_type(self.filters.company, "Output")
+ _class = GSTR11A11BData(self.filters, gst_accounts)
+
+ if type_of_business == "Advances":
+ query = _class.get_11A_query()
+ fields = (
+ _class.pe.name,
+ _class.pe.party,
+ _class.pe.posting_date,
+ _class.pe.company_gstin,
+ )
+ multipler = 1
+
+ elif type_of_business == "Adjustment":
+ query = _class.get_11B_query()
+ fields = (
+ _class.pe.name,
+ _class.pe.party,
+ _class.pe.posting_date,
+ _class.pe.company_gstin,
+ _class.pe_ref.reference_name,
+ )
+ multipler = -1
+
+ query = query.select(*fields)
+ data = query.run(as_dict=True)
+
+ for row in data:
+ self.process_data_for_advances_received_or_adjusted(
+ row, advances_data, multipler
+ )
+
+ return advances_data
diff --git a/india_compliance/gst_india/utils/gstr_1/test_gstr_1_json_map.py b/india_compliance/gst_india/utils/gstr_1/test_gstr_1_json_map.py
new file mode 100644
index 0000000000..a602bcbb0f
--- /dev/null
+++ b/india_compliance/gst_india/utils/gstr_1/test_gstr_1_json_map.py
@@ -0,0 +1,1366 @@
+import copy
+
+from frappe.tests.utils import FrappeTestCase
+
+from india_compliance.gst_india.doctype.gstr_1_log.gstr_1_log import GenerateGSTR1
+from india_compliance.gst_india.utils import get_party_for_gstin as _get_party_for_gstin
+from india_compliance.gst_india.utils.gstr_1 import (
+ SUB_CATEGORY_GOV_CATEGORY_MAPPING,
+ GovDataField,
+ GSTR1_B2B_InvoiceType,
+ GSTR1_DataField,
+ GSTR1_ItemField,
+ GSTR1_SubCategory,
+)
+from india_compliance.gst_india.utils.gstr_1.gstr_1_json_map import (
+ AT,
+ B2B,
+ B2CL,
+ B2CS,
+ CDNR,
+ CDNUR,
+ DOC_ISSUE,
+ HSNSUM,
+ SUPECOM,
+ TXPD,
+ Exports,
+ NilRated,
+ get_category_wise_data,
+)
+
+
+def get_party_for_gstin(gstin):
+ return _get_party_for_gstin(gstin, "Customer") or "Unknown"
+
+
+def normalize_data(data):
+ return GenerateGSTR1().normalize_data(data)
+
+
+def process_mapped_data(data):
+ return list(
+ get_category_wise_data(
+ normalize_data(copy.deepcopy(data)), SUB_CATEGORY_GOV_CATEGORY_MAPPING
+ ).values()
+ )[0]
+
+
+class TestB2B(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = [
+ {
+ GovDataField.CUST_GSTIN.value: "24AANFA2641L1ZF",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "S008400",
+ GovDataField.DOC_DATE.value: "24-11-2016",
+ GovDataField.DOC_VALUE.value: 729248.16,
+ GovDataField.POS.value: "06",
+ GovDataField.REVERSE_CHARGE.value: "N",
+ GovDataField.INVOICE_TYPE.value: "R",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ ],
+ },
+ {
+ GovDataField.DOC_NUMBER.value: "S008401",
+ GovDataField.DOC_DATE.value: "24-11-2016",
+ GovDataField.DOC_VALUE.value: 729248.16,
+ GovDataField.POS.value: "06",
+ GovDataField.REVERSE_CHARGE.value: "Y",
+ GovDataField.INVOICE_TYPE.value: "R",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ }
+ ],
+ },
+ ],
+ },
+ {
+ GovDataField.CUST_GSTIN.value: "29AABCR1718E1ZL",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "S008402",
+ GovDataField.DOC_DATE.value: "24-11-2016",
+ GovDataField.DOC_VALUE.value: 729248.16,
+ GovDataField.POS.value: "06",
+ GovDataField.REVERSE_CHARGE.value: "N",
+ GovDataField.INVOICE_TYPE.value: "SEWP",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ }
+ ],
+ },
+ {
+ GovDataField.DOC_NUMBER.value: "S008403",
+ GovDataField.DOC_DATE.value: "24-11-2016",
+ GovDataField.DOC_VALUE.value: 729248.16,
+ GovDataField.POS.value: "06",
+ GovDataField.REVERSE_CHARGE.value: "N",
+ GovDataField.INVOICE_TYPE.value: "DE",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ }
+ ],
+ },
+ ],
+ },
+ ]
+ cls.mapped_data = {
+ GSTR1_SubCategory.B2B_REGULAR.value: {
+ "S008400": {
+ GSTR1_DataField.CUST_GSTIN.value: "24AANFA2641L1ZF",
+ GSTR1_DataField.CUST_NAME.value: get_party_for_gstin(
+ "24AANFA2641L1ZF"
+ ),
+ GSTR1_DataField.DOC_NUMBER.value: "S008400",
+ GSTR1_DataField.DOC_DATE.value: "2016-11-24",
+ GSTR1_DataField.DOC_VALUE.value: 729248.16,
+ GSTR1_DataField.POS.value: "06-Haryana",
+ GSTR1_DataField.REVERSE_CHARGE.value: "N",
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_B2B_InvoiceType.R.value,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 20000,
+ GSTR1_DataField.IGST.value: 650,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 1000,
+ }
+ },
+ GSTR1_SubCategory.B2B_REVERSE_CHARGE.value: {
+ "S008401": {
+ GSTR1_DataField.CUST_GSTIN.value: "24AANFA2641L1ZF",
+ GSTR1_DataField.CUST_NAME.value: get_party_for_gstin(
+ "24AANFA2641L1ZF"
+ ),
+ GSTR1_DataField.DOC_NUMBER.value: "S008401",
+ GSTR1_DataField.DOC_DATE.value: "2016-11-24",
+ GSTR1_DataField.DOC_VALUE.value: 729248.16,
+ GSTR1_DataField.POS.value: "06-Haryana",
+ GSTR1_DataField.REVERSE_CHARGE.value: "Y",
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_B2B_InvoiceType.R.value,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 325,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 500,
+ }
+ },
+ GSTR1_SubCategory.SEZWP.value: {
+ "S008402": {
+ GSTR1_DataField.CUST_GSTIN.value: "29AABCR1718E1ZL",
+ GSTR1_DataField.CUST_NAME.value: get_party_for_gstin(
+ "29AABCR1718E1ZL"
+ ),
+ GSTR1_DataField.DOC_NUMBER.value: "S008402",
+ GSTR1_DataField.DOC_DATE.value: "2016-11-24",
+ GSTR1_DataField.DOC_VALUE.value: 729248.16,
+ GSTR1_DataField.POS.value: "06-Haryana",
+ GSTR1_DataField.REVERSE_CHARGE.value: "N",
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_B2B_InvoiceType.SEWP.value,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 325,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 500,
+ }
+ },
+ GSTR1_SubCategory.DE.value: {
+ "S008403": {
+ GSTR1_DataField.CUST_GSTIN.value: "29AABCR1718E1ZL",
+ GSTR1_DataField.CUST_NAME.value: get_party_for_gstin(
+ "29AABCR1718E1ZL"
+ ),
+ GSTR1_DataField.DOC_NUMBER.value: "S008403",
+ GSTR1_DataField.DOC_DATE.value: "2016-11-24",
+ GSTR1_DataField.DOC_VALUE.value: 729248.16,
+ GSTR1_DataField.POS.value: "06-Haryana",
+ GSTR1_DataField.REVERSE_CHARGE.value: "N",
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_B2B_InvoiceType.DE.value,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 325,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: 500,
+ }
+ },
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = B2B().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = B2B().convert_to_gov_data_format(process_mapped_data(self.mapped_data))
+ self.assertListEqual(self.json_data, output)
+
+
+class TestB2CL(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = [
+ {
+ GovDataField.POS.value: "05",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "92661",
+ GovDataField.DOC_DATE.value: "10-01-2016",
+ GovDataField.DOC_VALUE.value: 784586.33,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ ],
+ },
+ {
+ GovDataField.DOC_NUMBER.value: "92662",
+ GovDataField.DOC_DATE.value: "10-01-2016",
+ GovDataField.DOC_VALUE.value: 784586.33,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ }
+ ],
+ },
+ ],
+ },
+ {
+ GovDataField.POS.value: "24",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "92663",
+ GovDataField.DOC_DATE.value: "10-01-2016",
+ GovDataField.DOC_VALUE.value: 784586.33,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ },
+ ],
+ },
+ {
+ GovDataField.DOC_NUMBER.value: "92664",
+ GovDataField.DOC_DATE.value: "10-01-2016",
+ GovDataField.DOC_VALUE.value: 784586.33,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.IGST.value: 325,
+ GovDataField.CESS.value: 500,
+ },
+ }
+ ],
+ },
+ ],
+ },
+ ]
+ cls.mapped_data = {
+ GSTR1_SubCategory.B2CL.value: {
+ "92661": {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DOC_TYPE.value: "B2C (Large)",
+ GSTR1_DataField.DOC_NUMBER.value: "92661",
+ GSTR1_DataField.DOC_DATE.value: "2016-01-10",
+ GSTR1_DataField.DOC_VALUE.value: 784586.33,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 20000,
+ GSTR1_DataField.IGST.value: 650,
+ GSTR1_DataField.CESS.value: 1000,
+ },
+ "92662": {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DOC_TYPE.value: "B2C (Large)",
+ GSTR1_DataField.DOC_NUMBER.value: "92662",
+ GSTR1_DataField.DOC_DATE.value: "2016-01-10",
+ GSTR1_DataField.DOC_VALUE.value: 784586.33,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 325,
+ GSTR1_DataField.CESS.value: 500,
+ },
+ "92663": {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DOC_TYPE.value: "B2C (Large)",
+ GSTR1_DataField.DOC_NUMBER.value: "92663",
+ GSTR1_DataField.DOC_DATE.value: "2016-01-10",
+ GSTR1_DataField.DOC_VALUE.value: 784586.33,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 20000,
+ GSTR1_DataField.IGST.value: 650,
+ GSTR1_DataField.CESS.value: 1000,
+ },
+ "92664": {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DOC_TYPE.value: "B2C (Large)",
+ GSTR1_DataField.DOC_NUMBER.value: "92664",
+ GSTR1_DataField.DOC_DATE.value: "2016-01-10",
+ GSTR1_DataField.DOC_VALUE.value: 784586.33,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 325,
+ GSTR1_ItemField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 325,
+ GSTR1_DataField.CESS.value: 500,
+ },
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = B2CL().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = B2CL().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestExports(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = [
+ {
+ GovDataField.EXPORT_TYPE.value: "WPAY",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "81542",
+ GovDataField.DOC_DATE.value: "12-02-2016",
+ GovDataField.DOC_VALUE.value: 995048.36,
+ GovDataField.SHIPPING_PORT_CODE.value: "ASB991",
+ GovDataField.SHIPPING_BILL_NUMBER.value: "7896542",
+ GovDataField.SHIPPING_BILL_DATE.value: "04-10-2016",
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.IGST.value: 833.33,
+ GovDataField.CESS.value: 100,
+ }
+ ],
+ }
+ ],
+ },
+ {
+ GovDataField.EXPORT_TYPE.value: "WOPAY",
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.DOC_NUMBER.value: "81543",
+ GovDataField.DOC_DATE.value: "12-02-2016",
+ GovDataField.DOC_VALUE.value: 995048.36,
+ GovDataField.SHIPPING_PORT_CODE.value: "ASB981",
+ GovDataField.SHIPPING_BILL_NUMBER.value: "7896542",
+ GovDataField.SHIPPING_BILL_DATE.value: "04-10-2016",
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAXABLE_VALUE.value: 10000,
+ GovDataField.TAX_RATE.value: 0,
+ GovDataField.IGST.value: 0,
+ GovDataField.CESS.value: 100,
+ }
+ ],
+ }
+ ],
+ },
+ ]
+ cls.mapped_data = {
+ GSTR1_SubCategory.EXPWP.value: {
+ "81542": {
+ GSTR1_DataField.DOC_TYPE.value: "WPAY",
+ GSTR1_DataField.DOC_NUMBER.value: "81542",
+ GSTR1_DataField.DOC_DATE.value: "2016-02-12",
+ GSTR1_DataField.DOC_VALUE.value: 995048.36,
+ GSTR1_DataField.SHIPPING_PORT_CODE.value: "ASB991",
+ GSTR1_DataField.SHIPPING_BILL_NUMBER.value: "7896542",
+ GSTR1_DataField.SHIPPING_BILL_DATE.value: "2016-10-04",
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 833.33,
+ GSTR1_ItemField.CESS.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 833.33,
+ GSTR1_DataField.CESS.value: 100,
+ }
+ },
+ GSTR1_SubCategory.EXPWOP.value: {
+ "81543": {
+ GSTR1_DataField.DOC_TYPE.value: "WOPAY",
+ GSTR1_DataField.DOC_NUMBER.value: "81543",
+ GSTR1_DataField.DOC_DATE.value: "2016-02-12",
+ GSTR1_DataField.DOC_VALUE.value: 995048.36,
+ GSTR1_DataField.SHIPPING_PORT_CODE.value: "ASB981",
+ GSTR1_DataField.SHIPPING_BILL_NUMBER.value: "7896542",
+ GSTR1_DataField.SHIPPING_BILL_DATE.value: "2016-10-04",
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 0,
+ GSTR1_ItemField.CESS.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 0,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_DataField.IGST.value: 0,
+ GSTR1_DataField.CESS.value: 100,
+ }
+ },
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = Exports().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = Exports().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestB2CS(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = [
+ {
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TYPE.value: "OE",
+ GovDataField.POS.value: "05",
+ GovDataField.TAXABLE_VALUE.value: 110,
+ GovDataField.IGST.value: 10,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 10,
+ },
+ {
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.TYPE.value: "OE",
+ GovDataField.TAXABLE_VALUE.value: 100,
+ GovDataField.IGST.value: 10,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 10,
+ GovDataField.POS.value: "06",
+ },
+ ]
+ cls.mapped_data = {
+ GSTR1_SubCategory.B2CS.value: {
+ "05-Uttarakhand - 5.0": [
+ {
+ GSTR1_DataField.TAXABLE_VALUE.value: 110,
+ GSTR1_DataField.DOC_TYPE.value: "OE",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.TAX_RATE.value: 5,
+ GSTR1_DataField.IGST.value: 10,
+ GSTR1_DataField.CESS.value: 10,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ },
+ ],
+ "06-Haryana - 5.0": [
+ {
+ GSTR1_DataField.TAXABLE_VALUE.value: 100,
+ GSTR1_DataField.DOC_TYPE.value: "OE",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.POS.value: "06-Haryana",
+ GSTR1_DataField.TAX_RATE.value: 5,
+ GSTR1_DataField.IGST.value: 10,
+ GSTR1_DataField.CESS.value: 10,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ }
+ ],
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = B2CS().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = B2CS().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestNilRated(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = {
+ GovDataField.INVOICES.value: [
+ {
+ GovDataField.SUPPLY_TYPE.value: "INTRB2B",
+ GovDataField.EXEMPTED_AMOUNT.value: 123.45,
+ GovDataField.NIL_RATED_AMOUNT.value: 1470.85,
+ GovDataField.NON_GST_AMOUNT.value: 1258.5,
+ },
+ {
+ GovDataField.SUPPLY_TYPE.value: "INTRB2C",
+ GovDataField.EXEMPTED_AMOUNT.value: 123.45,
+ GovDataField.NIL_RATED_AMOUNT.value: 1470.85,
+ GovDataField.NON_GST_AMOUNT.value: 1258.5,
+ },
+ ]
+ }
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.NIL_EXEMPT.value: {
+ "Inter-State supplies to registered persons": [
+ {
+ GSTR1_DataField.DOC_TYPE.value: "Inter-State supplies to registered persons",
+ GSTR1_DataField.EXEMPTED_AMOUNT.value: 123.45,
+ GSTR1_DataField.NIL_RATED_AMOUNT.value: 1470.85,
+ GSTR1_DataField.NON_GST_AMOUNT.value: 1258.5,
+ GSTR1_DataField.TAXABLE_VALUE.value: 2852.8,
+ }
+ ],
+ "Inter-State supplies to unregistered persons": [
+ {
+ GSTR1_DataField.DOC_TYPE.value: "Inter-State supplies to unregistered persons",
+ GSTR1_DataField.EXEMPTED_AMOUNT.value: 123.45,
+ GSTR1_DataField.NIL_RATED_AMOUNT.value: 1470.85,
+ GSTR1_DataField.NON_GST_AMOUNT.value: 1258.5,
+ GSTR1_DataField.TAXABLE_VALUE.value: 2852.8,
+ }
+ ],
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = NilRated().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = NilRated().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertDictEqual(self.json_data, output)
+
+
+class TestCDNR(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = [
+ {
+ GovDataField.CUST_GSTIN.value: "24AANFA2641L1ZF",
+ GovDataField.NOTE_DETAILS.value: [
+ {
+ GovDataField.NOTE_TYPE.value: "C",
+ GovDataField.NOTE_NUMBER.value: "533515",
+ GovDataField.NOTE_DATE.value: "23-09-2016",
+ GovDataField.POS.value: "03",
+ GovDataField.REVERSE_CHARGE.value: "Y",
+ GovDataField.INVOICE_TYPE.value: "DE",
+ GovDataField.DOC_VALUE.value: 123123,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 10,
+ GovDataField.TAXABLE_VALUE.value: 5225.28,
+ GovDataField.SGST.value: 0,
+ GovDataField.CGST.value: 0,
+ GovDataField.IGST.value: 339.64,
+ GovDataField.CESS.value: 789.52,
+ },
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 10,
+ GovDataField.TAXABLE_VALUE.value: 5225.28,
+ GovDataField.SGST.value: 0,
+ GovDataField.CGST.value: 0,
+ GovDataField.IGST.value: 339.64,
+ GovDataField.CESS.value: 789.52,
+ },
+ },
+ ],
+ },
+ ],
+ }
+ ]
+ cls.mapped_data = {
+ GSTR1_SubCategory.CDNR.value: {
+ "533515": {
+ GSTR1_DataField.CUST_GSTIN.value: "24AANFA2641L1ZF",
+ GSTR1_DataField.CUST_NAME.value: get_party_for_gstin(
+ "24AANFA2641L1ZF"
+ ),
+ GSTR1_DataField.TRANSACTION_TYPE.value: "Credit Note",
+ GSTR1_DataField.DOC_NUMBER.value: "533515",
+ GSTR1_DataField.DOC_DATE.value: "2016-09-23",
+ GSTR1_DataField.POS.value: "03-Punjab",
+ GSTR1_DataField.REVERSE_CHARGE.value: "Y",
+ GSTR1_DataField.DOC_TYPE.value: "Deemed Exports",
+ GSTR1_DataField.DOC_VALUE.value: -123123,
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: -5225.28,
+ GSTR1_ItemField.IGST.value: -339.64,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: -789.52,
+ GSTR1_DataField.TAX_RATE.value: 10,
+ },
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: -5225.28,
+ GSTR1_ItemField.IGST.value: -339.64,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: -789.52,
+ GSTR1_DataField.TAX_RATE.value: 10,
+ },
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: -10450.56,
+ GSTR1_DataField.IGST.value: -679.28,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.CESS.value: -1579.04,
+ }
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = CDNR().convert_to_internal_data_format(copy.deepcopy(self.json_data))
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = CDNR().convert_to_gov_data_format(
+ process_mapped_data(copy.deepcopy(self.mapped_data))
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestCDNUR(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+
+ cls.json_data = [
+ {
+ GovDataField.TYPE.value: "B2CL",
+ GovDataField.NOTE_TYPE.value: "C",
+ GovDataField.NOTE_NUMBER.value: "533515",
+ GovDataField.NOTE_DATE.value: "23-09-2016",
+ GovDataField.POS.value: "03",
+ GovDataField.DOC_VALUE.value: 64646,
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.ITEM_DETAILS.value: {
+ GovDataField.TAX_RATE.value: 10,
+ GovDataField.TAXABLE_VALUE.value: 5225.28,
+ GovDataField.IGST.value: 339.64,
+ GovDataField.CESS.value: 789.52,
+ },
+ }
+ ],
+ }
+ ]
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.CDNUR.value: {
+ "533515": {
+ GSTR1_DataField.TRANSACTION_TYPE.value: "Credit Note",
+ GSTR1_DataField.DOC_TYPE.value: "B2CL",
+ GSTR1_DataField.DOC_NUMBER.value: "533515",
+ GSTR1_DataField.DOC_DATE.value: "2016-09-23",
+ GSTR1_DataField.DOC_VALUE.value: -64646,
+ GSTR1_DataField.POS.value: "03-Punjab",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.ITEMS.value: [
+ {
+ GSTR1_ItemField.TAXABLE_VALUE.value: -5225.28,
+ GSTR1_ItemField.IGST.value: -339.64,
+ GSTR1_ItemField.CESS.value: -789.52,
+ GSTR1_DataField.TAX_RATE.value: 10,
+ }
+ ],
+ GSTR1_DataField.TAXABLE_VALUE.value: -5225.28,
+ GSTR1_DataField.IGST.value: -339.64,
+ GSTR1_DataField.CESS.value: -789.52,
+ }
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = CDNUR().convert_to_internal_data_format(copy.deepcopy(self.json_data))
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = CDNUR().convert_to_gov_data_format(
+ process_mapped_data(copy.deepcopy(self.mapped_data))
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestHSNSUM(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+
+ cls.json_data = {
+ GovDataField.HSN_DATA.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.HSN_CODE.value: "1010",
+ GovDataField.DESCRIPTION.value: "Goods Description",
+ GovDataField.UOM.value: "KGS",
+ GovDataField.QUANTITY.value: 2.05,
+ GovDataField.TAXABLE_VALUE.value: 10.23,
+ GovDataField.IGST.value: 14.52,
+ GovDataField.CESS.value: 500,
+ GovDataField.TAX_RATE.value: 0.1,
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.HSN_CODE.value: "1011",
+ GovDataField.DESCRIPTION.value: "Goods Description",
+ GovDataField.UOM.value: "NOS",
+ GovDataField.QUANTITY.value: 2.05,
+ GovDataField.TAXABLE_VALUE.value: 10.23,
+ GovDataField.IGST.value: 14.52,
+ GovDataField.CESS.value: 500,
+ GovDataField.TAX_RATE.value: 5.0,
+ },
+ ]
+ }
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.HSN.value: {
+ "1010 - KGS-KILOGRAMS - 0.1": {
+ GSTR1_DataField.HSN_CODE.value: "1010",
+ GSTR1_DataField.DESCRIPTION.value: "Goods Description",
+ GSTR1_DataField.UOM.value: "KGS-KILOGRAMS",
+ GSTR1_DataField.QUANTITY.value: 2.05,
+ GSTR1_DataField.TAXABLE_VALUE.value: 10.23,
+ GSTR1_DataField.IGST.value: 14.52,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 0.1,
+ GSTR1_DataField.DOC_VALUE.value: 524.75,
+ },
+ "1011 - NOS-NUMBERS - 5.0": {
+ GSTR1_DataField.HSN_CODE.value: "1011",
+ GSTR1_DataField.DESCRIPTION.value: "Goods Description",
+ GSTR1_DataField.UOM.value: "NOS-NUMBERS",
+ GSTR1_DataField.QUANTITY.value: 2.05,
+ GSTR1_DataField.TAXABLE_VALUE.value: 10.23,
+ GSTR1_DataField.IGST.value: 14.52,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ GSTR1_DataField.DOC_VALUE.value: 524.75,
+ },
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = HSNSUM().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = HSNSUM().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertDictEqual(self.json_data, output)
+
+
+class TestAT(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+
+ cls.json_data = [
+ {
+ GovDataField.POS.value: "05",
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ {
+ GovDataField.TAX_RATE.value: 6,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ ],
+ },
+ {
+ GovDataField.POS.value: "24",
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ {
+ GovDataField.TAX_RATE.value: 6,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ ],
+ },
+ ]
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.AT.value: {
+ "05-Uttarakhand - 5.0": [
+ {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: 9400,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ ],
+ "05-Uttarakhand - 6.0": [
+ {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: 9400,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 6,
+ }
+ ],
+ "24-Gujarat - 5.0": [
+ {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: 9400,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ "24-Gujarat - 6.0": [
+ {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: 9400,
+ GSTR1_DataField.CESS.value: 500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: 100,
+ GSTR1_DataField.TAX_RATE.value: 6,
+ }
+ ],
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = AT().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = AT().convert_to_gov_data_format(process_mapped_data(self.mapped_data))
+ self.assertListEqual(self.json_data, output)
+
+
+class TestTXPD(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+
+ cls.json_data = [
+ {
+ GovDataField.POS.value: "05",
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ {
+ GovDataField.TAX_RATE.value: 6,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ ],
+ },
+ {
+ GovDataField.POS.value: "24",
+ GovDataField.SUPPLY_TYPE.value: "INTER",
+ GovDataField.DIFF_PERCENTAGE.value: 0.65,
+ GovDataField.ITEMS.value: [
+ {
+ GovDataField.TAX_RATE.value: 5,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ {
+ GovDataField.TAX_RATE.value: 6,
+ GovDataField.ADVANCE_AMOUNT.value: 100,
+ GovDataField.IGST.value: 9400,
+ GovDataField.CGST.value: 0,
+ GovDataField.SGST.value: 0,
+ GovDataField.CESS.value: 500,
+ },
+ ],
+ },
+ ]
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.TXP.value: {
+ "05-Uttarakhand - 5.0": [
+ {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: -9400,
+ GSTR1_DataField.CESS.value: -500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: -100,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ },
+ ],
+ "05-Uttarakhand - 6.0": [
+ {
+ GSTR1_DataField.POS.value: "05-Uttarakhand",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: -9400,
+ GSTR1_DataField.CESS.value: -500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: -100,
+ GSTR1_DataField.TAX_RATE.value: 6,
+ }
+ ],
+ "24-Gujarat - 5.0": [
+ {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: -9400,
+ GSTR1_DataField.CESS.value: -500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: -100,
+ GSTR1_DataField.TAX_RATE.value: 5,
+ }
+ ],
+ "24-Gujarat - 6.0": [
+ {
+ GSTR1_DataField.POS.value: "24-Gujarat",
+ GSTR1_DataField.DIFF_PERCENTAGE.value: 0.65,
+ GSTR1_DataField.IGST.value: -9400,
+ GSTR1_DataField.CESS.value: -500,
+ GSTR1_DataField.CGST.value: 0,
+ GSTR1_DataField.SGST.value: 0,
+ GSTR1_DataField.TAXABLE_VALUE.value: -100,
+ GSTR1_DataField.TAX_RATE.value: 6,
+ }
+ ],
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = TXPD().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = TXPD().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertListEqual(self.json_data, output)
+
+
+class TestDOC_ISSUE(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = {
+ GovDataField.DOC_ISSUE_DETAILS.value: [
+ {
+ GovDataField.DOC_ISSUE_NUMBER.value: 1,
+ GovDataField.DOC_ISSUE_LIST.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.FROM_SR.value: "1",
+ GovDataField.TO_SR.value: "10",
+ GovDataField.TOTAL_COUNT.value: 10,
+ GovDataField.CANCELLED_COUNT.value: 0,
+ GovDataField.NET_ISSUE.value: 10,
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.FROM_SR.value: "11",
+ GovDataField.TO_SR.value: "20",
+ GovDataField.TOTAL_COUNT.value: 10,
+ GovDataField.CANCELLED_COUNT.value: 0,
+ GovDataField.NET_ISSUE.value: 10,
+ },
+ ],
+ },
+ {
+ GovDataField.DOC_ISSUE_NUMBER.value: 2,
+ GovDataField.DOC_ISSUE_LIST.value: [
+ {
+ GovDataField.INDEX.value: 1,
+ GovDataField.FROM_SR.value: "1",
+ GovDataField.TO_SR.value: "10",
+ GovDataField.TOTAL_COUNT.value: 10,
+ GovDataField.CANCELLED_COUNT.value: 0,
+ GovDataField.NET_ISSUE.value: 10,
+ },
+ {
+ GovDataField.INDEX.value: 2,
+ GovDataField.FROM_SR.value: "11",
+ GovDataField.TO_SR.value: "20",
+ GovDataField.TOTAL_COUNT.value: 10,
+ GovDataField.CANCELLED_COUNT.value: 0,
+ GovDataField.NET_ISSUE.value: 10,
+ },
+ ],
+ },
+ ]
+ }
+ cls.mapped_data = {
+ GSTR1_SubCategory.DOC_ISSUE.value: {
+ "Invoices for outward supply - 1": {
+ GSTR1_DataField.DOC_TYPE.value: "Invoices for outward supply",
+ GSTR1_DataField.FROM_SR.value: "1",
+ GSTR1_DataField.TO_SR.value: "10",
+ GSTR1_DataField.TOTAL_COUNT.value: 10,
+ GSTR1_DataField.CANCELLED_COUNT.value: 0,
+ "net_issue": 10,
+ },
+ "Invoices for outward supply - 11": {
+ GSTR1_DataField.DOC_TYPE.value: "Invoices for outward supply",
+ GSTR1_DataField.FROM_SR.value: "11",
+ GSTR1_DataField.TO_SR.value: "20",
+ GSTR1_DataField.TOTAL_COUNT.value: 10,
+ GSTR1_DataField.CANCELLED_COUNT.value: 0,
+ "net_issue": 10,
+ },
+ "Invoices for inward supply from unregistered person - 1": {
+ GSTR1_DataField.DOC_TYPE.value: "Invoices for inward supply from unregistered person",
+ GSTR1_DataField.FROM_SR.value: "1",
+ GSTR1_DataField.TO_SR.value: "10",
+ GSTR1_DataField.TOTAL_COUNT.value: 10,
+ GSTR1_DataField.CANCELLED_COUNT.value: 0,
+ "net_issue": 10,
+ },
+ "Invoices for inward supply from unregistered person - 11": {
+ GSTR1_DataField.DOC_TYPE.value: "Invoices for inward supply from unregistered person",
+ GSTR1_DataField.FROM_SR.value: "11",
+ GSTR1_DataField.TO_SR.value: "20",
+ GSTR1_DataField.TOTAL_COUNT.value: 10,
+ GSTR1_DataField.CANCELLED_COUNT.value: 0,
+ "net_issue": 10,
+ },
+ }
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = DOC_ISSUE().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = DOC_ISSUE().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertDictEqual(self.json_data, output)
+
+
+class TestSUPECOM(FrappeTestCase):
+ @classmethod
+ def setUpClass(cls):
+ super().setUpClass()
+ cls.json_data = {
+ GovDataField.SUPECOM_52.value: [
+ {
+ GovDataField.ECOMMERCE_GSTIN.value: "20ALYPD6528PQC5",
+ GovDataField.NET_TAXABLE_VALUE.value: 10000,
+ "igst": 1000,
+ "cgst": 0,
+ "sgst": 0,
+ "cess": 0,
+ }
+ ],
+ GovDataField.SUPECOM_9_5.value: [
+ {
+ GovDataField.ECOMMERCE_GSTIN.value: "20ALYPD6528PQC5",
+ GovDataField.NET_TAXABLE_VALUE.value: 10000,
+ "igst": 1000,
+ "cgst": 0,
+ "sgst": 0,
+ "cess": 0,
+ }
+ ],
+ }
+
+ cls.mapped_data = {
+ GSTR1_SubCategory.SUPECOM_52.value: {
+ "20ALYPD6528PQC5": {
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_SubCategory.SUPECOM_52.value,
+ GSTR1_DataField.ECOMMERCE_GSTIN.value: "20ALYPD6528PQC5",
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 1000,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+ },
+ GSTR1_SubCategory.SUPECOM_9_5.value: {
+ "20ALYPD6528PQC5": {
+ GSTR1_DataField.DOC_TYPE.value: GSTR1_SubCategory.SUPECOM_9_5.value,
+ GSTR1_DataField.ECOMMERCE_GSTIN.value: "20ALYPD6528PQC5",
+ GSTR1_DataField.TAXABLE_VALUE.value: 10000,
+ GSTR1_ItemField.IGST.value: 1000,
+ GSTR1_ItemField.CGST.value: 0,
+ GSTR1_ItemField.SGST.value: 0,
+ GSTR1_ItemField.CESS.value: 0,
+ }
+ },
+ }
+
+ def test_convert_to_internal_data_format(self):
+ output = SUPECOM().convert_to_internal_data_format(self.json_data)
+ self.assertDictEqual(self.mapped_data, output)
+
+ def test_convert_to_gov_data_format(self):
+ output = SUPECOM().convert_to_gov_data_format(
+ process_mapped_data(self.mapped_data)
+ )
+ self.assertDictEqual(self.json_data, output)
diff --git a/india_compliance/gst_india/utils/gstr/__init__.py b/india_compliance/gst_india/utils/gstr_2/__init__.py
similarity index 84%
rename from india_compliance/gst_india/utils/gstr/__init__.py
rename to india_compliance/gst_india/utils/gstr_2/__init__.py
index ee3ccc1a0a..9c258d9b7c 100644
--- a/india_compliance/gst_india/utils/gstr/__init__.py
+++ b/india_compliance/gst_india/utils/gstr_2/__init__.py
@@ -5,22 +5,13 @@
from frappe.query_builder.terms import Criterion
from frappe.utils import cint
-from india_compliance.gst_india.api_classes.returns import (
- GSTR2aAPI,
- GSTR2bAPI,
- ReturnsAPI,
-)
+from india_compliance.gst_india.api_classes.returns import GSTR2aAPI, GSTR2bAPI
from india_compliance.gst_india.doctype.gstr_import_log.gstr_import_log import (
create_import_log,
- toggle_scheduled_jobs,
)
from india_compliance.gst_india.utils import get_party_for_gstin
-from india_compliance.gst_india.utils.gstr import gstr_2a, gstr_2b
-
-
-class ReturnType(Enum):
- GSTR2A = "GSTR2a"
- GSTR2B = "GSTR2b"
+from india_compliance.gst_india.utils.gstr_2 import gstr_2a, gstr_2b
+from india_compliance.gst_india.utils.gstr_utils import ReturnType
class GSTRCategory(Enum):
@@ -334,68 +325,6 @@ def _download_gstr_2a(gstin, return_period, json_data):
save_gstr_2a(gstin, return_period, json_data)
-GSTR_FUNCTIONS = {
- ReturnType.GSTR2A.value: _download_gstr_2a,
- ReturnType.GSTR2B.value: save_gstr_2b,
-}
-
-
-def download_queued_request():
- queued_requests = frappe.get_all(
- "GSTR Import Log",
- filters={"request_id": ["is", "set"]},
- fields=[
- "name",
- "gstin",
- "return_type",
- "classification",
- "return_period",
- "request_id",
- "request_time",
- ],
- )
-
- if not queued_requests:
- return toggle_scheduled_jobs(stopped=True)
-
- for doc in queued_requests:
- frappe.enqueue(_download_queued_request, queue="long", doc=doc)
-
-
-def _download_queued_request(doc):
- try:
- api = ReturnsAPI(doc.gstin)
- response = api.download_files(
- doc.return_period,
- doc.request_id,
- )
-
- except Exception as e:
- frappe.db.delete("GSTR Import Log", {"name": doc.name})
- raise e
-
- if response.error_type in ["otp_requested", "invalid_otp"]:
- return toggle_scheduled_jobs(stopped=True)
-
- if response.error_type == "no_docs_found":
- return create_import_log(
- doc.gstin,
- doc.return_type,
- doc.return_period,
- doc.classification,
- data_not_found=True,
- )
-
- if response.error_type == "queued":
- return
-
- if response.error_type:
- return frappe.db.delete("GSTR Import Log", {"name": doc.name})
-
- frappe.db.set_value("GSTR Import Log", doc.name, "request_id", None)
- GSTR_FUNCTIONS[doc.return_type](doc.gstin, doc.return_period, response)
-
-
def show_queued_message():
frappe.msgprint(
_(
diff --git a/india_compliance/gst_india/utils/gstr/gstr.py b/india_compliance/gst_india/utils/gstr_2/gstr.py
similarity index 65%
rename from india_compliance/gst_india/utils/gstr/gstr.py
rename to india_compliance/gst_india/utils/gstr_2/gstr.py
index 06305f8404..f9979cc595 100644
--- a/india_compliance/gst_india/utils/gstr/gstr.py
+++ b/india_compliance/gst_india/utils/gstr_2/gstr.py
@@ -1,13 +1,9 @@
import frappe
-from frappe import _
-from frappe.utils import add_to_date, now_datetime
from india_compliance.gst_india.constants import STATE_NUMBERS
from india_compliance.gst_india.doctype.gst_inward_supply.gst_inward_supply import (
create_inward_supply,
)
-from india_compliance.gst_india.utils import get_gstin_list
-from india_compliance.gst_india.utils.gstr import ReturnsAPI
def get_mapped_value(value, mapping):
@@ -146,85 +142,3 @@ def set_key(self, key, value):
def update_gstins(self):
pass
-
-
-@frappe.whitelist()
-def validate_company_gstins(company=None, company_gstin=None):
- """
- Checks the validity of the company's GSTIN authentication.
-
- Args:
- company_gstin (str): The GSTIN of the company to validate.
-
- Returns:
- dict: A dictionary where the keys are the GSTINs and the values are booleans indicating whether the authentication is valid.
- """
- frappe.has_permission("GST Settings", throw=True)
-
- credentials = get_company_gstin_credentials(company, company_gstin)
-
- if company_gstin and not credentials:
- frappe.throw(
- _("Missing GSTIN credentials for GSTIN: {gstin}.").format(
- gstin=company_gstin
- )
- )
-
- if not credentials:
- frappe.throw(_("Missing credentials in GST Settings"))
-
- if company and not company_gstin:
- missing_credentials = set(get_gstin_list(company)) - set(
- credential.gstin for credential in credentials
- )
-
- if missing_credentials:
- frappe.throw(
- _("Missing GSTIN credentials for GSTIN(s): {gstins}.").format(
- gstins=", ".join(missing_credentials),
- )
- )
-
- gstin_authentication_status = {
- credential.gstin: (
- credential.session_expiry
- and credential.auth_token
- and credential.session_expiry > add_to_date(now_datetime(), minutes=30)
- )
- for credential in credentials
- }
-
- return gstin_authentication_status
-
-
-def get_company_gstin_credentials(company=None, company_gstin=None):
- filters = {"service": "Returns"}
-
- if company:
- filters["company"] = company
-
- if company_gstin:
- filters["gstin"] = company_gstin
-
- return frappe.get_all(
- "GST Credential",
- filters=filters,
- fields=["gstin", "session_expiry", "auth_token"],
- )
-
-
-@frappe.whitelist()
-def request_otp(company_gstin):
- frappe.has_permission("GST Settings", throw=True)
-
- return ReturnsAPI(company_gstin).request_otp()
-
-
-@frappe.whitelist()
-def authenticate_otp(company_gstin, otp):
- frappe.has_permission("GST Settings", throw=True)
-
- api = ReturnsAPI(company_gstin)
- response = api.autheticate_with_otp(otp)
-
- return api.process_response(response)
diff --git a/india_compliance/gst_india/utils/gstr/gstr_2a.py b/india_compliance/gst_india/utils/gstr_2/gstr_2a.py
similarity index 99%
rename from india_compliance/gst_india/utils/gstr/gstr_2a.py
rename to india_compliance/gst_india/utils/gstr_2/gstr_2a.py
index 770a0c7eee..f9636ff9e5 100644
--- a/india_compliance/gst_india/utils/gstr/gstr_2a.py
+++ b/india_compliance/gst_india/utils/gstr_2/gstr_2a.py
@@ -3,7 +3,7 @@
import frappe
from india_compliance.gst_india.utils import get_datetime, parse_datetime
-from india_compliance.gst_india.utils.gstr.gstr import GSTR, get_mapped_value
+from india_compliance.gst_india.utils.gstr_2.gstr import GSTR, get_mapped_value
def map_date_format(date_str, source_format, target_format):
diff --git a/india_compliance/gst_india/utils/gstr/gstr_2b.py b/india_compliance/gst_india/utils/gstr_2/gstr_2b.py
similarity index 98%
rename from india_compliance/gst_india/utils/gstr/gstr_2b.py
rename to india_compliance/gst_india/utils/gstr_2/gstr_2b.py
index a1165623ec..1e9b348b1d 100644
--- a/india_compliance/gst_india/utils/gstr/gstr_2b.py
+++ b/india_compliance/gst_india/utils/gstr_2/gstr_2b.py
@@ -1,7 +1,7 @@
import frappe
from india_compliance.gst_india.utils import parse_datetime
-from india_compliance.gst_india.utils.gstr.gstr import GSTR, get_mapped_value
+from india_compliance.gst_india.utils.gstr_2.gstr import GSTR, get_mapped_value
class GSTR2b(GSTR):
diff --git a/india_compliance/gst_india/utils/gstr/test_gstr_2a.py b/india_compliance/gst_india/utils/gstr_2/test_gstr_2a.py
similarity index 98%
rename from india_compliance/gst_india/utils/gstr/test_gstr_2a.py
rename to india_compliance/gst_india/utils/gstr_2/test_gstr_2a.py
index a58cafc54f..4993aef91b 100644
--- a/india_compliance/gst_india/utils/gstr/test_gstr_2a.py
+++ b/india_compliance/gst_india/utils/gstr_2/test_gstr_2a.py
@@ -7,7 +7,7 @@
from frappe.utils import get_datetime
from india_compliance.gst_india.utils import get_data_file_path
-from india_compliance.gst_india.utils.gstr import (
+from india_compliance.gst_india.utils.gstr_2 import (
GSTRCategory,
ReturnType,
download_gstr_2a,
@@ -64,8 +64,8 @@ def tearDownClass(cls):
frappe.db.delete(cls.doctype, {"company_gstin": cls.gstin})
frappe.db.delete(cls.log_doctype, {"gstin": cls.gstin})
- @patch("india_compliance.gst_india.utils.gstr.save_gstr")
- @patch("india_compliance.gst_india.utils.gstr.GSTR2aAPI")
+ @patch("india_compliance.gst_india.utils.gstr_2.save_gstr")
+ @patch("india_compliance.gst_india.utils.gstr_2.GSTR2aAPI")
def test_download_gstr_2a(self, mock_gstr_2a_api, mock_save_gstr):
def mock_get_data(action, return_period, otp):
if action in ["B2B", "B2BA", "CDN", "CDNA"]:
diff --git a/india_compliance/gst_india/utils/gstr/test_gstr_2b.py b/india_compliance/gst_india/utils/gstr_2/test_gstr_2b.py
similarity index 98%
rename from india_compliance/gst_india/utils/gstr/test_gstr_2b.py
rename to india_compliance/gst_india/utils/gstr_2/test_gstr_2b.py
index 3d7002b6ba..ec940318ad 100644
--- a/india_compliance/gst_india/utils/gstr/test_gstr_2b.py
+++ b/india_compliance/gst_india/utils/gstr_2/test_gstr_2b.py
@@ -5,8 +5,8 @@
from frappe.tests.utils import FrappeTestCase
from india_compliance.gst_india.utils import get_data_file_path
-from india_compliance.gst_india.utils.gstr import GSTRCategory, save_gstr_2b
-from india_compliance.gst_india.utils.gstr.test_gstr_2a import TestGSTRMixin
+from india_compliance.gst_india.utils.gstr_2 import GSTRCategory, save_gstr_2b
+from india_compliance.gst_india.utils.gstr_2.test_gstr_2a import TestGSTRMixin
class TestGSTR2b(FrappeTestCase, TestGSTRMixin):
diff --git a/india_compliance/gst_india/utils/gstr_utils.py b/india_compliance/gst_india/utils/gstr_utils.py
new file mode 100644
index 0000000000..2e9c0a2c1c
--- /dev/null
+++ b/india_compliance/gst_india/utils/gstr_utils.py
@@ -0,0 +1,170 @@
+from enum import Enum
+
+import frappe
+from frappe import _
+from frappe.utils import add_to_date, now_datetime
+
+from india_compliance.gst_india.api_classes.returns import ReturnsAPI
+from india_compliance.gst_india.doctype.gstr_import_log.gstr_import_log import (
+ create_import_log,
+ toggle_scheduled_jobs,
+)
+from india_compliance.gst_india.utils import get_gstin_list
+from india_compliance.gst_india.utils.gstr_1.gstr_1_download import (
+ save_gstr_1_filed_data,
+ save_gstr_1_unfiled_data,
+)
+
+
+class ReturnType(Enum):
+ GSTR2A = "GSTR2a"
+ GSTR2B = "GSTR2b"
+ GSTR1 = "GSTR1"
+ UnfiledGSTR1 = "Unfiled GSTR1"
+
+
+@frappe.whitelist()
+def validate_company_gstins(company=None, company_gstin=None):
+ """
+ Checks the validity of the company's GSTIN authentication.
+
+ Args:
+ company_gstin (str): The GSTIN of the company to validate.
+
+ Returns:
+ dict: A dictionary where the keys are the GSTINs and the values are booleans indicating whether the authentication is valid.
+ """
+ frappe.has_permission("GST Settings", throw=True)
+
+ credentials = get_company_gstin_credentials(company, company_gstin)
+
+ if company_gstin and not credentials:
+ frappe.throw(
+ _("Missing GSTIN credentials for GSTIN: {gstin}.").format(
+ gstin=company_gstin
+ )
+ )
+
+ if not credentials:
+ frappe.throw(_("Missing credentials in GST Settings"))
+
+ if company and not company_gstin:
+ missing_credentials = set(get_gstin_list(company)) - set(
+ credential.gstin for credential in credentials
+ )
+
+ if missing_credentials:
+ frappe.throw(
+ _("Missing GSTIN credentials for GSTIN(s): {gstins}.").format(
+ gstins=", ".join(missing_credentials),
+ )
+ )
+
+ gstin_authentication_status = {
+ credential.gstin: (
+ credential.session_expiry
+ and credential.auth_token
+ and credential.session_expiry > add_to_date(now_datetime(), minutes=30)
+ )
+ for credential in credentials
+ }
+
+ return gstin_authentication_status
+
+
+def get_company_gstin_credentials(company=None, company_gstin=None):
+ filters = {"service": "Returns"}
+
+ if company:
+ filters["company"] = company
+
+ if company_gstin:
+ filters["gstin"] = company_gstin
+
+ return frappe.get_all(
+ "GST Credential",
+ filters=filters,
+ fields=["gstin", "session_expiry", "auth_token"],
+ )
+
+
+@frappe.whitelist()
+def request_otp(company_gstin):
+ frappe.has_permission("GST Settings", throw=True)
+
+ return ReturnsAPI(company_gstin).request_otp()
+
+
+@frappe.whitelist()
+def authenticate_otp(company_gstin, otp):
+ frappe.has_permission("GST Settings", throw=True)
+
+ api = ReturnsAPI(company_gstin)
+ response = api.autheticate_with_otp(otp)
+
+ return api.process_response(response)
+
+
+def download_queued_request():
+ queued_requests = frappe.get_all(
+ "GSTR Import Log",
+ filters={"request_id": ["is", "set"]},
+ fields=[
+ "name",
+ "gstin",
+ "return_type",
+ "classification",
+ "return_period",
+ "request_id",
+ "request_time",
+ ],
+ )
+
+ if not queued_requests:
+ return toggle_scheduled_jobs(stopped=True)
+
+ for doc in queued_requests:
+ frappe.enqueue(_download_queued_request, queue="long", doc=doc)
+
+
+def _download_queued_request(doc):
+ from india_compliance.gst_india.utils.gstr_2 import _download_gstr_2a, save_gstr_2b
+
+ GSTR_FUNCTIONS = {
+ ReturnType.GSTR2A.value: _download_gstr_2a,
+ ReturnType.GSTR2B.value: save_gstr_2b,
+ ReturnType.GSTR1.value: save_gstr_1_filed_data,
+ ReturnType.UnfiledGSTR1.value: save_gstr_1_unfiled_data,
+ }
+
+ try:
+ api = ReturnsAPI(doc.gstin)
+ response = api.download_files(
+ doc.return_period,
+ doc.request_id,
+ )
+
+ except Exception as e:
+ frappe.db.delete("GSTR Import Log", doc.name)
+ raise e
+
+ if response.error_type in ["otp_requested", "invalid_otp"]:
+ return toggle_scheduled_jobs(stopped=True)
+
+ if response.error_type == "no_docs_found":
+ return create_import_log(
+ doc.gstin,
+ doc.return_type,
+ doc.return_period,
+ doc.classification,
+ data_not_found=True,
+ )
+
+ if response.error_type == "queued":
+ return
+
+ if response.error_type:
+ return frappe.db.delete("GSTR Import Log", {"name": doc.name})
+
+ frappe.db.set_value("GSTR Import Log", doc.name, "request_id", None)
+ GSTR_FUNCTIONS[doc.return_type](doc.gstin, doc.return_period, response)
diff --git a/india_compliance/gst_india/workspace/gst_india/gst_india.json b/india_compliance/gst_india/workspace/gst_india/gst_india.json
index 35379c901e..0bf634fe11 100644
--- a/india_compliance/gst_india/workspace/gst_india/gst_india.json
+++ b/india_compliance/gst_india/workspace/gst_india/gst_india.json
@@ -1,6 +1,6 @@
{
"charts": [],
- "content": "[{\"id\":\"Xz6h3FH8sZ\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Pending e-Waybills\",\"col\":4}},{\"id\":\"5wHVnb2VB-\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Pending e-Invoices\",\"col\":4}},{\"id\":\"FT76_s4_M1\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Invoice Cancelled, e-Invoice Active\",\"col\":4}},{\"id\":\"ROouz1137a\",\"type\":\"spacer\",\"data\":{\"col\":12}},{\"id\":\"pSkSY7vg5b\",\"type\":\"header\",\"data\":{\"text\":\"Shortcuts\",\"col\":12}},{\"id\":\"kJscjOHSLN\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GST Settings\",\"col\":3}},{\"id\":\"BT3ZIyt2_1\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GSTR-1\",\"col\":3}},{\"id\":\"fVgN1kK1r6\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GSTR-3B\",\"col\":3}},{\"id\":\"kxE1gBYYUW\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"Purchase Reconciliation Tool\",\"col\":3}},{\"id\":\"-G2gVutaOP\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"HSN Code\",\"col\":3}},{\"id\":\"sEPHpNM3bl\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"India Compliance Account\",\"col\":3}},{\"id\":\"bVd6Hbw3Yp\",\"type\":\"spacer\",\"data\":{\"col\":12}},{\"id\":\"oN0FbtHAdc\",\"type\":\"header\",\"data\":{\"text\":\"Reports and Masters\",\"col\":12}},{\"id\":\"doCaNmtmY8\",\"type\":\"card\",\"data\":{\"card_name\":\"Sales and Purchase Reports\",\"col\":4}},{\"id\":\"Emy7VbsSYq\",\"type\":\"card\",\"data\":{\"card_name\":\"Logs\",\"col\":4}},{\"id\":\"2FUZmTJKVp\",\"type\":\"card\",\"data\":{\"card_name\":\"Other GST Reports\",\"col\":4}},{\"id\":\"PhR9LKvBdc\",\"type\":\"card\",\"data\":{\"card_name\":\"New Reports\",\"col\":3}}]",
+ "content": "[{\"id\":\"Xz6h3FH8sZ\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Pending e-Waybills\",\"col\":4}},{\"id\":\"5wHVnb2VB-\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Pending e-Invoices\",\"col\":4}},{\"id\":\"FT76_s4_M1\",\"type\":\"number_card\",\"data\":{\"number_card_name\":\"Invoice Cancelled, e-Invoice Active\",\"col\":4}},{\"id\":\"ROouz1137a\",\"type\":\"spacer\",\"data\":{\"col\":12}},{\"id\":\"pSkSY7vg5b\",\"type\":\"header\",\"data\":{\"text\":\"Shortcuts\",\"col\":12}},{\"id\":\"kJscjOHSLN\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GST Settings\",\"col\":3}},{\"id\":\"BT3ZIyt2_1\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GSTR-1 Beta\",\"col\":3}},{\"id\":\"fVgN1kK1r6\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"GSTR-3B\",\"col\":3}},{\"id\":\"kxE1gBYYUW\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"Purchase Reconciliation Tool\",\"col\":3}},{\"id\":\"-G2gVutaOP\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"HSN Code\",\"col\":3}},{\"id\":\"sEPHpNM3bl\",\"type\":\"shortcut\",\"data\":{\"shortcut_name\":\"India Compliance Account\",\"col\":3}},{\"id\":\"bVd6Hbw3Yp\",\"type\":\"spacer\",\"data\":{\"col\":12}},{\"id\":\"oN0FbtHAdc\",\"type\":\"header\",\"data\":{\"text\":\"Reports and Masters\",\"col\":12}},{\"id\":\"doCaNmtmY8\",\"type\":\"card\",\"data\":{\"card_name\":\"Sales and Purchase Reports\",\"col\":4}},{\"id\":\"Emy7VbsSYq\",\"type\":\"card\",\"data\":{\"card_name\":\"Logs\",\"col\":4}},{\"id\":\"2FUZmTJKVp\",\"type\":\"card\",\"data\":{\"card_name\":\"Other GST Reports\",\"col\":4}},{\"id\":\"PhR9LKvBdc\",\"type\":\"card\",\"data\":{\"card_name\":\"New Reports\",\"col\":3}}]",
"creation": "2022-02-28 19:04:58.655348",
"custom_blocks": [],
"docstatus": 0,
@@ -12,14 +12,44 @@
"is_hidden": 0,
"label": "GST India",
"links": [
+ {
+ "hidden": 0,
+ "is_query_report": 0,
+ "label": "New Reports",
+ "link_count": 1,
+ "link_type": "DocType",
+ "onboard": 0,
+ "type": "Card Break"
+ },
+ {
+ "hidden": 0,
+ "is_query_report": 1,
+ "label": "GST Sales Register Beta",
+ "link_count": 0,
+ "link_to": "GST Sales Register Beta",
+ "link_type": "Report",
+ "onboard": 0,
+ "type": "Link"
+ },
{
"hidden": 0,
"is_query_report": 0,
"label": "Sales and Purchase Reports",
- "link_count": 5,
+ "link_count": 6,
+ "link_type": "DocType",
"onboard": 0,
"type": "Card Break"
},
+ {
+ "hidden": 0,
+ "is_query_report": 0,
+ "label": "GSTR-1",
+ "link_count": 0,
+ "link_to": "GSTR-1",
+ "link_type": "Report",
+ "onboard": 0,
+ "type": "Link"
+ },
{
"hidden": 0,
"is_query_report": 1,
@@ -73,130 +103,133 @@
{
"hidden": 0,
"is_query_report": 0,
- "label": "Other GST Reports",
+ "label": "Logs",
"link_count": 5,
+ "link_type": "DocType",
"onboard": 0,
"type": "Card Break"
},
{
"hidden": 0,
- "is_query_report": 1,
- "label": "e-Invoice Summary",
+ "is_query_report": 0,
+ "label": "e-Waybill Log",
"link_count": 0,
- "link_to": "e-Invoice Summary",
- "link_type": "Report",
+ "link_to": "e-Waybill Log",
+ "link_type": "DocType",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "GSTR-3B Details",
+ "label": "e-Invoice Log",
"link_count": 0,
- "link_to": "GSTR-3B Details",
- "link_type": "Report",
+ "link_to": "e-Invoice Log",
+ "link_type": "DocType",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "GST Balance",
+ "label": "GSTR-1 Log",
"link_count": 0,
- "link_to": "GST Balance",
- "link_type": "Report",
+ "link_to": "GSTR-1 Log",
+ "link_type": "DocType",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
- "is_query_report": 1,
- "label": "HSN-wise Summary of Outward Supplies",
+ "is_query_report": 0,
+ "label": "GST Inward Supply",
"link_count": 0,
- "link_to": "HSN-wise-summary of outward supplies",
- "link_type": "Report",
+ "link_to": "GST Inward Supply",
+ "link_type": "DocType",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "Audit Trail",
+ "label": "Integration Request",
"link_count": 0,
- "link_to": "Audit Trail",
- "link_type": "Report",
+ "link_to": "Integration Request",
+ "link_type": "DocType",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "Logs",
- "link_count": 4,
+ "label": "Other GST Reports",
+ "link_count": 6,
+ "link_type": "DocType",
"onboard": 0,
"type": "Card Break"
},
{
"hidden": 0,
- "is_query_report": 0,
- "label": "e-Waybill Log",
+ "is_query_report": 1,
+ "label": "e-Invoice Summary",
"link_count": 0,
- "link_to": "e-Waybill Log",
- "link_type": "DocType",
+ "link_to": "e-Invoice Summary",
+ "link_type": "Report",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "e-Invoice Log",
+ "label": "GSTR-3B Details",
"link_count": 0,
- "link_to": "e-Invoice Log",
- "link_type": "DocType",
+ "link_to": "GSTR-3B Details",
+ "link_type": "Report",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "GST Inward Supply",
+ "label": "GST Balance",
"link_count": 0,
- "link_to": "GST Inward Supply",
- "link_type": "DocType",
+ "link_to": "GST Balance",
+ "link_type": "Report",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
- "is_query_report": 0,
- "label": "Integration Request",
+ "is_query_report": 1,
+ "label": "HSN-wise Summary of Outward Supplies",
"link_count": 0,
- "link_to": "Integration Request",
- "link_type": "DocType",
+ "link_to": "HSN-wise-summary of outward supplies",
+ "link_type": "Report",
"onboard": 0,
"type": "Link"
},
{
"hidden": 0,
"is_query_report": 0,
- "label": "New Reports",
- "link_count": 1,
- "link_type": "DocType",
+ "label": "Audit Trail",
+ "link_count": 0,
+ "link_to": "Audit Trail",
+ "link_type": "Report",
"onboard": 0,
- "type": "Card Break"
+ "type": "Link"
},
{
"hidden": 0,
- "is_query_report": 1,
- "label": "GST Sales Register Beta",
+ "is_query_report": 0,
+ "label": "GST Advance Detail",
"link_count": 0,
- "link_to": "GST Sales Register Beta",
+ "link_to": "GST Advance Detail",
"link_type": "Report",
"onboard": 0,
"type": "Link"
}
],
- "modified": "2024-03-26 16:43:35.237283",
+ "modified": "2024-06-08 16:54:55.015726",
"modified_by": "Administrator",
"module": "GST India",
"name": "GST India",
@@ -229,9 +262,9 @@
{
"color": "Grey",
"doc_view": "List",
- "label": "GSTR-1",
- "link_to": "GSTR-1",
- "type": "Report"
+ "label": "GSTR-1 Beta",
+ "link_to": "GSTR-1 Beta",
+ "type": "DocType"
},
{
"color": "Grey",
diff --git a/india_compliance/hooks.py b/india_compliance/hooks.py
index cde17bc6ae..c9e6cc3695 100644
--- a/india_compliance/hooks.py
+++ b/india_compliance/hooks.py
@@ -129,6 +129,7 @@
"validate": "india_compliance.gst_india.overrides.payment_entry.validate",
"on_submit": "india_compliance.gst_india.overrides.payment_entry.on_submit",
"on_update_after_submit": "india_compliance.gst_india.overrides.payment_entry.on_update_after_submit",
+ "before_cancel": "india_compliance.gst_india.overrides.payment_entry.before_cancel",
},
"Purchase Invoice": {
"onload": [
@@ -396,7 +397,7 @@
"cron": {
"*/5 * * * *": [
"india_compliance.gst_india.utils.e_invoice.retry_e_invoice_e_waybill_generation",
- "india_compliance.gst_india.utils.gstr.download_queued_request",
+ "india_compliance.gst_india.utils.gstr_utils.download_queued_request",
"india_compliance.gst_india.doctype.purchase_reconciliation_tool.purchase_reconciliation_tool.auto_refresh_authtoken",
],
"0 2 * * *": [
diff --git a/india_compliance/patches.txt b/india_compliance/patches.txt
index f49f742101..db3cd122e3 100644
--- a/india_compliance/patches.txt
+++ b/india_compliance/patches.txt
@@ -46,6 +46,7 @@ execute:from erpnext.stock.doctype.repost_item_valuation.repost_item_valuation i
india_compliance.patches.v14.set_item_details_from_purchase_invoice_to_bill_of_entry
india_compliance.patches.v14.update_item_gst_details_and_gst_trearment_in_bill_of_entry
india_compliance.patches.v14.update_default_auto_reconciliation_settings
+india_compliance.patches.v14.update_default_gstr1_settings
india_compliance.patches.v14.add_match_found_in_purchase_reconciliation_status
india_compliance.patches.v14.unset_inward_supply_link_for_cancelled_purchase
india_compliance.patches.v14.delete_not_generated_gstr_import_log
diff --git a/india_compliance/patches/post_install/improve_item_tax_template.py b/india_compliance/patches/post_install/improve_item_tax_template.py
index 36baa6ca2b..bf3dcd1412 100644
--- a/india_compliance/patches/post_install/improve_item_tax_template.py
+++ b/india_compliance/patches/post_install/improve_item_tax_template.py
@@ -275,38 +275,48 @@ def update_gst_treatment_for_transactions():
table = frappe.qb.DocType(item_doctype)
query = frappe.qb.update(table)
+ doctype = item_doctype.replace(" Item", "")
- (
- query.set(
- table.gst_treatment,
- Case()
- .when(table.is_nil_exempt == 1, "Nil-Rated")
- .when(table.is_non_gst == 1, "Non-GST")
- .else_("Taxable"),
- )
- .where(IfNull(table.gst_treatment, "") == "")
- .run()
+ update_gst_treatment_for_nil_exempt_and_non_gst(table, query, item_doctype)
+ update_gst_treatment_for_zero_rated(table, query, doctype)
+
+
+def update_gst_treatment_for_nil_exempt_and_non_gst(table, query, item_doctype):
+ if not frappe.db.has_column(item_doctype, "is_nil_exempt"):
+ return
+
+ (
+ query.set(
+ table.gst_treatment,
+ Case()
+ .when(table.is_nil_exempt == 1, "Nil-Rated")
+ .when(table.is_non_gst == 1, "Non-GST")
+ .else_("Taxable"),
)
+ .where(IfNull(table.gst_treatment, "") == "")
+ .run()
+ )
- doctype = item_doctype.replace(" Item", "")
- if doctype not in SALES_DOCTYPES:
- continue
- doc = frappe.qb.DocType(doctype)
-
- (
- query.join(doc)
- .on(doc.name == table.parent)
- .set(table.gst_treatment, "Zero-Rated")
- .where(
- (doc.gst_category == "SEZ")
- | (
- (doc.gst_category == "Overseas")
- & (doc.place_of_supply == "96-Other Countries")
- )
+def update_gst_treatment_for_zero_rated(table, query, doctype):
+ if doctype not in SALES_DOCTYPES:
+ return
+
+ doc = frappe.qb.DocType(doctype)
+
+ (
+ query.join(doc)
+ .on(doc.name == table.parent)
+ .set(table.gst_treatment, "Zero-Rated")
+ .where(
+ (doc.gst_category == "SEZ")
+ | (
+ (doc.gst_category == "Overseas")
+ & (doc.place_of_supply == "96-Other Countries")
)
- .run()
)
+ .run()
+ )
def update_gst_details_for_transactions(companies):
diff --git a/india_compliance/patches/v14/update_default_gstr1_settings.py b/india_compliance/patches/v14/update_default_gstr1_settings.py
new file mode 100644
index 0000000000..e8f4ef9865
--- /dev/null
+++ b/india_compliance/patches/v14/update_default_gstr1_settings.py
@@ -0,0 +1,12 @@
+import frappe
+
+
+def execute():
+ frappe.db.set_single_value(
+ "GST Settings",
+ {
+ "compare_gstr_1_data": 1,
+ "freeze_transactions": 1,
+ "filing_frequency": "Monthly",
+ },
+ )
diff --git a/india_compliance/public/js/purchase_reconciliation_tool/data_table_manager.js b/india_compliance/public/js/components/data_table_manager.js
similarity index 93%
rename from india_compliance/public/js/purchase_reconciliation_tool/data_table_manager.js
rename to india_compliance/public/js/components/data_table_manager.js
index a269009453..bcd375e13a 100644
--- a/india_compliance/public/js/purchase_reconciliation_tool/data_table_manager.js
+++ b/india_compliance/public/js/components/data_table_manager.js
@@ -78,12 +78,7 @@ india_compliance.DataTableManager = class DataTableManager {
value = column._value(value, column, data);
}
- return frappe.form.get_formatter(column.docfield.fieldtype)(
- value,
- column.docfield,
- { always_show_decimals: true },
- data
- );
+ return frappe.format(value, column, { always_show_decimals: true }, data);
};
return {
@@ -125,7 +120,7 @@ india_compliance.DataTableManager = class DataTableManager {
dynamicRowHeight: true,
checkboxColumn: true,
inlineFilters: true,
- noDataMessage: "No Matching Data Found!",
+ noDataMessage: __("No Matching Data Found!"),
// clusterize: false,
events: {
onCheckRow: () => {
diff --git a/india_compliance/public/js/purchase_reconciliation_tool/filter_group.js b/india_compliance/public/js/components/filter_group.js
similarity index 100%
rename from india_compliance/public/js/purchase_reconciliation_tool/filter_group.js
rename to india_compliance/public/js/components/filter_group.js
diff --git a/india_compliance/public/js/purchase_reconciliation_tool/number_card.js b/india_compliance/public/js/components/number_card.js
similarity index 100%
rename from india_compliance/public/js/purchase_reconciliation_tool/number_card.js
rename to india_compliance/public/js/components/number_card.js
diff --git a/india_compliance/public/js/components/set_gstin_options.js b/india_compliance/public/js/components/set_gstin_options.js
new file mode 100644
index 0000000000..e2307ebda9
--- /dev/null
+++ b/india_compliance/public/js/components/set_gstin_options.js
@@ -0,0 +1,14 @@
+frappe.provide("india_compliance");
+
+india_compliance.set_gstin_options = async function (frm) {
+ const { query, params } = india_compliance.get_gstin_query(frm.doc.company);
+ const { message } = await frappe.call({
+ method: query,
+ args: params,
+ });
+
+ if (!message) return [];
+ const gstin_field = frm.get_field("company_gstin");
+ gstin_field.set_data(message);
+ return message;
+}
diff --git a/india_compliance/public/js/components/view_group.js b/india_compliance/public/js/components/view_group.js
new file mode 100644
index 0000000000..1a33d7173b
--- /dev/null
+++ b/india_compliance/public/js/components/view_group.js
@@ -0,0 +1,81 @@
+frappe.provide("india_compliance");
+
+india_compliance.ViewGroup = class ViewGroup {
+ constructor(options) {
+ Object.assign(this, options);
+ this.views = {};
+ this.render();
+ }
+
+ render() {
+ $(this.$wrapper).append(
+ `
+
+ `
+ );
+
+ this.view_group_container = $(`
+
+ `).appendTo(this.$wrapper.find(`.view-switch`));
+
+ this.make_views();
+ this.setup_events();
+ }
+
+ set_active_view(view) {
+ this.active_view = view;
+ this.views[`${view}_view`].children().tab("show");
+ }
+
+ make_views() {
+ this.view_names.forEach(view => {
+ this.views[`${view}_view`] = $(
+ `
+
+
+ ${frappe.unscrub(view)}
+
+
+ `
+ ).appendTo(this.view_group_container);
+ });
+ }
+
+ setup_events() {
+ this.view_group_container.off("click").on("click", ".nav-link", e => {
+ e.preventDefault();
+ e.stopImmediatePropagation();
+
+ this.target = $(e.currentTarget);
+ const target_view = this.target.attr("data-fieldname");
+
+ this.set_active_view(target_view);
+ this.callback && this.callback(target_view);
+ });
+ }
+
+ disable_view(view, title) {
+ this.views[`${view}_view`].attr("title", title);
+ this.views[`${view}_view`].find(".nav-link").addClass("disabled");
+ }
+
+ enable_view(view) {
+ this.views[`${view}_view`].removeAttr("title");
+ this.views[`${view}_view`].find(".nav-link").removeClass("disabled");
+ }
+}
\ No newline at end of file
diff --git a/india_compliance/public/js/gstr1.bundle.js b/india_compliance/public/js/gstr1.bundle.js
new file mode 100644
index 0000000000..759400355b
--- /dev/null
+++ b/india_compliance/public/js/gstr1.bundle.js
@@ -0,0 +1,4 @@
+import "./components/filter_group";
+import "./components/data_table_manager";
+import "./components/set_gstin_options";
+import "./components/view_group";
\ No newline at end of file
diff --git a/india_compliance/public/js/purchase_reconciliation_tool.bundle.js b/india_compliance/public/js/purchase_reconciliation_tool.bundle.js
new file mode 100644
index 0000000000..e236f74221
--- /dev/null
+++ b/india_compliance/public/js/purchase_reconciliation_tool.bundle.js
@@ -0,0 +1,4 @@
+import "./components/data_table_manager";
+import "./components/filter_group";
+import "./components/number_card";
+import "./components/set_gstin_options";
diff --git a/india_compliance/public/js/purchase_reconciliation_tool/purchase_reconciliation_tool.bundle.js b/india_compliance/public/js/purchase_reconciliation_tool/purchase_reconciliation_tool.bundle.js
deleted file mode 100644
index 88785c2ba5..0000000000
--- a/india_compliance/public/js/purchase_reconciliation_tool/purchase_reconciliation_tool.bundle.js
+++ /dev/null
@@ -1,3 +0,0 @@
-import "./data_table_manager";
-import "./filter_group";
-import "./number_card";
diff --git a/india_compliance/public/js/utils.js b/india_compliance/public/js/utils.js
index f207172d81..7f1dc236a1 100644
--- a/india_compliance/public/js/utils.js
+++ b/india_compliance/public/js/utils.js
@@ -13,6 +13,40 @@ frappe.provide("india_compliance");
window.gst_settings = frappe.boot.gst_settings;
Object.assign(india_compliance, {
+ MONTH: [
+ "January",
+ "February",
+ "March",
+ "April",
+ "May",
+ "June",
+ "July",
+ "August",
+ "September",
+ "October",
+ "November",
+ "December",
+ ],
+
+ QUARTER: ["Jan-Mar", "Apr-Jun", "Jul-Sep", "Oct-Dec"],
+
+ get_month_year_from_period(period) {
+ /**
+ * Returns month or quarter and year from the period
+ * Month or quarter depends on the filing frequency set in GST Settings
+ *
+ * @param {String} period - period in format MMYYYY
+ * @returns {Array} - [month_or_quarter, year]
+ */
+
+ const { filing_frequency } = gst_settings;
+ const month_number = period.slice(0, 2);
+ const year = period.slice(2);
+
+ if (filing_frequency === "Monthly") return [this.MONTH[month_number - 1], year];
+ else return [this.QUARTER[Math.floor(month_number / 3)], year];
+ },
+
get_gstin_query(party, party_type = "Company") {
if (!party) {
frappe.show_alert({
@@ -148,7 +182,7 @@ Object.assign(india_compliance, {
validate_gstin(gstin) {
if (!gstin || gstin.length !== 15) {
- frappe.msgprint(__("GSTIN must be 15 characters long"))
+ frappe.msgprint(__("GSTIN must be 15 characters long"));
return;
}
@@ -162,8 +196,7 @@ Object.assign(india_compliance, {
},
get_gstin_otp(error_type, company_gstin) {
- let description =
- `An OTP has been sent to the registered mobile/email for GSTIN ${company_gstin} for further authentication. Please provide OTP.`;
+ let description = `An OTP has been sent to the registered mobile/email for GSTIN ${company_gstin} for further authentication. Please provide OTP.`;
if (error_type === "invalid_otp")
description = `Invalid OTP was provided for GSTIN ${company_gstin}. Please try again.`;
@@ -187,7 +220,7 @@ Object.assign(india_compliance, {
secondary_action_label: __("Resend OTP"),
secondary_action() {
frappe.call({
- method: "india_compliance.gst_india.utils.gstr.gstr.request_otp",
+ method: "india_compliance.gst_india.utils.gstr_utils.request_otp",
args: { company_gstin },
callback: function () {
frappe.show_alert({
@@ -331,25 +364,30 @@ Object.assign(india_compliance, {
async authenticate_company_gstins(company, company_gstin) {
const { message: gstin_authentication_status } = await frappe.call({
- method: "india_compliance.gst_india.utils.gstr.gstr.validate_company_gstins",
+ method: "india_compliance.gst_india.utils.gstr_utils.validate_company_gstins",
args: { company: company, company_gstin: company_gstin },
});
for (let gstin of Object.keys(gstin_authentication_status)) {
if (gstin_authentication_status[gstin]) continue;
- gstin_authentication_status[gstin] = await this.authenticate_otp(gstin);
+ gstin_authentication_status[gstin] =
+ await this.request_and_authenticate_otp(gstin);
}
return Object.keys(gstin_authentication_status);
},
- async authenticate_otp(gstin) {
+ async request_and_authenticate_otp(gstin) {
await frappe.call({
- method: "india_compliance.gst_india.utils.gstr.gstr.request_otp",
+ method: "india_compliance.gst_india.utils.gstr_utils.request_otp",
args: { company_gstin: gstin },
});
+ this.authenticate_otp(gstin);
+ },
+
+ async authenticate_otp(gstin) {
let error_type = "otp_requested";
let is_authenticated = false;
@@ -357,11 +395,14 @@ Object.assign(india_compliance, {
const otp = await this.get_gstin_otp(error_type, gstin);
const { message } = await frappe.call({
- method: "india_compliance.gst_india.utils.gstr.gstr.authenticate_otp",
+ method: "india_compliance.gst_india.utils.gstr_utils.authenticate_otp",
args: { company_gstin: gstin, otp: otp },
});
- if (message && ["otp_requested", "invalid_otp"].includes(message.error_type)) {
+ if (
+ message &&
+ ["otp_requested", "invalid_otp"].includes(message.error_type)
+ ) {
error_type = message.error_type;
continue;
}
@@ -369,7 +410,35 @@ Object.assign(india_compliance, {
is_authenticated = true;
return true;
}
- }
+ },
+
+ show_dismissable_alert(wrapper, message, alert_type = "primary", on_close = null) {
+ const alert = $(`
+
+ `).prependTo(wrapper);
+
+ alert.on("closed.bs.alert", () => {
+ if (on_close) on_close();
+ });
+
+ return alert;
+ },
});
function is_gstin_check_digit_valid(gstin) {
diff --git a/india_compliance/tests/__init__.py b/india_compliance/tests/__init__.py
index 99d601b854..f0fa8f60f1 100644
--- a/india_compliance/tests/__init__.py
+++ b/india_compliance/tests/__init__.py
@@ -62,7 +62,7 @@ def create_test_records():
)
for doctype, data in test_records.items():
- make_test_objects(doctype, data, reset=True)
+ make_test_objects(doctype, data)
if doctype == "Company":
add_companies_to_fiscal_year(data)