2023-03-30 06:02:39 +00:00
|
|
|
from collections import defaultdict
|
|
|
|
|
2023-03-13 09:21:43 +00:00
|
|
|
import frappe
|
|
|
|
from frappe.query_builder.functions import CombineDatetime, Sum
|
|
|
|
from frappe.utils import flt
|
2023-03-24 04:44:09 +00:00
|
|
|
from frappe.utils.deprecations import deprecated
|
2023-04-11 07:52:15 +00:00
|
|
|
from pypika import Order
|
2023-03-13 09:21:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class DeprecatedSerialNoValuation:
|
2023-03-24 04:44:09 +00:00
|
|
|
@deprecated
|
2023-03-13 09:21:43 +00:00
|
|
|
def calculate_stock_value_from_deprecarated_ledgers(self):
|
|
|
|
serial_nos = list(
|
|
|
|
filter(lambda x: x not in self.serial_no_incoming_rate and x, self.get_serial_nos())
|
|
|
|
)
|
|
|
|
|
|
|
|
actual_qty = flt(self.sle.actual_qty)
|
|
|
|
|
|
|
|
stock_value_change = 0
|
|
|
|
if actual_qty < 0:
|
|
|
|
if not self.sle.is_cancelled:
|
|
|
|
outgoing_value = self.get_incoming_value_for_serial_nos(serial_nos)
|
|
|
|
stock_value_change = -1 * outgoing_value
|
|
|
|
|
|
|
|
self.stock_value_change += stock_value_change
|
|
|
|
|
2023-03-24 04:44:09 +00:00
|
|
|
@deprecated
|
2023-03-13 09:21:43 +00:00
|
|
|
def get_incoming_value_for_serial_nos(self, serial_nos):
|
|
|
|
# get rate from serial nos within same company
|
|
|
|
all_serial_nos = frappe.get_all(
|
|
|
|
"Serial No", fields=["purchase_rate", "name", "company"], filters={"name": ("in", serial_nos)}
|
|
|
|
)
|
|
|
|
|
|
|
|
incoming_values = 0.0
|
|
|
|
for d in all_serial_nos:
|
|
|
|
if d.company == self.sle.company:
|
2023-03-31 03:33:54 +00:00
|
|
|
self.serial_no_incoming_rate[d.name] += flt(d.purchase_rate)
|
2023-03-13 09:21:43 +00:00
|
|
|
incoming_values += flt(d.purchase_rate)
|
|
|
|
|
|
|
|
# Get rate for serial nos which has been transferred to other company
|
|
|
|
invalid_serial_nos = [d.name for d in all_serial_nos if d.company != self.sle.company]
|
|
|
|
for serial_no in invalid_serial_nos:
|
2023-04-11 07:52:15 +00:00
|
|
|
table = frappe.qb.DocType("Stock Ledger Entry")
|
|
|
|
incoming_rate = (
|
|
|
|
frappe.qb.from_(table)
|
|
|
|
.select(table.incoming_rate)
|
|
|
|
.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%"))
|
2023-03-13 09:21:43 +00:00
|
|
|
)
|
2023-04-11 07:52:15 +00:00
|
|
|
& (table.company == self.sle.company)
|
|
|
|
& (table.serial_and_batch_bundle.isnull())
|
|
|
|
& (table.actual_qty > 0)
|
|
|
|
& (table.is_cancelled == 0)
|
|
|
|
)
|
|
|
|
.orderby(table.posting_date, order=Order.desc)
|
|
|
|
.limit(1)
|
|
|
|
).run()
|
2023-03-13 09:21:43 +00:00
|
|
|
|
2023-03-31 03:33:54 +00:00
|
|
|
self.serial_no_incoming_rate[serial_no] += flt(incoming_rate[0][0]) if incoming_rate else 0
|
2023-03-13 09:21:43 +00:00
|
|
|
incoming_values += self.serial_no_incoming_rate[serial_no]
|
|
|
|
|
|
|
|
return incoming_values
|
|
|
|
|
|
|
|
|
|
|
|
class DeprecatedBatchNoValuation:
|
2023-03-24 04:44:09 +00:00
|
|
|
@deprecated
|
2023-03-13 09:21:43 +00:00
|
|
|
def calculate_avg_rate_from_deprecarated_ledgers(self):
|
2023-03-21 05:24:41 +00:00
|
|
|
entries = self.get_sle_for_batches()
|
|
|
|
for ledger in entries:
|
2023-03-30 06:02:39 +00:00
|
|
|
self.stock_value_differece[ledger.batch_no] += flt(ledger.batch_value)
|
2023-03-20 08:45:34 +00:00
|
|
|
self.available_qty[ledger.batch_no] += flt(ledger.batch_qty)
|
2023-03-13 09:21:43 +00:00
|
|
|
|
2023-03-24 04:44:09 +00:00
|
|
|
@deprecated
|
2023-03-13 09:21:43 +00:00
|
|
|
def get_sle_for_batches(self):
|
2023-03-30 06:02:39 +00:00
|
|
|
if not self.batchwise_valuation_batches:
|
|
|
|
return []
|
|
|
|
|
2023-03-13 09:21:43 +00:00
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
|
|
|
|
|
|
timestamp_condition = CombineDatetime(sle.posting_date, sle.posting_time) < CombineDatetime(
|
|
|
|
self.sle.posting_date, self.sle.posting_time
|
|
|
|
)
|
|
|
|
if self.sle.creation:
|
|
|
|
timestamp_condition |= (
|
|
|
|
CombineDatetime(sle.posting_date, sle.posting_time)
|
|
|
|
== CombineDatetime(self.sle.posting_date, self.sle.posting_time)
|
|
|
|
) & (sle.creation < self.sle.creation)
|
|
|
|
|
2023-03-29 06:10:36 +00:00
|
|
|
query = (
|
2023-03-13 09:21:43 +00:00
|
|
|
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)
|
2023-03-30 06:02:39 +00:00
|
|
|
& (sle.batch_no.isin(self.batchwise_valuation_batches))
|
|
|
|
& (sle.batch_no.isnotnull())
|
2023-03-13 09:21:43 +00:00
|
|
|
& (sle.is_cancelled == 0)
|
|
|
|
)
|
|
|
|
.where(timestamp_condition)
|
|
|
|
.groupby(sle.batch_no)
|
2023-03-29 06:10:36 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if self.sle.name:
|
|
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
|
|
|
|
return query.run(as_dict=True)
|
2023-03-30 06:02:39 +00:00
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def calculate_avg_rate_for_non_batchwise_valuation(self):
|
|
|
|
if not self.non_batchwise_valuation_batches:
|
|
|
|
return
|
|
|
|
|
|
|
|
avg_rate = self.get_avg_rate_for_non_batchwise_valuation_batches()
|
|
|
|
avilable_qty = self.get_available_qty_for_non_batchwise_valuation_batches()
|
|
|
|
|
|
|
|
for batch_no in self.non_batchwise_valuation_batches:
|
|
|
|
self.stock_value_differece[batch_no] = avg_rate
|
|
|
|
self.available_qty[batch_no] = avilable_qty.get(batch_no, 0)
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def get_avg_rate_for_non_batchwise_valuation_batches(self):
|
|
|
|
stock_value, qty = self.get_balance_value_and_qty_from_sl_entries()
|
|
|
|
stock_value, qty = self.get_balance_value_and_qty_from_bundle(stock_value, qty)
|
|
|
|
|
|
|
|
return stock_value / qty if qty else 0
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def get_balance_value_and_qty_from_sl_entries(self):
|
|
|
|
stock_value_difference = 0.0
|
|
|
|
available_qty = 0.0
|
|
|
|
|
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
|
|
batch = frappe.qb.DocType("Batch")
|
|
|
|
|
|
|
|
timestamp_condition = CombineDatetime(sle.posting_date, sle.posting_time) < CombineDatetime(
|
|
|
|
self.sle.posting_date, self.sle.posting_time
|
|
|
|
)
|
|
|
|
if self.sle.creation:
|
|
|
|
timestamp_condition |= (
|
|
|
|
CombineDatetime(sle.posting_date, sle.posting_time)
|
|
|
|
== CombineDatetime(self.sle.posting_date, self.sle.posting_time)
|
|
|
|
) & (sle.creation < self.sle.creation)
|
|
|
|
|
|
|
|
query = (
|
|
|
|
frappe.qb.from_(sle)
|
|
|
|
.inner_join(batch)
|
|
|
|
.on(sle.batch_no == batch.name)
|
|
|
|
.select(
|
|
|
|
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.isnotnull())
|
|
|
|
& (batch.use_batchwise_valuation == 0)
|
|
|
|
& (sle.is_cancelled == 0)
|
|
|
|
)
|
|
|
|
.where(timestamp_condition)
|
|
|
|
)
|
|
|
|
|
|
|
|
if self.sle.name:
|
|
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
|
|
|
|
for d in query.run(as_dict=True):
|
|
|
|
stock_value_difference += flt(d.batch_value)
|
|
|
|
available_qty += flt(d.batch_qty)
|
|
|
|
|
|
|
|
return stock_value_difference, available_qty
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def get_balance_value_and_qty_from_bundle(self, stock_value, qty):
|
|
|
|
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(
|
|
|
|
Sum(bundle_child.stock_value_difference).as_("batch_value"),
|
|
|
|
Sum(bundle_child.qty).as_("batch_qty"),
|
|
|
|
)
|
|
|
|
.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)
|
2023-04-02 07:43:42 +00:00
|
|
|
& (bundle.type_of_transaction.isin(["Inward", "Outward"]))
|
2023-03-30 06:02:39 +00:00
|
|
|
)
|
|
|
|
.where(timestamp_condition)
|
|
|
|
)
|
|
|
|
|
|
|
|
if self.sle.serial_and_batch_bundle:
|
|
|
|
query = query.where(bundle.name != self.sle.serial_and_batch_bundle)
|
|
|
|
|
|
|
|
for d in query.run(as_dict=True):
|
|
|
|
stock_value += flt(d.batch_value)
|
|
|
|
qty += flt(d.batch_qty)
|
|
|
|
|
|
|
|
return stock_value, qty
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def get_available_qty_for_non_batchwise_valuation_batches(self):
|
|
|
|
available_qty = defaultdict(float)
|
|
|
|
self.set_available_qty_for_non_batchwise_valuation_batches_from_sle(available_qty)
|
|
|
|
self.set_available_qty_for_non_batchwise_valuation_batches_from_bundle(available_qty)
|
|
|
|
|
|
|
|
return available_qty
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def set_available_qty_for_non_batchwise_valuation_batches_from_sle(self, available_qty):
|
|
|
|
sle = frappe.qb.DocType("Stock Ledger Entry")
|
|
|
|
|
|
|
|
timestamp_condition = CombineDatetime(sle.posting_date, sle.posting_time) < CombineDatetime(
|
|
|
|
self.sle.posting_date, self.sle.posting_time
|
|
|
|
)
|
|
|
|
if self.sle.creation:
|
|
|
|
timestamp_condition |= (
|
|
|
|
CombineDatetime(sle.posting_date, sle.posting_time)
|
|
|
|
== CombineDatetime(self.sle.posting_date, self.sle.posting_time)
|
|
|
|
) & (sle.creation < self.sle.creation)
|
|
|
|
|
|
|
|
query = (
|
|
|
|
frappe.qb.from_(sle)
|
|
|
|
.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.isin(self.non_batchwise_valuation_batches))
|
|
|
|
& (sle.is_cancelled == 0)
|
|
|
|
)
|
|
|
|
.where(timestamp_condition)
|
|
|
|
.groupby(sle.batch_no)
|
|
|
|
)
|
|
|
|
|
|
|
|
if self.sle.name:
|
|
|
|
query = query.where(sle.name != self.sle.name)
|
|
|
|
|
|
|
|
for d in query.run(as_dict=True):
|
|
|
|
available_qty[d.batch_no] += flt(d.batch_qty)
|
|
|
|
|
|
|
|
@deprecated
|
|
|
|
def set_available_qty_for_non_batchwise_valuation_batches_from_bundle(self, available_qty):
|
|
|
|
bundle = frappe.qb.DocType("Serial and Batch Bundle")
|
|
|
|
bundle_child = frappe.qb.DocType("Serial and Batch Entry")
|
|
|
|
|
|
|
|
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)
|
|
|
|
.select(
|
|
|
|
bundle_child.batch_no,
|
|
|
|
Sum(bundle_child.qty).as_("batch_qty"),
|
|
|
|
)
|
|
|
|
.where(
|
|
|
|
(bundle.item_code == self.sle.item_code)
|
|
|
|
& (bundle.warehouse == self.sle.warehouse)
|
|
|
|
& (bundle_child.batch_no.isin(self.non_batchwise_valuation_batches))
|
|
|
|
& (bundle.is_cancelled == 0)
|
|
|
|
)
|
|
|
|
.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)
|
|
|
|
|
|
|
|
for d in query.run(as_dict=True):
|
|
|
|
available_qty[d.batch_no] += flt(d.batch_qty)
|