445 lines
13 KiB
Python
445 lines
13 KiB
Python
import datetime
|
|
from collections import defaultdict
|
|
|
|
import frappe
|
|
from frappe.query_builder.functions import CombineDatetime, Sum
|
|
from frappe.utils import flt, nowtime
|
|
from pypika import Order
|
|
|
|
from erpnext.deprecation_dumpster import deprecated
|
|
|
|
|
|
class DeprecatedSerialNoValuation:
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.SerialNoValuation.calculate_stock_value_from_deprecarated_ledgers",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def calculate_stock_value_from_deprecarated_ledgers(self):
|
|
if not has_sle_for_serial_nos(self.sle.item_code):
|
|
return
|
|
|
|
serial_nos = self.get_filterd_serial_nos()
|
|
if not serial_nos:
|
|
return
|
|
|
|
stock_value_change = 0
|
|
if not self.sle.is_cancelled:
|
|
stock_value_change = self.get_incoming_value_for_serial_nos(serial_nos)
|
|
|
|
self.stock_value_change += flt(stock_value_change)
|
|
|
|
def get_filterd_serial_nos(self):
|
|
serial_nos = []
|
|
non_filtered_serial_nos = self.get_serial_nos()
|
|
|
|
# If the serial no inwarded using the Serial and Batch Bundle, then the serial no should not be considered
|
|
for serial_no in non_filtered_serial_nos:
|
|
if serial_no and serial_no not in self.serial_no_incoming_rate:
|
|
serial_nos.append(serial_no)
|
|
|
|
return serial_nos
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.SerialNoValuation.get_incoming_value_for_serial_nos",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def get_incoming_value_for_serial_nos(self, serial_nos):
|
|
from erpnext.stock.utils import get_combine_datetime
|
|
|
|
# get rate from serial nos within same company
|
|
incoming_values = 0.0
|
|
for serial_no in serial_nos:
|
|
sn_details = frappe.db.get_value("Serial No", serial_no, ["purchase_rate", "company"], as_dict=1)
|
|
if sn_details and sn_details.purchase_rate and sn_details.company == self.sle.company:
|
|
self.serial_no_incoming_rate[serial_no] += flt(sn_details.purchase_rate)
|
|
incoming_values += self.serial_no_incoming_rate[serial_no]
|
|
continue
|
|
|
|
table = frappe.qb.DocType("Stock Ledger Entry")
|
|
stock_ledgers = (
|
|
frappe.qb.from_(table)
|
|
.select(table.incoming_rate, table.actual_qty, table.stock_value_difference)
|
|
.where(
|
|
(
|
|
(table.serial_no == serial_no)
|
|
| (table.serial_no.like(serial_no + "\n%"))
|
|
| (table.serial_no.like("%\n" + serial_no))
|
|
| (table.serial_no.like("%\n" + serial_no + "\n%"))
|
|
)
|
|
& (table.company == self.sle.company)
|
|
& (table.warehouse == self.sle.warehouse)
|
|
& (table.serial_and_batch_bundle.isnull())
|
|
& (table.actual_qty > 0)
|
|
& (table.is_cancelled == 0)
|
|
& table.posting_datetime
|
|
<= get_combine_datetime(self.sle.posting_date, self.sle.posting_time)
|
|
)
|
|
.orderby(table.posting_datetime, order=Order.desc)
|
|
.limit(1)
|
|
).run(as_dict=1)
|
|
|
|
for sle in stock_ledgers:
|
|
self.serial_no_incoming_rate[serial_no] += flt(sle.incoming_rate)
|
|
incoming_values += self.serial_no_incoming_rate[serial_no]
|
|
|
|
return incoming_values
|
|
|
|
|
|
@frappe.request_cache
|
|
def has_sle_for_serial_nos(item_code):
|
|
serial_nos = frappe.db.get_all(
|
|
"Stock Ledger Entry",
|
|
fields=["name"],
|
|
filters={"serial_no": ("is", "set"), "is_cancelled": 0, "item_code": item_code},
|
|
limit=1,
|
|
)
|
|
if serial_nos:
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
class DeprecatedBatchNoValuation:
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.calculate_avg_rate_from_deprecarated_ledgers",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def calculate_avg_rate_from_deprecarated_ledgers(self):
|
|
entries = self.get_sle_for_batches()
|
|
for ledger in entries:
|
|
self.stock_value_differece[ledger.batch_no] += flt(ledger.batch_value)
|
|
self.available_qty[ledger.batch_no] += flt(ledger.batch_qty)
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.get_sle_for_batches",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def get_sle_for_batches(self):
|
|
from erpnext.stock.utils import get_combine_datetime
|
|
|
|
if not self.batchwise_valuation_batches:
|
|
return []
|
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
|
|
timestamp_condition = None
|
|
if self.sle.posting_date:
|
|
if self.sle.posting_time is None:
|
|
self.sle.posting_time = nowtime()
|
|
|
|
posting_datetime = get_combine_datetime(self.sle.posting_date, self.sle.posting_time)
|
|
if not self.sle.creation:
|
|
posting_datetime = posting_datetime + datetime.timedelta(milliseconds=1)
|
|
|
|
timestamp_condition = sle.posting_datetime < posting_datetime
|
|
|
|
if self.sle.creation:
|
|
timestamp_condition |= (sle.posting_datetime == posting_datetime) & (
|
|
sle.creation < self.sle.creation
|
|
)
|
|
|
|
query = (
|
|
frappe.qb.from_(sle)
|
|
.select(
|
|
sle.batch_no,
|
|
Sum(sle.stock_value_difference).as_("batch_value"),
|
|
Sum(sle.actual_qty).as_("batch_qty"),
|
|
)
|
|
.where(
|
|
(sle.item_code == self.sle.item_code)
|
|
& (sle.warehouse == self.sle.warehouse)
|
|
& (sle.batch_no.isin(self.batchwise_valuation_batches))
|
|
& (sle.batch_no.isnotnull())
|
|
& (sle.is_cancelled == 0)
|
|
)
|
|
.groupby(sle.batch_no)
|
|
)
|
|
|
|
if timestamp_condition:
|
|
query = query.where(timestamp_condition)
|
|
|
|
if self.sle.name:
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
return query.run(as_dict=True)
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.calculate_avg_rate_for_non_batchwise_valuation",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def calculate_avg_rate_for_non_batchwise_valuation(self):
|
|
if not self.non_batchwise_valuation_batches:
|
|
return
|
|
|
|
self.non_batchwise_balance_value = defaultdict(float)
|
|
self.non_batchwise_balance_qty = defaultdict(float)
|
|
|
|
self.set_balance_value_for_non_batchwise_valuation_batches()
|
|
|
|
for batch_no, ledger in self.batch_nos.items():
|
|
if batch_no not in self.non_batchwise_valuation_batches:
|
|
continue
|
|
|
|
if not self.non_batchwise_balance_qty:
|
|
continue
|
|
|
|
if not self.non_batchwise_balance_qty.get(batch_no):
|
|
self.batch_avg_rate[batch_no] = 0.0
|
|
self.stock_value_differece[batch_no] = 0.0
|
|
else:
|
|
self.batch_avg_rate[batch_no] = (
|
|
self.non_batchwise_balance_value[batch_no] / self.non_batchwise_balance_qty[batch_no]
|
|
)
|
|
self.stock_value_differece[batch_no] = self.non_batchwise_balance_value
|
|
|
|
stock_value_change = self.batch_avg_rate[batch_no] * ledger.qty
|
|
self.stock_value_change += stock_value_change
|
|
|
|
self.non_batchwise_balance_value[batch_no] -= stock_value_change
|
|
self.non_batchwise_balance_qty[batch_no] -= ledger.qty
|
|
|
|
frappe.db.set_value(
|
|
"Serial and Batch Entry",
|
|
ledger.name,
|
|
{
|
|
"stock_value_difference": stock_value_change,
|
|
"incoming_rate": self.batch_avg_rate[batch_no],
|
|
},
|
|
)
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.set_balance_value_for_non_batchwise_valuation_batches",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def set_balance_value_for_non_batchwise_valuation_batches(self):
|
|
self.set_balance_value_from_sl_entries()
|
|
self.set_balance_value_from_bundle()
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.set_balance_value_from_sl_entries",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def set_balance_value_from_sl_entries(self) -> None:
|
|
from erpnext.stock.utils import get_combine_datetime
|
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
batch = frappe.qb.DocType("Batch")
|
|
|
|
posting_datetime = get_combine_datetime(self.sle.posting_date, self.sle.posting_time)
|
|
if not self.sle.creation:
|
|
posting_datetime = posting_datetime + datetime.timedelta(milliseconds=1)
|
|
|
|
timestamp_condition = sle.posting_datetime < posting_datetime
|
|
|
|
if self.sle.creation:
|
|
timestamp_condition |= (sle.posting_datetime == posting_datetime) & (
|
|
sle.creation < self.sle.creation
|
|
)
|
|
|
|
query = (
|
|
frappe.qb.from_(sle)
|
|
.inner_join(batch)
|
|
.on(sle.batch_no == batch.name)
|
|
.select(
|
|
sle.batch_no,
|
|
Sum(sle.actual_qty).as_("batch_qty"),
|
|
)
|
|
.where(
|
|
(sle.item_code == self.sle.item_code)
|
|
& (sle.warehouse == self.sle.warehouse)
|
|
& (sle.batch_no.isnotnull())
|
|
& (batch.use_batchwise_valuation == 0)
|
|
& (sle.is_cancelled == 0)
|
|
& (sle.batch_no.isin(self.non_batchwise_valuation_batches))
|
|
)
|
|
.where(timestamp_condition)
|
|
.groupby(sle.batch_no)
|
|
)
|
|
|
|
if self.sle.name:
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
batch_data = query.run(as_dict=True)
|
|
for d in batch_data:
|
|
self.available_qty[d.batch_no] += flt(d.batch_qty)
|
|
|
|
last_sle = self.get_last_sle_for_non_batch()
|
|
for d in batch_data:
|
|
if self.available_qty.get(d.batch_no):
|
|
self.non_batchwise_balance_value[d.batch_no] += flt(last_sle.stock_value)
|
|
self.non_batchwise_balance_qty[d.batch_no] += flt(last_sle.qty_after_transaction)
|
|
|
|
def get_last_sle_for_non_batch(self):
|
|
from erpnext.stock.utils import get_combine_datetime
|
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
batch = frappe.qb.DocType("Batch")
|
|
|
|
posting_datetime = get_combine_datetime(self.sle.posting_date, self.sle.posting_time)
|
|
if not self.sle.creation:
|
|
posting_datetime = posting_datetime + datetime.timedelta(milliseconds=1)
|
|
|
|
timestamp_condition = sle.posting_datetime < posting_datetime
|
|
|
|
if self.sle.creation:
|
|
timestamp_condition |= (sle.posting_datetime == posting_datetime) & (
|
|
sle.creation < self.sle.creation
|
|
)
|
|
|
|
query = (
|
|
frappe.qb.from_(sle)
|
|
.inner_join(batch)
|
|
.on(sle.batch_no == batch.name)
|
|
.select(
|
|
sle.stock_value,
|
|
sle.qty_after_transaction,
|
|
)
|
|
.where(
|
|
(sle.item_code == self.sle.item_code)
|
|
& (sle.warehouse == self.sle.warehouse)
|
|
& (sle.batch_no.isnotnull())
|
|
& (batch.use_batchwise_valuation == 0)
|
|
& (sle.is_cancelled == 0)
|
|
)
|
|
.where(timestamp_condition)
|
|
.orderby(sle.posting_datetime, order=Order.desc)
|
|
.orderby(sle.creation, order=Order.desc)
|
|
.limit(1)
|
|
)
|
|
|
|
if self.sle.name:
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
data = query.run(as_dict=True)
|
|
return data[0] if data else {}
|
|
|
|
def get_last_sle_for_sabb_no_batchwise_valuation(self):
|
|
sabb = frappe.qb.DocType("Serial and Batch Bundle")
|
|
sabb_entry = frappe.qb.DocType("Serial and Batch Entry")
|
|
batch = frappe.qb.DocType("Batch")
|
|
|
|
posting_datetime = CombineDatetime(self.sle.posting_date, self.sle.posting_time)
|
|
timestamp_condition = CombineDatetime(sabb.posting_date, sabb.posting_time) < posting_datetime
|
|
|
|
if self.sle.creation:
|
|
timestamp_condition |= (
|
|
CombineDatetime(sabb.posting_date, sabb.posting_time) == posting_datetime
|
|
) & (sabb.creation < self.sle.creation)
|
|
|
|
query = (
|
|
frappe.qb.from_(sabb)
|
|
.inner_join(sabb_entry)
|
|
.on(sabb.name == sabb_entry.parent)
|
|
.inner_join(batch)
|
|
.on(sabb_entry.batch_no == batch.name)
|
|
.select(sabb.name)
|
|
.where(
|
|
(sabb.item_code == self.sle.item_code)
|
|
& (sabb.warehouse == self.sle.warehouse)
|
|
& (sabb_entry.batch_no.isnotnull())
|
|
& (batch.use_batchwise_valuation == 0)
|
|
& (sabb.is_cancelled == 0)
|
|
& (sabb.docstatus == 1)
|
|
)
|
|
.where(timestamp_condition)
|
|
.orderby(sabb.posting_date, order=Order.desc)
|
|
.orderby(sabb.posting_time, order=Order.desc)
|
|
.orderby(sabb.creation, order=Order.desc)
|
|
.limit(1)
|
|
)
|
|
|
|
if self.sle.voucher_detail_no:
|
|
query = query.where(sabb.voucher_detail_no != self.sle.voucher_detail_no)
|
|
|
|
query = query.where(sabb.voucher_type != "Pick List")
|
|
|
|
data = query.run(as_dict=True)
|
|
if not data:
|
|
return {}
|
|
|
|
sle = frappe.db.get_value(
|
|
"Stock Ledger Entry",
|
|
{"serial_and_batch_bundle": data[0].name},
|
|
["stock_value", "qty_after_transaction"],
|
|
as_dict=1,
|
|
)
|
|
|
|
return sle if sle else {}
|
|
|
|
@deprecated(
|
|
"erpnext.stock.serial_batch_bundle.BatchNoValuation.set_balance_value_from_bundle",
|
|
"unknown",
|
|
"v16",
|
|
"No known instructions.",
|
|
)
|
|
def set_balance_value_from_bundle(self) -> None:
|
|
bundle = frappe.qb.DocType("Serial and Batch Bundle")
|
|
bundle_child = frappe.qb.DocType("Serial and Batch Entry")
|
|
batch = frappe.qb.DocType("Batch")
|
|
|
|
timestamp_condition = CombineDatetime(bundle.posting_date, bundle.posting_time) < CombineDatetime(
|
|
self.sle.posting_date, self.sle.posting_time
|
|
)
|
|
|
|
if self.sle.creation:
|
|
timestamp_condition |= (
|
|
CombineDatetime(bundle.posting_date, bundle.posting_time)
|
|
== CombineDatetime(self.sle.posting_date, self.sle.posting_time)
|
|
) & (bundle.creation < self.sle.creation)
|
|
|
|
query = (
|
|
frappe.qb.from_(bundle)
|
|
.inner_join(bundle_child)
|
|
.on(bundle.name == bundle_child.parent)
|
|
.inner_join(batch)
|
|
.on(bundle_child.batch_no == batch.name)
|
|
.select(
|
|
bundle_child.batch_no,
|
|
Sum(bundle_child.qty).as_("batch_qty"),
|
|
Sum(bundle_child.stock_value_difference).as_("batch_value"),
|
|
)
|
|
.where(
|
|
(bundle.item_code == self.sle.item_code)
|
|
& (bundle.warehouse == self.sle.warehouse)
|
|
& (bundle_child.batch_no.isnotnull())
|
|
& (batch.use_batchwise_valuation == 0)
|
|
& (bundle.is_cancelled == 0)
|
|
& (bundle.docstatus == 1)
|
|
& (bundle.type_of_transaction.isin(["Inward", "Outward"]))
|
|
& (bundle_child.batch_no.isin(self.non_batchwise_valuation_batches))
|
|
)
|
|
.where(timestamp_condition)
|
|
.groupby(bundle_child.batch_no)
|
|
)
|
|
|
|
if self.sle.serial_and_batch_bundle:
|
|
query = query.where(bundle.name != self.sle.serial_and_batch_bundle)
|
|
|
|
query = query.where(bundle.voucher_type != "Pick List")
|
|
|
|
batch_data = query.run(as_dict=True)
|
|
for d in batch_data:
|
|
self.available_qty[d.batch_no] += flt(d.batch_qty)
|
|
|
|
last_sle = self.get_last_sle_for_sabb_no_batchwise_valuation()
|
|
if not last_sle:
|
|
return
|
|
|
|
for batch_no in self.available_qty:
|
|
self.non_batchwise_balance_value[batch_no] = flt(last_sle.stock_value)
|
|
self.non_batchwise_balance_qty[batch_no] = flt(last_sle.qty_after_transaction)
|