2019-03-29 20:44:23 +05:30
|
|
|
# Copyright (c) 2013, Frappe Technologies Pvt. Ltd. and contributors
|
|
|
|
# For license information, please see license.txt
|
|
|
|
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
import frappe
|
|
|
|
from frappe import _
|
|
|
|
|
|
|
|
def execute(filters=None):
|
2019-04-28 17:11:02 +05:30
|
|
|
columns = get_columns(filters)
|
|
|
|
data = get_data(filters)
|
2019-03-29 20:44:23 +05:30
|
|
|
return columns, data
|
|
|
|
|
2019-04-28 17:11:02 +05:30
|
|
|
def get_columns(filters):
|
2019-03-29 20:44:23 +05:30
|
|
|
columns = [
|
|
|
|
{
|
2019-04-21 23:44:21 +05:30
|
|
|
"label": _("Material Request Date"),
|
|
|
|
"fieldname": "material_request_date",
|
2019-03-29 20:44:23 +05:30
|
|
|
"fieldtype": "Date",
|
|
|
|
"width": 140
|
|
|
|
},
|
2019-04-26 01:29:36 +05:30
|
|
|
{
|
|
|
|
"label": _("Material Request No"),
|
|
|
|
"options": "Material Request",
|
|
|
|
"fieldname": "material_request_no",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
2019-03-29 20:44:23 +05:30
|
|
|
{
|
2019-04-21 23:44:21 +05:30
|
|
|
"label": _("Cost Center"),
|
|
|
|
"options": "Cost Center",
|
|
|
|
"fieldname": "cost_center",
|
|
|
|
"fieldtype": "Link",
|
2019-03-29 20:44:23 +05:30
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
2019-04-21 23:44:21 +05:30
|
|
|
"label": _("Project"),
|
|
|
|
"options": "Project",
|
|
|
|
"fieldname": "project",
|
2019-03-29 20:44:23 +05:30
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Requesting Site"),
|
|
|
|
"options": "Warehouse",
|
|
|
|
"fieldname": "requesting_site",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Requestor"),
|
|
|
|
"options": "Employee",
|
|
|
|
"fieldname": "requestor",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Description"),
|
|
|
|
"fieldname": "description",
|
|
|
|
"fieldtype": "Data",
|
|
|
|
"width": 200
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Quantity"),
|
|
|
|
"fieldname": "quantity",
|
|
|
|
"fieldtype": "Int",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Unit of Measure"),
|
|
|
|
"options": "UOM",
|
|
|
|
"fieldname": "unit_of_measurement",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Status"),
|
|
|
|
"fieldname": "status",
|
|
|
|
"fieldtype": "data",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Purchase Order Date"),
|
|
|
|
"fieldname": "purchase_order_date",
|
|
|
|
"fieldtype": "Date",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Purchase Order"),
|
|
|
|
"options": "Purchase Order",
|
|
|
|
"fieldname": "purchase_order",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Supplier"),
|
|
|
|
"options": "Supplier",
|
|
|
|
"fieldname": "supplier",
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Estimated Cost"),
|
|
|
|
"fieldname": "estimated_cost",
|
|
|
|
"fieldtype": "Float",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Actual Cost"),
|
|
|
|
"fieldname": "actual_cost",
|
|
|
|
"fieldtype": "Float",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Purchase Order Amount"),
|
2019-04-21 23:44:21 +05:30
|
|
|
"fieldname": "purchase_order_amt",
|
2019-03-29 20:44:23 +05:30
|
|
|
"fieldtype": "Float",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
2019-04-21 23:44:21 +05:30
|
|
|
"label": _("Purchase Order Amount(Company Currency)"),
|
|
|
|
"fieldname": "purchase_order_amt_usd",
|
2019-03-29 20:44:23 +05:30
|
|
|
"fieldtype": "Float",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Expected Delivery Date"),
|
|
|
|
"fieldname": "expected_delivery_date",
|
|
|
|
"fieldtype": "Date",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Actual Delivery Date"),
|
|
|
|
"fieldname": "actual_delivery_date",
|
|
|
|
"fieldtype": "Date",
|
|
|
|
"width": 140
|
|
|
|
},
|
|
|
|
]
|
|
|
|
return columns
|
|
|
|
|
2019-04-28 17:11:02 +05:30
|
|
|
def get_conditions(filters):
|
|
|
|
conditions = ""
|
|
|
|
|
|
|
|
if filters.get("company"):
|
|
|
|
conditions += " AND company='%s'"% filters.get('company')
|
2019-05-03 16:00:02 +05:30
|
|
|
|
2019-04-28 17:11:02 +05:30
|
|
|
if filters.get("cost_center") or filters.get("project"):
|
|
|
|
conditions += """
|
|
|
|
AND (cost_center='%s'
|
|
|
|
OR project='%s')
|
|
|
|
"""% (filters.get('cost_center'), filters.get('project'))
|
2019-05-03 16:00:02 +05:30
|
|
|
|
|
|
|
if filters.get("from_date"):
|
2019-07-22 11:16:40 +05:30
|
|
|
conditions += " AND transaction_date>=%s"% filters.get('from_date')
|
2019-05-03 16:00:02 +05:30
|
|
|
|
|
|
|
if filters.get("to_date"):
|
2019-07-22 11:16:40 +05:30
|
|
|
conditions += " AND transaction_date<=%s"% filters.get('to_date')
|
2019-04-28 17:11:02 +05:30
|
|
|
return conditions
|
|
|
|
|
|
|
|
def get_data(filters):
|
|
|
|
conditions = get_conditions(filters)
|
|
|
|
purchase_order_entry = get_po_entries(conditions)
|
|
|
|
mr_records, procurement_record_against_mr = get_mapped_mr_details(conditions)
|
2019-04-26 01:29:36 +05:30
|
|
|
pr_records = get_mapped_pr_records()
|
|
|
|
pi_records = get_mapped_pi_records()
|
2019-05-06 12:53:52 +05:30
|
|
|
print(pi_records)
|
2019-04-01 21:14:05 +05:30
|
|
|
|
2019-04-26 01:29:36 +05:30
|
|
|
procurement_record=[]
|
|
|
|
if procurement_record_against_mr:
|
|
|
|
procurement_record += procurement_record_against_mr
|
|
|
|
for po in purchase_order_entry:
|
|
|
|
# fetch material records linked to the purchase order item
|
|
|
|
mr_record = mr_records.get(po.material_request_item, [{}])[0]
|
|
|
|
procurement_detail = {
|
2019-04-28 17:11:02 +05:30
|
|
|
"material_request_date": mr_record.get('transaction_date'),
|
2019-04-26 01:29:36 +05:30
|
|
|
"cost_center": po.cost_center,
|
|
|
|
"project": po.project,
|
|
|
|
"requesting_site": po.warehouse,
|
|
|
|
"requestor": po.owner,
|
|
|
|
"material_request_no": po.material_request,
|
|
|
|
"description": po.description,
|
|
|
|
"quantity": po.qty,
|
|
|
|
"unit_of_measurement": po.stock_uom,
|
|
|
|
"status": po.status,
|
|
|
|
"purchase_order_date": po.transaction_date,
|
|
|
|
"purchase_order": po.parent,
|
|
|
|
"supplier": po.supplier,
|
|
|
|
"estimated_cost": mr_record.get('amount'),
|
2019-04-26 16:02:25 +05:30
|
|
|
"actual_cost": pi_records.get(po.name),
|
2019-04-26 01:29:36 +05:30
|
|
|
"purchase_order_amt": po.amount,
|
|
|
|
"purchase_order_amt_in_company_currency": po.base_amount,
|
|
|
|
"expected_delivery_date": po.schedule_date,
|
2019-04-26 16:02:25 +05:30
|
|
|
"actual_delivery_date": pr_records.get(po.name)
|
2019-04-26 01:29:36 +05:30
|
|
|
}
|
|
|
|
procurement_record.append(procurement_detail)
|
|
|
|
return procurement_record
|
|
|
|
|
2019-04-28 17:11:02 +05:30
|
|
|
def get_mapped_mr_details(conditions):
|
2019-04-26 01:29:36 +05:30
|
|
|
mr_records = {}
|
2019-04-21 23:44:21 +05:30
|
|
|
mr_details = frappe.db.sql("""
|
2019-04-01 21:14:05 +05:30
|
|
|
SELECT
|
2019-04-21 23:44:21 +05:30
|
|
|
mr.transaction_date,
|
2019-04-26 01:29:36 +05:30
|
|
|
mr.per_ordered,
|
2019-04-21 23:44:21 +05:30
|
|
|
mr_item.name,
|
|
|
|
mr_item.parent,
|
|
|
|
mr_item.amount
|
|
|
|
FROM `tabMaterial Request` mr, `tabMaterial Request Item` mr_item
|
2019-04-01 21:14:05 +05:30
|
|
|
WHERE
|
2019-04-26 01:29:36 +05:30
|
|
|
mr.per_ordered>=0
|
|
|
|
AND mr.name=mr_item.parent
|
|
|
|
AND mr.docstatus=1
|
2019-04-28 17:11:02 +05:30
|
|
|
{conditions}
|
|
|
|
""".format(conditions=conditions), as_dict=1) #nosec
|
2019-04-03 19:45:08 +05:30
|
|
|
|
2019-04-26 01:29:36 +05:30
|
|
|
procurement_record_against_mr = []
|
|
|
|
for record in mr_details:
|
|
|
|
if record.per_ordered:
|
|
|
|
mr_records.setdefault(record.name, []).append(frappe._dict(record))
|
|
|
|
else:
|
|
|
|
procurement_record_details = dict(
|
|
|
|
material_request_date=record.transaction_date,
|
|
|
|
material_request_no=record.parent,
|
|
|
|
estimated_cost=record.amount
|
|
|
|
)
|
2019-04-26 16:02:25 +05:30
|
|
|
procurement_record_against_mr.append(procurement_record_details)
|
2019-04-26 01:29:36 +05:30
|
|
|
return mr_records, procurement_record_against_mr
|
|
|
|
|
|
|
|
def get_mapped_pi_records():
|
|
|
|
return frappe._dict(frappe.db.sql("""
|
2019-04-03 19:45:08 +05:30
|
|
|
SELECT
|
2019-05-03 16:41:04 +05:30
|
|
|
pi_item.po_detail,
|
|
|
|
pi_item.base_amount
|
2019-05-06 12:53:52 +05:30
|
|
|
FROM `tabPurchase Invoice Item` as pi_item
|
|
|
|
INNER JOIN `tabPurchase Order` as po
|
|
|
|
ON pi_item.`purchase_order` = po.`name`
|
2019-04-03 19:45:08 +05:30
|
|
|
WHERE
|
2019-05-06 12:53:52 +05:30
|
|
|
pi_item.docstatus = 1
|
|
|
|
AND po.status not in ("Closed","Completed","Cancelled")
|
2019-05-03 16:41:04 +05:30
|
|
|
AND pi_item.po_detail IS NOT NULL
|
2019-04-03 19:45:08 +05:30
|
|
|
"""))
|
|
|
|
|
2019-04-26 01:29:36 +05:30
|
|
|
def get_mapped_pr_records():
|
|
|
|
return frappe._dict(frappe.db.sql("""
|
2019-04-03 19:45:08 +05:30
|
|
|
SELECT
|
2019-04-21 23:44:21 +05:30
|
|
|
pr_item.purchase_order_item,
|
|
|
|
pr.posting_date
|
|
|
|
FROM `tabPurchase Receipt` pr, `tabPurchase Receipt Item` pr_item
|
2019-04-03 19:45:08 +05:30
|
|
|
WHERE
|
2019-04-21 23:44:21 +05:30
|
|
|
pr.docstatus=1
|
|
|
|
AND pr.name=pr_item.parent
|
|
|
|
AND pr_item.purchase_order_item IS NOT NULL
|
2019-05-03 16:41:04 +05:30
|
|
|
AND pr.status not in ("Closed","Completed","Cancelled")
|
2019-04-21 23:44:21 +05:30
|
|
|
"""))
|
|
|
|
|
2019-04-28 17:11:02 +05:30
|
|
|
def get_po_entries(conditions):
|
2019-04-26 01:29:36 +05:30
|
|
|
return frappe.db.sql("""
|
|
|
|
SELECT
|
|
|
|
po_item.name,
|
|
|
|
po_item.parent,
|
|
|
|
po_item.cost_center,
|
|
|
|
po_item.project,
|
|
|
|
po_item.warehouse,
|
|
|
|
po_item.material_request,
|
|
|
|
po_item.material_request_item,
|
|
|
|
po_item.description,
|
|
|
|
po_item.stock_uom,
|
|
|
|
po_item.qty,
|
|
|
|
po_item.amount,
|
|
|
|
po_item.base_amount,
|
|
|
|
po_item.schedule_date,
|
|
|
|
po.transaction_date,
|
|
|
|
po.supplier,
|
|
|
|
po.status,
|
|
|
|
po.owner
|
|
|
|
FROM `tabPurchase Order` po, `tabPurchase Order Item` po_item
|
|
|
|
WHERE
|
|
|
|
po.docstatus = 1
|
|
|
|
AND po.name = po_item.parent
|
|
|
|
AND po.status not in ("Closed","Completed","Cancelled")
|
2019-04-28 17:11:02 +05:30
|
|
|
{conditions}
|
2019-04-26 01:29:36 +05:30
|
|
|
GROUP BY
|
|
|
|
po.name,po_item.item_code
|
2019-04-28 17:11:02 +05:30
|
|
|
""".format(conditions=conditions), as_dict=1) #nosec
|