feat: Customer Records page + CRM data import (7,966 records)

This commit is contained in:
Westech Admin
2026-05-20 22:14:10 +00:00
parent 0128d1aae7
commit 6a5f2a3ebb
13 changed files with 455 additions and 0 deletions
Binary file not shown.
+129
View File
@@ -0,0 +1,129 @@
import frappe
import json
def run():
print("=== CRM Import Status Check ===")
print("Customer Record DocType exists:", bool(frappe.db.exists("DocType", "Customer Record")))
if not frappe.db.exists("DocType", "Customer Record"):
print("Customer Record DocType does not exist! Creating it...")
doctype_def = {
"name": "Customer Record",
"doctype": "DocType",
"module": "Westech R2",
"custom": 0,
"autoname": "field:record_number",
"naming_rule": "By fieldname",
"engine": "InnoDB",
"document_type": "Document",
"fields": [
{"fieldname": "record_number", "fieldtype": "Data", "label": "Record Number", "reqd": 1, "unique": 1, "in_list_view": 1},
{"fieldname": "company_name", "fieldtype": "Data", "label": "Company Name", "in_list_view": 1},
{"fieldname": "supplier", "fieldtype": "Link", "label": "Supplier", "options": "Supplier", "in_list_view": 1},
{"fieldname": "customer_number", "fieldtype": "Data", "label": "Customer Number", "in_list_view": 1},
{"fieldname": "contact_persons", "fieldtype": "Text", "label": "Contact Persons"},
{"fieldname": "email_address", "fieldtype": "Text", "label": "Email Addresses"},
{"fieldname": "phone_numbers", "fieldtype": "Text", "label": "Phone Numbers"},
{"fieldname": "customer_address", "fieldtype": "Data", "label": "Address"},
{"fieldname": "city", "fieldtype": "Data", "label": "City"},
{"fieldname": "state", "fieldtype": "Data", "label": "State"},
{"fieldname": "zip", "fieldtype": "Data", "label": "Zip"},
{"fieldname": "date_created", "fieldtype": "Date", "label": "Date Created"},
{"fieldname": "contacted_date", "fieldtype": "Date", "label": "Contacted Date"},
{"fieldname": "follow_up_date", "fieldtype": "Date", "label": "Follow Up Date"},
{"fieldname": "last_pu_date", "fieldtype": "Date", "label": "Last PU Date"},
{"fieldname": "notes", "fieldtype": "Text", "label": "Notes"},
{"fieldname": "comments", "fieldtype": "Text", "label": "Comments"},
{"fieldname": "hours_operation", "fieldtype": "Data", "label": "Hours of Operation"},
],
"permissions": [
{"role": "System Manager", "read": 1, "write": 1, "create": 1, "delete": 1, "export": 1, "print": 1, "share": 1, "email": 1},
{"role": "All", "read": 1, "write": 1, "create": 1, "delete": 0, "export": 0, "print": 1, "share": 0, "email": 0}
],
"quick_entry": 0,
"track_changes": 1,
}
doc = frappe.get_doc(doctype_def)
doc.insert(ignore_permissions=True)
frappe.db.commit()
print("Customer Record DocType created!")
# Check existing count
existing = frappe.db.count("Customer Record")
print(f"Records in Customer Record: {existing}")
if existing > 0:
print("Data already imported. Skipping.")
return {"status": "already_done", "count": existing}
# Load parsed data
with open("/tmp/crm_records.json") as f:
records = json.load(f)
print(f"Parsed records from xlsx: {len(records)}")
# Get supplier names
supplier_names = set(frappe.get_all("Supplier", fields=["name"], pluck="name"))
print(f"Total suppliers: {len(supplier_names)}")
# Count matches
matched = sum(1 for r in records if r["record_number"] in supplier_names)
print(f"Records matching suppliers: {matched}")
print(f"Importing {len(records)} records...")
created = 0
errors = []
for rec in records:
try:
doc = frappe.new_doc("Customer Record")
doc.record_number = rec["record_number"]
doc.company_name = rec.get("company_name") or ""
# Match supplier
if rec["record_number"] in supplier_names:
doc.supplier = rec["record_number"]
doc.customer_number = rec["record_number"]
doc.contact_persons = "\n".join(rec.get("contact_persons", []))
doc.email_address = "\n".join(rec.get("emails", []))
doc.phone_numbers = "\n".join(rec.get("phone_numbers", []))
doc.customer_address = rec.get("address") or ""
doc.city = rec.get("city") or ""
doc.state = rec.get("state") or ""
doc.zip = str(rec.get("zip") or "")
if rec.get("date_created"):
doc.date_created = rec["date_created"]
if rec.get("contacted_date"):
doc.contacted_date = rec["contacted_date"]
if rec.get("follow_up_date"):
doc.follow_up_date = rec["follow_up_date"]
if rec.get("last_pu_date"):
doc.last_pu_date = rec["last_pu_date"]
doc.notes = rec.get("notes") or ""
doc.comments = rec.get("comments") or ""
doc.hours_operation = rec.get("hours_operation") or ""
doc.save(ignore_permissions=True)
created += 1
if created % 500 == 0:
print(f" ... {created} records imported")
frappe.db.commit()
except Exception as e:
errors.append(f"{rec['record_number']}: {str(e)}")
if len(errors) > 10:
print(f"Too many errors, stopping. Last error: {e}")
break
frappe.db.commit()
print(f"\nImport complete: {created} records created")
if errors:
print(f"Errors ({len(errors)}):")
for e in errors[:10]:
print(f" {e}")
return {"status": "ok", "created": created, "errors": len(errors), "matched": matched}
+142
View File
@@ -0,0 +1,142 @@
import frappe
import json
from datetime import datetime
def parse_date(val):
"""Parse various date formats from the CRM spreadsheet"""
if not val:
return None
if isinstance(val, datetime):
return val.date()
if isinstance(val, str):
val = val.strip()
if not val or val.lower() in ['nan', 'none', 'null']:
return None
# Handle format like "9/8/0206" -> year 2006
if '/' in val:
parts = val.split('/')
if len(parts) == 3:
month, day, year = parts
year = year.strip()
# Fix 4-digit years starting with 02xx -> 20xx
if len(year) == 4 and year.startswith('02'):
year = '20' + year[2:]
# Fix 2-digit years
elif len(year) == 2:
year_int = int(year)
if year_int < 50:
year = '20' + year
else:
year = '19' + year
try:
return datetime(int(year), int(month), int(day)).date()
except:
return None
return None
def run():
print("=== CRM Import Fix & Continue ===")
# Load parsed data
with open("/tmp/crm_records.json") as f:
records = json.load(f)
print(f"Total parsed records: {len(records)}")
# Get already imported records
existing = frappe.get_all("Customer Record", fields=["name"], pluck="name")
existing_set = set(existing)
print(f"Already imported: {len(existing_set)}")
# Get supplier names
supplier_names = set(frappe.get_all("Supplier", fields=["name"], pluck="name"))
# Fix existing bad records
print("\nFixing existing records with bad dates...")
fixed = 0
for rec_num in existing_set:
doc = frappe.get_doc("Customer Record", rec_num)
changed = False
# Check and fix dates
date_fields = ['date_created', 'contacted_date', 'follow_up_date', 'last_pu_date']
for df in date_fields:
val = doc.get(df)
if val and isinstance(val, str):
parsed = parse_date(val)
if parsed:
doc.set(df, parsed)
changed = True
else:
# Can't parse, clear it
doc.set(df, None)
changed = True
if changed:
doc.save(ignore_permissions=True)
fixed += 1
if fixed:
frappe.db.commit()
print(f"Fixed {fixed} existing records")
# Import remaining records
remaining = [r for r in records if r["record_number"] not in existing_set]
print(f"\nImporting remaining {len(remaining)} records...")
created = 0
errors = []
for rec in remaining:
try:
doc = frappe.new_doc("Customer Record")
doc.record_number = rec["record_number"]
doc.company_name = rec.get("company_name") or ""
# Match supplier
if rec["record_number"] in supplier_names:
doc.supplier = rec["record_number"]
doc.customer_number = rec["record_number"]
doc.contact_persons = "\n".join(rec.get("contact_persons", []))
doc.email_address = "\n".join(rec.get("emails", []))
doc.phone_numbers = "\n".join(rec.get("phone_numbers", []))
doc.customer_address = rec.get("address") or ""
doc.city = rec.get("city") or ""
doc.state = rec.get("state") or ""
doc.zip = str(rec.get("zip") or "")
# Parse dates properly
doc.date_created = parse_date(rec.get("date_created"))
doc.contacted_date = parse_date(rec.get("contacted_date"))
doc.follow_up_date = parse_date(rec.get("follow_up_date"))
doc.last_pu_date = parse_date(rec.get("last_pu_date"))
doc.notes = rec.get("notes") or ""
doc.comments = rec.get("comments") or ""
doc.hours_operation = rec.get("hours_operation") or ""
doc.save(ignore_permissions=True)
created += 1
if created % 500 == 0:
print(f" ... {created} remaining records imported")
frappe.db.commit()
except Exception as e:
errors.append(f"{rec['record_number']}: {str(e)}")
if len(errors) > 20:
print(f"Too many errors ({len(errors)}), stopping. Last: {e}")
break
frappe.db.commit()
print(f"\nImport complete: {created} additional records created")
if errors:
print(f"Errors ({len(errors)}):")
for e in errors[:10]:
print(f" {e}")
total = frappe.db.count("Customer Record")
print(f"Total Customer Records now: {total}")
return {"status": "ok", "created": created, "errors": len(errors), "total": total}
@@ -0,0 +1,8 @@
// Copyright (c) 2026, Westech and contributors
// For license information, please see license.txt
// frappe.ui.form.on("Customer Record", {
// refresh(frm) {
// },
// });
@@ -0,0 +1,158 @@
{
"actions": [],
"allow_rename": 1,
"autoname": "field:record_number",
"creation": "2026-05-20 15:10:58.374067",
"doctype": "DocType",
"document_type": "Document",
"editable_grid": 1,
"engine": "InnoDB",
"field_order": [
"record_number",
"company_name",
"supplier",
"customer_number",
"contact_persons",
"email_address",
"phone_numbers",
"customer_address",
"city",
"state",
"zip",
"date_created",
"contacted_date",
"follow_up_date",
"last_pu_date",
"notes",
"comments",
"hours_operation"
],
"fields": [
{
"fieldname": "record_number",
"fieldtype": "Data",
"in_list_view": 1,
"label": "Record Number",
"reqd": 1,
"unique": 1
},
{
"fieldname": "company_name",
"fieldtype": "Data",
"in_list_view": 1,
"label": "Company Name"
},
{
"fieldname": "supplier",
"fieldtype": "Link",
"in_list_view": 1,
"label": "Supplier",
"options": "Supplier"
},
{
"fieldname": "customer_number",
"fieldtype": "Data",
"in_list_view": 1,
"label": "Customer Number"
},
{
"fieldname": "contact_persons",
"fieldtype": "Text",
"label": "Contact Persons"
},
{
"fieldname": "email_address",
"fieldtype": "Text",
"label": "Email Addresses"
},
{
"fieldname": "phone_numbers",
"fieldtype": "Text",
"label": "Phone Numbers"
},
{
"fieldname": "customer_address",
"fieldtype": "Data",
"label": "Address"
},
{
"fieldname": "city",
"fieldtype": "Data",
"label": "City"
},
{
"fieldname": "state",
"fieldtype": "Data",
"label": "State"
},
{
"fieldname": "zip",
"fieldtype": "Data",
"label": "Zip"
},
{
"fieldname": "date_created",
"fieldtype": "Date",
"label": "Date Created"
},
{
"fieldname": "contacted_date",
"fieldtype": "Date",
"label": "Contacted Date"
},
{
"fieldname": "follow_up_date",
"fieldtype": "Date",
"label": "Follow Up Date"
},
{
"fieldname": "last_pu_date",
"fieldtype": "Date",
"label": "Last PU Date"
},
{
"fieldname": "notes",
"fieldtype": "Text",
"label": "Notes"
},
{
"fieldname": "comments",
"fieldtype": "Text",
"label": "Comments"
},
{
"fieldname": "hours_operation",
"fieldtype": "Data",
"label": "Hours of Operation"
}
],
"grid_page_length": 50,
"index_web_pages_for_search": 1,
"links": [],
"modified": "2026-05-20 15:10:58.374067",
"modified_by": "Administrator",
"module": "Westech R2",
"name": "Customer Record",
"naming_rule": "By fieldname",
"owner": "Administrator",
"permissions": [
{
"create": 1,
"delete": 1,
"email": 1,
"export": 1,
"print": 1,
"read": 1,
"report": 1,
"role": "System Manager",
"share": 1,
"write": 1
}
],
"row_format": "Dynamic",
"rows_threshold_for_grid_search": 20,
"sort_field": "modified",
"sort_order": "DESC",
"states": [],
"track_changes": 1
}
@@ -0,0 +1,9 @@
# Copyright (c) 2026, Westech and contributors
# For license information, please see license.txt
# import frappe
from frappe.model.document import Document
class CustomerRecord(Document):
pass
@@ -0,0 +1,9 @@
# Copyright (c) 2026, Westech and Contributors
# See license.txt
# import frappe
from frappe.tests.utils import FrappeTestCase
class TestCustomerRecord(FrappeTestCase):
pass