blob: b5598f8d0b2cc62f9e7d7659d1d2aaddbacd7198 [file] [log] [blame]
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +05301# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
2# License: GNU General Public License v3. See license.txt
3
4from __future__ import unicode_literals
5import frappe
6from frappe.utils import flt, cstr, cint
7from frappe import _
mbauskarc482aed2017-05-02 12:53:12 +05308import json
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +05309
10from erpnext.stock.doctype.item.item import get_last_purchase_details
11from erpnext.stock.doctype.item.item import validate_end_of_life
12
13def update_last_purchase_rate(doc, is_submit):
14 """updates last_purchase_rate in item table for each item"""
15
16 import frappe.utils
17 this_purchase_date = frappe.utils.getdate(doc.get('posting_date') or doc.get('transaction_date'))
18
19 for d in doc.get("items"):
20 # get last purchase details
21 last_purchase_details = get_last_purchase_details(d.item_code, doc.name)
22
23 # compare last purchase date and this transaction's date
24 last_purchase_rate = None
25 if last_purchase_details and \
26 (last_purchase_details.purchase_date > this_purchase_date):
Saqib290253f2019-11-22 12:12:29 +053027 last_purchase_rate = last_purchase_details['base_net_rate']
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +053028 elif is_submit == 1:
29 # even if this transaction is the latest one, it should be submitted
30 # for it to be considered for latest purchase rate
31 if flt(d.conversion_factor):
Saqib290253f2019-11-22 12:12:29 +053032 last_purchase_rate = flt(d.base_net_rate) / flt(d.conversion_factor)
Deepesh Garg8b0302b2019-08-05 10:14:19 +053033 # Check if item code is present
34 # Conversion factor should not be mandatory for non itemized items
35 elif d.item_code:
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +053036 frappe.throw(_("UOM Conversion factor is required in row {0}").format(d.idx))
37
38 # update last purchsae rate
39 if last_purchase_rate:
40 frappe.db.sql("""update `tabItem` set last_purchase_rate = %s where name = %s""",
41 (flt(last_purchase_rate), d.item_code))
42
43def validate_for_items(doc):
44 items = []
45 for d in doc.get("items"):
46 if not d.qty:
47 if doc.doctype == "Purchase Receipt" and d.rejected_qty:
48 continue
49 frappe.throw(_("Please enter quantity for Item {0}").format(d.item_code))
50
51 # update with latest quantities
52 bin = frappe.db.sql("""select projected_qty from `tabBin` where
53 item_code = %s and warehouse = %s""", (d.item_code, d.warehouse), as_dict=1)
54
55 f_lst ={'projected_qty': bin and flt(bin[0]['projected_qty']) or 0, 'ordered_qty': 0, 'received_qty' : 0}
56 if d.doctype in ('Purchase Receipt Item', 'Purchase Invoice Item'):
57 f_lst.pop('received_qty')
58 for x in f_lst :
59 if d.meta.get_field(x):
60 d.set(x, f_lst[x])
61
62 item = frappe.db.sql("""select is_stock_item,
63 is_sub_contracted_item, end_of_life, disabled from `tabItem` where name=%s""",
64 d.item_code, as_dict=1)[0]
65
66 validate_end_of_life(d.item_code, item.end_of_life, item.disabled)
67
68 # validate stock item
Makarand Bauskarc4ec9372017-05-05 11:52:17 +053069 if item.is_stock_item==1 and d.qty and not d.warehouse and not d.get("delivered_by_supplier"):
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +053070 frappe.throw(_("Warehouse is mandatory for stock Item {0} in row {1}").format(d.item_code, d.idx))
71
72 items.append(cstr(d.item_code))
73
74 if items and len(items) != len(set(items)) and \
75 not cint(frappe.db.get_single_value("Buying Settings", "allow_multiple_items") or 0):
76 frappe.throw(_("Same item cannot be entered multiple times."))
77
Mangesh-Khairnara1afd782019-03-11 17:15:05 +053078def check_on_hold_or_closed_status(doctype, docname):
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +053079 status = frappe.db.get_value(doctype, docname, "status")
80
Mangesh-Khairnara1afd782019-03-11 17:15:05 +053081 if status in ("Closed", "On Hold"):
Rushabh Mehtacc8b2b22017-03-31 12:44:29 +053082 frappe.throw(_("{0} {1} status is {2}").format(doctype, docname, status), frappe.InvalidStatusError)
83
mbauskarc482aed2017-05-02 12:53:12 +053084@frappe.whitelist()
85def get_linked_material_requests(items):
86 items = json.loads(items)
87 mr_list = []
88 for item in items:
Deepesh Garg8b0302b2019-08-05 10:14:19 +053089 material_request = frappe.db.sql("""SELECT distinct mr.name AS mr_name,
90 (mr_item.qty - mr_item.ordered_qty) AS qty,
mbauskarc482aed2017-05-02 12:53:12 +053091 mr_item.item_code AS item_code,
Deepesh Garg8b0302b2019-08-05 10:14:19 +053092 mr_item.name AS mr_item
mbauskarc482aed2017-05-02 12:53:12 +053093 FROM `tabMaterial Request` mr, `tabMaterial Request Item` mr_item
94 WHERE mr.name = mr_item.parent
Deepesh Garg8b0302b2019-08-05 10:14:19 +053095 AND mr_item.item_code = %(item)s
mbauskarc482aed2017-05-02 12:53:12 +053096 AND mr.material_request_type = 'Purchase'
97 AND mr.per_ordered < 99.99
98 AND mr.docstatus = 1
99 AND mr.status != 'Stopped'
100 ORDER BY mr_item.item_code ASC""",{"item": item}, as_dict=1)
101 if material_request:
102 mr_list.append(material_request)
Deepesh Garg8b0302b2019-08-05 10:14:19 +0530103
mbauskarc482aed2017-05-02 12:53:12 +0530104 return mr_list
105