2015-03-03 09:25:30 +00:00
|
|
|
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
2013-08-05 09:29:54 +00:00
|
|
|
# License: GNU General Public License v3. See license.txt
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
from __future__ import unicode_literals
|
2014-02-14 10:17:51 +00:00
|
|
|
import frappe
|
2014-04-15 09:06:12 +00:00
|
|
|
from frappe import _
|
2015-03-26 09:07:55 +00:00
|
|
|
from frappe.utils import flt, getdate
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
def execute(filters=None):
|
|
|
|
if not filters: filters = {}
|
2016-03-14 09:29:21 +00:00
|
|
|
|
|
|
|
validate_filters(filters)
|
2014-04-15 09:06:12 +00:00
|
|
|
|
2016-03-07 07:41:59 +00:00
|
|
|
columns = get_columns()
|
2013-06-05 06:06:24 +00:00
|
|
|
item_map = get_item_details(filters)
|
|
|
|
iwb_map = get_item_warehouse_map(filters)
|
2014-04-15 09:06:12 +00:00
|
|
|
|
2013-06-05 06:06:24 +00:00
|
|
|
data = []
|
2016-02-11 15:08:01 +00:00
|
|
|
for (company, item, warehouse) in sorted(iwb_map):
|
|
|
|
qty_dict = iwb_map[(company, item, warehouse)]
|
|
|
|
data.append([item, item_map[item]["item_name"],
|
|
|
|
item_map[item]["item_group"],
|
|
|
|
item_map[item]["brand"],
|
|
|
|
item_map[item]["description"], warehouse,
|
|
|
|
item_map[item]["stock_uom"], qty_dict.opening_qty,
|
|
|
|
qty_dict.opening_val, qty_dict.in_qty,
|
|
|
|
qty_dict.in_val, qty_dict.out_qty,
|
|
|
|
qty_dict.out_val, qty_dict.bal_qty,
|
|
|
|
qty_dict.bal_val, qty_dict.val_rate,
|
|
|
|
company
|
|
|
|
])
|
2014-04-15 09:06:12 +00:00
|
|
|
|
2013-06-05 06:06:24 +00:00
|
|
|
return columns, data
|
|
|
|
|
2016-03-07 07:41:59 +00:00
|
|
|
def get_columns():
|
|
|
|
"""return columns"""
|
2014-04-15 09:06:12 +00:00
|
|
|
|
2016-02-04 11:52:52 +00:00
|
|
|
columns = [
|
2016-02-11 15:08:01 +00:00
|
|
|
_("Item")+":Link/Item:100",
|
|
|
|
_("Item Name")+"::150",
|
|
|
|
_("Item Group")+"::100",
|
|
|
|
_("Brand")+"::90",
|
|
|
|
_("Description")+"::140",
|
|
|
|
_("Warehouse")+":Link/Warehouse:100",
|
|
|
|
_("Stock UOM")+":Link/UOM:90",
|
|
|
|
_("Opening Qty")+":Float:100",
|
|
|
|
_("Opening Value")+":Float:110",
|
|
|
|
_("In Qty")+":Float:80",
|
|
|
|
_("In Value")+":Float:80",
|
|
|
|
_("Out Qty")+":Float:80",
|
|
|
|
_("Out Value")+":Float:80",
|
|
|
|
_("Balance Qty")+":Float:100",
|
|
|
|
_("Balance Value")+":Float:100",
|
|
|
|
_("Valuation Rate")+":Float:90",
|
2016-02-04 11:52:52 +00:00
|
|
|
_("Company")+":Link/Company:100"
|
|
|
|
]
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
return columns
|
|
|
|
|
|
|
|
def get_conditions(filters):
|
|
|
|
conditions = ""
|
|
|
|
if not filters.get("from_date"):
|
2014-04-15 09:06:12 +00:00
|
|
|
frappe.throw(_("'From Date' is required"))
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
if filters.get("to_date"):
|
2016-02-11 15:08:01 +00:00
|
|
|
conditions += " and posting_date <= '%s'" % frappe.db.escape(filters["to_date"])
|
2013-06-05 06:06:24 +00:00
|
|
|
else:
|
2014-04-15 09:06:12 +00:00
|
|
|
frappe.throw(_("'To Date' is required"))
|
|
|
|
|
2015-02-18 14:21:48 +00:00
|
|
|
if filters.get("item_code"):
|
2016-02-19 07:15:57 +00:00
|
|
|
conditions += " and item_code = '%s'" % frappe.db.escape(filters.get("item_code"), percent=False)
|
2015-02-18 14:21:48 +00:00
|
|
|
|
2016-03-07 07:41:59 +00:00
|
|
|
if filters.get("warehouse"):
|
2016-06-23 07:14:06 +00:00
|
|
|
lft, rgt = frappe.db.get_value("Warehouse", filters.get("warehouse"), ["lft", "rgt"])
|
|
|
|
conditions += " and exists (select name from `tabWarehouse` wh \
|
|
|
|
where wh.lft >= %s and wh.rgt <= %s and sle.warehouse = wh.name)"%(lft, rgt)
|
2016-03-07 07:41:59 +00:00
|
|
|
|
2013-06-05 06:06:24 +00:00
|
|
|
return conditions
|
|
|
|
|
|
|
|
def get_stock_ledger_entries(filters):
|
|
|
|
conditions = get_conditions(filters)
|
2014-10-14 06:11:44 +00:00
|
|
|
return frappe.db.sql("""select item_code, warehouse, posting_date, actual_qty, valuation_rate,
|
2016-02-11 15:08:01 +00:00
|
|
|
company, voucher_type, qty_after_transaction, stock_value_difference
|
2016-06-23 07:14:06 +00:00
|
|
|
from `tabStock Ledger Entry` sle force index (posting_sort_index)
|
2014-10-10 12:32:23 +00:00
|
|
|
where docstatus < 2 %s order by posting_date, posting_time, name""" %
|
2016-06-23 07:14:06 +00:00
|
|
|
conditions, as_dict=1, debug=1)
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
def get_item_warehouse_map(filters):
|
|
|
|
iwb_map = {}
|
2016-02-11 15:08:01 +00:00
|
|
|
from_date = getdate(filters["from_date"])
|
|
|
|
to_date = getdate(filters["to_date"])
|
|
|
|
|
|
|
|
sle = get_stock_ledger_entries(filters)
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
for d in sle:
|
2016-02-11 15:08:01 +00:00
|
|
|
key = (d.company, d.item_code, d.warehouse)
|
|
|
|
if key not in iwb_map:
|
|
|
|
iwb_map[key] = frappe._dict({
|
2014-09-29 06:05:52 +00:00
|
|
|
"opening_qty": 0.0, "opening_val": 0.0,
|
|
|
|
"in_qty": 0.0, "in_val": 0.0,
|
|
|
|
"out_qty": 0.0, "out_val": 0.0,
|
|
|
|
"bal_qty": 0.0, "bal_val": 0.0,
|
2014-09-11 11:17:19 +00:00
|
|
|
"val_rate": 0.0, "uom": None
|
2016-02-11 15:08:01 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
qty_dict = iwb_map[(d.company, d.item_code, d.warehouse)]
|
2014-09-11 11:17:19 +00:00
|
|
|
|
2014-10-10 12:32:23 +00:00
|
|
|
if d.voucher_type == "Stock Reconciliation":
|
|
|
|
qty_diff = flt(d.qty_after_transaction) - qty_dict.bal_qty
|
|
|
|
else:
|
|
|
|
qty_diff = flt(d.actual_qty)
|
2014-10-14 06:11:44 +00:00
|
|
|
|
|
|
|
value_diff = flt(d.stock_value_difference)
|
2016-02-11 15:08:01 +00:00
|
|
|
|
|
|
|
if d.posting_date < from_date:
|
2014-10-10 12:32:23 +00:00
|
|
|
qty_dict.opening_qty += qty_diff
|
|
|
|
qty_dict.opening_val += value_diff
|
2016-02-11 15:08:01 +00:00
|
|
|
|
|
|
|
elif d.posting_date >= from_date and d.posting_date <= to_date:
|
2014-10-10 12:32:23 +00:00
|
|
|
if qty_diff > 0:
|
|
|
|
qty_dict.in_qty += qty_diff
|
|
|
|
qty_dict.in_val += value_diff
|
2013-06-05 06:06:24 +00:00
|
|
|
else:
|
2014-10-10 12:32:23 +00:00
|
|
|
qty_dict.out_qty += abs(qty_diff)
|
|
|
|
qty_dict.out_val += abs(value_diff)
|
2016-02-11 15:08:01 +00:00
|
|
|
|
2015-10-06 13:16:56 +00:00
|
|
|
qty_dict.val_rate = d.valuation_rate
|
2014-10-10 12:32:23 +00:00
|
|
|
qty_dict.bal_qty += qty_diff
|
|
|
|
qty_dict.bal_val += value_diff
|
2013-06-05 06:06:24 +00:00
|
|
|
|
|
|
|
return iwb_map
|
|
|
|
|
|
|
|
def get_item_details(filters):
|
2016-03-07 07:41:59 +00:00
|
|
|
condition = ''
|
|
|
|
value = ()
|
|
|
|
if filters.get("item_code"):
|
|
|
|
condition = "where item_code=%s"
|
|
|
|
value = (filters["item_code"],)
|
|
|
|
|
|
|
|
items = frappe.db.sql("""select name, item_name, stock_uom, item_group, brand, description
|
|
|
|
from tabItem {condition}""".format(condition=condition), value, as_dict=1)
|
2013-06-05 06:06:24 +00:00
|
|
|
|
2016-03-07 07:41:59 +00:00
|
|
|
return dict((d.name, d) for d in items)
|
2016-03-14 09:29:21 +00:00
|
|
|
|
|
|
|
def validate_filters(filters):
|
|
|
|
if not (filters.get("item_code") or filters.get("warehouse")):
|
|
|
|
sle_count = flt(frappe.db.sql("""select count(name) from `tabStock Ledger Entry`""")[0][0])
|
|
|
|
if sle_count > 500000:
|
|
|
|
frappe.throw(_("Please set filter based on Item or Warehouse"))
|
|
|
|
|