Merge pull request #52311 from frappe/mergify/bp/version-16-hotfix/pr-52246

This commit is contained in:
Mihir Kandoi
2026-02-02 21:01:34 +05:30
committed by GitHub
2 changed files with 30 additions and 45 deletions

View File

@@ -296,7 +296,7 @@ class SellingController(StockController):
_( _(
"""Row #{0}: Selling rate for item {1} is lower than its {2}. """Row #{0}: Selling rate for item {1} is lower than its {2}.
Selling {3} should be atleast {4}.<br><br>Alternatively, Selling {3} should be atleast {4}.<br><br>Alternatively,
you can disable selling price validation in {5} to bypass you can disable '{5}' in {6} to bypass
this validation.""" this validation."""
).format( ).format(
idx, idx,
@@ -304,7 +304,8 @@ class SellingController(StockController):
bold(ref_rate_field), bold(ref_rate_field),
bold("net rate"), bold("net rate"),
bold(rate), bold(rate),
get_link_to_form("Selling Settings", "Selling Settings"), bold(frappe.get_meta("Selling Settings").get_label("validate_selling_price")),
get_link_to_form("Selling Settings"),
), ),
title=_("Invalid Selling Price"), title=_("Invalid Selling Price"),
) )
@@ -313,7 +314,6 @@ class SellingController(StockController):
return return
is_internal_customer = self.get("is_internal_customer") is_internal_customer = self.get("is_internal_customer")
valuation_rate_map = {}
for item in self.items: for item in self.items:
if not item.item_code or item.is_free_item: if not item.item_code or item.is_free_item:
@@ -323,7 +323,9 @@ class SellingController(StockController):
"Item", item.item_code, ("last_purchase_rate", "is_stock_item") "Item", item.item_code, ("last_purchase_rate", "is_stock_item")
) )
last_purchase_rate_in_sales_uom = last_purchase_rate * (item.conversion_factor or 1) last_purchase_rate_in_sales_uom = flt(
last_purchase_rate * (item.conversion_factor or 1), item.precision("base_net_rate")
)
if flt(item.base_net_rate) < flt(last_purchase_rate_in_sales_uom): if flt(item.base_net_rate) < flt(last_purchase_rate_in_sales_uom):
throw_message(item.idx, item.item_name, last_purchase_rate_in_sales_uom, "last purchase rate") throw_message(item.idx, item.item_name, last_purchase_rate_in_sales_uom, "last purchase rate")
@@ -331,50 +333,16 @@ class SellingController(StockController):
if is_internal_customer or not is_stock_item: if is_internal_customer or not is_stock_item:
continue continue
valuation_rate_map[(item.item_code, item.warehouse)] = None if item.get("incoming_rate") and item.base_net_rate < (
valuation_rate := flt(
if not valuation_rate_map: item.incoming_rate * (item.conversion_factor or 1), item.precision("base_net_rate")
return )
):
or_conditions = (
f"""(item_code = {frappe.db.escape(valuation_rate[0])}
and warehouse = {frappe.db.escape(valuation_rate[1])})"""
for valuation_rate in valuation_rate_map
)
valuation_rates = frappe.db.sql(
f"""
select
item_code, warehouse, valuation_rate
from
`tabBin`
where
({" or ".join(or_conditions)})
and valuation_rate > 0
""",
as_dict=True,
)
for rate in valuation_rates:
valuation_rate_map[(rate.item_code, rate.warehouse)] = rate.valuation_rate
for item in self.items:
if not item.item_code or item.is_free_item:
continue
last_valuation_rate = valuation_rate_map.get((item.item_code, item.warehouse))
if not last_valuation_rate:
continue
last_valuation_rate_in_sales_uom = last_valuation_rate * (item.conversion_factor or 1)
if flt(item.base_net_rate) < flt(last_valuation_rate_in_sales_uom):
throw_message( throw_message(
item.idx, item.idx,
item.item_name, item.item_name,
last_valuation_rate_in_sales_uom, valuation_rate,
"valuation rate (Moving Average)", "valuation rate",
) )
def get_item_list(self): def get_item_list(self):

View File

@@ -2864,6 +2864,23 @@ class TestDeliveryNote(IntegrationTestCase):
for entry in sabb.entries: for entry in sabb.entries:
self.assertEqual(entry.incoming_rate, 200) self.assertEqual(entry.incoming_rate, 200)
@IntegrationTestCase.change_settings("Selling Settings", {"validate_selling_price": 1})
def test_validate_selling_price(self):
item_code = make_item("VSP Item", properties={"is_stock_item": 1}).name
make_stock_entry(item_code=item_code, target="_Test Warehouse - _TC", qty=1, basic_rate=10)
make_stock_entry(item_code=item_code, target="_Test Warehouse - _TC", qty=1, basic_rate=1)
dn = create_delivery_note(
item_code=item_code,
qty=1,
rate=9,
do_not_save=True,
)
self.assertRaises(frappe.ValidationError, dn.save)
dn.items[0].incoming_rate = 0
dn.items[0].stock_qty = 2
dn.save()
def create_delivery_note(**args): def create_delivery_note(**args):
dn = frappe.new_doc("Delivery Note") dn = frappe.new_doc("Delivery Note")