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
2023-04-30 16:35:12 +05:30
from collections import OrderedDict
2021-09-02 16:44:59 +05:30
2014-02-14 15:47:51 +05:30
import frappe
2022-10-27 10:55:04 +05:30
from frappe import _ , qb , scrub
from frappe . query_builder import Order
2022-05-31 10:46:56 -03:00
from frappe . utils import cint , flt , formatdate
2021-09-02 16:44:59 +05:30
2017-03-14 14:56:24 +05:30
from erpnext . controllers . queries import get_match_cond
2022-08-15 09:14:23 -03:00
from erpnext . stock . report . stock_ledger . stock_ledger import get_item_group_condition
2019-03-04 12:56:27 +05:30
from erpnext . stock . utils import get_incoming_rate
2013-02-25 18:17:51 +05:30
2016-08-22 14:14:52 +05:30
2013-02-25 18:17:51 +05:30
def execute ( filters = None ) :
2016-01-19 17:27:06 +05:30
if not filters :
filters = frappe . _dict ( )
2018-08-08 16:37:31 +05:30
filters . currency = frappe . get_cached_value ( " Company " , filters . company , " default_currency " )
2016-03-09 13:15:38 +05:30
2014-11-24 15:12:37 +05:30
gross_profit_data = GrossProfitGenerator ( filters )
data = [ ]
group_wise_columns = frappe . _dict (
{
2021-11-22 12:34:46 +05:30
" invoice " : [
" invoice_or_item " ,
" customer " ,
" customer_group " ,
" posting_date " ,
" item_code " ,
" item_name " ,
" item_group " ,
" brand " ,
" description " ,
2014-11-24 15:12:37 +05:30
" warehouse " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
2016-03-09 17:02:59 +05:30
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
" project " ,
2016-03-09 13:15:38 +05:30
] ,
" item_code " : [
" item_code " ,
" item_name " ,
" brand " ,
" description " ,
" qty " ,
" base_rate " ,
2014-11-24 15:12:37 +05:30
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" warehouse " : [
" warehouse " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" brand " : [
" brand " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" item_group " : [
" item_group " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" customer " : [
" customer " ,
" customer_group " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" customer_group " : [
" customer_group " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" sales_person " : [
" sales_person " ,
" allocated_amount " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
2016-03-09 17:02:59 +05:30
" project " : [ " project " , " base_amount " , " buying_amount " , " gross_profit " , " gross_profit_percent " ] ,
2014-11-24 15:12:37 +05:30
" territory " : [
" territory " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
2022-03-28 18:52:46 +05:30
] ,
2022-05-31 10:46:56 -03:00
" monthly " : [
" monthly " ,
" qty " ,
" base_rate " ,
" buying_rate " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
" payment_term " : [
" payment_term " ,
" base_amount " ,
" buying_amount " ,
" gross_profit " ,
" gross_profit_percent " ,
] ,
2014-11-24 15:12:37 +05:30
}
)
columns = get_columns ( group_wise_columns , filters )
2021-08-20 02:16:49 +05:30
if filters . group_by == " Invoice " :
2021-08-25 02:52:18 +05:30
get_data_when_grouped_by_invoice ( columns , gross_profit_data , filters , group_wise_columns , data )
2016-03-09 13:15:38 +05:30
2021-08-25 02:52:18 +05:30
else :
get_data_when_not_grouped_by_invoice ( gross_profit_data , filters , group_wise_columns , data )
2021-09-02 16:44:59 +05:30
2021-08-25 02:52:18 +05:30
return columns , data
2021-08-20 18:25:36 +05:30
2022-03-28 18:52:46 +05:30
2021-08-25 02:52:18 +05:30
def get_data_when_grouped_by_invoice (
columns , gross_profit_data , filters , group_wise_columns , data
) :
column_names = get_column_names ( )
2014-11-24 15:12:37 +05:30
2021-08-25 02:52:18 +05:30
# to display item as Item Code: Item Name
columns [ 0 ] = " Sales Invoice:Link/Item:300 "
# removing Item Code and Item Name columns
del columns [ 4 : 6 ]
2021-08-20 02:16:49 +05:30
2021-08-25 02:52:18 +05:30
for src in gross_profit_data . si_list :
row = frappe . _dict ( )
row . indent = src . indent
row . parent_invoice = src . parent_invoice
row . currency = filters . currency
2021-08-20 02:16:49 +05:30
2021-08-25 02:52:18 +05:30
for col in group_wise_columns . get ( scrub ( filters . group_by ) ) :
row [ column_names [ col ] ] = src . get ( col )
data . append ( row )
2022-03-28 18:52:46 +05:30
2021-08-25 02:52:18 +05:30
def get_data_when_not_grouped_by_invoice ( gross_profit_data , filters , group_wise_columns , data ) :
2022-02-11 14:48:39 +05:30
for src in gross_profit_data . grouped_data :
2021-08-25 02:52:18 +05:30
row = [ ]
for col in group_wise_columns . get ( scrub ( filters . group_by ) ) :
row . append ( src . get ( col ) )
row . append ( filters . currency )
2021-11-16 03:22:04 +05:30
2021-08-25 02:52:18 +05:30
data . append ( row )
2014-11-24 15:12:37 +05:30
2022-03-28 18:52:46 +05:30
2014-11-24 15:12:37 +05:30
def get_columns ( group_wise_columns , filters ) :
columns = [ ]
column_map = frappe . _dict (
{
2022-02-11 14:48:39 +05:30
" parent " : {
" label " : _ ( " Sales Invoice " ) ,
" fieldname " : " parent_invoice " ,
" fieldtype " : " Link " ,
" options " : " Sales Invoice " ,
" width " : 120 ,
} ,
" invoice_or_item " : {
" label " : _ ( " Sales Invoice " ) ,
" fieldtype " : " Link " ,
" options " : " Sales Invoice " ,
" width " : 120 ,
} ,
" posting_date " : {
" label " : _ ( " Posting Date " ) ,
" fieldname " : " posting_date " ,
" fieldtype " : " Date " ,
" width " : 100 ,
} ,
" posting_time " : {
" label " : _ ( " Posting Time " ) ,
" fieldname " : " posting_time " ,
" fieldtype " : " Data " ,
" width " : 100 ,
} ,
" item_code " : {
" label " : _ ( " Item Code " ) ,
" fieldname " : " item_code " ,
" fieldtype " : " Link " ,
" options " : " Item " ,
" width " : 100 ,
} ,
" item_name " : {
" label " : _ ( " Item Name " ) ,
" fieldname " : " item_name " ,
" fieldtype " : " Data " ,
" width " : 100 ,
} ,
" item_group " : {
" label " : _ ( " Item Group " ) ,
" fieldname " : " item_group " ,
" fieldtype " : " Link " ,
" options " : " Item Group " ,
" width " : 100 ,
2022-03-28 18:52:46 +05:30
} ,
2022-02-11 14:48:39 +05:30
" brand " : { " label " : _ ( " Brand " ) , " fieldtype " : " Link " , " options " : " Brand " , " width " : 100 } ,
" description " : {
" label " : _ ( " Description " ) ,
" fieldname " : " description " ,
" fieldtype " : " Data " ,
" width " : 100 ,
} ,
" warehouse " : {
" label " : _ ( " Warehouse " ) ,
" fieldname " : " warehouse " ,
" fieldtype " : " Link " ,
2023-04-27 15:04:50 +05:30
" options " : " Warehouse " ,
2022-02-11 14:48:39 +05:30
" width " : 100 ,
2022-03-28 18:52:46 +05:30
} ,
2022-02-11 14:48:39 +05:30
" qty " : { " label " : _ ( " Qty " ) , " fieldname " : " qty " , " fieldtype " : " Float " , " width " : 80 } ,
" base_rate " : {
" label " : _ ( " Avg. Selling Rate " ) ,
" fieldname " : " avg._selling_rate " ,
" fieldtype " : " Currency " ,
" options " : " currency " ,
" width " : 100 ,
} ,
" buying_rate " : {
" label " : _ ( " Valuation Rate " ) ,
" fieldname " : " valuation_rate " ,
" fieldtype " : " Currency " ,
" options " : " currency " ,
" width " : 100 ,
} ,
" base_amount " : {
" label " : _ ( " Selling Amount " ) ,
" fieldname " : " selling_amount " ,
" fieldtype " : " Currency " ,
" options " : " currency " ,
" width " : 100 ,
} ,
" buying_amount " : {
" label " : _ ( " Buying Amount " ) ,
" fieldname " : " buying_amount " ,
" fieldtype " : " Currency " ,
" options " : " currency " ,
" width " : 100 ,
} ,
" gross_profit " : {
" label " : _ ( " Gross Profit " ) ,
" fieldname " : " gross_profit " ,
" fieldtype " : " Currency " ,
2020-04-28 11:17:45 +05:30
" options " : " currency " ,
2022-02-11 14:48:39 +05:30
" width " : 100 ,
2022-03-28 18:52:46 +05:30
} ,
2022-02-11 14:48:39 +05:30
" gross_profit_percent " : {
" label " : _ ( " Gross Profit Percent " ) ,
" fieldname " : " gross_profit_ % " ,
" fieldtype " : " Percent " ,
" width " : 100 ,
} ,
" project " : {
" label " : _ ( " Project " ) ,
" fieldname " : " project " ,
" fieldtype " : " Link " ,
" options " : " Project " ,
" width " : 100 ,
} ,
" sales_person " : {
" label " : _ ( " Sales Person " ) ,
" fieldname " : " sales_person " ,
2023-04-27 15:04:50 +05:30
" fieldtype " : " Link " ,
" options " : " Sales Person " ,
2022-02-11 14:48:39 +05:30
" width " : 100 ,
} ,
" allocated_amount " : {
" label " : _ ( " Allocated Amount " ) ,
" fieldname " : " allocated_amount " ,
" fieldtype " : " Currency " ,
" options " : " currency " ,
" width " : 100 ,
} ,
" customer " : {
" label " : _ ( " Customer " ) ,
" fieldname " : " customer " ,
" fieldtype " : " Link " ,
" options " : " Customer " ,
" width " : 100 ,
} ,
" customer_group " : {
" label " : _ ( " Customer Group " ) ,
" fieldname " : " customer_group " ,
" fieldtype " : " Link " ,
2023-04-27 15:04:50 +05:30
" options " : " Customer Group " ,
2022-02-11 14:48:39 +05:30
" width " : 100 ,
} ,
" territory " : {
" label " : _ ( " Territory " ) ,
" fieldname " : " territory " ,
2016-01-19 17:27:06 +05:30
" fieldtype " : " Link " ,
2023-04-27 15:04:50 +05:30
" options " : " Territory " ,
2022-02-11 14:48:39 +05:30
" width " : 100 ,
2014-11-24 15:12:37 +05:30
} ,
2022-05-31 10:46:56 -03:00
" monthly " : {
" label " : _ ( " Monthly " ) ,
" fieldname " : " monthly " ,
" fieldtype " : " Data " ,
" width " : 100 ,
} ,
" payment_term " : {
" label " : _ ( " Payment Term " ) ,
" fieldname " : " payment_term " ,
" fieldtype " : " Link " ,
" options " : " Payment Term " ,
" width " : 170 ,
} ,
2022-03-28 18:52:46 +05:30
}
2014-11-24 15:12:37 +05:30
)
for col in group_wise_columns . get ( scrub ( filters . group_by ) ) :
columns . append ( column_map . get ( col ) )
2016-03-09 13:15:38 +05:30
2016-01-19 17:27:06 +05:30
columns . append (
{
" fieldname " : " currency " ,
" label " : _ ( " Currency " ) ,
" fieldtype " : " Link " ,
2020-04-28 11:17:45 +05:30
" options " : " Currency " ,
" hidden " : 1 ,
2016-01-19 17:27:06 +05:30
}
)
2014-11-24 15:12:37 +05:30
return columns
2022-03-28 18:52:46 +05:30
2021-08-20 02:16:49 +05:30
def get_column_names ( ) :
return frappe . _dict (
{
2021-11-22 12:34:46 +05:30
" invoice_or_item " : " sales_invoice " ,
2021-08-20 02:16:49 +05:30
" customer " : " customer " ,
" customer_group " : " customer_group " ,
" posting_date " : " posting_date " ,
" item_code " : " item_code " ,
" item_name " : " item_name " ,
" item_group " : " item_group " ,
" brand " : " brand " ,
" description " : " description " ,
" warehouse " : " warehouse " ,
" qty " : " qty " ,
" base_rate " : " avg._selling_rate " ,
" buying_rate " : " valuation_rate " ,
" base_amount " : " selling_amount " ,
" buying_amount " : " buying_amount " ,
" gross_profit " : " gross_profit " ,
" gross_profit_percent " : " gross_profit_ % " ,
" project " : " project " ,
}
)
2022-03-28 18:52:46 +05:30
2021-08-20 02:16:49 +05:30
2014-11-24 15:12:37 +05:30
class GrossProfitGenerator ( object ) :
def __init__ ( self , filters = None ) :
2023-02-03 16:24:52 +05:30
self . sle = { }
2016-08-22 17:47:13 +05:30
self . data = [ ]
self . average_buying_rate = { }
self . filters = frappe . _dict ( filters )
self . load_invoice_items ( )
2022-10-27 10:55:04 +05:30
self . get_delivery_notes ( )
2021-08-20 02:16:49 +05:30
2021-08-20 02:20:18 +05:30
if filters . group_by == " Invoice " :
self . group_items_by_invoice ( )
2021-08-20 02:16:49 +05:30
2016-08-22 17:47:13 +05:30
self . load_product_bundle ( )
self . load_non_stock_items ( )
2017-09-18 19:13:45 +05:30
self . get_returned_invoice_items ( )
2016-08-22 17:47:13 +05:30
self . process ( )
2014-11-24 15:12:37 +05:30
def process ( self ) :
self . grouped = { }
2017-10-16 11:27:22 +05:30
self . grouped_data = [ ]
2019-03-04 12:56:27 +05:30
self . currency_precision = cint ( frappe . db . get_default ( " currency_precision " ) ) or 3
self . float_precision = cint ( frappe . db . get_default ( " float_precision " ) ) or 2
2021-08-23 21:17:32 +05:30
grouped_by_invoice = True if self . filters . get ( " group_by " ) == " Invoice " else False
if grouped_by_invoice :
buying_amount = 0
for row in reversed ( self . si_list ) :
2022-05-31 10:46:56 -03:00
if self . filters . get ( " group_by " ) == " Monthly " :
row . monthly = formatdate ( row . posting_date , " MMM YYYY " )
2022-02-12 19:05:03 +05:30
if self . skip_row ( row ) :
2014-11-24 15:12:37 +05:30
continue
2019-03-04 12:56:27 +05:30
row . base_amount = flt ( row . base_net_amount , self . currency_precision )
2014-11-24 15:12:37 +05:30
2015-11-12 11:07:32 +05:30
product_bundles = [ ]
2015-11-05 13:13:02 +05:30
if row . update_stock :
product_bundles = self . product_bundles . get ( row . parenttype , { } ) . get ( row . parent , frappe . _dict ( ) )
elif row . dn_detail :
product_bundles = self . product_bundles . get ( " Delivery Note " , { } ) . get (
row . delivery_note , frappe . _dict ( )
2022-03-28 18:52:46 +05:30
)
2015-11-05 13:13:02 +05:30
row . item_row = row . dn_detail
2023-01-09 22:45:43 +05:30
# Update warehouse and base_amount from 'Packed Item' List
if product_bundles and not row . parent :
# For Packed Items, row.parent_invoice will be the Bundle name
product_bundle = product_bundles . get ( row . parent_invoice )
if product_bundle :
for packed_item in product_bundle :
if (
packed_item . get ( " item_code " ) == row . item_code
and packed_item . get ( " parent_detail_docname " ) == row . item_row
) :
row . warehouse = packed_item . warehouse
row . base_amount = packed_item . base_amount
2015-11-12 11:07:32 +05:30
2014-11-24 15:12:37 +05:30
# get buying amount
2015-07-07 13:59:23 +05:30
if row . item_code in product_bundles :
2019-03-17 09:49:24 +05:30
row . buying_amount = flt (
self . get_buying_amount_from_product_bundle ( row , product_bundles [ row . item_code ] ) ,
self . currency_precision ,
)
2014-11-24 15:12:37 +05:30
else :
2019-03-17 09:49:24 +05:30
row . buying_amount = flt ( self . get_buying_amount ( row , row . item_code ) , self . currency_precision )
2014-11-24 15:12:37 +05:30
2021-08-23 21:17:32 +05:30
if grouped_by_invoice :
if row . indent == 1.0 :
2021-08-25 00:09:07 +05:30
buying_amount + = row . buying_amount
2021-08-23 21:17:32 +05:30
elif row . indent == 0.0 :
row . buying_amount = buying_amount
buying_amount = 0
2014-11-24 15:12:37 +05:30
# get buying rate
2021-08-24 23:30:50 +05:30
if flt ( row . qty ) :
row . buying_rate = flt ( row . buying_amount / flt ( row . qty ) , self . float_precision )
row . base_rate = flt ( row . base_amount / flt ( row . qty ) , self . float_precision )
2014-11-24 15:12:37 +05:30
else :
2021-08-25 00:33:55 +05:30
if self . is_not_invoice_row ( row ) :
2021-08-24 23:30:50 +05:30
row . buying_rate , row . base_rate = 0.0 , 0.0
2014-11-24 15:12:37 +05:30
# calculate gross profit
2019-03-04 12:56:27 +05:30
row . gross_profit = flt ( row . base_amount - row . buying_amount , self . currency_precision )
2015-04-13 15:31:24 +05:30
if row . base_amount :
2019-03-04 12:56:27 +05:30
row . gross_profit_percent = flt (
( row . gross_profit / row . base_amount ) * 100.0 , self . currency_precision
)
2014-11-24 15:12:37 +05:30
else :
row . gross_profit_percent = 0.0
# add to grouped
2017-09-18 19:13:45 +05:30
self . grouped . setdefault ( row . get ( scrub ( self . filters . group_by ) ) , [ ] ) . append ( row )
2014-11-24 15:12:37 +05:30
if self . grouped :
2016-04-12 17:01:39 +05:30
self . get_average_rate_based_on_group_by ( )
2014-11-24 15:12:37 +05:30
2016-04-12 17:01:39 +05:30
def get_average_rate_based_on_group_by ( self ) :
2018-05-23 01:01:24 -05:00
for key in list ( self . grouped ) :
2022-05-31 10:46:56 -03:00
if self . filters . get ( " group_by " ) == " Invoice " :
2017-09-18 19:13:45 +05:30
for i , row in enumerate ( self . grouped [ key ] ) :
2021-11-16 02:33:51 +05:30
if row . indent == 1.0 :
if (
row . parent in self . returned_invoices and row . item_code in self . returned_invoices [ row . parent ]
) :
returned_item_rows = self . returned_invoices [ row . parent ] [ row . item_code ]
for returned_item_row in returned_item_rows :
2023-02-05 13:09:34 +05:30
# returned_items 'qty' should be stateful
if returned_item_row . qty != 0 :
if row . qty > = abs ( returned_item_row . qty ) :
row . qty + = returned_item_row . qty
returned_item_row . qty = 0
else :
row . qty = 0
returned_item_row . qty + = row . qty
2021-11-16 02:33:51 +05:30
row . base_amount + = flt ( returned_item_row . base_amount , self . currency_precision )
row . buying_amount = flt ( flt ( row . qty ) * flt ( row . buying_rate ) , self . currency_precision )
2021-11-16 02:36:58 +05:30
if flt ( row . qty ) or row . base_amount :
2021-11-16 02:33:51 +05:30
row = self . set_average_rate ( row )
self . grouped_data . append ( row )
2022-05-31 10:46:56 -03:00
elif self . filters . get ( " group_by " ) == " Payment Term " :
for i , row in enumerate ( self . grouped [ key ] ) :
invoice_portion = 0
if row . is_return :
invoice_portion = 100
elif row . invoice_portion :
invoice_portion = row . invoice_portion
2022-12-14 10:46:11 +05:30
elif row . payment_amount :
2022-05-31 10:46:56 -03:00
invoice_portion = row . payment_amount * 100 / row . base_net_amount
if i == 0 :
new_row = row
self . set_average_based_on_payment_term_portion ( new_row , row , invoice_portion )
else :
new_row . qty + = flt ( row . qty )
self . set_average_based_on_payment_term_portion ( new_row , row , invoice_portion , True )
new_row = self . set_average_rate ( new_row )
self . grouped_data . append ( new_row )
else :
for i , row in enumerate ( self . grouped [ key ] ) :
if i == 0 :
new_row = row
else :
new_row . qty + = flt ( row . qty )
new_row . buying_amount + = flt ( row . buying_amount , self . currency_precision )
new_row . base_amount + = flt ( row . base_amount , self . currency_precision )
2023-10-10 10:30:09 +00:00
if self . filters . get ( " group_by " ) == " Sales Person " :
new_row . allocated_amount + = flt ( row . allocated_amount , self . currency_precision )
2022-05-31 10:46:56 -03:00
new_row = self . set_average_rate ( new_row )
self . grouped_data . append ( new_row )
def set_average_based_on_payment_term_portion ( self , new_row , row , invoice_portion , aggr = False ) :
cols = [ " base_amount " , " buying_amount " , " gross_profit " ]
for col in cols :
if aggr :
new_row [ col ] + = row [ col ] * invoice_portion / 100
else :
new_row [ col ] = row [ col ] * invoice_portion / 100
2017-09-18 19:13:45 +05:30
2021-08-25 00:33:55 +05:30
def is_not_invoice_row ( self , row ) :
return ( self . filters . get ( " group_by " ) == " Invoice " and row . indent != 0.0 ) or self . filters . get (
" group_by "
) != " Invoice "
2017-09-18 19:13:45 +05:30
def set_average_rate ( self , new_row ) :
2021-08-24 17:35:01 +05:30
self . set_average_gross_profit ( new_row )
new_row . buying_rate = (
flt ( new_row . buying_amount / new_row . qty , self . float_precision ) if new_row . qty else 0
2022-03-28 18:52:46 +05:30
)
2021-08-24 17:35:01 +05:30
new_row . base_rate = (
flt ( new_row . base_amount / new_row . qty , self . float_precision ) if new_row . qty else 0
2022-03-28 18:52:46 +05:30
)
2021-08-24 17:35:01 +05:30
return new_row
def set_average_gross_profit ( self , new_row ) :
2019-03-04 12:56:27 +05:30
new_row . gross_profit = flt ( new_row . base_amount - new_row . buying_amount , self . currency_precision )
new_row . gross_profit_percent = (
flt ( ( ( new_row . gross_profit / new_row . base_amount ) * 100.0 ) , self . currency_precision )
2017-09-18 19:13:45 +05:30
if new_row . base_amount
else 0
2022-03-28 18:52:46 +05:30
)
2019-03-04 12:56:27 +05:30
2017-09-18 19:13:45 +05:30
def get_returned_invoice_items ( self ) :
returned_invoices = frappe . db . sql (
"""
select
2019-01-16 14:36:55 +05:00
si . name , si_item . item_code , si_item . stock_qty as qty , si_item . base_net_amount as base_amount , si . return_against
2017-09-18 19:13:45 +05:30
from
` tabSales Invoice ` si , ` tabSales Invoice Item ` si_item
where
si . name = si_item . parent
and si . docstatus = 1
and si . is_return = 1
""" ,
as_dict = 1 ,
)
self . returned_invoices = frappe . _dict ( )
for inv in returned_invoices :
self . returned_invoices . setdefault ( inv . return_against , frappe . _dict ( ) ) . setdefault (
inv . item_code , [ ]
) . append ( inv )
2014-11-24 15:12:37 +05:30
2022-02-12 19:05:03 +05:30
def skip_row ( self , row ) :
2017-09-18 19:13:45 +05:30
if self . filters . get ( " group_by " ) != " Invoice " :
2019-01-02 16:22:22 +05:00
if not row . get ( scrub ( self . filters . get ( " group_by " , " " ) ) ) :
2017-09-18 19:13:45 +05:30
return True
2022-02-14 22:14:17 +05:30
return False
2014-11-24 15:12:37 +05:30
2015-07-07 13:59:23 +05:30
def get_buying_amount_from_product_bundle ( self , row , product_bundle ) :
2014-11-24 15:12:37 +05:30
buying_amount = 0.0
2015-11-05 13:13:02 +05:30
for packed_item in product_bundle :
if packed_item . get ( " parent_detail_docname " ) == row . item_row :
2023-01-09 22:45:43 +05:30
packed_item_row = row . copy ( )
packed_item_row . warehouse = packed_item . warehouse
buying_amount + = self . get_buying_amount ( packed_item_row , packed_item . item_code )
2014-11-24 15:12:37 +05:30
2019-03-17 09:49:24 +05:30
return flt ( buying_amount , self . currency_precision )
2014-11-24 15:12:37 +05:30
2022-10-27 10:55:04 +05:30
def calculate_buying_amount_from_sle ( self , row , my_sle , parenttype , parent , item_row , item_code ) :
for i , sle in enumerate ( my_sle ) :
# find the stock valution rate from stock ledger entry
if (
sle . voucher_type == parenttype
and parent == sle . voucher_no
and sle . voucher_detail_no == item_row
) :
previous_stock_value = len ( my_sle ) > i + 1 and flt ( my_sle [ i + 1 ] . stock_value ) or 0.0
if previous_stock_value :
return abs ( previous_stock_value - flt ( sle . stock_value ) ) * flt ( row . qty ) / abs ( flt ( sle . qty ) )
else :
return flt ( row . qty ) * self . get_average_buying_rate ( row , item_code )
2022-12-21 12:11:49 +05:30
return 0.0
2022-10-27 10:55:04 +05:30
2014-11-24 15:12:37 +05:30
def get_buying_amount ( self , row , item_code ) :
# IMP NOTE
# stock_ledger_entries should already be filtered by item_code and warehouse and
# sorted by posting_date desc, posting_time desc
2020-07-24 09:49:17 +05:30
if item_code in self . non_stock_items and ( row . project or row . cost_center ) :
2016-08-22 17:27:22 +05:30
# Issue 6089-Get last purchasing rate for non-stock item
2020-07-24 09:49:17 +05:30
item_rate = self . get_last_purchase_rate ( item_code , row )
2015-04-13 15:31:24 +05:30
return flt ( row . qty ) * item_rate
2014-11-24 15:12:37 +05:30
else :
2023-02-03 16:24:52 +05:30
my_sle = self . get_stock_ledger_entries ( item_code , row . warehouse )
2015-07-23 17:08:44 +05:30
if ( row . update_stock or row . dn_detail ) and my_sle :
2015-11-05 13:13:02 +05:30
parenttype , parent = row . parenttype , row . parent
2015-05-25 12:23:03 +05:30
if row . dn_detail :
2015-11-05 13:13:02 +05:30
parenttype , parent = " Delivery Note " , row . delivery_note
2015-11-12 11:07:32 +05:30
2022-10-27 10:55:04 +05:30
return self . calculate_buying_amount_from_sle (
row , my_sle , parenttype , parent , row . item_row , item_code
)
elif self . delivery_notes . get ( ( row . parent , row . item_code ) , None ) :
# check if Invoice has delivery notes
dn = self . delivery_notes . get ( ( row . parent , row . item_code ) )
parenttype , parent , item_row , warehouse = (
" Delivery Note " ,
dn [ " delivery_note " ] ,
dn [ " item_row " ] ,
dn [ " warehouse " ] ,
)
2023-02-03 16:24:52 +05:30
my_sle = self . get_stock_ledger_entries ( item_code , row . warehouse )
2022-10-27 10:55:04 +05:30
return self . calculate_buying_amount_from_sle (
row , my_sle , parenttype , parent , item_row , item_code
)
2023-01-19 09:39:43 +00:00
elif row . sales_order and row . so_detail :
incoming_amount = self . get_buying_amount_from_so_dn ( row . sales_order , row . so_detail , item_code )
if incoming_amount :
return incoming_amount
2015-04-13 15:31:24 +05:30
else :
2017-02-01 13:34:06 +05:30
return flt ( row . qty ) * self . get_average_buying_rate ( row , item_code )
2014-11-24 15:12:37 +05:30
2023-01-19 09:39:43 +00:00
return flt ( row . qty ) * self . get_average_buying_rate ( row , item_code )
def get_buying_amount_from_so_dn ( self , sales_order , so_detail , item_code ) :
from frappe . query_builder . functions import Sum
2023-01-19 10:25:05 +00:00
2023-01-19 09:39:43 +00:00
delivery_note_item = frappe . qb . DocType ( " Delivery Note Item " )
query = (
2023-02-23 11:15:56 +05:30
frappe . qb . from_ ( delivery_note_item )
2023-01-19 10:25:05 +00:00
. select ( Sum ( delivery_note_item . incoming_rate * delivery_note_item . stock_qty ) )
2023-02-23 11:15:56 +05:30
. where ( delivery_note_item . docstatus == 1 )
2023-01-19 09:39:43 +00:00
. where ( delivery_note_item . item_code == item_code )
. where ( delivery_note_item . against_sales_order == sales_order )
. where ( delivery_note_item . so_detail == so_detail )
. groupby ( delivery_note_item . item_code )
)
incoming_amount = query . run ( )
return flt ( incoming_amount [ 0 ] [ 0 ] ) if incoming_amount else 0
2014-11-24 15:12:37 +05:30
2017-02-01 13:34:06 +05:30
def get_average_buying_rate ( self , row , item_code ) :
2018-01-17 14:40:59 +05:30
args = row
2015-04-13 15:31:24 +05:30
if not item_code in self . average_buying_rate :
2020-07-24 09:49:17 +05:30
args . update (
{
" voucher_type " : row . parenttype ,
" voucher_no " : row . parent ,
" allow_zero_valuation " : True ,
" company " : self . filters . company ,
}
)
2018-01-17 14:40:59 +05:30
2023-03-31 09:03:54 +05:30
if row . serial_and_batch_bundle :
args . update ( { " serial_and_batch_bundle " : row . serial_and_batch_bundle } )
2020-07-24 09:49:17 +05:30
average_buying_rate = get_incoming_rate ( args )
self . average_buying_rate [ item_code ] = flt ( average_buying_rate )
2015-04-13 15:31:24 +05:30
return self . average_buying_rate [ item_code ]
2020-07-24 09:49:17 +05:30
def get_last_purchase_rate ( self , item_code , row ) :
2022-02-10 13:07:51 +05:30
purchase_invoice = frappe . qb . DocType ( " Purchase Invoice " )
purchase_invoice_item = frappe . qb . DocType ( " Purchase Invoice Item " )
query = (
frappe . qb . from_ ( purchase_invoice_item )
. inner_join ( purchase_invoice )
. on ( purchase_invoice . name == purchase_invoice_item . parent )
. select ( purchase_invoice_item . base_rate / purchase_invoice_item . conversion_factor )
. where ( purchase_invoice . docstatus == 1 )
. where ( purchase_invoice . posting_date < = self . filters . to_date )
. where ( purchase_invoice_item . item_code == item_code )
)
2020-07-24 09:49:17 +05:30
if row . project :
2022-02-11 14:48:39 +05:30
query . where ( purchase_invoice_item . project == row . project )
2022-02-10 13:07:51 +05:30
if row . cost_center :
query . where ( purchase_invoice_item . cost_center == row . cost_center )
query . orderby ( purchase_invoice . posting_date , order = frappe . qb . desc )
query . limit ( 1 )
last_purchase_rate = query . run ( )
2020-07-24 09:49:17 +05:30
2016-08-22 17:47:13 +05:30
return flt ( last_purchase_rate [ 0 ] [ 0 ] ) if last_purchase_rate else 0
2016-08-22 14:14:52 +05:30
2014-11-24 15:12:37 +05:30
def load_invoice_items ( self ) :
conditions = " "
if self . filters . company :
2023-05-25 14:18:41 +05:30
conditions + = " and `tabSales Invoice`.company = %(company)s "
2014-11-24 15:12:37 +05:30
if self . filters . from_date :
conditions + = " and posting_date >= %(from_date)s "
if self . filters . to_date :
conditions + = " and posting_date <= %(to_date)s "
2017-03-31 12:44:29 +05:30
2023-03-14 16:22:49 +05:30
conditions + = " and (is_return = 0 or (is_return=1 and return_against is null)) "
2022-08-15 09:14:23 -03:00
if self . filters . item_group :
conditions + = " and {0} " . format ( get_item_group_condition ( self . filters . item_group ) )
if self . filters . sales_person :
2022-08-15 09:23:56 -03:00
conditions + = """
and exists ( select 1
from ` tabSales Team ` st
2022-08-15 09:14:23 -03:00
where st . parent = ` tabSales Invoice ` . name
and st . sales_person = % ( sales_person ) s )
"""
2017-03-12 17:10:41 +05:30
if self . filters . group_by == " Sales Person " :
sales_person_cols = " , sales.sales_person, sales.allocated_amount, sales.incentives "
2017-03-14 14:56:24 +05:30
sales_team_table = " left join `tabSales Team` sales on sales.parent = `tabSales Invoice`.name "
2017-03-12 17:10:41 +05:30
else :
sales_person_cols = " "
sales_team_table = " "
2014-11-24 15:12:37 +05:30
2022-05-31 10:46:56 -03:00
if self . filters . group_by == " Payment Term " :
payment_term_cols = """ ,if(`tabSales Invoice`.is_return = 1,
' {0} ' ,
coalesce ( schedule . payment_term , ' {1} ' ) ) as payment_term ,
schedule . invoice_portion ,
schedule . payment_amount """ .format(
_ ( " Sales Return " ) , _ ( " No Terms " )
)
payment_term_table = """ left join `tabPayment Schedule` schedule on schedule.parent = `tabSales Invoice`.name and
` tabSales Invoice ` . is_return = 0 """
else :
payment_term_cols = " "
payment_term_table = " "
2019-04-23 18:33:01 +05:30
if self . filters . get ( " sales_invoice " ) :
conditions + = " and `tabSales Invoice`.name = %(sales_invoice)s "
if self . filters . get ( " item_code " ) :
conditions + = " and `tabSales Invoice Item`.item_code = %(item_code)s "
2023-02-01 18:44:15 +05:30
if self . filters . get ( " warehouse " ) :
warehouse_details = frappe . db . get_value (
" Warehouse " , self . filters . get ( " warehouse " ) , [ " lft " , " rgt " ] , as_dict = 1
)
if warehouse_details :
conditions + = f " and `tabSales Invoice Item`.warehouse in (select name from `tabWarehouse` wh where wh.lft >= { warehouse_details . lft } and wh.rgt <= { warehouse_details . rgt } and warehouse = wh.name) "
2017-09-18 19:13:45 +05:30
self . si_list = frappe . db . sql (
"""
select
` tabSales Invoice Item ` . parenttype , ` tabSales Invoice Item ` . parent ,
` tabSales Invoice ` . posting_date , ` tabSales Invoice ` . posting_time ,
` tabSales Invoice ` . project , ` tabSales Invoice ` . update_stock ,
` tabSales Invoice ` . customer , ` tabSales Invoice ` . customer_group ,
` tabSales Invoice ` . territory , ` tabSales Invoice Item ` . item_code ,
` tabSales Invoice Item ` . item_name , ` tabSales Invoice Item ` . description ,
` tabSales Invoice Item ` . warehouse , ` tabSales Invoice Item ` . item_group ,
2023-01-19 09:39:43 +00:00
` tabSales Invoice Item ` . brand , ` tabSales Invoice Item ` . so_detail ,
` tabSales Invoice Item ` . sales_order , ` tabSales Invoice Item ` . dn_detail ,
2017-09-18 19:13:45 +05:30
` tabSales Invoice Item ` . delivery_note , ` tabSales Invoice Item ` . stock_qty as qty ,
` tabSales Invoice Item ` . base_net_rate , ` tabSales Invoice Item ` . base_net_amount ,
2020-07-24 09:49:17 +05:30
` tabSales Invoice Item ` . name as " item_row " , ` tabSales Invoice ` . is_return ,
2023-03-31 09:03:54 +05:30
` tabSales Invoice Item ` . cost_center , ` tabSales Invoice Item ` . serial_and_batch_bundle
2017-03-12 17:10:41 +05:30
{ sales_person_cols }
2022-05-31 10:46:56 -03:00
{ payment_term_cols }
2017-03-31 12:44:29 +05:30
from
2017-09-18 19:13:45 +05:30
` tabSales Invoice ` inner join ` tabSales Invoice Item `
on ` tabSales Invoice Item ` . parent = ` tabSales Invoice ` . name
2022-08-15 09:14:23 -03:00
join ` tabItem ` item on item . name = ` tabSales Invoice Item ` . item_code
2017-03-12 17:10:41 +05:30
{ sales_team_table }
2022-05-31 10:46:56 -03:00
{ payment_term_table }
2014-11-24 15:12:37 +05:30
where
2019-01-02 16:09:34 +05:00
` tabSales Invoice ` . docstatus = 1 and ` tabSales Invoice ` . is_opening != ' Yes ' { conditions } { match_cond }
2014-11-24 15:12:37 +05:30
order by
2017-03-14 14:56:24 +05:30
` tabSales Invoice ` . posting_date desc , ` tabSales Invoice ` . posting_time desc """ .format(
2017-03-31 12:44:29 +05:30
conditions = conditions ,
sales_person_cols = sales_person_cols ,
2017-03-14 14:56:24 +05:30
sales_team_table = sales_team_table ,
2022-05-31 10:46:56 -03:00
payment_term_cols = payment_term_cols ,
payment_term_table = payment_term_table ,
2017-03-14 14:56:24 +05:30
match_cond = get_match_cond ( " Sales Invoice " ) ,
2022-03-28 18:52:46 +05:30
) ,
2017-03-14 14:56:24 +05:30
self . filters ,
as_dict = 1 ,
)
2014-11-24 15:12:37 +05:30
2022-10-27 10:55:04 +05:30
def get_delivery_notes ( self ) :
self . delivery_notes = frappe . _dict ( { } )
if self . si_list :
invoices = [ x . parent for x in self . si_list ]
dni = qb . DocType ( " Delivery Note Item " )
delivery_notes = (
qb . from_ ( dni )
. select (
dni . against_sales_invoice . as_ ( " sales_invoice " ) ,
dni . item_code ,
dni . warehouse ,
dni . parent . as_ ( " delivery_note " ) ,
dni . name . as_ ( " item_row " ) ,
)
. where ( ( dni . docstatus == 1 ) & ( dni . against_sales_invoice . isin ( invoices ) ) )
. groupby ( dni . against_sales_invoice , dni . item_code )
. orderby ( dni . creation , order = Order . desc )
. run ( as_dict = True )
)
for entry in delivery_notes :
self . delivery_notes [ ( entry . sales_invoice , entry . item_code ) ] = entry
2021-08-20 02:20:18 +05:30
def group_items_by_invoice ( self ) :
2021-08-20 06:07:53 +05:30
"""
Turns list of Sales Invoice Items to a tree of Sales Invoices with their Items as children .
"""
2023-04-30 16:35:12 +05:30
grouped = OrderedDict ( )
2021-08-20 02:20:18 +05:30
for row in self . si_list :
2023-04-30 16:35:12 +05:30
# initialize list with a header row for each new parent
grouped . setdefault ( row . parent , [ self . get_invoice_row ( row ) ] ) . append (
row . update (
{ " indent " : 1.0 , " parent_invoice " : row . parent , " invoice_or_item " : row . item_code }
) # descendant rows will have indent: 1.0 or greater
)
2021-08-20 02:20:18 +05:30
2023-04-30 16:35:12 +05:30
# if item is a bundle, add it's components as seperate rows
if frappe . db . exists ( " Product Bundle " , row . item_code ) :
bundled_items = self . get_bundle_items ( row )
for x in bundled_items :
bundle_item = self . get_bundle_item_row ( row , x )
grouped . get ( row . parent ) . append ( bundle_item )
2021-08-20 02:20:18 +05:30
2023-04-30 16:35:12 +05:30
self . si_list . clear ( )
2021-08-20 06:07:53 +05:30
2023-04-30 16:35:12 +05:30
for items in grouped . values ( ) :
self . si_list . extend ( items )
2021-08-25 15:33:53 +05:30
def get_invoice_row ( self , row ) :
2023-04-30 16:35:12 +05:30
# header row format
2021-08-25 15:33:53 +05:30
return frappe . _dict (
{
" parent_invoice " : " " ,
" indent " : 0.0 ,
2021-11-22 12:34:46 +05:30
" invoice_or_item " : row . parent ,
" parent " : None ,
2021-08-25 15:33:53 +05:30
" posting_date " : row . posting_date ,
" posting_time " : row . posting_time ,
" project " : row . project ,
" update_stock " : row . update_stock ,
" customer " : row . customer ,
" customer_group " : row . customer_group ,
" item_code " : None ,
" item_name " : None ,
" description " : None ,
" warehouse " : None ,
" item_group " : None ,
" brand " : None ,
" dn_detail " : None ,
" delivery_note " : None ,
" qty " : None ,
" item_row " : None ,
" is_return " : row . is_return ,
" cost_center " : row . cost_center ,
" base_net_amount " : frappe . db . get_value ( " Sales Invoice " , row . parent , " base_net_total " ) ,
}
)
2021-09-02 16:44:59 +05:30
2021-08-25 15:33:53 +05:30
def get_bundle_items ( self , product_bundle ) :
return frappe . get_all (
2021-08-20 06:07:53 +05:30
" Product Bundle Item " , filters = { " parent " : product_bundle . item_code } , fields = [ " item_code " , " qty " ]
)
2021-08-25 15:33:53 +05:30
def get_bundle_item_row ( self , product_bundle , item ) :
item_name , description , item_group , brand = self . get_bundle_item_details ( item . item_code )
return frappe . _dict (
{
" parent_invoice " : product_bundle . item_code ,
" indent " : product_bundle . indent + 1 ,
2021-11-22 12:34:46 +05:30
" parent " : None ,
" invoice_or_item " : item . item_code ,
2021-08-25 15:33:53 +05:30
" posting_date " : product_bundle . posting_date ,
" posting_time " : product_bundle . posting_time ,
" project " : product_bundle . project ,
" customer " : product_bundle . customer ,
" customer_group " : product_bundle . customer_group ,
" item_code " : item . item_code ,
" item_name " : item_name ,
" description " : description ,
" warehouse " : product_bundle . warehouse ,
" item_group " : item_group ,
" brand " : brand ,
" dn_detail " : product_bundle . dn_detail ,
" delivery_note " : product_bundle . delivery_note ,
" qty " : ( flt ( product_bundle . qty ) * flt ( item . qty ) ) ,
" item_row " : None ,
" is_return " : product_bundle . is_return ,
" cost_center " : product_bundle . cost_center ,
}
)
2021-08-20 06:07:53 +05:30
2021-08-24 23:38:24 +05:30
def get_bundle_item_details ( self , item_code ) :
return frappe . db . get_value (
" Item " , item_code , [ " item_name " , " description " , " item_group " , " brand " ]
)
2023-02-03 16:24:52 +05:30
def get_stock_ledger_entries ( self , item_code , warehouse ) :
if item_code and warehouse :
if ( item_code , warehouse ) not in self . sle :
sle = qb . DocType ( " Stock Ledger Entry " )
res = (
qb . from_ ( sle )
. select (
sle . item_code ,
sle . voucher_type ,
sle . voucher_no ,
sle . voucher_detail_no ,
sle . stock_value ,
sle . warehouse ,
sle . actual_qty . as_ ( " qty " ) ,
)
. where (
( sle . company == self . filters . company )
& ( sle . item_code == item_code )
& ( sle . warehouse == warehouse )
& ( sle . is_cancelled == 0 )
)
. orderby ( sle . item_code )
. orderby ( sle . warehouse , sle . posting_date , sle . posting_time , sle . creation , order = Order . desc )
. run ( as_dict = True )
)
self . sle [ ( item_code , warehouse ) ] = res
2014-11-24 15:12:37 +05:30
2023-02-03 16:24:52 +05:30
return self . sle [ ( item_code , warehouse ) ]
return [ ]
2014-11-24 15:12:37 +05:30
2015-07-07 13:59:23 +05:30
def load_product_bundle ( self ) :
self . product_bundles = { }
2014-11-24 15:12:37 +05:30
2023-01-09 22:45:43 +05:30
pki = qb . DocType ( " Packed Item " )
pki_query = (
frappe . qb . from_ ( pki )
. select (
pki . parenttype ,
pki . parent ,
pki . parent_item ,
pki . item_code ,
pki . warehouse ,
( - 1 * pki . qty ) . as_ ( " total_qty " ) ,
pki . rate ,
( pki . rate * pki . qty ) . as_ ( " base_amount " ) ,
pki . parent_detail_docname ,
)
. where ( pki . docstatus == 1 )
)
for d in pki_query . run ( as_dict = True ) :
2015-07-07 13:59:23 +05:30
self . product_bundles . setdefault ( d . parenttype , frappe . _dict ( ) ) . setdefault (
2014-11-24 15:12:37 +05:30
d . parent , frappe . _dict ( )
) . setdefault ( d . parent_item , [ ] ) . append ( d )
def load_non_stock_items ( self ) :
self . non_stock_items = frappe . db . sql_list (
""" select name from tabItem
2015-07-24 15:16:25 +05:30
where is_stock_item = 0 """
)