import frappe class ItemVariantsCacheManager: def __init__(self, item_code): self.item_code = item_code def get_item_variants_data(self): val = frappe.cache().hget("item_variants_data", self.item_code) if not val: self.build_cache() return frappe.cache().hget("item_variants_data", self.item_code) def get_attribute_value_item_map(self): val = frappe.cache().hget("attribute_value_item_map", self.item_code) if not val: self.build_cache() return frappe.cache().hget("attribute_value_item_map", self.item_code) def get_item_attribute_value_map(self): val = frappe.cache().hget("item_attribute_value_map", self.item_code) if not val: self.build_cache() return frappe.cache().hget("item_attribute_value_map", self.item_code) def get_optional_attributes(self): val = frappe.cache().hget("optional_attributes", self.item_code) if not val: self.build_cache() return frappe.cache().hget("optional_attributes", self.item_code) def get_ordered_attribute_values(self): val = frappe.cache().get_value("ordered_attribute_values_map") if val: return val all_attribute_values = frappe.get_all( "Item Attribute Value", ["attribute_value", "idx", "parent"], order_by="idx asc" ) ordered_attribute_values_map = frappe._dict({}) for d in all_attribute_values: ordered_attribute_values_map.setdefault(d.parent, []).append(d.attribute_value) frappe.cache().set_value("ordered_attribute_values_map", ordered_attribute_values_map) return ordered_attribute_values_map def build_cache(self): parent_item_code = self.item_code attributes = [ a.attribute for a in frappe.get_all( "Item Variant Attribute", {"parent": parent_item_code}, ["attribute"], order_by="idx asc" ) ] # Get Variants and tehir Attributes that are not disabled iva = frappe.qb.DocType("Item Variant Attribute") item = frappe.qb.DocType("Item") query = ( frappe.qb.from_(iva) .join(item) .on(item.name == iva.parent) .select(iva.parent, iva.attribute, iva.attribute_value) .where((iva.variant_of == parent_item_code) & (item.disabled == 0)) .orderby(iva.name) ) item_variants_data = query.run() attribute_value_item_map = frappe._dict() item_attribute_value_map = frappe._dict() for row in item_variants_data: item_code, attribute, attribute_value = row # (attr, value) => [item1, item2] attribute_value_item_map.setdefault((attribute, attribute_value), []).append(item_code) # item => {attr1: value1, attr2: value2} item_attribute_value_map.setdefault(item_code, {})[attribute] = attribute_value optional_attributes = set() for item_code, attr_dict in item_attribute_value_map.items(): for attribute in attributes: if attribute not in attr_dict: optional_attributes.add(attribute) frappe.cache().hset("attribute_value_item_map", parent_item_code, attribute_value_item_map) frappe.cache().hset("item_attribute_value_map", parent_item_code, item_attribute_value_map) frappe.cache().hset("item_variants_data", parent_item_code, item_variants_data) frappe.cache().hset("optional_attributes", parent_item_code, optional_attributes) def clear_cache(self): keys = [ "attribute_value_item_map", "item_attribute_value_map", "item_variants_data", "optional_attributes", ] for key in keys: frappe.cache().hdel(key, self.item_code) def rebuild_cache(self): self.clear_cache() enqueue_build_cache(self.item_code) def build_cache(item_code): frappe.cache().hset("item_cache_build_in_progress", item_code, 1) i = ItemVariantsCacheManager(item_code) i.build_cache() frappe.cache().hset("item_cache_build_in_progress", item_code, 0) def enqueue_build_cache(item_code): if frappe.cache().hget("item_cache_build_in_progress", item_code): return frappe.enqueue( "erpnext.e_commerce.variant_selector.item_variants_cache.build_cache", item_code=item_code, queue="long", )