2015-03-03 14:55:30 +05:30
# Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
2013-08-05 14:59:54 +05:30
# License: GNU General Public License v3. See license.txt
2013-03-19 12:01:24 +05:30
from __future__ import unicode_literals
2017-06-19 12:54:59 +05:30
import frappe , erpnext
2020-04-28 13:01:43 +05:30
from frappe . utils import cint , flt , cstr , get_link_to_form , today , getdate
2019-01-23 00:28:37 +05:30
from frappe import _
2014-02-14 15:47:51 +05:30
import frappe . defaults
2020-12-07 21:35:49 +05:30
from collections import defaultdict
2020-12-25 18:12:35 +05:30
from erpnext . accounts . utils import get_fiscal_year , check_if_stock_and_account_balance_synced
2020-04-30 10:38:58 +05:30
from erpnext . accounts . general_ledger import make_gl_entries , make_reverse_gl_entries , process_gl_map
2015-08-24 14:32:38 +05:30
from erpnext . controllers . accounts_controller import AccountsController
2017-01-25 18:47:53 +05:30
from erpnext . stock . stock_ledger import get_valuation_rate
2017-06-15 11:09:27 +05:30
from erpnext . stock import get_warehouse_account_map
2013-03-19 12:01:24 +05:30
2018-12-24 14:54:42 +05:30
class QualityInspectionRequiredError ( frappe . ValidationError ) : pass
class QualityInspectionRejectedError ( frappe . ValidationError ) : pass
2018-12-28 16:53:00 +05:30
class QualityInspectionNotSubmittedError ( frappe . ValidationError ) : pass
2018-12-24 14:54:42 +05:30
2013-03-19 12:01:24 +05:30
class StockController ( AccountsController ) :
2016-11-16 17:21:59 +05:30
def validate ( self ) :
super ( StockController , self ) . validate ( )
2020-06-12 12:30:59 +05:30
if not self . get ( ' is_return ' ) :
2020-06-11 16:39:03 +05:30
self . validate_inspection ( )
2019-12-30 13:26:47 +05:30
self . validate_serialized_batch ( )
2020-04-03 15:46:48 +05:30
self . validate_customer_provided_item ( )
2021-01-28 13:09:56 +05:30
self . validate_internal_transfer ( )
2020-12-07 21:35:49 +05:30
self . validate_putaway_capacity ( )
2017-01-16 17:23:20 +05:30
2020-12-21 14:45:50 +05:30
def make_gl_entries ( self , gl_entries = None , from_repost = False ) :
2014-03-28 13:55:00 +05:30
if self . docstatus == 2 :
2020-04-30 10:38:58 +05:30
make_reverse_gl_entries ( voucher_type = self . doctype , voucher_no = self . name )
2014-04-07 18:51:58 +05:30
2017-06-19 12:54:59 +05:30
if cint ( erpnext . is_perpetual_inventory_enabled ( self . company ) ) :
2019-03-08 11:13:35 +05:30
warehouse_account = get_warehouse_account_map ( self . company )
2014-04-07 18:51:58 +05:30
2014-03-28 13:55:00 +05:30
if self . docstatus == 1 :
2016-12-30 16:21:35 +05:30
if not gl_entries :
gl_entries = self . get_gl_entries ( warehouse_account )
2020-12-21 14:45:50 +05:30
make_gl_entries ( gl_entries , from_repost = from_repost )
2013-10-22 23:51:41 +05:30
2018-06-12 13:54:40 +05:30
elif self . doctype in [ ' Purchase Receipt ' , ' Purchase Invoice ' ] and self . docstatus == 1 :
2018-05-16 18:16:08 +05:30
gl_entries = [ ]
gl_entries = self . get_asset_gl_entry ( gl_entries )
2020-12-21 14:45:50 +05:30
make_gl_entries ( gl_entries , from_repost = from_repost )
2014-04-07 18:51:58 +05:30
2019-12-30 13:26:47 +05:30
def validate_serialized_batch ( self ) :
from erpnext . stock . doctype . serial_no . serial_no import get_serial_nos
for d in self . get ( " items " ) :
if hasattr ( d , ' serial_no ' ) and hasattr ( d , ' batch_no ' ) and d . serial_no and d . batch_no :
serial_nos = get_serial_nos ( d . serial_no )
for serial_no_data in frappe . get_all ( " Serial No " ,
filters = { " name " : ( " in " , serial_nos ) } , fields = [ " batch_no " , " name " ] ) :
if serial_no_data . batch_no != d . batch_no :
frappe . throw ( _ ( " Row # {0} : Serial No {1} does not belong to Batch {2} " )
. format ( d . idx , serial_no_data . name , d . batch_no ) )
2020-10-20 11:59:06 +05:30
if flt ( d . qty ) > 0.0 and d . get ( " batch_no " ) and self . get ( " posting_date " ) and self . docstatus < 2 :
2020-04-28 13:01:43 +05:30
expiry_date = frappe . get_cached_value ( " Batch " , d . get ( " batch_no " ) , " expiry_date " )
if expiry_date and getdate ( expiry_date ) < getdate ( self . posting_date ) :
frappe . throw ( _ ( " Row # {0} : The batch {1} has already expired. " )
. format ( d . idx , get_link_to_form ( " Batch " , d . get ( " batch_no " ) ) ) )
2013-11-14 18:40:08 +05:30
def get_gl_entries ( self , warehouse_account = None , default_expense_account = None ,
2013-09-17 15:15:16 +05:30
default_cost_center = None ) :
2014-10-06 11:53:52 +05:30
2013-09-17 15:15:16 +05:30
if not warehouse_account :
2019-03-08 11:13:35 +05:30
warehouse_account = get_warehouse_account_map ( self . company )
2014-04-07 18:51:58 +05:30
2014-04-17 11:37:46 +05:30
sle_map = self . get_stock_ledger_details ( )
voucher_details = self . get_voucher_details ( default_expense_account , default_cost_center , sle_map )
2014-04-07 18:51:58 +05:30
2013-08-28 18:53:11 +05:30
gl_list = [ ]
2013-09-17 10:21:20 +05:30
warehouse_with_no_account = [ ]
2020-01-15 16:35:31 +05:30
precision = frappe . get_precision ( " GL Entry " , " debit_in_account_currency " )
2016-12-15 13:46:03 +05:30
for item_row in voucher_details :
2021-01-28 13:09:56 +05:30
2016-12-15 13:46:03 +05:30
sle_list = sle_map . get ( item_row . name )
2013-08-28 18:53:11 +05:30
if sle_list :
for sle in sle_list :
if warehouse_account . get ( sle . warehouse ) :
2020-12-11 21:30:39 +05:30
# from warehouse account
2017-01-16 17:23:20 +05:30
2016-12-15 13:46:03 +05:30
self . check_expense_account ( item_row )
2017-01-16 17:23:20 +05:30
2017-04-14 18:24:04 +08:00
# If the item does not have the allow zero valuation rate flag set
2017-02-06 17:13:39 +05:30
# and ( valuation rate not mentioned in an incoming entry
2017-03-31 12:44:29 +05:30
# or incoming entry not found while delivering the item),
2017-02-06 17:13:39 +05:30
# try to pick valuation rate from previous sle or Item master and update in SLE
# Otherwise, throw an exception
2017-02-07 01:23:26 +05:30
if not sle . stock_value_difference and self . doctype != " Stock Reconciliation " \
2017-04-14 18:24:04 +08:00
and not item_row . get ( " allow_zero_valuation_rate " ) :
2017-02-07 01:23:26 +05:30
2017-02-06 17:13:39 +05:30
sle = self . update_stock_ledger_entries ( sle )
2015-10-15 12:28:20 +05:30
2020-12-11 21:30:39 +05:30
# expense account/ target_warehouse / source_warehouse
if item_row . get ( ' target_warehouse ' ) :
warehouse = item_row . get ( ' target_warehouse ' )
expense_account = warehouse_account [ warehouse ] [ " account " ]
else :
expense_account = item_row . expense_account
2013-08-28 18:53:11 +05:30
gl_list . append ( self . get_gl_dict ( {
2017-06-15 11:09:27 +05:30
" account " : warehouse_account [ sle . warehouse ] [ " account " ] ,
2020-12-11 21:30:39 +05:30
" against " : expense_account ,
2016-12-15 13:46:03 +05:30
" cost_center " : item_row . cost_center ,
2020-06-19 12:12:08 +05:30
" project " : item_row . project or self . get ( ' project ' ) ,
2014-04-07 12:02:57 +05:30
" remarks " : self . get ( " remarks " ) or " Accounting Entry for Stock " ,
2020-01-15 16:35:31 +05:30
" debit " : flt ( sle . stock_value_difference , precision ) ,
2019-05-22 18:03:09 +05:30
" is_opening " : item_row . get ( " is_opening " ) or self . get ( " is_opening " ) or " No " ,
2019-05-19 00:02:01 +05:30
} , warehouse_account [ sle . warehouse ] [ " account_currency " ] , item = item_row ) )
2013-08-28 18:53:11 +05:30
gl_list . append ( self . get_gl_dict ( {
2020-12-11 21:30:39 +05:30
" account " : expense_account ,
2017-06-15 11:09:27 +05:30
" against " : warehouse_account [ sle . warehouse ] [ " account " ] ,
2016-12-15 13:46:03 +05:30
" cost_center " : item_row . cost_center ,
2020-06-19 12:12:08 +05:30
" project " : item_row . project or self . get ( ' project ' ) ,
2014-04-07 12:02:57 +05:30
" remarks " : self . get ( " remarks " ) or " Accounting Entry for Stock " ,
2020-01-15 16:35:31 +05:30
" credit " : flt ( sle . stock_value_difference , precision ) ,
2019-05-16 17:28:39 +05:30
" project " : item_row . get ( " project " ) or self . get ( " project " ) ,
2019-05-22 18:03:09 +05:30
" is_opening " : item_row . get ( " is_opening " ) or self . get ( " is_opening " ) or " No "
2019-05-19 00:02:01 +05:30
} , item = item_row ) )
2013-09-17 10:21:20 +05:30
elif sle . warehouse not in warehouse_with_no_account :
warehouse_with_no_account . append ( sle . warehouse )
2014-04-07 18:51:58 +05:30
if warehouse_with_no_account :
2016-10-24 18:17:57 +05:30
for wh in warehouse_with_no_account :
if frappe . db . get_value ( " Warehouse " , wh , " company " ) :
2020-12-21 14:45:50 +05:30
frappe . throw ( _ ( " Warehouse {0} is not linked to any account, please mention the account in the warehouse record or set default inventory account in company {1} . " ) . format ( wh , self . company ) )
2015-10-15 12:28:20 +05:30
2013-08-28 18:53:11 +05:30
return process_gl_map ( gl_list )
2017-01-16 17:23:20 +05:30
2017-01-25 18:47:53 +05:30
def update_stock_ledger_entries ( self , sle ) :
2017-03-31 12:44:29 +05:30
sle . valuation_rate = get_valuation_rate ( sle . item_code , sle . warehouse ,
2017-06-16 15:21:36 +05:30
self . doctype , self . name , currency = self . company_currency , company = self . company )
2017-02-06 17:13:39 +05:30
2017-01-25 19:20:35 +05:30
sle . stock_value = flt ( sle . qty_after_transaction ) * flt ( sle . valuation_rate )
2017-02-06 17:13:39 +05:30
sle . stock_value_difference = flt ( sle . actual_qty ) * flt ( sle . valuation_rate )
2017-03-31 12:44:29 +05:30
2017-01-28 13:18:49 +05:30
if sle . name :
2017-02-06 17:13:39 +05:30
frappe . db . sql ( """
2017-03-31 12:44:29 +05:30
update
` tabStock Ledger Entry `
set
2017-02-06 17:13:39 +05:30
stock_value = % ( stock_value ) s ,
2017-03-31 12:44:29 +05:30
valuation_rate = % ( valuation_rate ) s ,
stock_value_difference = % ( stock_value_difference ) s
where
2017-02-06 17:13:39 +05:30
name = % ( name ) s """ , (sle))
2017-03-31 12:44:29 +05:30
2017-02-06 17:13:39 +05:30
return sle
2017-03-31 12:44:29 +05:30
2014-04-17 11:37:46 +05:30
def get_voucher_details ( self , default_expense_account , default_cost_center , sle_map ) :
if self . doctype == " Stock Reconciliation " :
2019-05-16 17:28:39 +05:30
reconciliation_purpose = frappe . db . get_value ( self . doctype , self . name , " purpose " )
is_opening = " Yes " if reconciliation_purpose == " Opening Stock " else " No "
details = [ ]
2019-07-03 10:34:31 +05:30
for voucher_detail_no in sle_map :
2019-05-16 17:28:39 +05:30
details . append ( frappe . _dict ( {
" name " : voucher_detail_no ,
" expense_account " : default_expense_account ,
" cost_center " : default_cost_center ,
" is_opening " : is_opening
} ) )
return details
2014-04-17 11:37:46 +05:30
else :
2014-12-26 13:15:21 +05:30
details = self . get ( " items " )
2014-04-17 11:37:46 +05:30
if default_expense_account or default_cost_center :
for d in details :
if default_expense_account and not d . get ( " expense_account " ) :
d . expense_account = default_expense_account
if default_cost_center and not d . get ( " cost_center " ) :
d . cost_center = default_cost_center
return details
2014-04-07 18:51:58 +05:30
2014-09-26 14:22:18 +05:30
def get_items_and_warehouses ( self ) :
2014-03-27 17:18:29 +05:30
items , warehouses = [ ] , [ ]
2014-04-07 18:51:58 +05:30
2014-12-26 13:15:21 +05:30
if hasattr ( self , " items " ) :
item_doclist = self . get ( " items " )
2014-03-28 13:55:00 +05:30
elif self . doctype == " Stock Reconciliation " :
2014-03-27 17:18:29 +05:30
import json
item_doclist = [ ]
2014-03-28 13:55:00 +05:30
data = json . loads ( self . reconciliation_json )
2014-03-27 17:18:29 +05:30
for row in data [ data . index ( self . head_row ) + 1 : ] :
d = frappe . _dict ( zip ( [ " item_code " , " warehouse " , " qty " , " valuation_rate " ] , row ) )
item_doclist . append ( d )
2014-04-07 18:51:58 +05:30
2014-03-27 17:18:29 +05:30
if item_doclist :
for d in item_doclist :
if d . item_code and d . item_code not in items :
items . append ( d . item_code )
2014-04-07 18:51:58 +05:30
2014-04-07 12:02:57 +05:30
if d . get ( " warehouse " ) and d . warehouse not in warehouses :
2014-03-27 17:18:29 +05:30
warehouses . append ( d . warehouse )
2014-04-07 18:51:58 +05:30
2014-04-07 12:02:57 +05:30
if self . doctype == " Stock Entry " :
if d . get ( " s_warehouse " ) and d . s_warehouse not in warehouses :
warehouses . append ( d . s_warehouse )
if d . get ( " t_warehouse " ) and d . t_warehouse not in warehouses :
warehouses . append ( d . t_warehouse )
2014-03-27 17:18:29 +05:30
2014-09-26 14:22:18 +05:30
return items , warehouses
2014-04-07 18:51:58 +05:30
2013-08-28 18:53:11 +05:30
def get_stock_ledger_details ( self ) :
stock_ledger = { }
2017-02-06 17:13:39 +05:30
stock_ledger_entries = frappe . db . sql ( """
2017-03-31 12:44:29 +05:30
select
2017-02-07 01:23:26 +05:30
name , warehouse , stock_value_difference , valuation_rate ,
2017-03-31 12:44:29 +05:30
voucher_detail_no , item_code , posting_date , posting_time ,
2017-02-07 01:23:26 +05:30
actual_qty , qty_after_transaction
2017-02-06 17:13:39 +05:30
from
` tabStock Ledger Entry `
where
voucher_type = % s and voucher_no = % s
""" , (self.doctype, self.name), as_dict=True)
2017-02-07 01:23:26 +05:30
2017-02-06 17:13:39 +05:30
for sle in stock_ledger_entries :
2021-01-28 13:09:56 +05:30
stock_ledger . setdefault ( sle . voucher_detail_no , [ ] ) . append ( sle )
2013-08-28 18:53:11 +05:30
return stock_ledger
2014-04-07 18:51:58 +05:30
2017-04-21 12:40:19 +05:30
def make_batches ( self , warehouse_field ) :
2017-04-20 15:21:01 +05:30
''' Create batches if required. Called before submit '''
for d in self . items :
2017-04-21 12:40:19 +05:30
if d . get ( warehouse_field ) and not d . batch_no :
has_batch_no , create_new_batch = frappe . db . get_value ( ' Item ' , d . item_code , [ ' has_batch_no ' , ' create_new_batch ' ] )
if has_batch_no and create_new_batch :
d . batch_no = frappe . get_doc ( dict (
doctype = ' Batch ' ,
item = d . item_code ,
supplier = getattr ( self , ' supplier ' , None ) ,
reference_doctype = self . doctype ,
reference_name = self . name ) ) . insert ( ) . name
2017-04-20 15:21:01 +05:30
2013-08-26 16:53:30 +05:30
def check_expense_account ( self , item ) :
2014-04-16 19:20:11 +05:30
if not item . get ( " expense_account " ) :
2020-11-18 17:57:35 +05:30
msg = _ ( " Please set an Expense Account in the Items table " )
frappe . throw ( _ ( " Row # {0} : Expense Account not set for the Item {1} . {2} " )
. format ( item . idx , frappe . bold ( item . item_code ) , msg ) , title = _ ( " Expense Account Missing " ) )
2014-04-07 18:51:58 +05:30
2014-06-19 19:25:19 +05:30
else :
2014-06-25 19:12:24 +05:30
is_expense_account = frappe . db . get_value ( " Account " ,
item . get ( " expense_account " ) , " report_type " ) == " Profit and Loss "
2016-02-08 23:44:55 +07:00
if self . doctype not in ( " Purchase Receipt " , " Purchase Invoice " , " Stock Reconciliation " , " Stock Entry " ) and not is_expense_account :
2014-06-25 19:12:24 +05:30
frappe . throw ( _ ( " Expense / Difference account ( {0} ) must be a ' Profit or Loss ' account " )
. format ( item . get ( " expense_account " ) ) )
2014-06-19 19:25:19 +05:30
if is_expense_account and not item . get ( " cost_center " ) :
frappe . throw ( _ ( " {0} {1} : Cost Center is mandatory for Item {2} " ) . format (
_ ( self . doctype ) , self . name , item . get ( " item_code " ) ) )
2014-04-07 18:51:58 +05:30
2020-01-23 17:36:52 +05:30
def delete_auto_created_batches ( self ) :
for d in self . items :
if not d . batch_no : continue
2020-11-18 17:57:35 +05:30
serial_nos = [ sr . name for sr in frappe . get_all ( " Serial No " ,
{ ' batch_no ' : d . batch_no , ' status ' : ' Inactive ' } ) ]
2020-03-02 15:02:58 +05:30
if serial_nos :
frappe . db . set_value ( " Serial No " , { ' name ' : [ ' in ' , serial_nos ] } , " batch_no " , None )
2020-01-23 17:36:52 +05:30
d . batch_no = None
d . db_set ( " batch_no " , None )
for data in frappe . get_all ( " Batch " ,
{ ' reference_name ' : self . name , ' reference_doctype ' : self . doctype } ) :
frappe . delete_doc ( " Batch " , data . name )
2020-01-23 12:42:42 +05:30
2014-04-07 18:51:58 +05:30
def get_sl_entries ( self , d , args ) :
2014-11-03 15:08:21 +05:30
sl_dict = frappe . _dict ( {
2014-05-02 17:36:13 +05:30
" item_code " : d . get ( " item_code " , None ) ,
2014-04-07 12:02:57 +05:30
" warehouse " : d . get ( " warehouse " , None ) ,
2014-03-28 13:55:00 +05:30
" posting_date " : self . posting_date ,
" posting_time " : self . posting_time ,
2016-03-04 14:33:49 +05:30
' fiscal_year ' : get_fiscal_year ( self . posting_date , company = self . company ) [ 0 ] ,
2014-03-28 13:55:00 +05:30
" voucher_type " : self . doctype ,
" voucher_no " : self . name ,
2013-08-02 11:42:11 +05:30
" voucher_detail_no " : d . name ,
2014-04-07 12:02:57 +05:30
" actual_qty " : ( self . docstatus == 1 and 1 or - 1 ) * flt ( d . get ( " stock_qty " ) ) ,
2015-09-21 09:18:43 +05:30
" stock_uom " : frappe . db . get_value ( " Item " , args . get ( " item_code " ) or d . get ( " item_code " ) , " stock_uom " ) ,
2013-08-02 11:42:11 +05:30
" incoming_rate " : 0 ,
2014-03-28 13:55:00 +05:30
" company " : self . company ,
2014-05-02 17:36:13 +05:30
" batch_no " : cstr ( d . get ( " batch_no " ) ) . strip ( ) ,
" serial_no " : d . get ( " serial_no " ) ,
2018-03-21 17:52:41 +05:30
" project " : d . get ( " project " ) or self . get ( ' project ' ) ,
2020-04-30 10:38:58 +05:30
" is_cancelled " : 1 if self . docstatus == 2 else 0
2014-11-03 15:08:21 +05:30
} )
2014-04-07 18:51:58 +05:30
2013-08-02 11:42:11 +05:30
sl_dict . update ( args )
return sl_dict
2014-04-07 18:51:58 +05:30
2020-04-30 10:38:58 +05:30
def make_sl_entries ( self , sl_entries , allow_negative_stock = False ,
2015-03-27 15:38:31 +05:30
via_landed_cost_voucher = False ) :
2013-12-12 19:12:19 +05:30
from erpnext . stock . stock_ledger import make_sl_entries
2020-04-30 10:38:58 +05:30
make_sl_entries ( sl_entries , allow_negative_stock , via_landed_cost_voucher )
2014-04-07 18:51:58 +05:30
2020-04-30 10:38:58 +05:30
def make_gl_entries_on_cancel ( self ) :
2014-04-07 18:51:58 +05:30
if frappe . db . sql ( """ select name from `tabGL Entry` where voucher_type= %s
2014-03-28 13:55:00 +05:30
and voucher_no = % s """ , (self.doctype, self.name)):
2020-04-30 10:38:58 +05:30
self . make_gl_entries ( )
2014-04-07 18:51:58 +05:30
2014-06-25 13:31:02 +05:30
def get_serialized_items ( self ) :
serialized_items = [ ]
2014-12-26 13:15:21 +05:30
item_codes = list ( set ( [ d . item_code for d in self . get ( " items " ) ] ) )
2014-06-25 13:31:02 +05:30
if item_codes :
serialized_items = frappe . db . sql_list ( """ select name from `tabItem`
2015-07-24 15:16:25 +05:30
where has_serial_no = 1 and name in ( { } ) """ .format( " , " .join([ " %s " ]*len(item_codes))),
2014-06-25 13:31:02 +05:30
tuple ( item_codes ) )
return serialized_items
2015-08-03 16:13:33 +05:30
2015-11-18 17:03:33 +05:30
def validate_warehouse ( self ) :
from erpnext . stock . utils import validate_warehouse_company
warehouses = list ( set ( [ d . warehouse for d in
self . get ( " items " ) if getattr ( d , " warehouse " , None ) ] ) )
2020-02-18 12:28:41 +05:30
target_warehouses = list ( set ( [ d . target_warehouse for d in
self . get ( " items " ) if getattr ( d , " target_warehouse " , None ) ] ) )
warehouses . extend ( target_warehouses )
from_warehouse = list ( set ( [ d . from_warehouse for d in
self . get ( " items " ) if getattr ( d , " from_warehouse " , None ) ] ) )
warehouses . extend ( from_warehouse )
2015-11-18 17:03:33 +05:30
for w in warehouses :
validate_warehouse_company ( w , self . company )
2017-01-16 17:23:20 +05:30
2016-01-06 16:32:06 +05:30
def update_billing_percentage ( self , update_modified = True ) :
2020-11-02 15:07:48 +05:30
target_ref_field = " amount "
if self . doctype == " Delivery Note " :
target_ref_field = " amount - (returned_qty * rate) "
2015-12-30 19:08:11 +05:30
self . _update_percent_field ( {
" target_dt " : self . doctype + " Item " ,
" target_parent_dt " : self . doctype ,
" target_parent_field " : " per_billed " ,
2020-11-02 15:07:48 +05:30
" target_ref_field " : target_ref_field ,
2015-12-30 19:08:11 +05:30
" target_field " : " billed_amt " ,
" name " : self . name ,
2016-01-06 16:32:06 +05:30
} , update_modified )
2014-06-25 13:31:02 +05:30
2016-11-16 17:21:59 +05:30
def validate_inspection ( self ) :
2016-11-10 19:15:11 +05:30
''' Checks if quality inspection is set for Items that require inspection.
On submit , throw an exception '''
2016-11-18 14:52:13 +05:30
inspection_required_fieldname = None
2016-11-16 17:21:59 +05:30
if self . doctype in [ " Purchase Receipt " , " Purchase Invoice " ] :
inspection_required_fieldname = " inspection_required_before_purchase "
elif self . doctype in [ " Delivery Note " , " Sales Invoice " ] :
inspection_required_fieldname = " inspection_required_before_delivery "
2017-01-16 17:23:20 +05:30
2018-02-22 11:03:48 +05:30
if ( ( not inspection_required_fieldname and self . doctype != " Stock Entry " ) or
( self . doctype == " Stock Entry " and not self . inspection_required ) or
( self . doctype in [ " Sales Invoice " , " Purchase Invoice " ] and not self . update_stock ) ) :
2016-11-16 17:21:59 +05:30
return
2017-01-16 17:23:20 +05:30
2016-11-10 19:15:11 +05:30
for d in self . get ( ' items ' ) :
2018-12-24 14:54:42 +05:30
qa_required = False
2018-02-22 11:03:48 +05:30
if ( inspection_required_fieldname and not d . quality_inspection and
frappe . db . get_value ( " Item " , d . item_code , inspection_required_fieldname ) ) :
2018-12-24 14:54:42 +05:30
qa_required = True
2018-02-22 11:03:48 +05:30
elif self . doctype == " Stock Entry " and not d . quality_inspection and d . t_warehouse :
2018-12-24 14:54:42 +05:30
qa_required = True
2018-12-28 16:53:00 +05:30
if self . docstatus == 1 and d . quality_inspection :
qa_doc = frappe . get_doc ( " Quality Inspection " , d . quality_inspection )
if qa_doc . docstatus == 0 :
link = frappe . utils . get_link_to_form ( ' Quality Inspection ' , d . quality_inspection )
2019-01-01 15:00:16 +05:30
frappe . throw ( _ ( " Quality Inspection: {0} is not submitted for the item: {1} in row {2} " ) . format ( link , d . item_code , d . idx ) , QualityInspectionNotSubmittedError )
2018-12-28 16:53:00 +05:30
qa_failed = any ( [ r . status == " Rejected " for r in qa_doc . readings ] )
if qa_failed :
frappe . throw ( _ ( " Row {0} : Quality Inspection rejected for item {1} " )
. format ( d . idx , d . item_code ) , QualityInspectionRejectedError )
elif qa_required :
2019-07-03 10:34:31 +05:30
action = frappe . get_doc ( ' Stock Settings ' ) . action_if_quality_inspection_is_not_submitted
if self . docstatus == 1 and action == ' Stop ' :
frappe . throw ( _ ( " Quality Inspection required for Item {0} to submit " ) . format ( frappe . bold ( d . item_code ) ) ,
exc = QualityInspectionRequiredError )
else :
frappe . msgprint ( _ ( " Create Quality Inspection for Item {0} " ) . format ( frappe . bold ( d . item_code ) ) )
2016-11-10 19:15:11 +05:30
2018-06-14 15:54:34 +05:30
def update_blanket_order ( self ) :
2018-06-14 17:09:55 +05:30
blanket_orders = list ( set ( [ d . blanket_order for d in self . items if d . blanket_order ] ) )
2018-06-14 15:54:34 +05:30
for blanket_order in blanket_orders :
frappe . get_doc ( " Blanket Order " , blanket_order ) . update_ordered_qty ( )
2018-05-28 20:07:08 +05:30
2020-04-03 15:46:48 +05:30
def validate_customer_provided_item ( self ) :
for d in self . get ( ' items ' ) :
# Customer Provided parts will have zero valuation rate
if frappe . db . get_value ( ' Item ' , d . item_code , ' is_customer_provided_item ' ) :
d . allow_zero_valuation_rate = 1
2014-04-07 18:51:58 +05:30
2021-01-28 13:09:56 +05:30
def validate_internal_transfer ( self ) :
if self . doctype in ( ' Sales Invoice ' , ' Delivery Note ' , ' Purchase Invoice ' , ' Purchase Receipt ' ) \
and self . is_internal_transfer ( ) :
self . validate_in_transit_warehouses ( )
self . validate_multi_currency ( )
self . validate_packed_items ( )
def validate_in_transit_warehouses ( self ) :
if ( self . doctype == ' Sales Invoice ' and self . get ( ' update_stock ' ) ) or self . doctype == ' Delivery Note ' :
for item in self . get ( ' items ' ) :
if not item . target_warehouse :
frappe . throw ( _ ( " Row {0} : Target Warehouse is mandatory for internal transfers " ) . format ( item . idx ) )
if ( self . doctype == ' Purchase Invoice ' and self . get ( ' update_stock ' ) ) or self . doctype == ' Purchase Receipt ' :
for item in self . get ( ' items ' ) :
if not item . from_warehouse :
frappe . throw ( _ ( " Row {0} : From Warehouse is mandatory for internal transfers " ) . format ( item . idx ) )
def validate_multi_currency ( self ) :
if self . currency != self . company_currency :
frappe . throw ( _ ( " Internal transfers can only be done in company ' s default currency " ) )
def validate_packed_items ( self ) :
if self . doctype in ( ' Sales Invoice ' , ' Delivery Note Item ' ) and self . get ( ' packed_items ' ) :
frappe . throw ( _ ( " Packed Items cannot be transferred internally " ) )
2020-12-07 21:35:49 +05:30
def validate_putaway_capacity ( self ) :
# if over receipt is attempted while 'apply putaway rule' is disabled
# and if rule was applied on the transaction, validate it.
2021-01-18 23:47:24 +05:30
from erpnext . stock . doctype . putaway_rule . putaway_rule import get_available_putaway_capacity
valid_doctype = self . doctype in ( " Purchase Receipt " , " Stock Entry " , " Purchase Invoice " ,
" Stock Reconciliation " )
2020-12-07 21:35:49 +05:30
2021-01-18 23:47:24 +05:30
if self . doctype == " Purchase Invoice " and self . get ( " update_stock " ) == 0 :
valid_doctype = False
if valid_doctype :
2020-12-07 21:35:49 +05:30
rule_map = defaultdict ( dict )
for item in self . get ( " items " ) :
2021-01-18 23:47:24 +05:30
warehouse_field = " t_warehouse " if self . doctype == " Stock Entry " else " warehouse "
rule = frappe . db . get_value ( " Putaway Rule " ,
{
" item_code " : item . get ( " item_code " ) ,
" warehouse " : item . get ( warehouse_field )
} ,
[ " name " , " disable " ] , as_dict = True )
if rule :
if rule . get ( " disabled " ) : continue # dont validate for disabled rule
if self . doctype == " Stock Reconciliation " :
stock_qty = flt ( item . qty )
else :
stock_qty = flt ( item . transfer_qty ) if self . doctype == " Stock Entry " else flt ( item . stock_qty )
rule_name = rule . get ( " name " )
if not rule_map [ rule_name ] :
rule_map [ rule_name ] [ " warehouse " ] = item . get ( warehouse_field )
rule_map [ rule_name ] [ " item " ] = item . get ( " item_code " )
rule_map [ rule_name ] [ " qty_put " ] = 0
rule_map [ rule_name ] [ " capacity " ] = get_available_putaway_capacity ( rule_name )
rule_map [ rule_name ] [ " qty_put " ] + = flt ( stock_qty )
2020-12-07 21:35:49 +05:30
for rule , values in rule_map . items ( ) :
if flt ( values [ " qty_put " ] ) > flt ( values [ " capacity " ] ) :
2021-01-18 23:47:24 +05:30
message = self . prepare_over_receipt_message ( rule , values )
2020-12-07 21:35:49 +05:30
frappe . throw ( msg = message , title = _ ( " Over Receipt " ) )
2021-01-18 23:47:24 +05:30
def prepare_over_receipt_message ( self , rule , values ) :
message = _ ( " {0} qty of Item {1} is being received into Warehouse {2} with capacity {3} . " ) \
. format (
frappe . bold ( values [ " qty_put " ] ) , frappe . bold ( values [ " item " ] ) ,
frappe . bold ( values [ " warehouse " ] ) , frappe . bold ( values [ " capacity " ] )
)
message + = " <br><br> "
rule_link = frappe . utils . get_link_to_form ( " Putaway Rule " , rule )
message + = _ ( " Please adjust the qty or edit {0} to proceed. " ) . format ( rule_link )
return message
2020-12-07 21:35:49 +05:30
2020-12-21 14:45:50 +05:30
def repost_future_sle_and_gle ( self ) :
args = frappe . _dict ( {
" posting_date " : self . posting_date ,
" posting_time " : self . posting_time ,
" voucher_type " : self . doctype ,
" voucher_no " : self . name ,
" company " : self . company
} )
if check_if_future_sle_exists ( args ) :
create_repost_item_valuation_entry ( args )
2020-12-25 18:12:35 +05:30
elif not is_reposting_pending ( ) :
check_if_stock_and_account_balance_synced ( self . posting_date ,
self . company , self . doctype , self . name )
def is_reposting_pending ( ) :
return frappe . db . exists ( " Repost Item Valuation " ,
{ ' docstatus ' : 1 , ' status ' : [ ' in ' , [ ' Queued ' , ' In Progress ' ] ] } )
2020-12-21 14:45:50 +05:30
def check_if_future_sle_exists ( args ) :
sl_entries = frappe . db . get_all ( " Stock Ledger Entry " ,
filters = { " voucher_type " : args . voucher_type , " voucher_no " : args . voucher_no } ,
fields = [ " item_code " , " warehouse " ] ,
order_by = " creation asc " )
distinct_item_warehouses = list ( set ( [ ( d . item_code , d . warehouse ) for d in sl_entries ] ) )
sle_exists = False
for item_code , warehouse in distinct_item_warehouses :
args . update ( {
" item_code " : item_code ,
" warehouse " : warehouse
} )
if get_sle ( args ) :
sle_exists = True
2016-09-12 11:17:14 +08:00
break
2020-12-21 14:45:50 +05:30
return sle_exists
def get_sle ( args ) :
return frappe . db . sql ( """
select name
from ` tabStock Ledger Entry `
where
item_code = % ( item_code ) s
and warehouse = % ( warehouse ) s
and timestamp ( posting_date , posting_time ) > = timestamp ( % ( posting_date ) s , % ( posting_time ) s )
and voucher_no != % ( voucher_no ) s
and is_cancelled = 0
limit 1
""" , args)
def create_repost_item_valuation_entry ( args ) :
args = frappe . _dict ( args )
repost_entry = frappe . new_doc ( " Repost Item Valuation " )
repost_entry . based_on = args . based_on
if not args . based_on :
repost_entry . based_on = ' Transaction ' if args . voucher_no else " Item and Warehouse "
repost_entry . voucher_type = args . voucher_type
repost_entry . voucher_no = args . voucher_no
repost_entry . item_code = args . item_code
repost_entry . warehouse = args . warehouse
repost_entry . posting_date = args . posting_date
repost_entry . posting_time = args . posting_time
repost_entry . company = args . company
repost_entry . allow_zero_rate = args . allow_zero_rate
repost_entry . flags . ignore_links = True
repost_entry . save ( )
repost_entry . submit ( )