Merge pull request #33075 from frappe/version-13-hotfix
chore: release v13
This commit is contained in:
@@ -16,8 +16,8 @@ repos:
|
|||||||
- id: check-merge-conflict
|
- id: check-merge-conflict
|
||||||
- id: check-ast
|
- id: check-ast
|
||||||
|
|
||||||
- repo: https://gitlab.com/pycqa/flake8
|
- repo: https://github.com/PyCQA/flake8
|
||||||
rev: 3.9.2
|
rev: 5.0.4
|
||||||
hooks:
|
hooks:
|
||||||
- id: flake8
|
- id: flake8
|
||||||
additional_dependencies: [
|
additional_dependencies: [
|
||||||
|
|||||||
@@ -45,21 +45,6 @@ frappe.ui.form.on("Journal Entry Template", {
|
|||||||
|
|
||||||
frm.trigger("clear_child");
|
frm.trigger("clear_child");
|
||||||
switch(frm.doc.voucher_type){
|
switch(frm.doc.voucher_type){
|
||||||
case "Opening Entry":
|
|
||||||
frm.set_value("is_opening", "Yes");
|
|
||||||
frappe.call({
|
|
||||||
type:"GET",
|
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_opening_accounts",
|
|
||||||
args: {
|
|
||||||
"company": frm.doc.company
|
|
||||||
},
|
|
||||||
callback: function(r) {
|
|
||||||
if(r.message) {
|
|
||||||
add_accounts(frm.doc, r.message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
break;
|
|
||||||
case "Bank Entry":
|
case "Bank Entry":
|
||||||
case "Cash Entry":
|
case "Cash Entry":
|
||||||
frappe.call({
|
frappe.call({
|
||||||
|
|||||||
@@ -20,7 +20,6 @@ frappe.ui.form.on('Opening Invoice Creation Tool', {
|
|||||||
frm.dashboard.reset();
|
frm.dashboard.reset();
|
||||||
frm.doc.import_in_progress = true;
|
frm.doc.import_in_progress = true;
|
||||||
}
|
}
|
||||||
if (data.user != frappe.session.user) return;
|
|
||||||
if (data.count == data.total) {
|
if (data.count == data.total) {
|
||||||
setTimeout((title) => {
|
setTimeout((title) => {
|
||||||
frm.doc.import_in_progress = false;
|
frm.doc.import_in_progress = false;
|
||||||
|
|||||||
@@ -271,10 +271,10 @@ def publish(index, total, doctype):
|
|||||||
dict(
|
dict(
|
||||||
title=_("Opening Invoice Creation In Progress"),
|
title=_("Opening Invoice Creation In Progress"),
|
||||||
message=_("Creating {} out of {} {}").format(index + 1, total, doctype),
|
message=_("Creating {} out of {} {}").format(index + 1, total, doctype),
|
||||||
user=frappe.session.user,
|
|
||||||
count=index + 1,
|
count=index + 1,
|
||||||
total=total,
|
total=total,
|
||||||
),
|
),
|
||||||
|
user=frappe.session.user,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ frappe.ui.form.on('POS Closing Entry', {
|
|||||||
|
|
||||||
frappe.realtime.on('closing_process_complete', async function(data) {
|
frappe.realtime.on('closing_process_complete', async function(data) {
|
||||||
await frm.reload_doc();
|
await frm.reload_doc();
|
||||||
if (frm.doc.status == 'Failed' && frm.doc.error_message && data.user == frappe.session.user) {
|
if (frm.doc.status == 'Failed' && frm.doc.error_message) {
|
||||||
frappe.msgprint({
|
frappe.msgprint({
|
||||||
title: __('POS Closing Failed'),
|
title: __('POS Closing Failed'),
|
||||||
message: frm.doc.error_message,
|
message: frm.doc.error_message,
|
||||||
|
|||||||
@@ -432,7 +432,7 @@ def create_merge_logs(invoice_by_customer, closing_entry=None):
|
|||||||
|
|
||||||
finally:
|
finally:
|
||||||
frappe.db.commit()
|
frappe.db.commit()
|
||||||
frappe.publish_realtime("closing_process_complete", {"user": frappe.session.user})
|
frappe.publish_realtime("closing_process_complete", user=frappe.session.user)
|
||||||
|
|
||||||
|
|
||||||
def cancel_merge_logs(merge_logs, closing_entry=None):
|
def cancel_merge_logs(merge_logs, closing_entry=None):
|
||||||
@@ -459,7 +459,7 @@ def cancel_merge_logs(merge_logs, closing_entry=None):
|
|||||||
|
|
||||||
finally:
|
finally:
|
||||||
frappe.db.commit()
|
frappe.db.commit()
|
||||||
frappe.publish_realtime("closing_process_complete", {"user": frappe.session.user})
|
frappe.publish_realtime("closing_process_complete", user=frappe.session.user)
|
||||||
|
|
||||||
|
|
||||||
def enqueue_job(job, **kwargs):
|
def enqueue_job(job, **kwargs):
|
||||||
|
|||||||
@@ -3,7 +3,8 @@
|
|||||||
|
|
||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
from frappe import _, scrub
|
from frappe import _, qb, scrub
|
||||||
|
from frappe.query_builder import Order
|
||||||
from frappe.utils import cint, flt
|
from frappe.utils import cint, flt
|
||||||
|
|
||||||
from erpnext.controllers.queries import get_match_cond
|
from erpnext.controllers.queries import get_match_cond
|
||||||
@@ -367,6 +368,7 @@ class GrossProfitGenerator(object):
|
|||||||
self.average_buying_rate = {}
|
self.average_buying_rate = {}
|
||||||
self.filters = frappe._dict(filters)
|
self.filters = frappe._dict(filters)
|
||||||
self.load_invoice_items()
|
self.load_invoice_items()
|
||||||
|
self.get_delivery_notes()
|
||||||
|
|
||||||
if filters.group_by == "Invoice":
|
if filters.group_by == "Invoice":
|
||||||
self.group_items_by_invoice()
|
self.group_items_by_invoice()
|
||||||
@@ -535,6 +537,21 @@ class GrossProfitGenerator(object):
|
|||||||
|
|
||||||
return flt(buying_amount, self.currency_precision)
|
return flt(buying_amount, self.currency_precision)
|
||||||
|
|
||||||
|
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)
|
||||||
|
|
||||||
def get_buying_amount(self, row, item_code):
|
def get_buying_amount(self, row, item_code):
|
||||||
# IMP NOTE
|
# IMP NOTE
|
||||||
# stock_ledger_entries should already be filtered by item_code and warehouse and
|
# stock_ledger_entries should already be filtered by item_code and warehouse and
|
||||||
@@ -551,19 +568,22 @@ class GrossProfitGenerator(object):
|
|||||||
if row.dn_detail:
|
if row.dn_detail:
|
||||||
parenttype, parent = "Delivery Note", row.delivery_note
|
parenttype, parent = "Delivery Note", row.delivery_note
|
||||||
|
|
||||||
for i, sle in enumerate(my_sle):
|
return self.calculate_buying_amount_from_sle(
|
||||||
# find the stock valution rate from stock ledger entry
|
row, my_sle, parenttype, parent, row.item_row, item_code
|
||||||
if (
|
)
|
||||||
sle.voucher_type == parenttype
|
elif self.delivery_notes.get((row.parent, row.item_code), None):
|
||||||
and parent == sle.voucher_no
|
# check if Invoice has delivery notes
|
||||||
and sle.voucher_detail_no == row.item_row
|
dn = self.delivery_notes.get((row.parent, row.item_code))
|
||||||
):
|
parenttype, parent, item_row, warehouse = (
|
||||||
previous_stock_value = len(my_sle) > i + 1 and flt(my_sle[i + 1].stock_value) or 0.0
|
"Delivery Note",
|
||||||
|
dn["delivery_note"],
|
||||||
if previous_stock_value:
|
dn["item_row"],
|
||||||
return abs(previous_stock_value - flt(sle.stock_value)) * flt(row.qty) / abs(flt(sle.qty))
|
dn["warehouse"],
|
||||||
else:
|
)
|
||||||
return flt(row.qty) * self.get_average_buying_rate(row, item_code)
|
my_sle = self.sle.get((item_code, warehouse))
|
||||||
|
return self.calculate_buying_amount_from_sle(
|
||||||
|
row, my_sle, parenttype, parent, item_row, item_code
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
return flt(row.qty) * self.get_average_buying_rate(row, item_code)
|
return flt(row.qty) * self.get_average_buying_rate(row, item_code)
|
||||||
|
|
||||||
@@ -667,6 +687,29 @@ class GrossProfitGenerator(object):
|
|||||||
as_dict=1,
|
as_dict=1,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
def group_items_by_invoice(self):
|
def group_items_by_invoice(self):
|
||||||
"""
|
"""
|
||||||
Turns list of Sales Invoice Items to a tree of Sales Invoices with their Items as children.
|
Turns list of Sales Invoice Items to a tree of Sales Invoices with their Items as children.
|
||||||
|
|||||||
208
erpnext/accounts/report/gross_profit/test_gross_profit.py
Normal file
208
erpnext/accounts/report/gross_profit/test_gross_profit.py
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
import frappe
|
||||||
|
from frappe import qb
|
||||||
|
from frappe.tests.utils import FrappeTestCase
|
||||||
|
from frappe.utils import add_days, flt, nowdate
|
||||||
|
|
||||||
|
from erpnext.accounts.doctype.sales_invoice.sales_invoice import make_delivery_note
|
||||||
|
from erpnext.accounts.doctype.sales_invoice.test_sales_invoice import create_sales_invoice
|
||||||
|
from erpnext.accounts.report.gross_profit.gross_profit import execute
|
||||||
|
from erpnext.stock.doctype.item.test_item import create_item
|
||||||
|
from erpnext.stock.doctype.stock_entry.stock_entry_utils import make_stock_entry
|
||||||
|
|
||||||
|
|
||||||
|
class TestGrossProfit(FrappeTestCase):
|
||||||
|
def setUp(self):
|
||||||
|
self.create_company()
|
||||||
|
self.create_item()
|
||||||
|
self.create_customer()
|
||||||
|
self.create_sales_invoice()
|
||||||
|
self.clear_old_entries()
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
frappe.db.rollback()
|
||||||
|
|
||||||
|
def create_company(self):
|
||||||
|
company_name = "_Test Gross Profit"
|
||||||
|
abbr = "_GP"
|
||||||
|
if frappe.db.exists("Company", company_name):
|
||||||
|
company = frappe.get_doc("Company", company_name)
|
||||||
|
else:
|
||||||
|
company = frappe.get_doc(
|
||||||
|
{
|
||||||
|
"doctype": "Company",
|
||||||
|
"company_name": company_name,
|
||||||
|
"country": "India",
|
||||||
|
"default_currency": "INR",
|
||||||
|
"create_chart_of_accounts_based_on": "Standard Template",
|
||||||
|
"chart_of_accounts": "Standard",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
company = company.save()
|
||||||
|
|
||||||
|
self.company = company.name
|
||||||
|
self.cost_center = company.cost_center
|
||||||
|
self.warehouse = "Stores - " + abbr
|
||||||
|
self.income_account = "Sales - " + abbr
|
||||||
|
self.expense_account = "Cost of Goods Sold - " + abbr
|
||||||
|
self.debit_to = "Debtors - " + abbr
|
||||||
|
self.creditors = "Creditors - " + abbr
|
||||||
|
|
||||||
|
def create_item(self):
|
||||||
|
item = create_item(
|
||||||
|
item_code="_Test GP Item", is_stock_item=1, company=self.company, warehouse=self.warehouse
|
||||||
|
)
|
||||||
|
self.item = item if isinstance(item, str) else item.item_code
|
||||||
|
|
||||||
|
def create_customer(self):
|
||||||
|
name = "_Test GP Customer"
|
||||||
|
if frappe.db.exists("Customer", name):
|
||||||
|
self.customer = name
|
||||||
|
else:
|
||||||
|
customer = frappe.new_doc("Customer")
|
||||||
|
customer.customer_name = name
|
||||||
|
customer.type = "Individual"
|
||||||
|
customer.save()
|
||||||
|
self.customer = customer.name
|
||||||
|
|
||||||
|
def create_sales_invoice(
|
||||||
|
self, qty=1, rate=100, posting_date=nowdate(), do_not_save=False, do_not_submit=False
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Helper function to populate default values in sales invoice
|
||||||
|
"""
|
||||||
|
sinv = create_sales_invoice(
|
||||||
|
qty=qty,
|
||||||
|
rate=rate,
|
||||||
|
company=self.company,
|
||||||
|
customer=self.customer,
|
||||||
|
item_code=self.item,
|
||||||
|
item_name=self.item,
|
||||||
|
cost_center=self.cost_center,
|
||||||
|
warehouse=self.warehouse,
|
||||||
|
debit_to=self.debit_to,
|
||||||
|
parent_cost_center=self.cost_center,
|
||||||
|
update_stock=0,
|
||||||
|
currency="INR",
|
||||||
|
is_pos=0,
|
||||||
|
is_return=0,
|
||||||
|
return_against=None,
|
||||||
|
income_account=self.income_account,
|
||||||
|
expense_account=self.expense_account,
|
||||||
|
do_not_save=do_not_save,
|
||||||
|
do_not_submit=do_not_submit,
|
||||||
|
)
|
||||||
|
return sinv
|
||||||
|
|
||||||
|
def clear_old_entries(self):
|
||||||
|
doctype_list = [
|
||||||
|
"Sales Invoice",
|
||||||
|
"GL Entry",
|
||||||
|
"Stock Entry",
|
||||||
|
"Stock Ledger Entry",
|
||||||
|
"Delivery Note",
|
||||||
|
]
|
||||||
|
for doctype in doctype_list:
|
||||||
|
qb.from_(qb.DocType(doctype)).delete().where(qb.DocType(doctype).company == self.company).run()
|
||||||
|
|
||||||
|
def test_invoice_without_only_delivery_note(self):
|
||||||
|
"""
|
||||||
|
Test buying amount for Invoice without `update_stock` flag set but has Delivery Note
|
||||||
|
"""
|
||||||
|
se = make_stock_entry(
|
||||||
|
company=self.company,
|
||||||
|
item_code=self.item,
|
||||||
|
target=self.warehouse,
|
||||||
|
qty=1,
|
||||||
|
basic_rate=100,
|
||||||
|
do_not_submit=True,
|
||||||
|
)
|
||||||
|
item = se.items[0]
|
||||||
|
se.append(
|
||||||
|
"items",
|
||||||
|
{
|
||||||
|
"item_code": item.item_code,
|
||||||
|
"s_warehouse": item.s_warehouse,
|
||||||
|
"t_warehouse": item.t_warehouse,
|
||||||
|
"qty": 1,
|
||||||
|
"basic_rate": 200,
|
||||||
|
"conversion_factor": item.conversion_factor or 1.0,
|
||||||
|
"transfer_qty": flt(item.qty) * (flt(item.conversion_factor) or 1.0),
|
||||||
|
"serial_no": item.serial_no,
|
||||||
|
"batch_no": item.batch_no,
|
||||||
|
"cost_center": item.cost_center,
|
||||||
|
"expense_account": item.expense_account,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
se = se.save().submit()
|
||||||
|
|
||||||
|
sinv = create_sales_invoice(
|
||||||
|
qty=1,
|
||||||
|
rate=100,
|
||||||
|
company=self.company,
|
||||||
|
customer=self.customer,
|
||||||
|
item_code=self.item,
|
||||||
|
item_name=self.item,
|
||||||
|
cost_center=self.cost_center,
|
||||||
|
warehouse=self.warehouse,
|
||||||
|
debit_to=self.debit_to,
|
||||||
|
parent_cost_center=self.cost_center,
|
||||||
|
update_stock=0,
|
||||||
|
currency="INR",
|
||||||
|
income_account=self.income_account,
|
||||||
|
expense_account=self.expense_account,
|
||||||
|
)
|
||||||
|
|
||||||
|
filters = frappe._dict(
|
||||||
|
company=self.company, from_date=nowdate(), to_date=nowdate(), group_by="Invoice"
|
||||||
|
)
|
||||||
|
|
||||||
|
columns, data = execute(filters=filters)
|
||||||
|
|
||||||
|
# Without Delivery Note, buying rate should be 150
|
||||||
|
expected_entry_without_dn = {
|
||||||
|
"parent_invoice": sinv.name,
|
||||||
|
"currency": "INR",
|
||||||
|
"sales_invoice": self.item,
|
||||||
|
"customer": self.customer,
|
||||||
|
"posting_date": frappe.utils.datetime.date.fromisoformat(nowdate()),
|
||||||
|
"item_code": self.item,
|
||||||
|
"item_name": self.item,
|
||||||
|
"warehouse": "Stores - _GP",
|
||||||
|
"qty": 1.0,
|
||||||
|
"avg._selling_rate": 100.0,
|
||||||
|
"valuation_rate": 150.0,
|
||||||
|
"selling_amount": 100.0,
|
||||||
|
"buying_amount": 150.0,
|
||||||
|
"gross_profit": -50.0,
|
||||||
|
"gross_profit_%": -50.0,
|
||||||
|
}
|
||||||
|
gp_entry = [x for x in data if x.parent_invoice == sinv.name]
|
||||||
|
self.assertDictContainsSubset(expected_entry_without_dn, gp_entry[0])
|
||||||
|
|
||||||
|
# make delivery note
|
||||||
|
dn = make_delivery_note(sinv.name)
|
||||||
|
dn.items[0].qty = 1
|
||||||
|
dn = dn.save().submit()
|
||||||
|
|
||||||
|
columns, data = execute(filters=filters)
|
||||||
|
|
||||||
|
# Without Delivery Note, buying rate should be 100
|
||||||
|
expected_entry_with_dn = {
|
||||||
|
"parent_invoice": sinv.name,
|
||||||
|
"currency": "INR",
|
||||||
|
"sales_invoice": self.item,
|
||||||
|
"customer": self.customer,
|
||||||
|
"posting_date": frappe.utils.datetime.date.fromisoformat(nowdate()),
|
||||||
|
"item_code": self.item,
|
||||||
|
"item_name": self.item,
|
||||||
|
"warehouse": "Stores - _GP",
|
||||||
|
"qty": 1.0,
|
||||||
|
"avg._selling_rate": 100.0,
|
||||||
|
"valuation_rate": 100.0,
|
||||||
|
"selling_amount": 100.0,
|
||||||
|
"buying_amount": 100.0,
|
||||||
|
"gross_profit": 0.0,
|
||||||
|
"gross_profit_%": 0.0,
|
||||||
|
}
|
||||||
|
gp_entry = [x for x in data if x.parent_invoice == sinv.name]
|
||||||
|
self.assertDictContainsSubset(expected_entry_with_dn, gp_entry[0])
|
||||||
@@ -232,12 +232,12 @@ def get_conditions(filters):
|
|||||||
|
|
||||||
conditions += (
|
conditions += (
|
||||||
common_condition
|
common_condition
|
||||||
+ "and ifnull(`tabPurchase Invoice Item`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
+ "and ifnull(`tabPurchase Invoice`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
conditions += (
|
conditions += (
|
||||||
common_condition
|
common_condition
|
||||||
+ "and ifnull(`tabPurchase Invoice Item`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
+ "and ifnull(`tabPurchase Invoice`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
||||||
)
|
)
|
||||||
|
|
||||||
return conditions
|
return conditions
|
||||||
|
|||||||
@@ -390,12 +390,12 @@ def get_conditions(filters):
|
|||||||
|
|
||||||
conditions += (
|
conditions += (
|
||||||
common_condition
|
common_condition
|
||||||
+ "and ifnull(`tabSales Invoice Item`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
+ "and ifnull(`tabSales Invoice`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
conditions += (
|
conditions += (
|
||||||
common_condition
|
common_condition
|
||||||
+ "and ifnull(`tabSales Invoice Item`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
+ "and ifnull(`tabSales Invoice`.{0}, '') in %({0})s)".format(dimension.fieldname)
|
||||||
)
|
)
|
||||||
|
|
||||||
return conditions
|
return conditions
|
||||||
|
|||||||
@@ -200,11 +200,11 @@ def get_children(doctype, parent=None, location=None, is_root=False):
|
|||||||
name as value,
|
name as value,
|
||||||
is_group as expandable
|
is_group as expandable
|
||||||
from
|
from
|
||||||
`tab{doctype}` comp
|
`tabLocation` comp
|
||||||
where
|
where
|
||||||
ifnull(parent_location, "")={parent}
|
ifnull(parent_location, "")={parent}
|
||||||
""".format(
|
""".format(
|
||||||
doctype=doctype, parent=frappe.db.escape(parent)
|
parent=frappe.db.escape(parent)
|
||||||
),
|
),
|
||||||
as_dict=1,
|
as_dict=1,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1118,7 +1118,8 @@
|
|||||||
"fetch_from": "supplier.is_internal_supplier",
|
"fetch_from": "supplier.is_internal_supplier",
|
||||||
"fieldname": "is_internal_supplier",
|
"fieldname": "is_internal_supplier",
|
||||||
"fieldtype": "Check",
|
"fieldtype": "Check",
|
||||||
"label": "Is Internal Supplier"
|
"label": "Is Internal Supplier",
|
||||||
|
"read_only": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fetch_from": "supplier.represents_company",
|
"fetch_from": "supplier.represents_company",
|
||||||
@@ -1169,7 +1170,7 @@
|
|||||||
"idx": 105,
|
"idx": 105,
|
||||||
"is_submittable": 1,
|
"is_submittable": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2022-09-16 17:45:04.954055",
|
"modified": "2022-11-17 12:34:36.033363",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Buying",
|
"module": "Buying",
|
||||||
"name": "Purchase Order",
|
"name": "Purchase Order",
|
||||||
|
|||||||
@@ -479,7 +479,7 @@ def get_rfq_containing_supplier(doctype, txt, searchfield, start, page_len, filt
|
|||||||
conditions += "and rfq.transaction_date = '{0}'".format(filters.get("transaction_date"))
|
conditions += "and rfq.transaction_date = '{0}'".format(filters.get("transaction_date"))
|
||||||
|
|
||||||
rfq_data = frappe.db.sql(
|
rfq_data = frappe.db.sql(
|
||||||
"""
|
f"""
|
||||||
select
|
select
|
||||||
distinct rfq.name, rfq.transaction_date,
|
distinct rfq.name, rfq.transaction_date,
|
||||||
rfq.company
|
rfq.company
|
||||||
@@ -487,15 +487,18 @@ def get_rfq_containing_supplier(doctype, txt, searchfield, start, page_len, filt
|
|||||||
`tabRequest for Quotation` rfq, `tabRequest for Quotation Supplier` rfq_supplier
|
`tabRequest for Quotation` rfq, `tabRequest for Quotation Supplier` rfq_supplier
|
||||||
where
|
where
|
||||||
rfq.name = rfq_supplier.parent
|
rfq.name = rfq_supplier.parent
|
||||||
and rfq_supplier.supplier = '{0}'
|
and rfq_supplier.supplier = %(supplier)s
|
||||||
and rfq.docstatus = 1
|
and rfq.docstatus = 1
|
||||||
and rfq.company = '{1}'
|
and rfq.company = %(company)s
|
||||||
{2}
|
{conditions}
|
||||||
order by rfq.transaction_date ASC
|
order by rfq.transaction_date ASC
|
||||||
limit %(page_len)s offset %(start)s """.format(
|
limit %(page_len)s offset %(start)s """,
|
||||||
filters.get("supplier"), filters.get("company"), conditions
|
{
|
||||||
),
|
"page_len": page_len,
|
||||||
{"page_len": page_len, "start": start},
|
"start": start,
|
||||||
|
"company": filters.get("company"),
|
||||||
|
"supplier": filters.get("supplier"),
|
||||||
|
},
|
||||||
as_dict=1,
|
as_dict=1,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -1345,7 +1345,7 @@ class QuickBooksMigrator(Document):
|
|||||||
)[0]["name"]
|
)[0]["name"]
|
||||||
|
|
||||||
def _publish(self, *args, **kwargs):
|
def _publish(self, *args, **kwargs):
|
||||||
frappe.publish_realtime("quickbooks_progress_update", *args, **kwargs)
|
frappe.publish_realtime("quickbooks_progress_update", *args, **kwargs, user=self.modified_by)
|
||||||
|
|
||||||
def _get_unique_account_name(self, quickbooks_name, number=0):
|
def _get_unique_account_name(self, quickbooks_name, number=0):
|
||||||
if number:
|
if number:
|
||||||
|
|||||||
@@ -302,6 +302,7 @@ class TallyMigration(Document):
|
|||||||
frappe.publish_realtime(
|
frappe.publish_realtime(
|
||||||
"tally_migration_progress_update",
|
"tally_migration_progress_update",
|
||||||
{"title": title, "message": message, "count": count, "total": total},
|
{"title": title, "message": message, "count": count, "total": total},
|
||||||
|
user=self.modified_by,
|
||||||
)
|
)
|
||||||
|
|
||||||
def _import_master_data(self):
|
def _import_master_data(self):
|
||||||
|
|||||||
@@ -302,7 +302,7 @@ def check_for_nexus(doc, tax_dict):
|
|||||||
item.tax_collectable = flt(0)
|
item.tax_collectable = flt(0)
|
||||||
item.taxable_amount = flt(0)
|
item.taxable_amount = flt(0)
|
||||||
|
|
||||||
for tax in doc.taxes:
|
for tax in list(doc.taxes):
|
||||||
if tax.account_head == TAX_ACCOUNT_HEAD:
|
if tax.account_head == TAX_ACCOUNT_HEAD:
|
||||||
doc.taxes.remove(tax)
|
doc.taxes.remove(tax)
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -70,11 +70,11 @@ def get_children(doctype, parent=None, company=None, is_root=False):
|
|||||||
select
|
select
|
||||||
name as value,
|
name as value,
|
||||||
is_group as expandable
|
is_group as expandable
|
||||||
from `tab{doctype}`
|
from `tabDepartment`
|
||||||
where
|
where
|
||||||
{condition}
|
{condition}
|
||||||
order by name""".format(
|
order by name""".format(
|
||||||
doctype=doctype, condition=condition
|
condition=condition
|
||||||
),
|
),
|
||||||
var_dict,
|
var_dict,
|
||||||
as_dict=1,
|
as_dict=1,
|
||||||
|
|||||||
@@ -652,23 +652,13 @@ class ProductionPlan(Document):
|
|||||||
else:
|
else:
|
||||||
material_request = material_request_map[key]
|
material_request = material_request_map[key]
|
||||||
|
|
||||||
conversion_factor = 1.0
|
|
||||||
if (
|
|
||||||
material_request_type == "Purchase"
|
|
||||||
and item_doc.purchase_uom
|
|
||||||
and item_doc.purchase_uom != item_doc.stock_uom
|
|
||||||
):
|
|
||||||
conversion_factor = (
|
|
||||||
get_conversion_factor(item_doc.name, item_doc.purchase_uom).get("conversion_factor") or 1.0
|
|
||||||
)
|
|
||||||
|
|
||||||
# add item
|
# add item
|
||||||
material_request.append(
|
material_request.append(
|
||||||
"items",
|
"items",
|
||||||
{
|
{
|
||||||
"item_code": item.item_code,
|
"item_code": item.item_code,
|
||||||
"from_warehouse": item.from_warehouse,
|
"from_warehouse": item.from_warehouse,
|
||||||
"qty": item.quantity / conversion_factor,
|
"qty": item.quantity,
|
||||||
"schedule_date": schedule_date,
|
"schedule_date": schedule_date,
|
||||||
"warehouse": item.warehouse,
|
"warehouse": item.warehouse,
|
||||||
"sales_order": item.sales_order,
|
"sales_order": item.sales_order,
|
||||||
@@ -988,11 +978,25 @@ def get_material_request_items(
|
|||||||
if include_safety_stock:
|
if include_safety_stock:
|
||||||
required_qty += flt(row["safety_stock"])
|
required_qty += flt(row["safety_stock"])
|
||||||
|
|
||||||
|
item_details = frappe.get_cached_value(
|
||||||
|
"Item", row.item_code, ["purchase_uom", "stock_uom"], as_dict=1
|
||||||
|
)
|
||||||
|
|
||||||
|
conversion_factor = 1.0
|
||||||
|
if (
|
||||||
|
row.get("default_material_request_type") == "Purchase"
|
||||||
|
and item_details.purchase_uom
|
||||||
|
and item_details.purchase_uom != item_details.stock_uom
|
||||||
|
):
|
||||||
|
conversion_factor = (
|
||||||
|
get_conversion_factor(row.item_code, item_details.purchase_uom).get("conversion_factor") or 1.0
|
||||||
|
)
|
||||||
|
|
||||||
if required_qty > 0:
|
if required_qty > 0:
|
||||||
return {
|
return {
|
||||||
"item_code": row.item_code,
|
"item_code": row.item_code,
|
||||||
"item_name": row.item_name,
|
"item_name": row.item_name,
|
||||||
"quantity": required_qty,
|
"quantity": required_qty / conversion_factor,
|
||||||
"required_bom_qty": total_qty,
|
"required_bom_qty": total_qty,
|
||||||
"stock_uom": row.get("stock_uom"),
|
"stock_uom": row.get("stock_uom"),
|
||||||
"warehouse": warehouse
|
"warehouse": warehouse
|
||||||
|
|||||||
@@ -752,6 +752,11 @@ class TestProductionPlan(FrappeTestCase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
pln.make_material_request()
|
pln.make_material_request()
|
||||||
|
|
||||||
|
for row in pln.mr_items:
|
||||||
|
self.assertEqual(row.uom, "Nos")
|
||||||
|
self.assertEqual(row.quantity, 1)
|
||||||
|
|
||||||
for row in frappe.get_all(
|
for row in frappe.get_all(
|
||||||
"Material Request Item",
|
"Material Request Item",
|
||||||
filters={"production_plan": pln.name},
|
filters={"production_plan": pln.name},
|
||||||
|
|||||||
@@ -159,6 +159,12 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.item-img {
|
||||||
|
@extend .image;
|
||||||
|
border-radius: 8px 8px 0 0;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
|
||||||
> .item-detail {
|
> .item-detail {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|||||||
@@ -79,7 +79,7 @@ def get_children(doctype, parent=None, parent_quality_procedure=None, is_root=Fa
|
|||||||
]
|
]
|
||||||
else:
|
else:
|
||||||
return frappe.get_all(
|
return frappe.get_all(
|
||||||
doctype,
|
"Quality Procedure",
|
||||||
fields=["name as value", "is_group as expandable"],
|
fields=["name as value", "is_group as expandable"],
|
||||||
filters=dict(parent_quality_procedure=parent),
|
filters=dict(parent_quality_procedure=parent),
|
||||||
order_by="name asc",
|
order_by="name asc",
|
||||||
|
|||||||
@@ -146,7 +146,9 @@ class ImportSupplierInvoice(Document):
|
|||||||
|
|
||||||
def publish(self, title, message, count, total):
|
def publish(self, title, message, count, total):
|
||||||
frappe.publish_realtime(
|
frappe.publish_realtime(
|
||||||
"import_invoice_update", {"title": title, "message": message, "count": count, "total": total}
|
"import_invoice_update",
|
||||||
|
{"title": title, "message": message, "count": count, "total": total},
|
||||||
|
user=self.modified_by,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
import json
|
import json
|
||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
|
from frappe.utils import cint
|
||||||
from frappe.utils.nestedset import get_root_of
|
from frappe.utils.nestedset import get_root_of
|
||||||
|
|
||||||
from erpnext.accounts.doctype.pos_invoice.pos_invoice import get_stock_availability
|
from erpnext.accounts.doctype.pos_invoice.pos_invoice import get_stock_availability
|
||||||
@@ -105,11 +106,11 @@ def get_items(start, page_length, price_list, item_group, pos_profile, search_te
|
|||||||
ORDER BY
|
ORDER BY
|
||||||
item.name asc
|
item.name asc
|
||||||
LIMIT
|
LIMIT
|
||||||
{start}, {page_length}""".format(
|
{page_length} offset {start}""".format(
|
||||||
start=start,
|
start=cint(start),
|
||||||
page_length=page_length,
|
page_length=cint(page_length),
|
||||||
lft=lft,
|
lft=cint(lft),
|
||||||
rgt=rgt,
|
rgt=cint(rgt),
|
||||||
condition=condition,
|
condition=condition,
|
||||||
bin_join_selection=bin_join_selection,
|
bin_join_selection=bin_join_selection,
|
||||||
bin_join_condition=bin_join_condition,
|
bin_join_condition=bin_join_condition,
|
||||||
|
|||||||
@@ -103,9 +103,9 @@ erpnext.PointOfSale.ItemSelector = class {
|
|||||||
<div class="flex items-center justify-center h-32 border-b-grey text-6xl text-grey-100">
|
<div class="flex items-center justify-center h-32 border-b-grey text-6xl text-grey-100">
|
||||||
<img
|
<img
|
||||||
onerror="cur_pos.item_selector.handle_broken_image(this)"
|
onerror="cur_pos.item_selector.handle_broken_image(this)"
|
||||||
class="h-full" src="${item_image}"
|
class="h-full item-img" src="${item_image}"
|
||||||
alt="${frappe.get_abbr(item.item_name)}"
|
alt="${frappe.get_abbr(item.item_name)}"
|
||||||
style="object-fit: cover;">
|
>
|
||||||
</div>`;
|
</div>`;
|
||||||
} else {
|
} else {
|
||||||
return `<div class="item-qty-pill">
|
return `<div class="item-qty-pill">
|
||||||
|
|||||||
@@ -611,11 +611,11 @@ def get_children(doctype, parent=None, company=None, is_root=False):
|
|||||||
name as value,
|
name as value,
|
||||||
is_group as expandable
|
is_group as expandable
|
||||||
from
|
from
|
||||||
`tab{doctype}` comp
|
`tabCompany` comp
|
||||||
where
|
where
|
||||||
ifnull(parent_company, "")={parent}
|
ifnull(parent_company, "")={parent}
|
||||||
""".format(
|
""".format(
|
||||||
doctype=doctype, parent=frappe.db.escape(parent)
|
parent=frappe.db.escape(parent)
|
||||||
),
|
),
|
||||||
as_dict=1,
|
as_dict=1,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import json
|
|||||||
import frappe
|
import frappe
|
||||||
from frappe import _, msgprint
|
from frappe import _, msgprint
|
||||||
from frappe.model.mapper import get_mapped_doc
|
from frappe.model.mapper import get_mapped_doc
|
||||||
from frappe.utils import cstr, flt, get_link_to_form, getdate, new_line_sep, nowdate
|
from frappe.utils import cint, cstr, flt, get_link_to_form, getdate, new_line_sep, nowdate
|
||||||
from six import string_types
|
from six import string_types
|
||||||
|
|
||||||
from erpnext.buying.utils import check_on_hold_or_closed_status, validate_for_items
|
from erpnext.buying.utils import check_on_hold_or_closed_status, validate_for_items
|
||||||
@@ -504,13 +504,13 @@ def get_material_requests_based_on_supplier(doctype, txt, searchfield, start, pa
|
|||||||
and mr.per_ordered < 99.99
|
and mr.per_ordered < 99.99
|
||||||
and mr.docstatus = 1
|
and mr.docstatus = 1
|
||||||
and mr.status != 'Stopped'
|
and mr.status != 'Stopped'
|
||||||
and mr.company = '{1}'
|
and mr.company = %s
|
||||||
{2}
|
{1}
|
||||||
order by mr_item.item_code ASC
|
order by mr_item.item_code ASC
|
||||||
limit {3} offset {4} """.format(
|
limit {2} offset {3} """.format(
|
||||||
", ".join(["%s"] * len(supplier_items)), filters.get("company"), conditions, page_len, start
|
", ".join(["%s"] * len(supplier_items)), conditions, cint(page_len), cint(start)
|
||||||
),
|
),
|
||||||
tuple(supplier_items),
|
tuple(supplier_items) + (filters.get("company"),),
|
||||||
as_dict=1,
|
as_dict=1,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import frappe
|
|||||||
from frappe import _
|
from frappe import _
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.model.mapper import get_mapped_doc
|
from frappe.model.mapper import get_mapped_doc
|
||||||
from frappe.utils import cint, flt
|
from frappe.utils import cint, cstr, flt
|
||||||
|
|
||||||
from erpnext.stock.doctype.quality_inspection_template.quality_inspection_template import (
|
from erpnext.stock.doctype.quality_inspection_template.quality_inspection_template import (
|
||||||
get_template_details,
|
get_template_details,
|
||||||
@@ -219,29 +219,32 @@ class QualityInspection(Document):
|
|||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
@frappe.validate_and_sanitize_search_inputs
|
@frappe.validate_and_sanitize_search_inputs
|
||||||
def item_query(doctype, txt, searchfield, start, page_len, filters):
|
def item_query(doctype, txt, searchfield, start, page_len, filters):
|
||||||
if filters.get("from"):
|
|
||||||
from frappe.desk.reportview import get_match_cond
|
from frappe.desk.reportview import get_match_cond
|
||||||
|
|
||||||
mcond = get_match_cond(filters["from"])
|
from_doctype = cstr(filters.get("doctype"))
|
||||||
|
if not from_doctype or not frappe.db.exists("DocType", from_doctype):
|
||||||
|
return []
|
||||||
|
|
||||||
|
mcond = get_match_cond(from_doctype)
|
||||||
cond, qi_condition = "", "and (quality_inspection is null or quality_inspection = '')"
|
cond, qi_condition = "", "and (quality_inspection is null or quality_inspection = '')"
|
||||||
|
|
||||||
if filters.get("parent"):
|
if filters.get("parent"):
|
||||||
if (
|
if (
|
||||||
filters.get("from") in ["Purchase Invoice Item", "Purchase Receipt Item"]
|
from_doctype in ["Purchase Invoice Item", "Purchase Receipt Item"]
|
||||||
and filters.get("inspection_type") != "In Process"
|
and filters.get("inspection_type") != "In Process"
|
||||||
):
|
):
|
||||||
cond = """and item_code in (select name from `tabItem` where
|
cond = """and item_code in (select name from `tabItem` where
|
||||||
inspection_required_before_purchase = 1)"""
|
inspection_required_before_purchase = 1)"""
|
||||||
elif (
|
elif (
|
||||||
filters.get("from") in ["Sales Invoice Item", "Delivery Note Item"]
|
from_doctype in ["Sales Invoice Item", "Delivery Note Item"]
|
||||||
and filters.get("inspection_type") != "In Process"
|
and filters.get("inspection_type") != "In Process"
|
||||||
):
|
):
|
||||||
cond = """and item_code in (select name from `tabItem` where
|
cond = """and item_code in (select name from `tabItem` where
|
||||||
inspection_required_before_delivery = 1)"""
|
inspection_required_before_delivery = 1)"""
|
||||||
elif filters.get("from") == "Stock Entry Detail":
|
elif from_doctype == "Stock Entry Detail":
|
||||||
cond = """and s_warehouse is null"""
|
cond = """and s_warehouse is null"""
|
||||||
|
|
||||||
if filters.get("from") in ["Supplier Quotation Item"]:
|
if from_doctype in ["Supplier Quotation Item"]:
|
||||||
qi_condition = ""
|
qi_condition = ""
|
||||||
|
|
||||||
return frappe.db.sql(
|
return frappe.db.sql(
|
||||||
@@ -250,13 +253,13 @@ def item_query(doctype, txt, searchfield, start, page_len, filters):
|
|||||||
FROM `tab{doc}`
|
FROM `tab{doc}`
|
||||||
WHERE parent=%(parent)s and docstatus < 2 and item_code like %(txt)s
|
WHERE parent=%(parent)s and docstatus < 2 and item_code like %(txt)s
|
||||||
{qi_condition} {cond} {mcond}
|
{qi_condition} {cond} {mcond}
|
||||||
ORDER BY item_code limit {start}, {page_len}
|
ORDER BY item_code limit {page_len} offset {start}
|
||||||
""".format(
|
""".format(
|
||||||
doc=filters.get("from"),
|
doc=from_doctype,
|
||||||
cond=cond,
|
cond=cond,
|
||||||
mcond=mcond,
|
mcond=mcond,
|
||||||
start=start,
|
start=cint(start),
|
||||||
page_len=page_len,
|
page_len=cint(page_len),
|
||||||
qi_condition=qi_condition,
|
qi_condition=qi_condition,
|
||||||
),
|
),
|
||||||
{"parent": filters.get("parent"), "txt": "%%%s%%" % txt},
|
{"parent": filters.get("parent"), "txt": "%%%s%%" % txt},
|
||||||
@@ -270,13 +273,13 @@ def item_query(doctype, txt, searchfield, start, page_len, filters):
|
|||||||
WHERE name = %(reference_name)s and docstatus < 2 and production_item like %(txt)s
|
WHERE name = %(reference_name)s and docstatus < 2 and production_item like %(txt)s
|
||||||
{qi_condition} {cond} {mcond}
|
{qi_condition} {cond} {mcond}
|
||||||
ORDER BY production_item
|
ORDER BY production_item
|
||||||
LIMIT {start}, {page_len}
|
limit {page_len} offset {start}
|
||||||
""".format(
|
""".format(
|
||||||
doc=filters.get("from"),
|
doc=from_doctype,
|
||||||
cond=cond,
|
cond=cond,
|
||||||
mcond=mcond,
|
mcond=mcond,
|
||||||
start=start,
|
start=cint(start),
|
||||||
page_len=page_len,
|
page_len=cint(page_len),
|
||||||
qi_condition=qi_condition,
|
qi_condition=qi_condition,
|
||||||
),
|
),
|
||||||
{"reference_name": filters.get("reference_name"), "txt": "%%%s%%" % txt},
|
{"reference_name": filters.get("reference_name"), "txt": "%%%s%%" % txt},
|
||||||
|
|||||||
@@ -355,7 +355,7 @@ def get_opening_balance(filters, columns, sl_entries):
|
|||||||
)
|
)
|
||||||
|
|
||||||
# check if any SLEs are actually Opening Stock Reconciliation
|
# check if any SLEs are actually Opening Stock Reconciliation
|
||||||
for sle in sl_entries:
|
for sle in list(sl_entries):
|
||||||
if (
|
if (
|
||||||
sle.get("voucher_type") == "Stock Reconciliation"
|
sle.get("voucher_type") == "Stock Reconciliation"
|
||||||
and sle.get("date").split()[0] == filters.from_date
|
and sle.get("date").split()[0] == filters.from_date
|
||||||
|
|||||||
Reference in New Issue
Block a user