import traceback import frappe import taxjar from frappe import _ from frappe.contacts.doctype.address.address import get_company_address from frappe.utils import cint, flt from erpnext import get_default_company, get_region SUPPORTED_COUNTRY_CODES = [ "AT", "AU", "BE", "BG", "CA", "CY", "CZ", "DE", "DK", "EE", "ES", "FI", "FR", "GB", "GR", "HR", "HU", "IE", "IT", "LT", "LU", "LV", "MT", "NL", "PL", "PT", "RO", "SE", "SI", "SK", "US", ] SUPPORTED_STATE_CODES = [ "AL", "AK", "AZ", "AR", "CA", "CO", "CT", "DE", "DC", "FL", "GA", "HI", "ID", "IL", "IN", "IA", "KS", "KY", "LA", "ME", "MD", "MA", "MI", "MN", "MS", "MO", "MT", "NE", "NV", "NH", "NJ", "NM", "NY", "NC", "ND", "OH", "OK", "OR", "PA", "RI", "SC", "SD", "TN", "TX", "UT", "VT", "VA", "WA", "WV", "WI", "WY", ] def get_client(): taxjar_settings = frappe.get_single("TaxJar Settings") if not taxjar_settings.is_sandbox: api_key = taxjar_settings.api_key and taxjar_settings.get_password("api_key") api_url = taxjar.DEFAULT_API_URL else: api_key = taxjar_settings.sandbox_api_key and taxjar_settings.get_password("sandbox_api_key") api_url = taxjar.SANDBOX_API_URL if api_key and api_url: client = taxjar.Client(api_key=api_key, api_url=api_url) client.set_api_config("headers", {"x-api-version": "2022-01-24"}) return client def create_transaction(doc, method): TAXJAR_CREATE_TRANSACTIONS = frappe.db.get_single_value( "TaxJar Settings", "taxjar_create_transactions" ) """Create an order transaction in TaxJar""" if not TAXJAR_CREATE_TRANSACTIONS: return client = get_client() if not client: return TAX_ACCOUNT_HEAD = frappe.db.get_single_value("TaxJar Settings", "tax_account_head") sales_tax = sum([tax.tax_amount for tax in doc.taxes if tax.account_head == TAX_ACCOUNT_HEAD]) if not sales_tax: return tax_dict = get_tax_data(doc) if not tax_dict: return tax_dict["transaction_id"] = doc.name tax_dict["transaction_date"] = frappe.utils.today() tax_dict["sales_tax"] = sales_tax tax_dict["amount"] = doc.total + tax_dict["shipping"] try: if doc.is_return: client.create_refund(tax_dict) else: client.create_order(tax_dict) except taxjar.exceptions.TaxJarResponseError as err: frappe.throw(_(sanitize_error_response(err))) except Exception as ex: print(traceback.format_exc(ex)) def delete_transaction(doc, method): """Delete an existing TaxJar order transaction""" TAXJAR_CREATE_TRANSACTIONS = frappe.db.get_single_value( "TaxJar Settings", "taxjar_create_transactions" ) if not TAXJAR_CREATE_TRANSACTIONS: return client = get_client() if not client: return client.delete_order(doc.name) def get_tax_data(doc): SHIP_ACCOUNT_HEAD = frappe.db.get_single_value("TaxJar Settings", "shipping_account_head") from_address = get_company_address_details(doc) from_shipping_state = from_address.get("state") from_country_code = frappe.db.get_value("Country", from_address.country, "code") from_country_code = from_country_code.upper() to_address = get_shipping_address_details(doc) to_shipping_state = to_address.get("state") to_country_code = frappe.db.get_value("Country", to_address.country, "code") to_country_code = to_country_code.upper() shipping = sum([tax.tax_amount for tax in doc.taxes if tax.account_head == SHIP_ACCOUNT_HEAD]) line_items = [get_line_item_dict(item, doc.docstatus) for item in doc.items] if from_shipping_state not in SUPPORTED_STATE_CODES: from_shipping_state = get_state_code(from_address, "Company") if to_shipping_state not in SUPPORTED_STATE_CODES: to_shipping_state = get_state_code(to_address, "Shipping") tax_dict = { "from_country": from_country_code, "from_zip": from_address.pincode, "from_state": from_shipping_state, "from_city": from_address.city, "from_street": from_address.address_line1, "to_country": to_country_code, "to_zip": to_address.pincode, "to_city": to_address.city, "to_street": to_address.address_line1, "to_state": to_shipping_state, "shipping": shipping, "amount": doc.net_total, "plugin": "erpnext", "line_items": line_items, } return tax_dict def get_state_code(address, location): if address is not None: state_code = get_iso_3166_2_state_code(address) if state_code not in SUPPORTED_STATE_CODES: frappe.throw(_("Please enter a valid State in the {0} Address").format(location)) else: frappe.throw(_("Please enter a valid State in the {0} Address").format(location)) return state_code def get_line_item_dict(item, docstatus): tax_dict = dict( id=item.get("idx"), quantity=item.get("qty"), unit_price=item.get("rate"), product_tax_code=item.get("product_tax_category"), ) if docstatus == 1: tax_dict.update({"sales_tax": item.get("tax_collectable")}) return tax_dict def set_sales_tax(doc, method): TAX_ACCOUNT_HEAD = frappe.db.get_single_value("TaxJar Settings", "tax_account_head") TAXJAR_CALCULATE_TAX = frappe.db.get_single_value("TaxJar Settings", "taxjar_calculate_tax") if not TAXJAR_CALCULATE_TAX: return if get_region(doc.company) != "United States": return if not doc.items: return if check_sales_tax_exemption(doc): return tax_dict = get_tax_data(doc) if not tax_dict: # Remove existing tax rows if address is changed from a taxable state/country setattr(doc, "taxes", [tax for tax in doc.taxes if tax.account_head != TAX_ACCOUNT_HEAD]) return # check if delivering within a nexus check_for_nexus(doc, tax_dict) tax_data = validate_tax_request(tax_dict) if tax_data is not None: if not tax_data.amount_to_collect: setattr(doc, "taxes", [tax for tax in doc.taxes if tax.account_head != TAX_ACCOUNT_HEAD]) elif tax_data.amount_to_collect > 0: # Loop through tax rows for existing Sales Tax entry # If none are found, add a row with the tax amount for tax in doc.taxes: if tax.account_head == TAX_ACCOUNT_HEAD: tax.tax_amount = tax_data.amount_to_collect doc.run_method("calculate_taxes_and_totals") break else: doc.append( "taxes", { "charge_type": "Actual", "description": "Sales Tax", "account_head": TAX_ACCOUNT_HEAD, "tax_amount": tax_data.amount_to_collect, }, ) # Assigning values to tax_collectable and taxable_amount fields in sales item table for item in tax_data.breakdown.line_items: doc.get("items")[cint(item.id) - 1].tax_collectable = item.tax_collectable doc.get("items")[cint(item.id) - 1].taxable_amount = item.taxable_amount doc.run_method("calculate_taxes_and_totals") def check_for_nexus(doc, tax_dict): TAX_ACCOUNT_HEAD = frappe.db.get_single_value("TaxJar Settings", "tax_account_head") if not frappe.db.get_value("TaxJar Nexus", {"region_code": tax_dict["to_state"]}): for item in doc.get("items"): item.tax_collectable = flt(0) item.taxable_amount = flt(0) for tax in list(doc.taxes): if tax.account_head == TAX_ACCOUNT_HEAD: doc.taxes.remove(tax) return def check_sales_tax_exemption(doc): # if the party is exempt from sales tax, then set all tax account heads to zero TAX_ACCOUNT_HEAD = frappe.db.get_single_value("TaxJar Settings", "tax_account_head") sales_tax_exempted = ( hasattr(doc, "exempt_from_sales_tax") and doc.exempt_from_sales_tax or frappe.db.has_column("Customer", "exempt_from_sales_tax") and frappe.db.get_value("Customer", doc.customer, "exempt_from_sales_tax") ) if sales_tax_exempted: for tax in doc.taxes: if tax.account_head == TAX_ACCOUNT_HEAD: tax.tax_amount = 0 break doc.run_method("calculate_taxes_and_totals") return True else: return False def validate_tax_request(tax_dict): """Return the sales tax that should be collected for a given order.""" client = get_client() if not client: return try: tax_data = client.tax_for_order(tax_dict) except taxjar.exceptions.TaxJarResponseError as err: frappe.throw(_(sanitize_error_response(err))) else: return tax_data def get_company_address_details(doc): """Return default company address details""" company_address = get_company_address(get_default_company()).company_address if not company_address: frappe.throw(_("Please set a default company address")) company_address = frappe.get_doc("Address", company_address) return company_address def get_shipping_address_details(doc): """Return customer shipping address details""" if doc.shipping_address_name: shipping_address = frappe.get_doc("Address", doc.shipping_address_name) elif doc.customer_address: shipping_address = frappe.get_doc("Address", doc.customer_address) else: shipping_address = get_company_address_details(doc) return shipping_address def get_iso_3166_2_state_code(address): import pycountry country_code = frappe.db.get_value("Country", address.get("country"), "code") error_message = _( """{0} is not a valid state! Check for typos or enter the ISO code for your state.""" ).format(address.get("state")) state = address.get("state").upper().strip() # The max length for ISO state codes is 3, excluding the country code if len(state) <= 3: # PyCountry returns state code as {country_code}-{state-code} (e.g. US-FL) address_state = (country_code + "-" + state).upper() states = pycountry.subdivisions.get(country_code=country_code.upper()) states = [pystate.code for pystate in states] if address_state in states: return state frappe.throw(_(error_message)) else: try: lookup_state = pycountry.subdivisions.lookup(state) except LookupError: frappe.throw(_(error_message)) else: return lookup_state.code.split("-")[1] def sanitize_error_response(response): response = response.full_response.get("detail") response = response.replace("_", " ") sanitized_responses = { "to zip": "Zipcode", "to city": "City", "to state": "State", "to country": "Country", } for k, v in sanitized_responses.items(): response = response.replace(k, v) return response