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-01-30 12:49:08 +05:30
from __future__ import unicode_literals
2017-06-19 12:54:59 +05:30
import frappe , erpnext
2019-10-31 15:55:03 +05:30
from frappe . utils import flt , cstr , cint , comma_and
2014-02-14 15:47:51 +05:30
from frappe import _
2019-10-31 15:55:03 +05:30
from erpnext . accounts . utils import get_stock_and_account_balance
2015-05-28 13:00:37 +05:30
from frappe . model . meta import get_field_precision
2016-05-16 14:38:47 +05:30
from erpnext . accounts . doctype . budget . budget import validate_expense_against_budget
2019-05-19 00:02:01 +05:30
from erpnext . accounts . doctype . accounting_dimension . accounting_dimension import get_accounting_dimensions
2013-01-30 12:49:08 +05:30
2013-08-29 18:19:37 +05:30
2019-08-08 15:44:11 +05:30
class ClosedAccountingPeriod ( frappe . ValidationError ) : pass
2014-02-14 15:47:51 +05:30
class StockAccountInvalidTransaction ( frappe . ValidationError ) : pass
2019-10-31 15:55:03 +05:30
class StockValueAndAccountBalanceOutOfSync ( frappe . ValidationError ) : pass
2013-01-30 12:49:08 +05:30
2016-12-30 16:21:35 +05:30
def make_gl_entries ( gl_map , cancel = False , adv_adj = False , merge_entries = True , update_outstanding = ' Yes ' , from_repost = False ) :
2013-08-28 18:53:11 +05:30
if gl_map :
if not cancel :
2019-08-08 15:44:11 +05:30
validate_accounting_period ( gl_map )
2013-08-28 18:53:11 +05:30
gl_map = process_gl_map ( gl_map , merge_entries )
2014-07-29 18:06:18 +05:30
if gl_map and len ( gl_map ) > 1 :
2016-12-30 16:21:35 +05:30
save_entries ( gl_map , adv_adj , update_outstanding , from_repost )
2014-07-29 18:06:18 +05:30
else :
2014-07-30 14:28:24 +05:30
frappe . throw ( _ ( " Incorrect number of General Ledger Entries found. You might have selected a wrong Account in the transaction. " ) )
2013-08-28 18:53:11 +05:30
else :
delete_gl_entries ( gl_map , adv_adj = adv_adj , update_outstanding = update_outstanding )
2014-04-16 15:21:46 +05:30
2019-08-08 15:44:11 +05:30
def validate_accounting_period ( gl_map ) :
accounting_periods = frappe . db . sql ( """ SELECT
ap . name as name
FROM
` tabAccounting Period ` ap , ` tabClosed Document ` cd
WHERE
ap . name = cd . parent
AND ap . company = % ( company ) s
AND cd . closed = 1
AND cd . document_type = % ( voucher_type ) s
AND % ( date ) s between ap . start_date and ap . end_date
""" , {
' date ' : gl_map [ 0 ] . posting_date ,
' company ' : gl_map [ 0 ] . company ,
' voucher_type ' : gl_map [ 0 ] . voucher_type
} , as_dict = 1 )
if accounting_periods :
frappe . throw ( _ ( " You can ' t create accounting entries in the closed accounting period {0} " )
. format ( accounting_periods [ 0 ] . name ) , ClosedAccountingPeriod )
2013-08-26 16:53:30 +05:30
def process_gl_map ( gl_map , merge_entries = True ) :
2013-01-30 12:49:08 +05:30
if merge_entries :
gl_map = merge_similar_entries ( gl_map )
2013-08-26 16:53:30 +05:30
for entry in gl_map :
2015-11-16 19:05:46 +05:30
# toggle debit, credit if negative entry
2013-10-23 16:29:19 +05:30
if flt ( entry . debit ) < 0 :
entry . credit = flt ( entry . credit ) - flt ( entry . debit )
entry . debit = 0.0
2015-11-16 19:05:46 +05:30
2015-08-19 19:22:34 +05:30
if flt ( entry . debit_in_account_currency ) < 0 :
entry . credit_in_account_currency = \
flt ( entry . credit_in_account_currency ) - flt ( entry . debit_in_account_currency )
entry . debit_in_account_currency = 0.0
2015-11-16 19:05:46 +05:30
2013-10-23 16:29:19 +05:30
if flt ( entry . credit ) < 0 :
entry . debit = flt ( entry . debit ) - flt ( entry . credit )
entry . credit = 0.0
2015-11-16 19:05:46 +05:30
2015-08-19 19:22:34 +05:30
if flt ( entry . credit_in_account_currency ) < 0 :
entry . debit_in_account_currency = \
flt ( entry . debit_in_account_currency ) - flt ( entry . credit_in_account_currency )
entry . credit_in_account_currency = 0.0
2015-11-16 19:05:46 +05:30
2013-08-26 16:53:30 +05:30
return gl_map
2014-04-16 15:21:46 +05:30
2013-01-30 12:49:08 +05:30
def merge_similar_entries ( gl_map ) :
merged_gl_map = [ ]
2019-05-19 00:02:01 +05:30
accounting_dimensions = get_accounting_dimensions ( )
2013-01-30 12:49:08 +05:30
for entry in gl_map :
2014-04-16 15:21:46 +05:30
# if there is already an entry in this account then just add it
2013-01-30 12:49:08 +05:30
# to that entry
2019-05-19 00:02:01 +05:30
same_head = check_if_in_list ( entry , merged_gl_map , accounting_dimensions )
2013-01-30 12:49:08 +05:30
if same_head :
2013-08-28 18:53:11 +05:30
same_head . debit = flt ( same_head . debit ) + flt ( entry . debit )
2015-08-19 19:22:34 +05:30
same_head . debit_in_account_currency = \
flt ( same_head . debit_in_account_currency ) + flt ( entry . debit_in_account_currency )
2013-08-28 18:53:11 +05:30
same_head . credit = flt ( same_head . credit ) + flt ( entry . credit )
2015-08-19 19:22:34 +05:30
same_head . credit_in_account_currency = \
flt ( same_head . credit_in_account_currency ) + flt ( entry . credit_in_account_currency )
2013-01-30 12:49:08 +05:30
else :
merged_gl_map . append ( entry )
2014-04-16 15:21:46 +05:30
2013-08-07 17:00:01 +05:30
# filter zero debit and credit entries
2015-08-13 12:19:20 +05:30
merged_gl_map = filter ( lambda x : flt ( x . debit , 9 ) != 0 or flt ( x . credit , 9 ) != 0 , merged_gl_map )
2018-03-08 13:10:51 +05:30
merged_gl_map = list ( merged_gl_map )
2018-08-08 16:37:31 +05:30
2013-01-30 12:49:08 +05:30
return merged_gl_map
2019-05-19 00:02:01 +05:30
def check_if_in_list ( gle , gl_map , dimensions = None ) :
account_head_fieldnames = [ ' party_type ' , ' party ' , ' against_voucher ' , ' against_voucher_type ' ,
' cost_center ' , ' project ' ]
if dimensions :
account_head_fieldnames = account_head_fieldnames + dimensions
2013-08-29 18:19:37 +05:30
for e in gl_map :
2019-05-19 00:02:01 +05:30
same_head = True
if e . account != gle . account :
same_head = False
for fieldname in account_head_fieldnames :
if cstr ( e . get ( fieldname ) ) != cstr ( gle . get ( fieldname ) ) :
same_head = False
if same_head :
return e
2013-01-30 12:49:08 +05:30
2016-12-30 16:21:35 +05:30
def save_entries ( gl_map , adv_adj , update_outstanding , from_repost = False ) :
if not from_repost :
2019-08-05 10:18:57 +05:30
validate_cwip_accounts ( gl_map )
2018-08-08 16:37:31 +05:30
2015-05-28 13:00:37 +05:30
round_off_debit_credit ( gl_map )
2013-01-30 12:49:08 +05:30
for entry in gl_map :
2016-12-30 16:21:35 +05:30
make_entry ( entry , adv_adj , update_outstanding , from_repost )
2018-08-08 16:37:31 +05:30
2013-08-26 16:53:30 +05:30
# check against budget
2016-12-30 16:21:35 +05:30
if not from_repost :
validate_expense_against_budget ( entry )
2014-04-16 15:21:46 +05:30
2019-10-31 15:55:03 +05:30
if not from_repost :
validate_account_for_perpetual_inventory ( gl_map )
2016-12-30 16:21:35 +05:30
def make_entry ( args , adv_adj , update_outstanding , from_repost = False ) :
2013-08-23 15:17:36 +05:30
args . update ( { " doctype " : " GL Entry " } )
2014-04-04 12:16:26 +05:30
gle = frappe . get_doc ( args )
2015-02-10 14:41:27 +05:30
gle . flags . ignore_permissions = 1
2016-12-30 16:21:35 +05:30
gle . flags . from_repost = from_repost
2013-08-23 15:17:36 +05:30
gle . insert ( )
2016-12-30 16:21:35 +05:30
gle . run_method ( " on_update_with_args " , adv_adj , update_outstanding , from_repost )
2013-08-23 15:17:36 +05:30
gle . submit ( )
2014-04-16 15:21:46 +05:30
2017-06-16 15:21:36 +05:30
def validate_account_for_perpetual_inventory ( gl_map ) :
2019-10-31 15:55:03 +05:30
if cint ( erpnext . is_perpetual_inventory_enabled ( gl_map [ 0 ] . company ) ) :
account_list = [ gl_entries . account for gl_entries in gl_map ]
2014-04-16 15:21:46 +05:30
2019-10-31 15:55:03 +05:30
aii_accounts = [ d . name for d in frappe . get_all ( " Account " ,
filters = { ' account_type ' : ' Stock ' , ' is_group ' : 0 , ' company ' : gl_map [ 0 ] . company } ) ]
for account in account_list :
if account not in aii_accounts :
continue
account_bal , stock_bal , warehouse_list = get_stock_and_account_balance ( account ,
gl_map [ 0 ] . posting_date , gl_map [ 0 ] . company )
if gl_map [ 0 ] . voucher_type == " Journal Entry " :
# In case of Journal Entry, there are no corresponding SL entries,
# hence deducting currency amount
account_bal - = flt ( gl_map [ 0 ] . debit ) - flt ( gl_map [ 0 ] . credit )
if account_bal == stock_bal :
2015-06-22 07:31:49 +05:30
frappe . throw ( _ ( " Account: {0} can only be updated via Stock Transactions " )
2019-10-31 15:55:03 +05:30
. format ( account ) , StockAccountInvalidTransaction )
elif account_bal != stock_bal :
frappe . throw ( _ ( " Account Balance ( {0} ) and Stock Value ( {1} ) is out of sync for account {2} and linked warehouse ( {3} ). Please create adjustment Journal Entry for amount {4} . " )
. format ( account_bal , stock_bal , account , comma_and ( warehouse_list ) , stock_bal - account_bal ) ,
StockValueAndAccountBalanceOutOfSync )
2014-04-16 15:21:46 +05:30
2019-08-05 10:18:57 +05:30
def validate_cwip_accounts ( gl_map ) :
2019-11-12 19:17:43 +05:30
cwip_enabled = cint ( frappe . get_cached_value ( " Company " ,
gl_map [ 0 ] . company , " enable_cwip_accounting " ) )
if not cwip_enabled :
cwip_enabled = any ( [ cint ( ac . enable_cwip_accounting ) for ac in frappe . db . get_all ( " Asset Category " , " enable_cwip_accounting " ) ] )
if cwip_enabled and gl_map [ 0 ] . voucher_type == " Journal Entry " :
2019-08-05 10:18:57 +05:30
cwip_accounts = [ d [ 0 ] for d in frappe . db . sql ( """ select name from tabAccount
where account_type = ' Capital Work in Progress ' and is_group = 0 """ )]
for entry in gl_map :
if entry . account in cwip_accounts :
2019-11-12 19:17:43 +05:30
frappe . throw (
_ ( " Account: <b> {0} </b> is capital Work in progress and can not be updated by Journal Entry " ) . format ( entry . account ) )
2019-08-05 10:18:57 +05:30
2015-05-28 13:00:37 +05:30
def round_off_debit_credit ( gl_map ) :
precision = get_field_precision ( frappe . get_meta ( " GL Entry " ) . get_field ( " debit " ) ,
2018-08-08 16:37:31 +05:30
currency = frappe . get_cached_value ( ' Company ' , gl_map [ 0 ] . company , " default_currency " ) )
2015-11-16 19:05:46 +05:30
2015-05-28 13:00:37 +05:30
debit_credit_diff = 0.0
for entry in gl_map :
entry . debit = flt ( entry . debit , precision )
entry . credit = flt ( entry . credit , precision )
debit_credit_diff + = entry . debit - entry . credit
2015-11-16 19:05:46 +05:30
2015-05-28 13:00:37 +05:30
debit_credit_diff = flt ( debit_credit_diff , precision )
2018-08-08 16:37:31 +05:30
2017-05-05 10:41:16 +05:30
if gl_map [ 0 ] [ " voucher_type " ] in ( " Journal Entry " , " Payment Entry " ) :
2016-02-18 19:18:07 +05:30
allowance = 5.0 / ( 10 * * precision )
else :
2017-05-05 10:41:16 +05:30
allowance = .5
2018-08-08 16:37:31 +05:30
2016-02-18 19:18:07 +05:30
if abs ( debit_credit_diff ) > = allowance :
2015-05-28 13:00:37 +05:30
frappe . throw ( _ ( " Debit and Credit not equal for {0} # {1} . Difference is {2} . " )
. format ( gl_map [ 0 ] . voucher_type , gl_map [ 0 ] . voucher_no , debit_credit_diff ) )
2015-11-16 19:05:46 +05:30
2015-05-28 13:00:37 +05:30
elif abs ( debit_credit_diff ) > = ( 1.0 / ( 10 * * precision ) ) :
2019-07-03 10:34:31 +05:30
make_round_off_gle ( gl_map , debit_credit_diff , precision )
2015-11-16 19:05:46 +05:30
2019-07-03 10:34:31 +05:30
def make_round_off_gle ( gl_map , debit_credit_diff , precision ) :
2017-09-19 14:53:16 +05:30
round_off_account , round_off_cost_center = get_round_off_account_and_cost_center ( gl_map [ 0 ] . company )
2018-08-14 16:28:14 +05:30
round_off_account_exists = False
2015-05-28 19:19:59 +05:30
round_off_gle = frappe . _dict ( )
2018-08-14 16:28:14 +05:30
for d in gl_map :
if d . account == round_off_account :
round_off_gle = d
if d . debit_in_account_currency :
debit_credit_diff - = flt ( d . debit_in_account_currency )
else :
debit_credit_diff + = flt ( d . credit_in_account_currency )
round_off_account_exists = True
2019-07-03 10:34:31 +05:30
if round_off_account_exists and abs ( debit_credit_diff ) < = ( 1.0 / ( 10 * * precision ) ) :
gl_map . remove ( round_off_gle )
return
2018-08-14 16:28:14 +05:30
if not round_off_gle :
for k in [ " voucher_type " , " voucher_no " , " company " ,
" posting_date " , " remarks " , " is_opening " ] :
round_off_gle [ k ] = gl_map [ 0 ] [ k ]
2015-11-16 19:05:46 +05:30
2015-05-28 13:00:37 +05:30
round_off_gle . update ( {
" account " : round_off_account ,
2016-01-20 16:14:27 +05:30
" debit_in_account_currency " : abs ( debit_credit_diff ) if debit_credit_diff < 0 else 0 ,
" credit_in_account_currency " : debit_credit_diff if debit_credit_diff > 0 else 0 ,
2015-05-28 13:00:37 +05:30
" debit " : abs ( debit_credit_diff ) if debit_credit_diff < 0 else 0 ,
" credit " : debit_credit_diff if debit_credit_diff > 0 else 0 ,
2015-05-28 13:19:01 +05:30
" cost_center " : round_off_cost_center ,
" party_type " : None ,
" party " : None ,
" against_voucher_type " : None ,
" against_voucher " : None
2015-05-28 13:00:37 +05:30
} )
2015-11-16 19:05:46 +05:30
2018-08-14 16:28:14 +05:30
if not round_off_account_exists :
gl_map . append ( round_off_gle )
2015-05-28 13:00:37 +05:30
2017-09-19 14:53:16 +05:30
def get_round_off_account_and_cost_center ( company ) :
2018-08-08 16:37:31 +05:30
round_off_account , round_off_cost_center = frappe . get_cached_value ( ' Company ' , company ,
2017-09-19 14:53:16 +05:30
[ " round_off_account " , " round_off_cost_center " ] ) or [ None , None ]
if not round_off_account :
frappe . throw ( _ ( " Please mention Round Off Account in Company " ) )
if not round_off_cost_center :
frappe . throw ( _ ( " Please mention Round Off Cost Center in Company " ) )
return round_off_account , round_off_cost_center
2014-04-16 15:21:46 +05:30
def delete_gl_entries ( gl_entries = None , voucher_type = None , voucher_no = None ,
2013-08-28 18:53:11 +05:30
adv_adj = False , update_outstanding = " Yes " ) :
2014-04-16 15:21:46 +05:30
2014-03-03 19:18:17 +05:30
from erpnext . accounts . doctype . gl_entry . gl_entry import validate_balance_type , \
2013-09-24 14:36:55 +05:30
check_freezing_date , update_outstanding_amt , validate_frozen_account
2014-04-16 15:21:46 +05:30
2013-08-28 18:53:11 +05:30
if not gl_entries :
2016-07-01 15:58:39 +05:30
gl_entries = frappe . db . sql ( """
2016-10-03 12:25:04 +05:30
select account , posting_date , party_type , party , cost_center , fiscal_year , voucher_type ,
voucher_no , against_voucher_type , against_voucher , cost_center , company
2016-07-01 15:58:39 +05:30
from ` tabGL Entry `
2013-08-28 18:53:11 +05:30
where voucher_type = % s and voucher_no = % s """ , (voucher_type, voucher_no), as_dict=True)
2016-07-01 15:58:39 +05:30
2013-08-26 16:53:30 +05:30
if gl_entries :
check_freezing_date ( gl_entries [ 0 ] [ " posting_date " ] , adv_adj )
2014-04-16 15:21:46 +05:30
frappe . db . sql ( """ delete from `tabGL Entry` where voucher_type= %s and voucher_no= %s """ ,
2013-08-28 18:53:11 +05:30
( voucher_type or gl_entries [ 0 ] [ " voucher_type " ] , voucher_no or gl_entries [ 0 ] [ " voucher_no " ] ) )
2014-04-16 15:21:46 +05:30
2013-08-21 17:47:11 +05:30
for entry in gl_entries :
2013-09-24 14:36:55 +05:30
validate_frozen_account ( entry [ " account " ] , adv_adj )
2014-03-03 19:18:17 +05:30
validate_balance_type ( entry [ " account " ] , adv_adj )
2018-01-29 16:45:43 +05:30
if not adv_adj :
validate_expense_against_budget ( entry )
2018-08-08 16:37:31 +05:30
2018-01-29 16:45:43 +05:30
if entry . get ( " against_voucher " ) and update_outstanding == ' Yes ' and not adv_adj :
2014-08-29 11:18:32 +05:30
update_outstanding_amt ( entry [ " account " ] , entry . get ( " party_type " ) , entry . get ( " party " ) , entry . get ( " against_voucher_type " ) ,
2014-04-16 17:21:25 +05:30
entry . get ( " against_voucher " ) , on_cancel = True )