257 lines
8.0 KiB
Python
257 lines
8.0 KiB
Python
import base64
|
|
import hashlib
|
|
import hmac
|
|
import json
|
|
|
|
import frappe
|
|
from frappe import _
|
|
from frappe.utils import cstr
|
|
|
|
|
|
def verify_request():
|
|
woocommerce_settings = frappe.get_doc("Woocommerce Settings")
|
|
sig = base64.b64encode(
|
|
hmac.new(
|
|
woocommerce_settings.secret.encode("utf8"), frappe.request.data, hashlib.sha256
|
|
).digest()
|
|
)
|
|
|
|
if (
|
|
frappe.request.data
|
|
and not sig == frappe.get_request_header("X-Wc-Webhook-Signature", "").encode()
|
|
):
|
|
frappe.throw(_("Unverified Webhook Data"))
|
|
frappe.set_user(woocommerce_settings.creation_user)
|
|
|
|
|
|
@frappe.whitelist(allow_guest=True)
|
|
def order(*args, **kwargs):
|
|
try:
|
|
_order(*args, **kwargs)
|
|
except Exception:
|
|
error_message = (
|
|
frappe.get_traceback() + "\n\n Request Data: \n" + json.loads(frappe.request.data).__str__()
|
|
)
|
|
frappe.log_error("WooCommerce Error", error_message)
|
|
raise
|
|
|
|
|
|
def _order(*args, **kwargs):
|
|
woocommerce_settings = frappe.get_doc("Woocommerce Settings")
|
|
if frappe.flags.woocomm_test_order_data:
|
|
order = frappe.flags.woocomm_test_order_data
|
|
event = "created"
|
|
# Ignore the test ping issued during WooCommerce webhook configuration
|
|
# Ref: https://github.com/woocommerce/woocommerce/issues/15642
|
|
if frappe.request.data.decode("utf-8").startswith("webhook_id="):
|
|
return "success"
|
|
elif frappe.request and frappe.request.data:
|
|
verify_request()
|
|
try:
|
|
order = json.loads(frappe.request.data)
|
|
except ValueError:
|
|
# woocommerce returns 'webhook_id=value' for the first request which is not JSON
|
|
order = frappe.request.data
|
|
event = frappe.get_request_header("X-Wc-Webhook-Event")
|
|
|
|
else:
|
|
return "success"
|
|
|
|
if event == "created":
|
|
sys_lang = frappe.get_single("System Settings").language or "en"
|
|
raw_billing_data = order.get("billing")
|
|
raw_shipping_data = order.get("shipping")
|
|
customer_name = raw_billing_data.get("first_name") + " " + raw_billing_data.get("last_name")
|
|
link_customer_and_address(raw_billing_data, raw_shipping_data, customer_name)
|
|
link_items(order.get("line_items"), woocommerce_settings, sys_lang)
|
|
create_sales_order(order, woocommerce_settings, customer_name, sys_lang)
|
|
|
|
|
|
def link_customer_and_address(raw_billing_data, raw_shipping_data, customer_name):
|
|
customer_woo_com_email = raw_billing_data.get("email")
|
|
customer_exists = frappe.get_value("Customer", {"woocommerce_email": customer_woo_com_email})
|
|
if not customer_exists:
|
|
# Create Customer
|
|
customer = frappe.new_doc("Customer")
|
|
else:
|
|
# Edit Customer
|
|
customer = frappe.get_doc("Customer", {"woocommerce_email": customer_woo_com_email})
|
|
old_name = customer.customer_name
|
|
|
|
customer.customer_name = customer_name
|
|
customer.woocommerce_email = customer_woo_com_email
|
|
customer.flags.ignore_mandatory = True
|
|
customer.save()
|
|
|
|
if customer_exists:
|
|
# Fixes https://github.com/frappe/erpnext/issues/33708
|
|
if old_name != customer_name:
|
|
frappe.rename_doc("Customer", old_name, customer_name)
|
|
for address_type in (
|
|
"Billing",
|
|
"Shipping",
|
|
):
|
|
try:
|
|
address = frappe.get_doc(
|
|
"Address", {"woocommerce_email": customer_woo_com_email, "address_type": address_type}
|
|
)
|
|
rename_address(address, customer)
|
|
except (
|
|
frappe.DoesNotExistError,
|
|
frappe.DuplicateEntryError,
|
|
frappe.ValidationError,
|
|
):
|
|
pass
|
|
else:
|
|
create_address(raw_billing_data, customer, "Billing")
|
|
create_address(raw_shipping_data, customer, "Shipping")
|
|
create_contact(raw_billing_data, customer)
|
|
|
|
|
|
def create_contact(data, customer):
|
|
email = data.get("email", None)
|
|
phone = data.get("phone", None)
|
|
|
|
if not email and not phone:
|
|
return
|
|
|
|
contact = frappe.new_doc("Contact")
|
|
contact.first_name = data.get("first_name")
|
|
contact.last_name = data.get("last_name")
|
|
contact.is_primary_contact = 1
|
|
contact.is_billing_contact = 1
|
|
|
|
if phone:
|
|
contact.add_phone(phone, is_primary_mobile_no=1, is_primary_phone=1)
|
|
|
|
if email:
|
|
contact.add_email(email, is_primary=1)
|
|
|
|
contact.append("links", {"link_doctype": "Customer", "link_name": customer.name})
|
|
|
|
contact.flags.ignore_mandatory = True
|
|
contact.save()
|
|
|
|
|
|
def create_address(raw_data, customer, address_type):
|
|
address = frappe.new_doc("Address")
|
|
|
|
address.address_line1 = raw_data.get("address_1", "Not Provided")
|
|
address.address_line2 = raw_data.get("address_2", "Not Provided")
|
|
address.city = raw_data.get("city", "Not Provided")
|
|
address.woocommerce_email = customer.woocommerce_email
|
|
address.address_type = address_type
|
|
address.country = frappe.get_value("Country", {"code": raw_data.get("country", "IN").lower()})
|
|
address.state = raw_data.get("state")
|
|
address.pincode = raw_data.get("postcode")
|
|
address.phone = raw_data.get("phone")
|
|
address.email_id = customer.woocommerce_email
|
|
address.append("links", {"link_doctype": "Customer", "link_name": customer.name})
|
|
|
|
address.flags.ignore_mandatory = True
|
|
address.save()
|
|
|
|
|
|
def rename_address(address, customer):
|
|
old_address_title = address.name
|
|
new_address_title = customer.name + "-" + address.address_type
|
|
address.address_title = customer.customer_name
|
|
address.save()
|
|
|
|
frappe.rename_doc("Address", old_address_title, new_address_title)
|
|
|
|
|
|
def link_items(items_list, woocommerce_settings, sys_lang):
|
|
for item_data in items_list:
|
|
item_woo_com_id = cstr(item_data.get("product_id"))
|
|
|
|
if not frappe.db.get_value("Item", {"woocommerce_id": item_woo_com_id}, "name"):
|
|
# Create Item
|
|
item = frappe.new_doc("Item")
|
|
item.item_code = _("woocommerce - {0}", sys_lang).format(item_woo_com_id)
|
|
item.stock_uom = woocommerce_settings.uom or _("Nos", sys_lang)
|
|
item.item_group = _("WooCommerce Products", sys_lang)
|
|
|
|
item.item_name = item_data.get("name")
|
|
item.woocommerce_id = item_woo_com_id
|
|
item.flags.ignore_mandatory = True
|
|
item.save()
|
|
|
|
|
|
def create_sales_order(order, woocommerce_settings, customer_name, sys_lang):
|
|
new_sales_order = frappe.new_doc("Sales Order")
|
|
new_sales_order.customer = customer_name
|
|
|
|
new_sales_order.po_no = new_sales_order.woocommerce_id = order.get("id")
|
|
new_sales_order.naming_series = woocommerce_settings.sales_order_series or "SO-WOO-"
|
|
|
|
created_date = order.get("date_created").split("T")
|
|
new_sales_order.transaction_date = created_date[0]
|
|
delivery_after = woocommerce_settings.delivery_after_days or 7
|
|
new_sales_order.delivery_date = frappe.utils.add_days(created_date[0], delivery_after)
|
|
|
|
new_sales_order.company = woocommerce_settings.company
|
|
|
|
set_items_in_sales_order(new_sales_order, woocommerce_settings, order, sys_lang)
|
|
new_sales_order.flags.ignore_mandatory = True
|
|
new_sales_order.insert()
|
|
new_sales_order.submit()
|
|
|
|
frappe.db.commit()
|
|
|
|
|
|
def set_items_in_sales_order(new_sales_order, woocommerce_settings, order, sys_lang):
|
|
company_abbr = frappe.db.get_value("Company", woocommerce_settings.company, "abbr")
|
|
|
|
default_warehouse = _("Stores - {0}", sys_lang).format(company_abbr)
|
|
if not frappe.db.exists("Warehouse", default_warehouse) and not woocommerce_settings.warehouse:
|
|
frappe.throw(_("Please set Warehouse in Woocommerce Settings"))
|
|
|
|
for item in order.get("line_items"):
|
|
woocomm_item_id = item.get("product_id")
|
|
found_item = frappe.get_doc("Item", {"woocommerce_id": cstr(woocomm_item_id)})
|
|
|
|
ordered_items_tax = item.get("total_tax")
|
|
|
|
new_sales_order.append(
|
|
"items",
|
|
{
|
|
"item_code": found_item.name,
|
|
"item_name": found_item.item_name,
|
|
"description": found_item.item_name,
|
|
"delivery_date": new_sales_order.delivery_date,
|
|
"uom": woocommerce_settings.uom or _("Nos", sys_lang),
|
|
"qty": item.get("quantity"),
|
|
"rate": item.get("price"),
|
|
"warehouse": woocommerce_settings.warehouse or default_warehouse,
|
|
},
|
|
)
|
|
|
|
add_tax_details(
|
|
new_sales_order, ordered_items_tax, "Ordered Item tax", woocommerce_settings.tax_account
|
|
)
|
|
|
|
# shipping_details = order.get("shipping_lines") # used for detailed order
|
|
|
|
add_tax_details(
|
|
new_sales_order, order.get("shipping_tax"), "Shipping Tax", woocommerce_settings.f_n_f_account
|
|
)
|
|
add_tax_details(
|
|
new_sales_order,
|
|
order.get("shipping_total"),
|
|
"Shipping Total",
|
|
woocommerce_settings.f_n_f_account,
|
|
)
|
|
|
|
|
|
def add_tax_details(sales_order, price, desc, tax_account_head):
|
|
sales_order.append(
|
|
"taxes",
|
|
{
|
|
"charge_type": "Actual",
|
|
"account_head": tax_account_head,
|
|
"tax_amount": price,
|
|
"description": desc,
|
|
},
|
|
)
|