You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

224 lines
6.9 KiB

  1. # Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
  2. # License: GNU General Public License v3. See license.txt
  3. import frappe
  4. import frappe.share
  5. from frappe import _
  6. from frappe.utils import cint, flt, get_time, now_datetime
  7. from erpnext.controllers.status_updater import StatusUpdater
  8. class UOMMustBeIntegerError(frappe.ValidationError):
  9. pass
  10. class TransactionBase(StatusUpdater):
  11. def validate_posting_time(self):
  12. # set Edit Posting Date and Time to 1 while data import
  13. if frappe.flags.in_import and self.posting_date:
  14. self.set_posting_time = 1
  15. if not getattr(self, "set_posting_time", None):
  16. now = now_datetime()
  17. self.posting_date = now.strftime("%Y-%m-%d")
  18. self.posting_time = now.strftime("%H:%M:%S.%f")
  19. elif self.posting_time:
  20. try:
  21. get_time(self.posting_time)
  22. except ValueError:
  23. frappe.throw(_("Invalid Posting Time"))
  24. def validate_uom_is_integer(self, uom_field, qty_fields):
  25. validate_uom_is_integer(self, uom_field, qty_fields)
  26. def validate_with_previous_doc(self, ref):
  27. self.exclude_fields = ["conversion_factor", "uom"] if self.get("is_return") else []
  28. for key, val in ref.items():
  29. is_child = val.get("is_child_table")
  30. ref_doc = {}
  31. item_ref_dn = []
  32. for d in self.get_all_children(self.doctype + " Item"):
  33. ref_dn = d.get(val["ref_dn_field"])
  34. if ref_dn:
  35. if is_child:
  36. self.compare_values({key: [ref_dn]}, val["compare_fields"], d)
  37. if ref_dn not in item_ref_dn:
  38. item_ref_dn.append(ref_dn)
  39. elif not val.get("allow_duplicate_prev_row_id"):
  40. frappe.throw(_("Duplicate row {0} with same {1}").format(d.idx, key))
  41. elif ref_dn:
  42. ref_doc.setdefault(key, [])
  43. if ref_dn not in ref_doc[key]:
  44. ref_doc[key].append(ref_dn)
  45. if ref_doc:
  46. self.compare_values(ref_doc, val["compare_fields"])
  47. def compare_values(self, ref_doc, fields, doc=None):
  48. for reference_doctype, ref_dn_list in ref_doc.items():
  49. prev_doc_detail_map = self.get_prev_doc_reference_details(
  50. ref_dn_list, reference_doctype, fields
  51. )
  52. for reference_name in ref_dn_list:
  53. prevdoc_values = prev_doc_detail_map.get(reference_name)
  54. if not prevdoc_values:
  55. frappe.throw(_("Invalid reference {0} {1}").format(reference_doctype, reference_name))
  56. for field, condition in fields:
  57. if prevdoc_values[field] is not None and field not in self.exclude_fields:
  58. self.validate_value(field, condition, prevdoc_values[field], doc)
  59. def get_prev_doc_reference_details(self, reference_names, reference_doctype, fields):
  60. prev_doc_detail_map = {}
  61. details = frappe.get_all(
  62. reference_doctype,
  63. filters={"name": ("in", reference_names)},
  64. fields=["name"] + [d[0] for d in fields],
  65. )
  66. for d in details:
  67. prev_doc_detail_map.setdefault(d.name, d)
  68. return prev_doc_detail_map
  69. def validate_rate_with_reference_doc(self, ref_details):
  70. if self.get("is_internal_supplier"):
  71. return
  72. buying_doctypes = ["Purchase Order", "Purchase Invoice", "Purchase Receipt"]
  73. if self.doctype in buying_doctypes:
  74. action, role_allowed_to_override = frappe.get_cached_value(
  75. "Buying Settings", "None", ["maintain_same_rate_action", "role_to_override_stop_action"]
  76. )
  77. else:
  78. action, role_allowed_to_override = frappe.get_cached_value(
  79. "Selling Settings", "None", ["maintain_same_rate_action", "role_to_override_stop_action"]
  80. )
  81. for ref_dt, ref_dn_field, ref_link_field in ref_details:
  82. reference_names = [d.get(ref_link_field) for d in self.get("items") if d.get(ref_link_field)]
  83. reference_details = self.get_reference_details(reference_names, ref_dt + " Item")
  84. for d in self.get("items"):
  85. if d.get(ref_link_field):
  86. ref_rate = reference_details.get(d.get(ref_link_field))
  87. if abs(flt(d.rate - ref_rate, d.precision("rate"))) >= 0.01:
  88. if action == "Stop":
  89. if role_allowed_to_override not in frappe.get_roles():
  90. frappe.throw(
  91. _("Row #{0}: Rate must be same as {1}: {2} ({3} / {4})").format(
  92. d.idx, ref_dt, d.get(ref_dn_field), d.rate, ref_rate
  93. )
  94. )
  95. else:
  96. frappe.msgprint(
  97. _("Row #{0}: Rate must be same as {1}: {2} ({3} / {4})").format(
  98. d.idx, ref_dt, d.get(ref_dn_field), d.rate, ref_rate
  99. ),
  100. title=_("Warning"),
  101. indicator="orange",
  102. )
  103. def get_reference_details(self, reference_names, reference_doctype):
  104. return frappe._dict(
  105. frappe.get_all(
  106. reference_doctype,
  107. filters={"name": ("in", reference_names)},
  108. fields=["name", "rate"],
  109. as_list=1,
  110. )
  111. )
  112. def get_link_filters(self, for_doctype):
  113. if hasattr(self, "prev_link_mapper") and self.prev_link_mapper.get(for_doctype):
  114. fieldname = self.prev_link_mapper[for_doctype]["fieldname"]
  115. values = filter(None, tuple(item.as_dict()[fieldname] for item in self.items))
  116. if values:
  117. ret = {for_doctype: {"filters": [[for_doctype, "name", "in", values]]}}
  118. else:
  119. ret = None
  120. else:
  121. ret = None
  122. return ret
  123. def reset_default_field_value(self, default_field: str, child_table: str, child_table_field: str):
  124. """Reset "Set default X" fields on forms to avoid confusion.
  125. example:
  126. doc = {
  127. "set_from_warehouse": "Warehouse A",
  128. "items": [{"from_warehouse": "warehouse B"}, {"from_warehouse": "warehouse A"}],
  129. }
  130. Since this has dissimilar values in child table, the default field will be erased.
  131. doc.reset_default_field_value("set_from_warehouse", "items", "from_warehouse")
  132. """
  133. child_table_values = set()
  134. for row in self.get(child_table):
  135. child_table_values.add(row.get(child_table_field))
  136. if len(child_table_values) > 1:
  137. self.set(default_field, None)
  138. def delete_events(ref_type, ref_name):
  139. events = (
  140. frappe.db.sql_list(
  141. """ SELECT
  142. distinct `tabEvent`.name
  143. from
  144. `tabEvent`, `tabEvent Participants`
  145. where
  146. `tabEvent`.name = `tabEvent Participants`.parent
  147. and `tabEvent Participants`.reference_doctype = %s
  148. and `tabEvent Participants`.reference_docname = %s
  149. """,
  150. (ref_type, ref_name),
  151. )
  152. or []
  153. )
  154. if events:
  155. frappe.delete_doc("Event", events, for_reload=True)
  156. def validate_uom_is_integer(doc, uom_field, qty_fields, child_dt=None):
  157. if isinstance(qty_fields, str):
  158. qty_fields = [qty_fields]
  159. distinct_uoms = list(set(d.get(uom_field) for d in doc.get_all_children()))
  160. integer_uoms = list(
  161. filter(
  162. lambda uom: frappe.db.get_value("UOM", uom, "must_be_whole_number", cache=True) or None,
  163. distinct_uoms,
  164. )
  165. )
  166. if not integer_uoms:
  167. return
  168. for d in doc.get_all_children(parenttype=child_dt):
  169. if d.get(uom_field) in integer_uoms:
  170. for f in qty_fields:
  171. qty = d.get(f)
  172. if qty:
  173. if abs(cint(qty) - flt(qty, d.precision(f))) > 0.0000001:
  174. frappe.throw(
  175. _(
  176. "Row {1}: Quantity ({0}) cannot be a fraction. To allow this, disable '{2}' in UOM {3}."
  177. ).format(
  178. flt(qty, d.precision(f)),
  179. d.idx,
  180. frappe.bold(_("Must be Whole Number")),
  181. frappe.bold(d.get(uom_field)),
  182. ),
  183. UOMMustBeIntegerError,
  184. )