fix: merge conflict
This commit is contained in:
		
						commit
						470c7e773f
					
				| @ -301,17 +301,21 @@ def process_deferred_accounting(posting_date=None): | |||||||
| 	start_date = add_months(today(), -1) | 	start_date = add_months(today(), -1) | ||||||
| 	end_date = add_days(today(), -1) | 	end_date = add_days(today(), -1) | ||||||
| 
 | 
 | ||||||
| 	for record_type in ('Income', 'Expense'): | 	companies = frappe.get_all('Company') | ||||||
| 		doc = frappe.get_doc(dict( |  | ||||||
| 			doctype='Process Deferred Accounting', |  | ||||||
| 			posting_date=posting_date, |  | ||||||
| 			start_date=start_date, |  | ||||||
| 			end_date=end_date, |  | ||||||
| 			type=record_type |  | ||||||
| 		)) |  | ||||||
| 
 | 
 | ||||||
| 		doc.insert() | 	for company in companies: | ||||||
| 		doc.submit() | 		for record_type in ('Income', 'Expense'): | ||||||
|  | 			doc = frappe.get_doc(dict( | ||||||
|  | 				doctype='Process Deferred Accounting', | ||||||
|  | 				company=company.name, | ||||||
|  | 				posting_date=posting_date, | ||||||
|  | 				start_date=start_date, | ||||||
|  | 				end_date=end_date, | ||||||
|  | 				type=record_type | ||||||
|  | 			)) | ||||||
|  | 
 | ||||||
|  | 			doc.insert() | ||||||
|  | 			doc.submit() | ||||||
| 
 | 
 | ||||||
| def make_gl_entries(doc, credit_account, debit_account, against, | def make_gl_entries(doc, credit_account, debit_account, against, | ||||||
| 	amount, base_amount, posting_date, project, account_currency, cost_center, item, deferred_process=None): | 	amount, base_amount, posting_date, project, account_currency, cost_center, item, deferred_process=None): | ||||||
|  | |||||||
| @ -51,7 +51,7 @@ class BankStatementImport(DataImport): | |||||||
| 			self.import_file, self.google_sheets_url | 			self.import_file, self.google_sheets_url | ||||||
| 		) | 		) | ||||||
| 
 | 
 | ||||||
| 		if 'Bank Account' not in json.dumps(preview): | 		if 'Bank Account' not in json.dumps(preview['columns']): | ||||||
| 			frappe.throw(_("Please add the Bank Account column")) | 			frappe.throw(_("Please add the Bank Account column")) | ||||||
| 
 | 
 | ||||||
| 		from frappe.core.page.background_jobs.background_jobs import get_info | 		from frappe.core.page.background_jobs.background_jobs import get_info | ||||||
|  | |||||||
| @ -13,7 +13,8 @@ from erpnext.accounts.doctype.account.chart_of_accounts.chart_of_accounts import | |||||||
| from frappe.utils.xlsxutils import read_xlsx_file_from_attached_file, read_xls_file_from_attached_file | from frappe.utils.xlsxutils import read_xlsx_file_from_attached_file, read_xls_file_from_attached_file | ||||||
| 
 | 
 | ||||||
| class ChartofAccountsImporter(Document): | class ChartofAccountsImporter(Document): | ||||||
| 	pass | 	def validate(self): | ||||||
|  | 		validate_accounts(self.import_file) | ||||||
| 
 | 
 | ||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def validate_company(company): | def validate_company(company): | ||||||
| @ -301,28 +302,27 @@ def validate_accounts(file_name): | |||||||
| 		if account["parent_account"] and accounts_dict.get(account["parent_account"]): | 		if account["parent_account"] and accounts_dict.get(account["parent_account"]): | ||||||
| 			accounts_dict[account["parent_account"]]["is_group"] = 1 | 			accounts_dict[account["parent_account"]]["is_group"] = 1 | ||||||
| 
 | 
 | ||||||
| 	message = validate_root(accounts_dict) | 	validate_root(accounts_dict) | ||||||
| 	if message: return message | 
 | ||||||
| 	message = validate_account_types(accounts_dict) | 	validate_account_types(accounts_dict) | ||||||
| 	if message: return message |  | ||||||
| 
 | 
 | ||||||
| 	return [True, len(accounts)] | 	return [True, len(accounts)] | ||||||
| 
 | 
 | ||||||
| def validate_root(accounts): | def validate_root(accounts): | ||||||
| 	roots = [accounts[d] for d in accounts if not accounts[d].get('parent_account')] | 	roots = [accounts[d] for d in accounts if not accounts[d].get('parent_account')] | ||||||
| 	if len(roots) < 4: | 	if len(roots) < 4: | ||||||
| 		return _("Number of root accounts cannot be less than 4") | 		frappe.throw(_("Number of root accounts cannot be less than 4")) | ||||||
| 
 | 
 | ||||||
| 	error_messages = [] | 	error_messages = [] | ||||||
| 
 | 
 | ||||||
| 	for account in roots: | 	for account in roots: | ||||||
| 		if not account.get("root_type") and account.get("account_name"): | 		if not account.get("root_type") and account.get("account_name"): | ||||||
| 			error_messages.append("Please enter Root Type for account- {0}".format(account.get("account_name"))) | 			error_messages.append(_("Please enter Root Type for account- {0}").format(account.get("account_name"))) | ||||||
| 		elif account.get("root_type") not in get_root_types() and account.get("account_name"): | 		elif account.get("root_type") not in get_root_types() and account.get("account_name"): | ||||||
| 			error_messages.append("Root Type for {0} must be one of the Asset, Liability, Income, Expense and Equity".format(account.get("account_name"))) | 			error_messages.append(_("Root Type for {0} must be one of the Asset, Liability, Income, Expense and Equity").format(account.get("account_name"))) | ||||||
| 
 | 
 | ||||||
| 	if error_messages: | 	if error_messages: | ||||||
| 		return "<br>".join(error_messages) | 		frappe.throw("<br>".join(error_messages)) | ||||||
| 
 | 
 | ||||||
| def get_root_types(): | def get_root_types(): | ||||||
| 	return ('Asset', 'Liability', 'Expense', 'Income', 'Equity') | 	return ('Asset', 'Liability', 'Expense', 'Income', 'Equity') | ||||||
| @ -356,7 +356,7 @@ def validate_account_types(accounts): | |||||||
| 
 | 
 | ||||||
| 	missing = list(set(account_types_for_ledger) - set(account_types)) | 	missing = list(set(account_types_for_ledger) - set(account_types)) | ||||||
| 	if missing: | 	if missing: | ||||||
| 		return _("Please identify/create Account (Ledger) for type - {0}").format(' , '.join(missing)) | 		frappe.throw(_("Please identify/create Account (Ledger) for type - {0}").format(' , '.join(missing))) | ||||||
| 
 | 
 | ||||||
| 	account_types_for_group = ["Bank", "Cash", "Stock"] | 	account_types_for_group = ["Bank", "Cash", "Stock"] | ||||||
| 	# fix logic bug | 	# fix logic bug | ||||||
| @ -364,7 +364,7 @@ def validate_account_types(accounts): | |||||||
| 
 | 
 | ||||||
| 	missing = list(set(account_types_for_group) - set(account_groups)) | 	missing = list(set(account_types_for_group) - set(account_groups)) | ||||||
| 	if missing: | 	if missing: | ||||||
| 		return _("Please identify/create Account (Group) for type - {0}").format(' , '.join(missing)) | 		frappe.throw(_("Please identify/create Account (Group) for type - {0}").format(' , '.join(missing))) | ||||||
| 
 | 
 | ||||||
| def unset_existing_data(company): | def unset_existing_data(company): | ||||||
| 	linked = frappe.db.sql('''select fieldname from tabDocField | 	linked = frappe.db.sql('''select fieldname from tabDocField | ||||||
|  | |||||||
| @ -25,7 +25,7 @@ class Dunning(AccountsController): | |||||||
| 
 | 
 | ||||||
| 	def validate_amount(self): | 	def validate_amount(self): | ||||||
| 		amounts = calculate_interest_and_amount( | 		amounts = calculate_interest_and_amount( | ||||||
| 			self.posting_date, self.outstanding_amount, self.rate_of_interest, self.dunning_fee, self.overdue_days) | 			self.outstanding_amount, self.rate_of_interest, self.dunning_fee, self.overdue_days) | ||||||
| 		if self.interest_amount != amounts.get('interest_amount'): | 		if self.interest_amount != amounts.get('interest_amount'): | ||||||
| 			self.interest_amount = flt(amounts.get('interest_amount'), self.precision('interest_amount')) | 			self.interest_amount = flt(amounts.get('interest_amount'), self.precision('interest_amount')) | ||||||
| 		if self.dunning_amount != amounts.get('dunning_amount'): | 		if self.dunning_amount != amounts.get('dunning_amount'): | ||||||
| @ -91,13 +91,13 @@ def resolve_dunning(doc, state): | |||||||
| 			for dunning in dunnings: | 			for dunning in dunnings: | ||||||
| 				frappe.db.set_value("Dunning", dunning.name, "status", 'Resolved') | 				frappe.db.set_value("Dunning", dunning.name, "status", 'Resolved') | ||||||
| 
 | 
 | ||||||
| def calculate_interest_and_amount(posting_date, outstanding_amount, rate_of_interest, dunning_fee, overdue_days): | def calculate_interest_and_amount(outstanding_amount, rate_of_interest, dunning_fee, overdue_days): | ||||||
| 	interest_amount = 0 | 	interest_amount = 0 | ||||||
| 	grand_total = 0 | 	grand_total = flt(outstanding_amount) + flt(dunning_fee) | ||||||
| 	if rate_of_interest: | 	if rate_of_interest: | ||||||
| 		interest_per_year = flt(outstanding_amount) * flt(rate_of_interest) / 100 | 		interest_per_year = flt(outstanding_amount) * flt(rate_of_interest) / 100 | ||||||
| 		interest_amount = (interest_per_year * cint(overdue_days)) / 365 | 		interest_amount = (interest_per_year * cint(overdue_days)) / 365 | ||||||
| 		grand_total = flt(outstanding_amount) + flt(interest_amount) + flt(dunning_fee) | 		grand_total += flt(interest_amount) | ||||||
| 	dunning_amount = flt(interest_amount) + flt(dunning_fee) | 	dunning_amount = flt(interest_amount) + flt(dunning_fee) | ||||||
| 	return { | 	return { | ||||||
| 		'interest_amount': interest_amount, | 		'interest_amount': interest_amount, | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ class TestDunning(unittest.TestCase): | |||||||
| 	@classmethod | 	@classmethod | ||||||
| 	def setUpClass(self): | 	def setUpClass(self): | ||||||
| 		create_dunning_type() | 		create_dunning_type() | ||||||
|  | 		create_dunning_type_with_zero_interest_rate() | ||||||
| 		unlink_payment_on_cancel_of_invoice() | 		unlink_payment_on_cancel_of_invoice() | ||||||
| 
 | 
 | ||||||
| 	@classmethod | 	@classmethod | ||||||
| @ -25,11 +26,20 @@ class TestDunning(unittest.TestCase): | |||||||
| 	def test_dunning(self): | 	def test_dunning(self): | ||||||
| 		dunning = create_dunning() | 		dunning = create_dunning() | ||||||
| 		amounts = calculate_interest_and_amount( | 		amounts = calculate_interest_and_amount( | ||||||
| 			dunning.posting_date, dunning.outstanding_amount, dunning.rate_of_interest, dunning.dunning_fee, dunning.overdue_days) | 			dunning.outstanding_amount, dunning.rate_of_interest, dunning.dunning_fee, dunning.overdue_days) | ||||||
| 		self.assertEqual(round(amounts.get('interest_amount'), 2), 0.44) | 		self.assertEqual(round(amounts.get('interest_amount'), 2), 0.44) | ||||||
| 		self.assertEqual(round(amounts.get('dunning_amount'), 2), 20.44) | 		self.assertEqual(round(amounts.get('dunning_amount'), 2), 20.44) | ||||||
| 		self.assertEqual(round(amounts.get('grand_total'), 2), 120.44) | 		self.assertEqual(round(amounts.get('grand_total'), 2), 120.44) | ||||||
| 
 | 
 | ||||||
|  | 	def test_dunning_with_zero_interest_rate(self): | ||||||
|  | 		dunning = create_dunning_with_zero_interest_rate() | ||||||
|  | 		amounts = calculate_interest_and_amount( | ||||||
|  | 			dunning.outstanding_amount, dunning.rate_of_interest, dunning.dunning_fee, dunning.overdue_days) | ||||||
|  | 		self.assertEqual(round(amounts.get('interest_amount'), 2), 0) | ||||||
|  | 		self.assertEqual(round(amounts.get('dunning_amount'), 2), 20) | ||||||
|  | 		self.assertEqual(round(amounts.get('grand_total'), 2), 120) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| 	def test_gl_entries(self): | 	def test_gl_entries(self): | ||||||
| 		dunning = create_dunning() | 		dunning = create_dunning() | ||||||
| 		dunning.submit() | 		dunning.submit() | ||||||
| @ -83,6 +93,27 @@ def create_dunning(): | |||||||
| 	dunning.save() | 	dunning.save() | ||||||
| 	return dunning | 	return dunning | ||||||
| 
 | 
 | ||||||
|  | def create_dunning_with_zero_interest_rate(): | ||||||
|  | 	posting_date = add_days(today(), -20) | ||||||
|  | 	due_date = add_days(today(), -15) | ||||||
|  | 	sales_invoice = create_sales_invoice_against_cost_center( | ||||||
|  | 		posting_date=posting_date, due_date=due_date, status='Overdue') | ||||||
|  | 	dunning_type = frappe.get_doc("Dunning Type", 'First Notice with 0% Rate of Interest') | ||||||
|  | 	dunning = frappe.new_doc("Dunning") | ||||||
|  | 	dunning.sales_invoice = sales_invoice.name | ||||||
|  | 	dunning.customer_name = sales_invoice.customer_name | ||||||
|  | 	dunning.outstanding_amount = sales_invoice.outstanding_amount | ||||||
|  | 	dunning.debit_to = sales_invoice.debit_to | ||||||
|  | 	dunning.currency = sales_invoice.currency | ||||||
|  | 	dunning.company = sales_invoice.company | ||||||
|  | 	dunning.posting_date = nowdate() | ||||||
|  | 	dunning.due_date = sales_invoice.due_date | ||||||
|  | 	dunning.dunning_type = 'First Notice with 0% Rate of Interest' | ||||||
|  | 	dunning.rate_of_interest = dunning_type.rate_of_interest | ||||||
|  | 	dunning.dunning_fee = dunning_type.dunning_fee | ||||||
|  | 	dunning.save() | ||||||
|  | 	return dunning | ||||||
|  | 
 | ||||||
| def create_dunning_type(): | def create_dunning_type(): | ||||||
| 	dunning_type = frappe.new_doc("Dunning Type") | 	dunning_type = frappe.new_doc("Dunning Type") | ||||||
| 	dunning_type.dunning_type = 'First Notice' | 	dunning_type.dunning_type = 'First Notice' | ||||||
| @ -98,3 +129,19 @@ def create_dunning_type(): | |||||||
| 		} | 		} | ||||||
| 	) | 	) | ||||||
| 	dunning_type.save() | 	dunning_type.save() | ||||||
|  | 
 | ||||||
|  | def create_dunning_type_with_zero_interest_rate(): | ||||||
|  | 	dunning_type = frappe.new_doc("Dunning Type") | ||||||
|  | 	dunning_type.dunning_type = 'First Notice with 0% Rate of Interest' | ||||||
|  | 	dunning_type.start_day = 10 | ||||||
|  | 	dunning_type.end_day = 20 | ||||||
|  | 	dunning_type.dunning_fee = 20 | ||||||
|  | 	dunning_type.rate_of_interest = 0 | ||||||
|  | 	dunning_type.append( | ||||||
|  | 		"dunning_letter_text", { | ||||||
|  | 			'language': 'en', | ||||||
|  | 			'body_text': 'We have still not received payment for our invoice ', | ||||||
|  | 			'closing_text': 'We kindly request that you pay the outstanding amount immediately, and late fees.' | ||||||
|  | 		} | ||||||
|  | 	) | ||||||
|  | 	dunning_type.save()  | ||||||
| @ -1318,9 +1318,9 @@ def get_reference_details(reference_doctype, reference_name, party_account_curre | |||||||
| 
 | 
 | ||||||
| 	return frappe._dict({ | 	return frappe._dict({ | ||||||
| 		"due_date": ref_doc.get("due_date"), | 		"due_date": ref_doc.get("due_date"), | ||||||
| 		"total_amount": total_amount, | 		"total_amount": flt(total_amount), | ||||||
| 		"outstanding_amount": outstanding_amount, | 		"outstanding_amount": flt(outstanding_amount), | ||||||
| 		"exchange_rate": exchange_rate, | 		"exchange_rate": flt(exchange_rate), | ||||||
| 		"bill_no": bill_no | 		"bill_no": bill_no | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -589,9 +589,9 @@ class TestPaymentEntry(unittest.TestCase): | |||||||
| 		party_account_balance = get_balance_on(account=pe.paid_from, cost_center=pe.cost_center) | 		party_account_balance = get_balance_on(account=pe.paid_from, cost_center=pe.cost_center) | ||||||
| 
 | 
 | ||||||
| 		self.assertEqual(pe.cost_center, si.cost_center) | 		self.assertEqual(pe.cost_center, si.cost_center) | ||||||
| 		self.assertEqual(expected_account_balance, account_balance) | 		self.assertEqual(flt(expected_account_balance), account_balance) | ||||||
| 		self.assertEqual(expected_party_balance, party_balance) | 		self.assertEqual(flt(expected_party_balance), party_balance) | ||||||
| 		self.assertEqual(expected_party_account_balance, party_account_balance) | 		self.assertEqual(flt(expected_party_account_balance), party_account_balance) | ||||||
| 
 | 
 | ||||||
| def create_payment_terms_template(): | def create_payment_terms_template(): | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -207,10 +207,9 @@ def fetch_customers(customer_collection, collection_name, primary_mandatory): | |||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def get_customer_emails(customer_name, primary_mandatory, billing_and_primary=True): | def get_customer_emails(customer_name, primary_mandatory, billing_and_primary=True): | ||||||
| 	billing_email = frappe.db.sql(""" | 	billing_email = frappe.db.sql(""" | ||||||
| 		SELECT c.email_id FROM `tabContact` AS c JOIN `tabDynamic Link` AS l ON c.name=l.parent \ | 		SELECT c.email_id FROM `tabContact` AS c JOIN `tabDynamic Link` AS l ON c.name=l.parent | ||||||
| 		WHERE l.link_doctype='Customer' and l.link_name='""" + customer_name + """' and \ | 		WHERE l.link_doctype='Customer' and l.link_name=%s and c.is_billing_contact=1 | ||||||
| 		c.is_billing_contact=1 \ | 		order by c.creation desc""", customer_name) | ||||||
| 		order by c.creation desc""") |  | ||||||
| 
 | 
 | ||||||
| 	if len(billing_email) == 0 or (billing_email[0][0] is None): | 	if len(billing_email) == 0 or (billing_email[0][0] is None): | ||||||
| 		if billing_and_primary: | 		if billing_and_primary: | ||||||
|  | |||||||
| @ -1010,21 +1010,21 @@ class TestPurchaseInvoice(unittest.TestCase): | |||||||
| 		# Check GLE for Purchase Invoice | 		# Check GLE for Purchase Invoice | ||||||
| 		# Zero net effect on final TDS Payable on invoice | 		# Zero net effect on final TDS Payable on invoice | ||||||
| 		expected_gle = [ | 		expected_gle = [ | ||||||
| 			['_Test Account Cost for Goods Sold - _TC', 30000, 0], | 			['_Test Account Cost for Goods Sold - _TC', 30000], | ||||||
| 			['_Test Account Excise Duty - _TC', 0, 3000], | 			['_Test Account Excise Duty - _TC', -3000], | ||||||
| 			['Creditors - _TC', 0, 27000], | 			['Creditors - _TC', -27000], | ||||||
| 			['TDS Payable - _TC', 3000, 3000] | 			['TDS Payable - _TC', 0] | ||||||
| 		] | 		] | ||||||
| 
 | 
 | ||||||
| 		gl_entries = frappe.db.sql("""select account, debit, credit | 		gl_entries = frappe.db.sql("""select account, sum(debit - credit) as amount | ||||||
| 			from `tabGL Entry` | 			from `tabGL Entry` | ||||||
| 			where voucher_type='Purchase Invoice' and voucher_no=%s | 			where voucher_type='Purchase Invoice' and voucher_no=%s | ||||||
|  | 			group by account | ||||||
| 			order by account asc""", (purchase_invoice.name), as_dict=1) | 			order by account asc""", (purchase_invoice.name), as_dict=1) | ||||||
| 
 | 
 | ||||||
| 		for i, gle in enumerate(gl_entries): | 		for i, gle in enumerate(gl_entries): | ||||||
| 			self.assertEqual(expected_gle[i][0], gle.account) | 			self.assertEqual(expected_gle[i][0], gle.account) | ||||||
| 			self.assertEqual(expected_gle[i][1], gle.debit) | 			self.assertEqual(expected_gle[i][1], gle.amount) | ||||||
| 			self.assertEqual(expected_gle[i][2], gle.credit) |  | ||||||
| 
 | 
 | ||||||
| def update_tax_witholding_category(company, account, date): | def update_tax_witholding_category(company, account, date): | ||||||
| 	from erpnext.accounts.utils import get_fiscal_year | 	from erpnext.accounts.utils import get_fiscal_year | ||||||
|  | |||||||
| @ -1957,6 +1957,33 @@ class TestSalesInvoice(unittest.TestCase): | |||||||
| 		einvoice = make_einvoice(si) | 		einvoice = make_einvoice(si) | ||||||
| 		validate_totals(einvoice) | 		validate_totals(einvoice) | ||||||
| 
 | 
 | ||||||
|  | 	def test_item_tax_net_range(self): | ||||||
|  | 		item = create_item("T Shirt") | ||||||
|  | 
 | ||||||
|  | 		item.set('taxes', []) | ||||||
|  | 		item.append("taxes", { | ||||||
|  | 			"item_tax_template": "_Test Account Excise Duty @ 10 - _TC", | ||||||
|  | 			"minimum_net_rate": 0, | ||||||
|  | 			"maximum_net_rate": 500 | ||||||
|  | 		}) | ||||||
|  | 
 | ||||||
|  | 		item.append("taxes", { | ||||||
|  | 			"item_tax_template": "_Test Account Excise Duty @ 12 - _TC", | ||||||
|  | 			"minimum_net_rate": 501, | ||||||
|  | 			"maximum_net_rate": 1000 | ||||||
|  | 		}) | ||||||
|  | 
 | ||||||
|  | 		item.save() | ||||||
|  | 
 | ||||||
|  | 		sales_invoice = create_sales_invoice(item = "T Shirt", rate=700, do_not_submit=True) | ||||||
|  | 		self.assertEqual(sales_invoice.items[0].item_tax_template, "_Test Account Excise Duty @ 12 - _TC") | ||||||
|  | 
 | ||||||
|  | 		# Apply discount | ||||||
|  | 		sales_invoice.apply_discount_on = 'Net Total' | ||||||
|  | 		sales_invoice.discount_amount = 300 | ||||||
|  | 		sales_invoice.save() | ||||||
|  | 		self.assertEqual(sales_invoice.items[0].item_tax_template, "_Test Account Excise Duty @ 10 - _TC") | ||||||
|  | 
 | ||||||
| def get_sales_invoice_for_e_invoice(): | def get_sales_invoice_for_e_invoice(): | ||||||
| 	si = make_sales_invoice_for_ewaybill() | 	si = make_sales_invoice_for_ewaybill() | ||||||
| 	si.naming_series = 'INV-2020-.#####' | 	si.naming_series = 'INV-2020-.#####' | ||||||
| @ -1985,32 +2012,6 @@ def get_sales_invoice_for_e_invoice(): | |||||||
| 
 | 
 | ||||||
| 	return si | 	return si | ||||||
| 
 | 
 | ||||||
| 	def test_item_tax_net_range(self): |  | ||||||
| 		item = create_item("T Shirt") |  | ||||||
| 
 |  | ||||||
| 		item.set('taxes', []) |  | ||||||
| 		item.append("taxes", { |  | ||||||
| 			"item_tax_template": "_Test Account Excise Duty @ 10 - _TC", |  | ||||||
| 			"minimum_net_rate": 0, |  | ||||||
| 			"maximum_net_rate": 500 |  | ||||||
| 		}) |  | ||||||
| 
 |  | ||||||
| 		item.append("taxes", { |  | ||||||
| 			"item_tax_template": "_Test Account Excise Duty @ 12 - _TC", |  | ||||||
| 			"minimum_net_rate": 501, |  | ||||||
| 			"maximum_net_rate": 1000 |  | ||||||
| 		}) |  | ||||||
| 
 |  | ||||||
| 		item.save() |  | ||||||
| 
 |  | ||||||
| 		sales_invoice = create_sales_invoice(item = "T Shirt", rate=700, do_not_submit=True) |  | ||||||
| 		self.assertEqual(sales_invoice.items[0].item_tax_template, "_Test Account Excise Duty @ 12 - _TC") |  | ||||||
| 
 |  | ||||||
| 		# Apply discount |  | ||||||
| 		sales_invoice.apply_discount_on = 'Net Total' |  | ||||||
| 		sales_invoice.discount_amount = 300 |  | ||||||
| 		sales_invoice.save() |  | ||||||
| 		self.assertEqual(sales_invoice.items[0].item_tax_template, "_Test Account Excise Duty @ 10 - _TC") |  | ||||||
| 
 | 
 | ||||||
| def make_test_address_for_ewaybill(): | def make_test_address_for_ewaybill(): | ||||||
| 	if not frappe.db.exists('Address', '_Test Address for Eway bill-Billing'): | 	if not frappe.db.exists('Address', '_Test Address for Eway bill-Billing'): | ||||||
| @ -2087,9 +2088,9 @@ def make_sales_invoice_for_ewaybill(): | |||||||
| 	if not gst_account: | 	if not gst_account: | ||||||
| 		gst_settings.append("gst_accounts", { | 		gst_settings.append("gst_accounts", { | ||||||
| 			"company": "_Test Company", | 			"company": "_Test Company", | ||||||
| 			"cgst_account": "CGST - _TC", | 			"cgst_account": "Output Tax CGST - _TC", | ||||||
| 			"sgst_account": "SGST - _TC", | 			"sgst_account": "Output Tax SGST - _TC", | ||||||
| 			"igst_account": "IGST - _TC", | 			"igst_account": "Output Tax IGST - _TC", | ||||||
| 		}) | 		}) | ||||||
| 
 | 
 | ||||||
| 	gst_settings.save() | 	gst_settings.save() | ||||||
| @ -2106,7 +2107,7 @@ def make_sales_invoice_for_ewaybill(): | |||||||
| 
 | 
 | ||||||
| 	si.append("taxes", { | 	si.append("taxes", { | ||||||
| 		"charge_type": "On Net Total", | 		"charge_type": "On Net Total", | ||||||
| 		"account_head": "CGST - _TC", | 		"account_head": "Output Tax CGST - _TC", | ||||||
| 		"cost_center": "Main - _TC", | 		"cost_center": "Main - _TC", | ||||||
| 		"description": "CGST @ 9.0", | 		"description": "CGST @ 9.0", | ||||||
| 		"rate": 9 | 		"rate": 9 | ||||||
| @ -2114,7 +2115,7 @@ def make_sales_invoice_for_ewaybill(): | |||||||
| 
 | 
 | ||||||
| 	si.append("taxes", { | 	si.append("taxes", { | ||||||
| 		"charge_type": "On Net Total", | 		"charge_type": "On Net Total", | ||||||
| 		"account_head": "SGST - _TC", | 		"account_head": "Output Tax SGST - _TC", | ||||||
| 		"cost_center": "Main - _TC", | 		"cost_center": "Main - _TC", | ||||||
| 		"description": "SGST @ 9.0", | 		"description": "SGST @ 9.0", | ||||||
| 		"rate": 9 | 		"rate": 9 | ||||||
|  | |||||||
| @ -1,24 +1,6 @@ | |||||||
| // Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
 | // Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
 | ||||||
| // License: GNU General Public License v3. See license.txt
 | // License: GNU General Public License v3. See license.txt
 | ||||||
| 
 | 
 | ||||||
| cur_frm.add_fetch("customer", "customer_group", "customer_group" ); |  | ||||||
| cur_frm.add_fetch("supplier", "supplier_group_name", "supplier_group" ); |  | ||||||
| 
 |  | ||||||
| frappe.ui.form.on("Tax Rule", "tax_type", function(frm) { |  | ||||||
| 	frm.toggle_reqd("sales_tax_template", frm.doc.tax_type=="Sales"); |  | ||||||
| 	frm.toggle_reqd("purchase_tax_template", frm.doc.tax_type=="Purchase"); |  | ||||||
| }) |  | ||||||
| 
 |  | ||||||
| frappe.ui.form.on("Tax Rule", "onload", function(frm) { |  | ||||||
| 	if(frm.doc.__islocal) { |  | ||||||
| 		frm.set_value("use_for_shopping_cart", 1); |  | ||||||
| 	} |  | ||||||
| }) |  | ||||||
| 
 |  | ||||||
| frappe.ui.form.on("Tax Rule", "refresh", function(frm) { |  | ||||||
| 	frappe.ui.form.trigger("Tax Rule", "tax_type"); |  | ||||||
| }) |  | ||||||
| 
 |  | ||||||
| frappe.ui.form.on("Tax Rule", "customer", function(frm) { | frappe.ui.form.on("Tax Rule", "customer", function(frm) { | ||||||
| 	if(frm.doc.customer) { | 	if(frm.doc.customer) { | ||||||
| 		frappe.call({ | 		frappe.call({ | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -50,7 +50,7 @@ class TestTaxRule(unittest.TestCase): | |||||||
| 		tax_rule1 = make_tax_rule(customer_group= "All Customer Groups", | 		tax_rule1 = make_tax_rule(customer_group= "All Customer Groups", | ||||||
| 			sales_tax_template = "_Test Sales Taxes and Charges Template - _TC", priority = 1, from_date = "2015-01-01") | 			sales_tax_template = "_Test Sales Taxes and Charges Template - _TC", priority = 1, from_date = "2015-01-01") | ||||||
| 		tax_rule1.save() | 		tax_rule1.save() | ||||||
| 		self.assertEqual(get_tax_template("2015-01-01", {"customer_group" : "Commercial", "use_for_shopping_cart":0}), | 		self.assertEqual(get_tax_template("2015-01-01", {"customer_group" : "Commercial", "use_for_shopping_cart":1}), | ||||||
| 			"_Test Sales Taxes and Charges Template - _TC") | 			"_Test Sales Taxes and Charges Template - _TC") | ||||||
| 
 | 
 | ||||||
| 	def test_conflict_with_overlapping_dates(self): | 	def test_conflict_with_overlapping_dates(self): | ||||||
|  | |||||||
| @ -542,6 +542,7 @@ def get_dashboard_info(party_type, party, loyalty_program=None): | |||||||
| 		select company, sum(debit_in_account_currency) - sum(credit_in_account_currency) | 		select company, sum(debit_in_account_currency) - sum(credit_in_account_currency) | ||||||
| 		from `tabGL Entry` | 		from `tabGL Entry` | ||||||
| 		where party_type = %s and party=%s | 		where party_type = %s and party=%s | ||||||
|  | 		and is_cancelled = 0 | ||||||
| 		group by company""", (party_type, party))) | 		group by company""", (party_type, party))) | ||||||
| 
 | 
 | ||||||
| 	for d in companies: | 	for d in companies: | ||||||
|  | |||||||
| @ -397,6 +397,7 @@ def get_chart_data(filters, columns, data): | |||||||
| 				{'name': 'Budget', 'chartType': 'bar', 'values': budget_values}, | 				{'name': 'Budget', 'chartType': 'bar', 'values': budget_values}, | ||||||
| 				{'name': 'Actual Expense', 'chartType': 'bar', 'values': actual_values} | 				{'name': 'Actual Expense', 'chartType': 'bar', 'values': actual_values} | ||||||
| 			] | 			] | ||||||
| 		} | 		}, | ||||||
|  | 		'type' : 'bar' | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -380,7 +380,7 @@ def set_gl_entries_by_account(from_date, to_date, root_lft, root_rgt, filters, g | |||||||
| 		gl_entries = frappe.db.sql("""select gl.posting_date, gl.account, gl.debit, gl.credit, gl.is_opening, gl.company, | 		gl_entries = frappe.db.sql("""select gl.posting_date, gl.account, gl.debit, gl.credit, gl.is_opening, gl.company, | ||||||
| 			gl.fiscal_year, gl.debit_in_account_currency, gl.credit_in_account_currency, gl.account_currency, | 			gl.fiscal_year, gl.debit_in_account_currency, gl.credit_in_account_currency, gl.account_currency, | ||||||
| 			acc.account_name, acc.account_number | 			acc.account_name, acc.account_number | ||||||
| 			from `tabGL Entry` gl, `tabAccount` acc where acc.name = gl.account and gl.company = %(company)s | 			from `tabGL Entry` gl, `tabAccount` acc where acc.name = gl.account and gl.company = %(company)s and gl.is_cancelled = 0 | ||||||
| 			{additional_conditions} and gl.posting_date <= %(to_date)s and acc.lft >= %(lft)s and acc.rgt <= %(rgt)s | 			{additional_conditions} and gl.posting_date <= %(to_date)s and acc.lft >= %(lft)s and acc.rgt <= %(rgt)s | ||||||
| 			order by gl.account, gl.posting_date""".format(additional_conditions=additional_conditions), | 			order by gl.account, gl.posting_date""".format(additional_conditions=additional_conditions), | ||||||
| 			{ | 			{ | ||||||
|  | |||||||
| @ -48,13 +48,12 @@ def validate_filters(filters, account_details): | |||||||
| 
 | 
 | ||||||
| 	if not filters.get("from_date") and not filters.get("to_date"): | 	if not filters.get("from_date") and not filters.get("to_date"): | ||||||
| 		frappe.throw(_("{0} and {1} are mandatory").format(frappe.bold(_("From Date")), frappe.bold(_("To Date")))) | 		frappe.throw(_("{0} and {1} are mandatory").format(frappe.bold(_("From Date")), frappe.bold(_("To Date")))) | ||||||
| 
 |  | ||||||
| 	for account in filters.account: |  | ||||||
| 		if not account_details.get(account): |  | ||||||
| 			frappe.throw(_("Account {0} does not exists").format(account)) |  | ||||||
| 			 | 			 | ||||||
| 	if filters.get('account'): | 	if filters.get('account'): | ||||||
| 		filters.account = frappe.parse_json(filters.get('account')) | 		filters.account = frappe.parse_json(filters.get('account')) | ||||||
|  | 		for account in filters.account: | ||||||
|  | 			if not account_details.get(account): | ||||||
|  | 				frappe.throw(_("Account {0} does not exists").format(account)) | ||||||
| 
 | 
 | ||||||
| 	if (filters.get("account") and filters.get("group_by") == _('Group by Account') | 	if (filters.get("account") and filters.get("group_by") == _('Group by Account') | ||||||
| 		and account_details[filters.account].is_group == 0): | 		and account_details[filters.account].is_group == 0): | ||||||
|  | |||||||
| @ -168,21 +168,24 @@ def get_columns(filters): | |||||||
| 			"label": _("Income"), | 			"label": _("Income"), | ||||||
| 			"fieldtype": "Currency", | 			"fieldtype": "Currency", | ||||||
| 			"options": "currency", | 			"options": "currency", | ||||||
| 			"width": 120 | 			"width": 305 | ||||||
|  | 
 | ||||||
| 		}, | 		}, | ||||||
| 		{ | 		{ | ||||||
| 			"fieldname": "expense", | 			"fieldname": "expense", | ||||||
| 			"label": _("Expense"), | 			"label": _("Expense"), | ||||||
| 			"fieldtype": "Currency", | 			"fieldtype": "Currency", | ||||||
| 			"options": "currency", | 			"options": "currency", | ||||||
| 			"width": 120 | 			"width": 305 | ||||||
|  | 
 | ||||||
| 		}, | 		}, | ||||||
| 		{ | 		{ | ||||||
| 			"fieldname": "gross_profit_loss", | 			"fieldname": "gross_profit_loss", | ||||||
| 			"label": _("Gross Profit / Loss"), | 			"label": _("Gross Profit / Loss"), | ||||||
| 			"fieldtype": "Currency", | 			"fieldtype": "Currency", | ||||||
| 			"options": "currency", | 			"options": "currency", | ||||||
| 			"width": 120 | 			"width": 307 | ||||||
|  | 
 | ||||||
| 		} | 		} | ||||||
| 	] | 	] | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -784,7 +784,7 @@ def get_children(doctype, parent, company, is_root=False): | |||||||
| 	return acc | 	return acc | ||||||
| 
 | 
 | ||||||
| def create_payment_gateway_account(gateway, payment_channel="Email"): | def create_payment_gateway_account(gateway, payment_channel="Email"): | ||||||
| 	from erpnext.setup.setup_wizard.operations.company_setup import create_bank_account | 	from erpnext.setup.setup_wizard.operations.install_fixtures import create_bank_account | ||||||
| 
 | 
 | ||||||
| 	company = frappe.db.get_value("Global Defaults", None, "default_company") | 	company = frappe.db.get_value("Global Defaults", None, "default_company") | ||||||
| 	if not company: | 	if not company: | ||||||
|  | |||||||
| @ -97,6 +97,9 @@ | |||||||
|   "is_fixed_asset", |   "is_fixed_asset", | ||||||
|   "item_tax_rate", |   "item_tax_rate", | ||||||
|   "section_break_72", |   "section_break_72", | ||||||
|  |   "production_plan", | ||||||
|  |   "production_plan_item", | ||||||
|  |   "production_plan_sub_assembly_item", | ||||||
|   "page_break" |   "page_break" | ||||||
|  ], |  ], | ||||||
|  "fields": [ |  "fields": [ | ||||||
| @ -803,13 +806,37 @@ | |||||||
|    "options": "Company:company:default_currency", |    "options": "Company:company:default_currency", | ||||||
|    "print_hide": 1, |    "print_hide": 1, | ||||||
|    "read_only": 1 |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "production_plan", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Production Plan", | ||||||
|  |    "options": "Production Plan", | ||||||
|  |    "print_hide": 1, | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "production_plan_item", | ||||||
|  |    "fieldtype": "Data", | ||||||
|  |    "hidden": 1, | ||||||
|  |    "label": "Production Plan Item", | ||||||
|  |    "no_copy": 1, | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "production_plan_sub_assembly_item", | ||||||
|  |    "fieldtype": "Data", | ||||||
|  |    "hidden": 1, | ||||||
|  |    "label": "Production Plan Sub Assembly Item", | ||||||
|  |    "no_copy": 1, | ||||||
|  |    "read_only": 1 | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "idx": 1, |  "idx": 1, | ||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "istable": 1, |  "istable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-03-22 11:46:12.357435", |  "modified": "2021-06-28 19:22:22.715365", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Buying", |  "module": "Buying", | ||||||
|  "name": "Purchase Order Item", |  "name": "Purchase Order Item", | ||||||
|  | |||||||
| @ -99,9 +99,10 @@ def validate_returned_items(doc): | |||||||
| 								frappe.throw(_("Row # {0}: Serial No {1} does not match with {2} {3}") | 								frappe.throw(_("Row # {0}: Serial No {1} does not match with {2} {3}") | ||||||
| 									.format(d.idx, s, doc.doctype, doc.return_against)) | 									.format(d.idx, s, doc.doctype, doc.return_against)) | ||||||
| 
 | 
 | ||||||
| 				if warehouse_mandatory and frappe.db.get_value("Item", d.item_code, "is_stock_item") \ | 				if (warehouse_mandatory and not d.get("warehouse") and | ||||||
| 					and not d.get("warehouse"): | 					frappe.db.get_value("Item", d.item_code, "is_stock_item") | ||||||
| 						frappe.throw(_("Warehouse is mandatory")) | 				): | ||||||
|  | 					frappe.throw(_("Warehouse is mandatory")) | ||||||
| 
 | 
 | ||||||
| 			items_returned = True | 			items_returned = True | ||||||
| 
 | 
 | ||||||
| @ -462,4 +463,4 @@ def get_returned_serial_nos(child_doc, parent_doc): | |||||||
| 	for row in frappe.get_all(parent_doc.doctype, fields = fields, filters=filters): | 	for row in frappe.get_all(parent_doc.doctype, fields = fields, filters=filters): | ||||||
| 		serial_nos.extend(get_serial_nos(row.serial_no)) | 		serial_nos.extend(get_serial_nos(row.serial_no)) | ||||||
| 
 | 
 | ||||||
| 	return serial_nos | 	return serial_nos | ||||||
|  | |||||||
| @ -356,42 +356,68 @@ class StockController(AccountsController): | |||||||
| 		}, update_modified) | 		}, update_modified) | ||||||
| 
 | 
 | ||||||
| 	def validate_inspection(self): | 	def validate_inspection(self): | ||||||
| 		'''Checks if quality inspection is set for Items that require inspection. | 		"""Checks if quality inspection is set/ is valid for Items that require inspection.""" | ||||||
| 		On submit, throw an exception''' | 		inspection_fieldname_map = { | ||||||
| 		inspection_required_fieldname = None | 			"Purchase Receipt": "inspection_required_before_purchase", | ||||||
| 		if self.doctype in ["Purchase Receipt", "Purchase Invoice"]: | 			"Purchase Invoice": "inspection_required_before_purchase", | ||||||
| 			inspection_required_fieldname = "inspection_required_before_purchase" | 			"Sales Invoice": "inspection_required_before_delivery", | ||||||
| 		elif self.doctype in ["Delivery Note", "Sales Invoice"]: | 			"Delivery Note": "inspection_required_before_delivery" | ||||||
| 			inspection_required_fieldname = "inspection_required_before_delivery" | 		} | ||||||
|  | 		inspection_required_fieldname = inspection_fieldname_map.get(self.doctype) | ||||||
| 
 | 
 | ||||||
|  | 		# return if inspection is not required on document level | ||||||
| 		if ((not inspection_required_fieldname and self.doctype != "Stock Entry") or | 		if ((not inspection_required_fieldname and self.doctype != "Stock Entry") or | ||||||
| 			(self.doctype == "Stock Entry" and not self.inspection_required) or | 			(self.doctype == "Stock Entry" and not self.inspection_required) or | ||||||
| 			(self.doctype in ["Sales Invoice", "Purchase Invoice"] and not self.update_stock)): | 			(self.doctype in ["Sales Invoice", "Purchase Invoice"] and not self.update_stock)): | ||||||
| 				return | 				return | ||||||
| 
 | 
 | ||||||
| 		for d in self.get('items'): | 		for row in self.get('items'): | ||||||
| 			qa_required = False | 			qi_required = False | ||||||
| 			if (inspection_required_fieldname and not d.quality_inspection and | 			if (inspection_required_fieldname and frappe.db.get_value("Item", row.item_code, inspection_required_fieldname)): | ||||||
| 				frappe.db.get_value("Item", d.item_code, inspection_required_fieldname)): | 				qi_required = True | ||||||
| 				qa_required = True | 			elif self.doctype == "Stock Entry" and row.t_warehouse: | ||||||
| 			elif self.doctype == "Stock Entry" and not d.quality_inspection and d.t_warehouse: | 				qi_required = True # inward stock needs inspection | ||||||
| 				qa_required = True |  | ||||||
| 			if self.docstatus == 1 and d.quality_inspection: |  | ||||||
| 				qa_doc = frappe.get_doc("Quality Inspection", d.quality_inspection) |  | ||||||
| 				if qa_doc.docstatus == 0: |  | ||||||
| 					link = frappe.utils.get_link_to_form('Quality Inspection', d.quality_inspection) |  | ||||||
| 					frappe.throw(_("Quality Inspection: {0} is not submitted for the item: {1} in row {2}").format(link, d.item_code, d.idx), QualityInspectionNotSubmittedError) |  | ||||||
| 
 | 
 | ||||||
| 				if qa_doc.status != 'Accepted': | 			if qi_required: # validate row only if inspection is required on item level | ||||||
| 					frappe.throw(_("Row {0}: Quality Inspection rejected for item {1}") | 				self.validate_qi_presence(row) | ||||||
| 						.format(d.idx, d.item_code), QualityInspectionRejectedError) | 				if self.docstatus == 1: | ||||||
| 			elif qa_required : | 					self.validate_qi_submission(row) | ||||||
| 				action = frappe.get_doc('Stock Settings').action_if_quality_inspection_is_not_submitted | 					self.validate_qi_rejection(row) | ||||||
| 				if self.docstatus==1 and action == 'Stop': | 
 | ||||||
| 					frappe.throw(_("Quality Inspection required for Item {0} to submit").format(frappe.bold(d.item_code)), | 	def validate_qi_presence(self, row): | ||||||
| 						exc=QualityInspectionRequiredError) | 		"""Check if QI is present on row level. Warn on save and stop on submit if missing.""" | ||||||
| 				else: | 		if not row.quality_inspection: | ||||||
| 					frappe.msgprint(_("Create Quality Inspection for Item {0}").format(frappe.bold(d.item_code))) | 			msg = f"Row #{row.idx}: Quality Inspection is required for Item {frappe.bold(row.item_code)}" | ||||||
|  | 			if self.docstatus == 1: | ||||||
|  | 				frappe.throw(_(msg), title=_("Inspection Required"), exc=QualityInspectionRequiredError) | ||||||
|  | 			else: | ||||||
|  | 				frappe.msgprint(_(msg), title=_("Inspection Required"), indicator="blue") | ||||||
|  | 
 | ||||||
|  | 	def validate_qi_submission(self, row): | ||||||
|  | 		"""Check if QI is submitted on row level, during submission""" | ||||||
|  | 		action = frappe.db.get_single_value("Stock Settings", "action_if_quality_inspection_is_not_submitted") | ||||||
|  | 		qa_docstatus = frappe.db.get_value("Quality Inspection", row.quality_inspection, "docstatus") | ||||||
|  | 
 | ||||||
|  | 		if not qa_docstatus == 1: | ||||||
|  | 			link = frappe.utils.get_link_to_form('Quality Inspection', row.quality_inspection) | ||||||
|  | 			msg = f"Row #{row.idx}: Quality Inspection {link} is not submitted for the item: {row.item_code}" | ||||||
|  | 			if action == "Stop": | ||||||
|  | 				frappe.throw(_(msg), title=_("Inspection Submission"), exc=QualityInspectionNotSubmittedError) | ||||||
|  | 			else: | ||||||
|  | 				frappe.msgprint(_(msg), alert=True, indicator="orange") | ||||||
|  | 
 | ||||||
|  | 	def validate_qi_rejection(self, row): | ||||||
|  | 		"""Check if QI is rejected on row level, during submission""" | ||||||
|  | 		action = frappe.db.get_single_value("Stock Settings", "action_if_quality_inspection_is_rejected") | ||||||
|  | 		qa_status = frappe.db.get_value("Quality Inspection", row.quality_inspection, "status") | ||||||
|  | 
 | ||||||
|  | 		if qa_status == "Rejected": | ||||||
|  | 			link = frappe.utils.get_link_to_form('Quality Inspection', row.quality_inspection) | ||||||
|  | 			msg = f"Row #{row.idx}: Quality Inspection {link} was rejected for item {row.item_code}" | ||||||
|  | 			if action == "Stop": | ||||||
|  | 				frappe.throw(_(msg), title=_("Inspection Rejected"), exc=QualityInspectionRejectedError) | ||||||
|  | 			else: | ||||||
|  | 				frappe.msgprint(_(msg), alert=True, indicator="orange") | ||||||
| 
 | 
 | ||||||
| 	def update_blanket_order(self): | 	def update_blanket_order(self): | ||||||
| 		blanket_orders = list(set([d.blanket_order for d in self.items if d.blanket_order])) | 		blanket_orders = list(set([d.blanket_order for d in self.items if d.blanket_order])) | ||||||
|  | |||||||
| @ -102,7 +102,7 @@ | |||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2020-01-28 16:16:45.447213", |  "modified": "2021-06-29 18:27:02.832979", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "CRM", |  "module": "CRM", | ||||||
|  "name": "Appointment", |  "name": "Appointment", | ||||||
| @ -153,6 +153,18 @@ | |||||||
|    "role": "Sales User", |    "role": "Sales User", | ||||||
|    "share": 1, |    "share": 1, | ||||||
|    "write": 1 |    "write": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "create": 1, | ||||||
|  |    "delete": 1, | ||||||
|  |    "email": 1, | ||||||
|  |    "export": 1, | ||||||
|  |    "print": 1, | ||||||
|  |    "read": 1, | ||||||
|  |    "report": 1, | ||||||
|  |    "role": "Employee", | ||||||
|  |    "share": 1, | ||||||
|  |    "write": 1 | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "quick_entry": 1, |  "quick_entry": 1, | ||||||
|  | |||||||
| @ -168,12 +168,13 @@ class Lead(SellingController): | |||||||
| 		if self.phone: | 		if self.phone: | ||||||
| 			contact.append("phone_nos", { | 			contact.append("phone_nos", { | ||||||
| 				"phone": self.phone, | 				"phone": self.phone, | ||||||
| 				"is_primary": 1 | 				"is_primary_phone": 1 | ||||||
| 			}) | 			}) | ||||||
| 
 | 
 | ||||||
| 		if self.mobile_no: | 		if self.mobile_no: | ||||||
| 			contact.append("phone_nos", { | 			contact.append("phone_nos", { | ||||||
| 				"phone": self.mobile_no | 				"phone": self.mobile_no, | ||||||
|  | 				"is_primary_mobile_no":1 | ||||||
| 			}) | 			}) | ||||||
| 
 | 
 | ||||||
| 		contact.insert(ignore_permissions=True) | 		contact.insert(ignore_permissions=True) | ||||||
|  | |||||||
| @ -355,11 +355,11 @@ def get_or_create_course_enrollment(course, program): | |||||||
| 	student = get_current_student() | 	student = get_current_student() | ||||||
| 	course_enrollment = get_enrollment("course", course, student.name) | 	course_enrollment = get_enrollment("course", course, student.name) | ||||||
| 	if not course_enrollment: | 	if not course_enrollment: | ||||||
| 		program_enrollment = get_enrollment('program', program, student.name) | 		program_enrollment = get_enrollment('program', program.name, student.name) | ||||||
| 		if not program_enrollment: | 		if not program_enrollment: | ||||||
| 			frappe.throw(_("You are not enrolled in program {0}").format(program)) | 			frappe.throw(_("You are not enrolled in program {0}").format(program)) | ||||||
| 			return | 			return | ||||||
| 		return student.enroll_in_course(course_name=course, program_enrollment=get_enrollment('program', program, student.name)) | 		return student.enroll_in_course(course_name=course, program_enrollment=get_enrollment('program', program.name, student.name)) | ||||||
| 	else: | 	else: | ||||||
| 		return frappe.get_doc('Course Enrollment', course_enrollment) | 		return frappe.get_doc('Course Enrollment', course_enrollment) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -7,16 +7,21 @@ import frappe | |||||||
| import unittest | import unittest | ||||||
| from erpnext.erpnext_integrations.doctype.mpesa_settings.mpesa_settings import process_balance_info, verify_transaction | from erpnext.erpnext_integrations.doctype.mpesa_settings.mpesa_settings import process_balance_info, verify_transaction | ||||||
| from erpnext.accounts.doctype.pos_invoice.test_pos_invoice import create_pos_invoice | from erpnext.accounts.doctype.pos_invoice.test_pos_invoice import create_pos_invoice | ||||||
|  | from erpnext.erpnext_integrations.utils import create_mode_of_payment | ||||||
| 
 | 
 | ||||||
| class TestMpesaSettings(unittest.TestCase): | class TestMpesaSettings(unittest.TestCase): | ||||||
|  | 	def setUp(self): | ||||||
|  | 		# create payment gateway in setup | ||||||
|  | 		create_mpesa_settings(payment_gateway_name="_Test") | ||||||
|  | 		create_mpesa_settings(payment_gateway_name="_Account Balance") | ||||||
|  | 		create_mpesa_settings(payment_gateway_name="Payment") | ||||||
|  | 
 | ||||||
| 	def tearDown(self): | 	def tearDown(self): | ||||||
| 		frappe.db.sql('delete from `tabMpesa Settings`') | 		frappe.db.sql('delete from `tabMpesa Settings`') | ||||||
| 		frappe.db.sql('delete from `tabIntegration Request` where integration_request_service = "Mpesa"') | 		frappe.db.sql('delete from `tabIntegration Request` where integration_request_service = "Mpesa"') | ||||||
| 
 | 
 | ||||||
| 	def test_creation_of_payment_gateway(self): | 	def test_creation_of_payment_gateway(self): | ||||||
| 		create_mpesa_settings(payment_gateway_name="_Test") | 		mode_of_payment = create_mode_of_payment('Mpesa-_Test', payment_type="Phone") | ||||||
| 
 |  | ||||||
| 		mode_of_payment = frappe.get_doc("Mode of Payment", "Mpesa-_Test") |  | ||||||
| 		self.assertTrue(frappe.db.exists("Payment Gateway Account", {'payment_gateway': "Mpesa-_Test"})) | 		self.assertTrue(frappe.db.exists("Payment Gateway Account", {'payment_gateway': "Mpesa-_Test"})) | ||||||
| 		self.assertTrue(mode_of_payment.name) | 		self.assertTrue(mode_of_payment.name) | ||||||
| 		self.assertEqual(mode_of_payment.type, "Phone") | 		self.assertEqual(mode_of_payment.type, "Phone") | ||||||
| @ -47,7 +52,6 @@ class TestMpesaSettings(unittest.TestCase): | |||||||
| 		integration_request.delete() | 		integration_request.delete() | ||||||
| 
 | 
 | ||||||
| 	def test_processing_of_callback_payload(self): | 	def test_processing_of_callback_payload(self): | ||||||
| 		create_mpesa_settings(payment_gateway_name="Payment") |  | ||||||
| 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | ||||||
| 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | ||||||
| 		frappe.db.set_value("Customer", "_Test Customer", "default_currency", "KES") | 		frappe.db.set_value("Customer", "_Test Customer", "default_currency", "KES") | ||||||
| @ -90,7 +94,6 @@ class TestMpesaSettings(unittest.TestCase): | |||||||
| 		pos_invoice.delete() | 		pos_invoice.delete() | ||||||
| 
 | 
 | ||||||
| 	def test_processing_of_multiple_callback_payload(self): | 	def test_processing_of_multiple_callback_payload(self): | ||||||
| 		create_mpesa_settings(payment_gateway_name="Payment") |  | ||||||
| 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | ||||||
| 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | ||||||
| 		frappe.db.set_value("Mpesa Settings", "Payment", "transaction_limit", "500") | 		frappe.db.set_value("Mpesa Settings", "Payment", "transaction_limit", "500") | ||||||
| @ -141,7 +144,6 @@ class TestMpesaSettings(unittest.TestCase): | |||||||
| 		pos_invoice.delete() | 		pos_invoice.delete() | ||||||
| 
 | 
 | ||||||
| 	def test_processing_of_only_one_succes_callback_payload(self): | 	def test_processing_of_only_one_succes_callback_payload(self): | ||||||
| 		create_mpesa_settings(payment_gateway_name="Payment") |  | ||||||
| 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | 		mpesa_account = frappe.db.get_value("Payment Gateway Account", {"payment_gateway": 'Mpesa-Payment'}, "payment_account") | ||||||
| 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | 		frappe.db.set_value("Account", mpesa_account, "account_currency", "KES") | ||||||
| 		frappe.db.set_value("Mpesa Settings", "Payment", "transaction_limit", "500") | 		frappe.db.set_value("Mpesa Settings", "Payment", "transaction_limit", "500") | ||||||
| @ -202,6 +204,7 @@ def create_mpesa_settings(payment_gateway_name="Express"): | |||||||
| 
 | 
 | ||||||
| 	doc = frappe.get_doc(dict( #nosec | 	doc = frappe.get_doc(dict( #nosec | ||||||
| 		doctype="Mpesa Settings", | 		doctype="Mpesa Settings", | ||||||
|  | 		sandbox=1, | ||||||
| 		payment_gateway_name=payment_gateway_name, | 		payment_gateway_name=payment_gateway_name, | ||||||
| 		consumer_key="5sMu9LVI1oS3oBGPJfh3JyvLHwZOdTKn", | 		consumer_key="5sMu9LVI1oS3oBGPJfh3JyvLHwZOdTKn", | ||||||
| 		consumer_secret="VI1oS3oBGPJfh3JyvLHw", | 		consumer_secret="VI1oS3oBGPJfh3JyvLHw", | ||||||
|  | |||||||
| @ -52,7 +52,8 @@ def create_mode_of_payment(gateway, payment_type="General"): | |||||||
| 			"payment_gateway": gateway | 			"payment_gateway": gateway | ||||||
| 		}, ['payment_account']) | 		}, ['payment_account']) | ||||||
| 
 | 
 | ||||||
| 	if not frappe.db.exists("Mode of Payment", gateway) and payment_gateway_account: | 	mode_of_payment = frappe.db.exists("Mode of Payment", gateway)  | ||||||
|  | 	if not mode_of_payment and payment_gateway_account: | ||||||
| 		mode_of_payment = frappe.get_doc({ | 		mode_of_payment = frappe.get_doc({ | ||||||
| 			"doctype": "Mode of Payment", | 			"doctype": "Mode of Payment", | ||||||
| 			"mode_of_payment": gateway, | 			"mode_of_payment": gateway, | ||||||
| @ -66,6 +67,10 @@ def create_mode_of_payment(gateway, payment_type="General"): | |||||||
| 		}) | 		}) | ||||||
| 		mode_of_payment.insert(ignore_permissions=True) | 		mode_of_payment.insert(ignore_permissions=True) | ||||||
| 
 | 
 | ||||||
|  | 		return mode_of_payment | ||||||
|  | 	elif mode_of_payment: | ||||||
|  | 		return frappe.get_doc("Mode of Payment", mode_of_payment) | ||||||
|  | 
 | ||||||
| def get_tracking_url(carrier, tracking_number): | def get_tracking_url(carrier, tracking_number): | ||||||
| 	# Return the formatted Tracking URL. | 	# Return the formatted Tracking URL. | ||||||
| 	tracking_url = '' | 	tracking_url = '' | ||||||
|  | |||||||
| @ -157,6 +157,7 @@ website_route_rules = [ | |||||||
| 			"parents": [{"label": _("Material Request"), "route": "material-requests"}] | 			"parents": [{"label": _("Material Request"), "route": "material-requests"}] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
|  | 	{"from_route": "/project", "to_route": "Project"} | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| standard_portal_menu_items = [ | standard_portal_menu_items = [ | ||||||
|  | |||||||
| @ -72,7 +72,8 @@ class TestExpenseClaim(unittest.TestCase): | |||||||
| 	def test_expense_claim_gl_entry(self): | 	def test_expense_claim_gl_entry(self): | ||||||
| 		payable_account = get_payable_account(company_name) | 		payable_account = get_payable_account(company_name) | ||||||
| 		taxes = generate_taxes() | 		taxes = generate_taxes() | ||||||
| 		expense_claim = make_expense_claim(payable_account, 300, 200, company_name, "Travel Expenses - _TC4", do_not_submit=True, taxes=taxes) | 		expense_claim = make_expense_claim(payable_account, 300, 200, company_name, "Travel Expenses - _TC4",  | ||||||
|  | 			do_not_submit=True, taxes=taxes) | ||||||
| 		expense_claim.submit() | 		expense_claim.submit() | ||||||
| 
 | 
 | ||||||
| 		gl_entries = frappe.db.sql("""select account, debit, credit | 		gl_entries = frappe.db.sql("""select account, debit, credit | ||||||
| @ -82,7 +83,7 @@ class TestExpenseClaim(unittest.TestCase): | |||||||
| 		self.assertTrue(gl_entries) | 		self.assertTrue(gl_entries) | ||||||
| 
 | 
 | ||||||
| 		expected_values = dict((d[0], d) for d in [ | 		expected_values = dict((d[0], d) for d in [ | ||||||
| 			['CGST - _TC4',18.0, 0.0], | 			['Output Tax CGST - _TC4',18.0, 0.0], | ||||||
| 			[payable_account, 0.0, 218.0], | 			[payable_account, 0.0, 218.0], | ||||||
| 			["Travel Expenses - _TC4", 200.0, 0.0] | 			["Travel Expenses - _TC4", 200.0, 0.0] | ||||||
| 		]) | 		]) | ||||||
| @ -145,7 +146,7 @@ def generate_taxes(): | |||||||
| 	parent_account = frappe.db.get_value('Account', | 	parent_account = frappe.db.get_value('Account', | ||||||
| 		{'company': company_name, 'is_group':1, 'account_type': 'Tax'}, | 		{'company': company_name, 'is_group':1, 'account_type': 'Tax'}, | ||||||
| 		'name') | 		'name') | ||||||
| 	account = create_account(company=company_name, account_name="CGST", account_type="Tax", parent_account=parent_account) | 	account = create_account(company=company_name, account_name="Output Tax CGST", account_type="Tax", parent_account=parent_account) | ||||||
| 	return {'taxes':[{ | 	return {'taxes':[{ | ||||||
| 		"account_head": account, | 		"account_head": account, | ||||||
| 		"rate": 0, | 		"rate": 0, | ||||||
|  | |||||||
| @ -20,11 +20,10 @@ frappe.ui.form.on('Training Event', { | |||||||
| 				frappe.set_route("List", "Training Feedback"); | 				frappe.set_route("List", "Training Feedback"); | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
| 	} | 		frm.events.set_employee_query(frm); | ||||||
| }); | 	}, | ||||||
| 
 | 
 | ||||||
| frappe.ui.form.on("Training Event Employee", { | 	set_employee_query: function(frm) { | ||||||
| 	employee: function (frm) { |  | ||||||
| 		let emp = []; | 		let emp = []; | ||||||
| 		for (let d in frm.doc.employees) { | 		for (let d in frm.doc.employees) { | ||||||
| 			if (frm.doc.employees[d].employee) { | 			if (frm.doc.employees[d].employee) { | ||||||
| @ -34,9 +33,17 @@ frappe.ui.form.on("Training Event Employee", { | |||||||
| 		frm.set_query("employee", "employees", function () { | 		frm.set_query("employee", "employees", function () { | ||||||
| 			return { | 			return { | ||||||
| 				filters: { | 				filters: { | ||||||
| 					name: ["NOT IN", emp] | 					name: ["NOT IN", emp], | ||||||
|  | 					status: "Active" | ||||||
| 				} | 				} | ||||||
| 			}; | 			}; | ||||||
| 		}); | 		}); | ||||||
| 	} | 	} | ||||||
| }); | }); | ||||||
|  | 
 | ||||||
|  | frappe.ui.form.on("Training Event Employee", { | ||||||
|  | 	employee: function(frm) { | ||||||
|  | 		frm.events.set_employee_query(frm); | ||||||
|  | 	} | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | |||||||
| @ -19,6 +19,7 @@ | |||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
|    "in_list_view": 1, |    "in_list_view": 1, | ||||||
|    "label": "Employee", |    "label": "Employee", | ||||||
|  |    "no_copy": 1, | ||||||
|    "options": "Employee" |    "options": "Employee" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
| @ -68,7 +69,7 @@ | |||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "istable": 1, |  "istable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-05-21 12:41:59.336237", |  "modified": "2021-07-02 17:20:27.630176", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "HR", |  "module": "HR", | ||||||
|  "name": "Training Event Employee", |  "name": "Training Event Employee", | ||||||
|  | |||||||
| @ -35,7 +35,9 @@ | |||||||
|    "no_copy": 1, |    "no_copy": 1, | ||||||
|    "options": "Loan Security Pledge", |    "options": "Loan Security Pledge", | ||||||
|    "print_hide": 1, |    "print_hide": 1, | ||||||
|    "read_only": 1 |    "read_only": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fetch_from": "loan_application.applicant", |    "fetch_from": "loan_application.applicant", | ||||||
| @ -45,47 +47,63 @@ | |||||||
|    "in_standard_filter": 1, |    "in_standard_filter": 1, | ||||||
|    "label": "Applicant", |    "label": "Applicant", | ||||||
|    "options": "applicant_type", |    "options": "applicant_type", | ||||||
|    "reqd": 1 |    "reqd": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "loan_security_details_section", |    "fieldname": "loan_security_details_section", | ||||||
|    "fieldtype": "Section Break", |    "fieldtype": "Section Break", | ||||||
|    "label": "Loan Security Details" |    "label": "Loan Security Details", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "column_break_3", |    "fieldname": "column_break_3", | ||||||
|    "fieldtype": "Column Break" |    "fieldtype": "Column Break", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "loan", |    "fieldname": "loan", | ||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
|    "label": "Loan", |    "label": "Loan", | ||||||
|    "options": "Loan" |    "options": "Loan", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "loan_application", |    "fieldname": "loan_application", | ||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
|    "label": "Loan Application", |    "label": "Loan Application", | ||||||
|    "options": "Loan Application" |    "options": "Loan Application", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "total_security_value", |    "fieldname": "total_security_value", | ||||||
|    "fieldtype": "Currency", |    "fieldtype": "Currency", | ||||||
|    "label": "Total Security Value", |    "label": "Total Security Value", | ||||||
|    "options": "Company:company:default_currency", |    "options": "Company:company:default_currency", | ||||||
|    "read_only": 1 |    "read_only": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "maximum_loan_value", |    "fieldname": "maximum_loan_value", | ||||||
|    "fieldtype": "Currency", |    "fieldtype": "Currency", | ||||||
|    "label": "Maximum Loan Value", |    "label": "Maximum Loan Value", | ||||||
|    "options": "Company:company:default_currency", |    "options": "Company:company:default_currency", | ||||||
|    "read_only": 1 |    "read_only": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "loan_details_section", |    "fieldname": "loan_details_section", | ||||||
|    "fieldtype": "Section Break", |    "fieldtype": "Section Break", | ||||||
|    "label": "Loan  Details" |    "label": "Loan  Details", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "default": "Requested", |    "default": "Requested", | ||||||
| @ -94,37 +112,49 @@ | |||||||
|    "in_list_view": 1, |    "in_list_view": 1, | ||||||
|    "in_standard_filter": 1, |    "in_standard_filter": 1, | ||||||
|    "label": "Status", |    "label": "Status", | ||||||
|    "options": "Requested\nUnpledged\nPledged\nPartially Pledged", |    "options": "Requested\nUnpledged\nPledged\nPartially Pledged\nCancelled", | ||||||
|    "read_only": 1 |    "read_only": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "pledge_time", |    "fieldname": "pledge_time", | ||||||
|    "fieldtype": "Datetime", |    "fieldtype": "Datetime", | ||||||
|    "label": "Pledge Time", |    "label": "Pledge Time", | ||||||
|    "read_only": 1 |    "read_only": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "securities", |    "fieldname": "securities", | ||||||
|    "fieldtype": "Table", |    "fieldtype": "Table", | ||||||
|    "label": "Securities", |    "label": "Securities", | ||||||
|    "options": "Pledge", |    "options": "Pledge", | ||||||
|    "reqd": 1 |    "reqd": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "column_break_11", |    "fieldname": "column_break_11", | ||||||
|    "fieldtype": "Column Break" |    "fieldtype": "Column Break", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "section_break_10", |    "fieldname": "section_break_10", | ||||||
|    "fieldtype": "Section Break", |    "fieldtype": "Section Break", | ||||||
|    "label": "Totals" |    "label": "Totals", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "company", |    "fieldname": "company", | ||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
|    "label": "Company", |    "label": "Company", | ||||||
|    "options": "Company", |    "options": "Company", | ||||||
|    "reqd": 1 |    "reqd": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fetch_from": "loan.applicant_type", |    "fetch_from": "loan.applicant_type", | ||||||
| @ -132,35 +162,45 @@ | |||||||
|    "fieldtype": "Select", |    "fieldtype": "Select", | ||||||
|    "label": "Applicant Type", |    "label": "Applicant Type", | ||||||
|    "options": "Employee\nMember\nCustomer", |    "options": "Employee\nMember\nCustomer", | ||||||
|    "reqd": 1 |    "reqd": 1, | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "collapsible": 1, |    "collapsible": 1, | ||||||
|    "fieldname": "more_information_section", |    "fieldname": "more_information_section", | ||||||
|    "fieldtype": "Section Break", |    "fieldtype": "Section Break", | ||||||
|    "label": "More Information" |    "label": "More Information", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "allow_on_submit": 1, |    "allow_on_submit": 1, | ||||||
|    "fieldname": "reference_no", |    "fieldname": "reference_no", | ||||||
|    "fieldtype": "Data", |    "fieldtype": "Data", | ||||||
|    "label": "Reference No" |    "label": "Reference No", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "column_break_18", |    "fieldname": "column_break_18", | ||||||
|    "fieldtype": "Column Break" |    "fieldtype": "Column Break", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "allow_on_submit": 1, |    "allow_on_submit": 1, | ||||||
|    "fieldname": "description", |    "fieldname": "description", | ||||||
|    "fieldtype": "Text", |    "fieldtype": "Text", | ||||||
|    "label": "Description" |    "label": "Description", | ||||||
|  |    "show_days": 1, | ||||||
|  |    "show_seconds": 1 | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "is_submittable": 1, |  "is_submittable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-04-19 18:23:16.953305", |  "modified": "2021-06-29 17:15:16.082256", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Loan Management", |  "module": "Loan Management", | ||||||
|  "name": "Loan Security Pledge", |  "name": "Loan Security Pledge", | ||||||
|  | |||||||
| @ -23,6 +23,12 @@ class LoanSecurityPledge(Document): | |||||||
| 			update_shortfall_status(self.loan, self.total_security_value) | 			update_shortfall_status(self.loan, self.total_security_value) | ||||||
| 			update_loan(self.loan, self.maximum_loan_value) | 			update_loan(self.loan, self.maximum_loan_value) | ||||||
| 
 | 
 | ||||||
|  | 	def on_cancel(self): | ||||||
|  | 		if self.loan: | ||||||
|  | 			self.db_set("status", "Cancelled") | ||||||
|  | 			self.db_set("pledge_time", None) | ||||||
|  | 			update_loan(self.loan, self.maximum_loan_value, cancel=1) | ||||||
|  | 
 | ||||||
| 	def validate_duplicate_securities(self): | 	def validate_duplicate_securities(self): | ||||||
| 		security_list = [] | 		security_list = [] | ||||||
| 		for security in self.securities: | 		for security in self.securities: | ||||||
| @ -36,7 +42,7 @@ class LoanSecurityPledge(Document): | |||||||
| 		existing_pledge = '' | 		existing_pledge = '' | ||||||
| 
 | 
 | ||||||
| 		if self.loan: | 		if self.loan: | ||||||
| 			existing_pledge = frappe.db.get_value('Loan Security Pledge', {'loan': self.loan}, ['name']) | 			existing_pledge = frappe.db.get_value('Loan Security Pledge', {'loan': self.loan, 'docstatus': 1}, ['name']) | ||||||
| 
 | 
 | ||||||
| 		if existing_pledge: | 		if existing_pledge: | ||||||
| 			loan_security_type = frappe.db.get_value('Pledge', {'parent': existing_pledge}, ['loan_security_type']) | 			loan_security_type = frappe.db.get_value('Pledge', {'parent': existing_pledge}, ['loan_security_type']) | ||||||
| @ -77,8 +83,12 @@ class LoanSecurityPledge(Document): | |||||||
| 		self.total_security_value = total_security_value | 		self.total_security_value = total_security_value | ||||||
| 		self.maximum_loan_value = maximum_loan_value | 		self.maximum_loan_value = maximum_loan_value | ||||||
| 
 | 
 | ||||||
| def update_loan(loan, maximum_value_against_pledge): | def update_loan(loan, maximum_value_against_pledge, cancel=0): | ||||||
| 	maximum_loan_value = frappe.db.get_value('Loan', {'name': loan}, ['maximum_loan_amount']) | 	maximum_loan_value = frappe.db.get_value('Loan', {'name': loan}, ['maximum_loan_amount']) | ||||||
| 
 | 
 | ||||||
| 	frappe.db.sql(""" UPDATE `tabLoan` SET maximum_loan_amount=%s, is_secured_loan=1 | 	if cancel: | ||||||
| 		WHERE name=%s""", (maximum_loan_value + maximum_value_against_pledge, loan)) | 		frappe.db.sql(""" UPDATE `tabLoan` SET maximum_loan_amount=%s | ||||||
|  | 			WHERE name=%s""", (maximum_loan_value - maximum_value_against_pledge, loan)) | ||||||
|  | 	else: | ||||||
|  | 		frappe.db.sql(""" UPDATE `tabLoan` SET maximum_loan_amount=%s, is_secured_loan=1 | ||||||
|  | 			WHERE name=%s""", (maximum_loan_value + maximum_value_against_pledge, loan)) | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ frappe.ui.form.on('Blanket Order', { | |||||||
| 
 | 
 | ||||||
| 	refresh: function(frm) { | 	refresh: function(frm) { | ||||||
| 		erpnext.hide_company(); | 		erpnext.hide_company(); | ||||||
| 		if (frm.doc.customer && frm.doc.docstatus === 1) { | 		if (frm.doc.customer && frm.doc.docstatus === 1 && frm.doc.to_date > frappe.datetime.get_today()) { | ||||||
| 			frm.add_custom_button(__("Sales Order"), function() { | 			frm.add_custom_button(__("Sales Order"), function() { | ||||||
| 				frappe.model.open_mapped_doc({ | 				frappe.model.open_mapped_doc({ | ||||||
| 					method: "erpnext.manufacturing.doctype.blanket_order.blanket_order.make_order", | 					method: "erpnext.manufacturing.doctype.blanket_order.blanket_order.make_order", | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| { | { | ||||||
|  |  "actions": [], | ||||||
|  "autoname": "naming_series:", |  "autoname": "naming_series:", | ||||||
|  "creation": "2018-05-24 07:18:08.256060", |  "creation": "2018-05-24 07:18:08.256060", | ||||||
|  "doctype": "DocType", |  "doctype": "DocType", | ||||||
| @ -79,6 +80,7 @@ | |||||||
|    "reqd": 1 |    "reqd": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|  |    "allow_on_submit": 1, | ||||||
|    "fieldname": "to_date", |    "fieldname": "to_date", | ||||||
|    "fieldtype": "Date", |    "fieldtype": "Date", | ||||||
|    "label": "To Date", |    "label": "To Date", | ||||||
| @ -129,8 +131,10 @@ | |||||||
|    "label": "Terms and Conditions Details" |    "label": "Terms and Conditions Details" | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  |  "index_web_pages_for_search": 1, | ||||||
|  "is_submittable": 1, |  "is_submittable": 1, | ||||||
|  "modified": "2019-11-18 19:37:37.151686", |  "links": [], | ||||||
|  |  "modified": "2021-06-29 00:30:30.621636", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "Blanket Order", |  "name": "Blanket Order", | ||||||
|  | |||||||
| @ -36,6 +36,9 @@ | |||||||
|   "materials_section", |   "materials_section", | ||||||
|   "inspection_required", |   "inspection_required", | ||||||
|   "quality_inspection_template", |   "quality_inspection_template", | ||||||
|  |   "column_break_31", | ||||||
|  |   "bom_level", | ||||||
|  |   "section_break_33", | ||||||
|   "items", |   "items", | ||||||
|   "scrap_section", |   "scrap_section", | ||||||
|   "scrap_items", |   "scrap_items", | ||||||
| @ -513,6 +516,22 @@ | |||||||
|    "no_copy": 1, |    "no_copy": 1, | ||||||
|    "print_hide": 1, |    "print_hide": 1, | ||||||
|    "read_only": 1 |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "column_break_31", | ||||||
|  |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "default": "0", | ||||||
|  |    "fieldname": "bom_level", | ||||||
|  |    "fieldtype": "Int", | ||||||
|  |    "label": "BOM Level", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "section_break_33", | ||||||
|  |    "fieldtype": "Section Break", | ||||||
|  |    "hide_border": 1 | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "icon": "fa fa-sitemap", |  "icon": "fa fa-sitemap", | ||||||
| @ -520,7 +539,7 @@ | |||||||
|  "image_field": "image", |  "image_field": "image", | ||||||
|  "is_submittable": 1, |  "is_submittable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-03-16 12:25:09.081968", |  "modified": "2021-05-16 12:25:09.081968", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "BOM", |  "name": "BOM", | ||||||
|  | |||||||
| @ -154,6 +154,7 @@ class BOM(WebsiteGenerator): | |||||||
| 		self.calculate_cost() | 		self.calculate_cost() | ||||||
| 		self.update_stock_qty() | 		self.update_stock_qty() | ||||||
| 		self.update_cost(update_parent=False, from_child_bom=True, update_hour_rate = False, save=False) | 		self.update_cost(update_parent=False, from_child_bom=True, update_hour_rate = False, save=False) | ||||||
|  | 		self.set_bom_level() | ||||||
| 
 | 
 | ||||||
| 	def get_context(self, context): | 	def get_context(self, context): | ||||||
| 		context.parents = [{'name': 'boms', 'title': _('All BOMs') }] | 		context.parents = [{'name': 'boms', 'title': _('All BOMs') }] | ||||||
| @ -676,6 +677,19 @@ class BOM(WebsiteGenerator): | |||||||
| 		"""Get a complete tree representation preserving order of child items.""" | 		"""Get a complete tree representation preserving order of child items.""" | ||||||
| 		return BOMTree(self.name) | 		return BOMTree(self.name) | ||||||
| 
 | 
 | ||||||
|  | 	def set_bom_level(self, update=False): | ||||||
|  | 		levels = [] | ||||||
|  | 
 | ||||||
|  | 		self.bom_level = 0 | ||||||
|  | 		for row in self.items: | ||||||
|  | 			if row.bom_no: | ||||||
|  | 				levels.append(frappe.get_cached_value("BOM", row.bom_no, "bom_level") or 0) | ||||||
|  | 
 | ||||||
|  | 		if levels: | ||||||
|  | 			self.bom_level = max(levels) + 1 | ||||||
|  | 
 | ||||||
|  | 		if update: | ||||||
|  | 			self.db_set("bom_level", self.bom_level) | ||||||
| 
 | 
 | ||||||
| def get_bom_item_rate(args, bom_doc): | def get_bom_item_rate(args, bom_doc): | ||||||
| 	if bom_doc.rm_cost_as_per == 'Valuation Rate': | 	if bom_doc.rm_cost_as_per == 'Valuation Rate': | ||||||
| @ -860,7 +874,7 @@ def get_children(doctype, parent=None, is_root=False, **filters): | |||||||
| 		frappe.form_dict.parent = parent | 		frappe.form_dict.parent = parent | ||||||
| 
 | 
 | ||||||
| 	if frappe.form_dict.parent: | 	if frappe.form_dict.parent: | ||||||
| 		bom_doc = frappe.get_doc("BOM", frappe.form_dict.parent) | 		bom_doc = frappe.get_cached_doc("BOM", frappe.form_dict.parent) | ||||||
| 		frappe.has_permission("BOM", doc=bom_doc, throw=True) | 		frappe.has_permission("BOM", doc=bom_doc, throw=True) | ||||||
| 
 | 
 | ||||||
| 		bom_items = frappe.get_all('BOM Item', | 		bom_items = frappe.get_all('BOM Item', | ||||||
| @ -871,7 +885,7 @@ def get_children(doctype, parent=None, is_root=False, **filters): | |||||||
| 		item_names = tuple(d.get('item_code') for d in bom_items) | 		item_names = tuple(d.get('item_code') for d in bom_items) | ||||||
| 
 | 
 | ||||||
| 		items = frappe.get_list('Item', | 		items = frappe.get_list('Item', | ||||||
| 			fields=['image', 'description', 'name', 'stock_uom', 'item_name'], | 			fields=['image', 'description', 'name', 'stock_uom', 'item_name', 'is_sub_contracted_item'], | ||||||
| 			filters=[['name', 'in', item_names]]) # to get only required item dicts | 			filters=[['name', 'in', item_names]]) # to get only required item dicts | ||||||
| 
 | 
 | ||||||
| 		for bom_item in bom_items: | 		for bom_item in bom_items: | ||||||
| @ -884,6 +898,7 @@ def get_children(doctype, parent=None, is_root=False, **filters): | |||||||
| 
 | 
 | ||||||
| 			bom_item.parent_bom_qty = bom_doc.quantity | 			bom_item.parent_bom_qty = bom_doc.quantity | ||||||
| 			bom_item.expandable = 0 if bom_item.value in ('', None)  else 1 | 			bom_item.expandable = 0 if bom_item.value in ('', None)  else 1 | ||||||
|  | 			bom_item.image = frappe.db.escape(bom_item.image) | ||||||
| 
 | 
 | ||||||
| 		return bom_items | 		return bom_items | ||||||
| 
 | 
 | ||||||
| @ -1100,6 +1115,8 @@ def make_variant_bom(source_name, bom_no, item, variant_items, target_doc=None): | |||||||
| 		}, | 		}, | ||||||
| 		'BOM Item': { | 		'BOM Item': { | ||||||
| 			'doctype': 'BOM Item', | 			'doctype': 'BOM Item', | ||||||
|  | 			# stop get_mapped_doc copying parent bom_no to children | ||||||
|  | 			'field_no_map': ['bom_no'], | ||||||
| 			'condition': lambda doc: doc.has_variants == 0 | 			'condition': lambda doc: doc.has_variants == 0 | ||||||
| 		}, | 		}, | ||||||
| 	}, target_doc, postprocess) | 	}, target_doc, postprocess) | ||||||
|  | |||||||
| @ -1,13 +1,31 @@ | |||||||
| <div style="padding: 15px;"> | <div style="padding: 15px;"> | ||||||
| 	{% if data.image %} | 	<div class="row mb-5"> | ||||||
| 	<img class="responsive" src={{ data.image }}> | 		<div class="col-md-5" style="max-height: 500px"> | ||||||
| 	<hr style="margin: 15px -15px;"> | 			{% if data.image %} | ||||||
| 	{% endif %} | 				<div class="border image-field " style="overflow: hidden;border-color:#e6e6e6"> | ||||||
| 	<h4> | 					<img class="responsive" src={{ data.image }}> | ||||||
| 		{{ __("Description") }} | 				</div> | ||||||
| 	</h4> | 			{% endif %} | ||||||
| 	<div style="padding-top: 10px;"> | 		</div> | ||||||
| 		{{ data.description }} | 		<div class="col-md-7 h-500"> | ||||||
|  | 			<h4> | ||||||
|  | 				{{ __("Description") }} | ||||||
|  | 			</h4> | ||||||
|  | 			<div style="padding-top: 10px;"> | ||||||
|  | 				{{ data.description }} | ||||||
|  | 			</div> | ||||||
|  | 			<hr style="margin: 15px -15px;"> | ||||||
|  | 			<p> | ||||||
|  | 				{% if data.value %} | ||||||
|  | 				<a style="margin-right: 7px; margin-bottom: 7px" class="btn btn-default btn-xs" href="#Form/BOM/{{ data.value }}"> | ||||||
|  | 					{{ __("Open BOM {0}", [data.value.bold()]) }}</a> | ||||||
|  | 				{% endif %} | ||||||
|  | 				{% if data.item_code %} | ||||||
|  | 				<a class="btn btn-default btn-xs" href="#Form/Item/{{ data.item_code }}"> | ||||||
|  | 					{{ __("Open Item {0}", [data.item_code.bold()]) }}</a> | ||||||
|  | 				{% endif %} | ||||||
|  | 			</p> | ||||||
|  | 		</div> | ||||||
| 	</div> | 	</div> | ||||||
| 	<hr style="margin: 15px -15px;"> | 	<hr style="margin: 15px -15px;"> | ||||||
| 	<p> | 	<p> | ||||||
|  | |||||||
| @ -64,7 +64,7 @@ frappe.treeview_settings["BOM"] = { | |||||||
| 		if(node.is_root && node.data.value!="BOM") { | 		if(node.is_root && node.data.value!="BOM") { | ||||||
| 			frappe.model.with_doc("BOM", node.data.value, function() { | 			frappe.model.with_doc("BOM", node.data.value, function() { | ||||||
| 				var bom = frappe.model.get_doc("BOM", node.data.value); | 				var bom = frappe.model.get_doc("BOM", node.data.value); | ||||||
| 				node.data.image = bom.image || ""; | 				node.data.image = escape(bom.image) || ""; | ||||||
| 				node.data.description = bom.description || ""; | 				node.data.description = bom.description || ""; | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
|  | |||||||
| @ -192,15 +192,20 @@ class JobCard(Document): | |||||||
| 						"completed_qty": args.get("completed_qty") or 0.0 | 						"completed_qty": args.get("completed_qty") or 0.0 | ||||||
| 					}) | 					}) | ||||||
| 		elif args.get("start_time"): | 		elif args.get("start_time"): | ||||||
| 			for name in employees: | 			new_args = { | ||||||
| 				self.append("time_logs", { | 				"from_time": get_datetime(args.get("start_time")), | ||||||
| 					"from_time": get_datetime(args.get("start_time")), | 				"operation": args.get("sub_operation"), | ||||||
| 					"employee": name.get('employee'), | 				"completed_qty": 0.0 | ||||||
| 					"operation": args.get("sub_operation"), | 			} | ||||||
| 					"completed_qty": 0.0 |  | ||||||
| 				}) |  | ||||||
| 
 | 
 | ||||||
| 		if not self.employee: | 			if employees: | ||||||
|  | 				for name in employees: | ||||||
|  | 					new_args.employee = name.get('employee') | ||||||
|  | 					self.add_start_time_log(new_args) | ||||||
|  | 			else: | ||||||
|  | 				self.add_start_time_log(new_args) | ||||||
|  | 
 | ||||||
|  | 		if not self.employee and employees: | ||||||
| 			self.set_employees(employees) | 			self.set_employees(employees) | ||||||
| 
 | 
 | ||||||
| 		if self.status == "On Hold": | 		if self.status == "On Hold": | ||||||
| @ -208,6 +213,9 @@ class JobCard(Document): | |||||||
| 
 | 
 | ||||||
| 		self.save() | 		self.save() | ||||||
| 
 | 
 | ||||||
|  | 	def add_start_time_log(self, args): | ||||||
|  | 		self.append("time_logs", args) | ||||||
|  | 
 | ||||||
| 	def set_employees(self, employees): | 	def set_employees(self, employees): | ||||||
| 		for name in employees: | 		for name in employees: | ||||||
| 			self.append('employee', { | 			self.append('employee', { | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ | |||||||
| frappe.ui.form.on('Production Plan', { | frappe.ui.form.on('Production Plan', { | ||||||
| 	setup: function(frm) { | 	setup: function(frm) { | ||||||
| 		frm.custom_make_buttons = { | 		frm.custom_make_buttons = { | ||||||
| 			'Work Order': 'Work Order', | 			'Work Order': 'Work Order / Subcontract PO', | ||||||
| 			'Material Request': 'Material Request', | 			'Material Request': 'Material Request', | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| @ -68,17 +68,13 @@ frappe.ui.form.on('Production Plan', { | |||||||
| 			frm.trigger("show_progress"); | 			frm.trigger("show_progress"); | ||||||
| 
 | 
 | ||||||
| 			if (frm.doc.status !== "Completed") { | 			if (frm.doc.status !== "Completed") { | ||||||
| 				if (frm.doc.po_items && frm.doc.status !== "Closed") { | 				frm.add_custom_button(__("Work Order Tree"), ()=> { | ||||||
| 					frm.add_custom_button(__("Work Order"), ()=> { | 					frappe.set_route('Tree', 'Work Order', {production_plan: frm.doc.name}); | ||||||
| 						frm.trigger("make_work_order"); | 				}, __('View')); | ||||||
| 					}, __('Create')); |  | ||||||
| 				} |  | ||||||
| 
 | 
 | ||||||
| 				if (frm.doc.mr_items && !in_list(['Material Requested', 'Closed'], frm.doc.status)) { | 				frm.add_custom_button(__("Production Plan Summary"), ()=> { | ||||||
| 					frm.add_custom_button(__("Material Request"), ()=> { | 					frappe.set_route('query-report', 'Production Plan Summary', {production_plan: frm.doc.name}); | ||||||
| 						frm.trigger("make_material_request"); | 				}, __('View')); | ||||||
| 					}, __('Create')); |  | ||||||
| 				} |  | ||||||
| 
 | 
 | ||||||
| 				if  (frm.doc.status === "Closed") { | 				if  (frm.doc.status === "Closed") { | ||||||
| 					frm.add_custom_button(__("Re-open"), function() { | 					frm.add_custom_button(__("Re-open"), function() { | ||||||
| @ -89,6 +85,18 @@ frappe.ui.form.on('Production Plan', { | |||||||
| 						frm.events.close_open_production_plan(frm, true); | 						frm.events.close_open_production_plan(frm, true); | ||||||
| 					}, __("Status")); | 					}, __("Status")); | ||||||
| 				} | 				} | ||||||
|  | 
 | ||||||
|  | 				if (frm.doc.po_items && frm.doc.status !== "Closed") { | ||||||
|  | 					frm.add_custom_button(__("Work Order / Subcontract PO"), ()=> { | ||||||
|  | 						frm.trigger("make_work_order"); | ||||||
|  | 					}, __('Create')); | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				if (frm.doc.mr_items && !in_list(['Material Requested', 'Closed'], frm.doc.status)) { | ||||||
|  | 					frm.add_custom_button(__("Material Request"), ()=> { | ||||||
|  | 						frm.trigger("make_material_request"); | ||||||
|  | 					}, __('Create')); | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -233,6 +241,17 @@ frappe.ui.form.on('Production Plan', { | |||||||
| 		}); | 		}); | ||||||
| 	}, | 	}, | ||||||
| 
 | 
 | ||||||
|  | 	get_sub_assembly_items: function(frm) { | ||||||
|  | 		frappe.call({ | ||||||
|  | 			method: "get_sub_assembly_items", | ||||||
|  | 			freeze: true, | ||||||
|  | 			doc: frm.doc, | ||||||
|  | 			callback: function() { | ||||||
|  | 				refresh_field("sub_assembly_items"); | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	}, | ||||||
|  | 
 | ||||||
| 	get_items_for_mr: function(frm) { | 	get_items_for_mr: function(frm) { | ||||||
| 		if (!frm.doc.for_warehouse) { | 		if (!frm.doc.for_warehouse) { | ||||||
| 			frappe.throw(__("Select warehouse for material requests")); | 			frappe.throw(__("Select warehouse for material requests")); | ||||||
|  | |||||||
| @ -32,6 +32,9 @@ | |||||||
|   "po_items", |   "po_items", | ||||||
|   "section_break_25", |   "section_break_25", | ||||||
|   "prod_plan_references", |   "prod_plan_references", | ||||||
|  |   "section_break_24", | ||||||
|  |   "get_sub_assembly_items", | ||||||
|  |   "sub_assembly_items", | ||||||
|   "material_request_planning", |   "material_request_planning", | ||||||
|   "include_non_stock_items", |   "include_non_stock_items", | ||||||
|   "include_subcontracted_items", |   "include_subcontracted_items", | ||||||
| @ -187,7 +190,7 @@ | |||||||
|    "depends_on": "get_items_from", |    "depends_on": "get_items_from", | ||||||
|    "fieldname": "get_items", |    "fieldname": "get_items", | ||||||
|    "fieldtype": "Button", |    "fieldtype": "Button", | ||||||
|    "label": "Get Items For Work Order" |    "label": "Get Finished Goods for Manufacture" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "po_items", |    "fieldname": "po_items", | ||||||
| @ -199,7 +202,7 @@ | |||||||
|   { |   { | ||||||
|    "fieldname": "material_request_planning", |    "fieldname": "material_request_planning", | ||||||
|    "fieldtype": "Section Break", |    "fieldtype": "Section Break", | ||||||
|    "label": "Material Request Planning" |    "label": "Material Requirement Planning" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "default": "1", |    "default": "1", | ||||||
| @ -237,12 +240,13 @@ | |||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "section_break_27", |    "fieldname": "section_break_27", | ||||||
|    "fieldtype": "Section Break" |    "fieldtype": "Section Break", | ||||||
|  |    "hide_border": 1 | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|    "fieldname": "mr_items", |    "fieldname": "mr_items", | ||||||
|    "fieldtype": "Table", |    "fieldtype": "Table", | ||||||
|    "label": "Material Request Plan Item", |    "label": "Raw Materials", | ||||||
|    "no_copy": 1, |    "no_copy": 1, | ||||||
|    "options": "Material Request Plan Item" |    "options": "Material Request Plan Item" | ||||||
|   }, |   }, | ||||||
| @ -337,13 +341,30 @@ | |||||||
|    "hidden": 1, |    "hidden": 1, | ||||||
|    "label": "Production Plan Item Reference", |    "label": "Production Plan Item Reference", | ||||||
|    "options": "Production Plan Item Reference" |    "options": "Production Plan Item Reference" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "section_break_24", | ||||||
|  |    "fieldtype": "Section Break", | ||||||
|  |    "hide_border": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "sub_assembly_items", | ||||||
|  |    "fieldtype": "Table", | ||||||
|  |    "label": "Sub Assembly Items", | ||||||
|  |    "no_copy": 1, | ||||||
|  |    "options": "Production Plan Sub Assembly Item" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "get_sub_assembly_items", | ||||||
|  |    "fieldtype": "Button", | ||||||
|  |    "label": "Get Sub Assembly Items" | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "icon": "fa fa-calendar", |  "icon": "fa fa-calendar", | ||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "is_submittable": 1, |  "is_submittable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-05-24 16:59:03.643211", |  "modified": "2021-06-28 20:00:33.905114", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "Production Plan", |  "name": "Production Plan", | ||||||
|  | |||||||
| @ -5,10 +5,11 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
| import frappe, json, copy | import frappe, json, copy | ||||||
| from frappe import msgprint, _ | from frappe import msgprint, _ | ||||||
| from six import string_types, iteritems | from six import iteritems | ||||||
| 
 | 
 | ||||||
| from frappe.model.document import Document | from frappe.model.document import Document | ||||||
| from frappe.utils import cstr, flt, cint, nowdate, add_days, comma_and, now_datetime, ceil | from frappe.utils import (flt, cint, nowdate, add_days, comma_and, now_datetime, | ||||||
|  | 	ceil, get_link_to_form, getdate) | ||||||
| from frappe.utils.csvutils import build_csv_response | from frappe.utils.csvutils import build_csv_response | ||||||
| from erpnext.manufacturing.doctype.bom.bom import validate_bom_no, get_children | from erpnext.manufacturing.doctype.bom.bom import validate_bom_no, get_children | ||||||
| from erpnext.manufacturing.doctype.work_order.work_order import get_item_details | from erpnext.manufacturing.doctype.work_order.work_order import get_item_details | ||||||
| @ -349,49 +350,88 @@ class ProductionPlan(Document): | |||||||
| 
 | 
 | ||||||
| 	@frappe.whitelist() | 	@frappe.whitelist() | ||||||
| 	def make_work_order(self): | 	def make_work_order(self): | ||||||
| 		wo_list = [] | 		wo_list, po_list = [], [] | ||||||
|  | 		subcontracted_po = {} | ||||||
|  | 
 | ||||||
| 		self.validate_data() | 		self.validate_data() | ||||||
|  | 		self.make_work_order_for_finished_goods(wo_list) | ||||||
|  | 		self.make_work_order_for_subassembly_items(wo_list, subcontracted_po) | ||||||
|  | 		self.make_subcontracted_purchase_order(subcontracted_po, po_list) | ||||||
|  | 		self.show_list_created_message('Work Order', wo_list) | ||||||
|  | 		self.show_list_created_message('Purchase Order', po_list) | ||||||
|  | 
 | ||||||
|  | 	def make_work_order_for_finished_goods(self, wo_list): | ||||||
| 		items_data = self.get_production_items() | 		items_data = self.get_production_items() | ||||||
| 
 | 
 | ||||||
| 		for key, item in items_data.items(): | 		for key, item in items_data.items(): | ||||||
|  | 			if self.sub_assembly_items: | ||||||
|  | 				item['use_multi_level_bom'] = 0 | ||||||
|  | 
 | ||||||
| 			work_order = self.create_work_order(item) | 			work_order = self.create_work_order(item) | ||||||
| 			if work_order: | 			if work_order: | ||||||
| 				wo_list.append(work_order) | 				wo_list.append(work_order) | ||||||
| 
 | 
 | ||||||
| 			if item.get("make_work_order_for_sub_assembly_items"): | 	def make_work_order_for_subassembly_items(self, wo_list, subcontracted_po): | ||||||
| 				work_orders = self.make_work_order_for_sub_assembly_items(item) | 		for row in self.sub_assembly_items: | ||||||
| 				wo_list.extend(work_orders) | 			if row.type_of_manufacturing == 'Subcontract': | ||||||
|  | 				subcontracted_po.setdefault(row.supplier, []).append(row) | ||||||
|  | 				continue | ||||||
|  | 
 | ||||||
|  | 			args = {} | ||||||
|  | 			self.prepare_args_for_sub_assembly_items(row, args) | ||||||
|  | 			work_order = self.create_work_order(args) | ||||||
|  | 			if work_order: | ||||||
|  | 				wo_list.append(work_order) | ||||||
|  | 
 | ||||||
|  | 	def make_subcontracted_purchase_order(self, subcontracted_po, purchase_orders): | ||||||
|  | 		if not subcontracted_po: | ||||||
|  | 			return | ||||||
|  | 
 | ||||||
|  | 		for supplier, po_list in subcontracted_po.items(): | ||||||
|  | 			po = frappe.new_doc('Purchase Order') | ||||||
|  | 			po.supplier = supplier | ||||||
|  | 			po.schedule_date = getdate(po_list[0].schedule_date) if po_list[0].schedule_date else nowdate() | ||||||
|  | 			po.is_subcontracted_item = 'Yes' | ||||||
|  | 			for row in po_list: | ||||||
|  | 				args = { | ||||||
|  | 					'item_code': row.production_item, | ||||||
|  | 					'warehouse': row.fg_warehouse, | ||||||
|  | 					'production_plan_sub_assembly_item': row.name, | ||||||
|  | 					'bom': row.bom_no, | ||||||
|  | 					'production_plan': self.name | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				for field in ['schedule_date', 'qty', 'uom', 'stock_uom', 'item_name', | ||||||
|  | 					'description', 'production_plan_item']: | ||||||
|  | 					args[field] = row.get(field) | ||||||
|  | 
 | ||||||
|  | 				po.append('items', args) | ||||||
|  | 
 | ||||||
|  | 			po.set_missing_values() | ||||||
|  | 			po.flags.ignore_mandatory = True | ||||||
|  | 			po.flags.ignore_validate = True | ||||||
|  | 			po.insert() | ||||||
|  | 			purchase_orders.append(po.name) | ||||||
|  | 
 | ||||||
|  | 	def show_list_created_message(self, doctype, doc_list=None): | ||||||
|  | 		if not doc_list: | ||||||
|  | 			return | ||||||
| 
 | 
 | ||||||
| 		frappe.flags.mute_messages = False | 		frappe.flags.mute_messages = False | ||||||
|  | 		if doc_list: | ||||||
|  | 			doc_list = [get_link_to_form(doctype, p) for p in doc_list] | ||||||
|  | 			msgprint(_("{0} created").format(comma_and(doc_list))) | ||||||
| 
 | 
 | ||||||
| 		if wo_list: | 	def prepare_args_for_sub_assembly_items(self, row, args): | ||||||
| 			wo_list = ["""<a href="/app/Form/Work Order/%s" target="_blank">%s</a>""" % \ | 		for field in ["production_item", "item_name", "qty", "fg_warehouse", | ||||||
| 				(p, p) for p in wo_list] | 			"description", "bom_no", "stock_uom", "bom_level", "production_plan_item"]: | ||||||
| 			msgprint(_("{0} created").format(comma_and(wo_list))) | 			args[field] = row.get(field) | ||||||
| 		else : |  | ||||||
| 			msgprint(_("No Work Orders created")) |  | ||||||
| 
 | 
 | ||||||
| 	def make_work_order_for_sub_assembly_items(self, item): | 		args.update({ | ||||||
| 		work_orders = [] | 			"use_multi_level_bom": 0, | ||||||
| 		bom_data = {} | 			"production_plan": self.name, | ||||||
| 
 | 			"production_plan_sub_assembly_item": row.name | ||||||
| 		get_sub_assembly_items(item.get("bom_no"), bom_data, item.get("qty")) | 		}) | ||||||
| 
 |  | ||||||
| 		for key, data in bom_data.items(): |  | ||||||
| 			data.update({ |  | ||||||
| 				'qty': data.get("stock_qty"), |  | ||||||
| 				'production_plan': self.name, |  | ||||||
| 				'use_multi_level_bom': item.get("use_multi_level_bom"), |  | ||||||
| 				'company': self.company, |  | ||||||
| 				'fg_warehouse': item.get("fg_warehouse"), |  | ||||||
| 				'update_consumed_material_cost_in_project': 0 |  | ||||||
| 			}) |  | ||||||
| 
 |  | ||||||
| 			work_order = self.create_work_order(data) |  | ||||||
| 			if work_order: |  | ||||||
| 				work_orders.append(work_order) |  | ||||||
| 
 |  | ||||||
| 		return work_orders |  | ||||||
| 
 | 
 | ||||||
| 	def create_work_order(self, item): | 	def create_work_order(self, item): | ||||||
| 		from erpnext.manufacturing.doctype.work_order.work_order import OverProductionError, get_default_warehouse | 		from erpnext.manufacturing.doctype.work_order.work_order import OverProductionError, get_default_warehouse | ||||||
| @ -476,9 +516,32 @@ class ProductionPlan(Document): | |||||||
| 		else : | 		else : | ||||||
| 			msgprint(_("No material request created")) | 			msgprint(_("No material request created")) | ||||||
| 
 | 
 | ||||||
|  | 	@frappe.whitelist() | ||||||
|  | 	def get_sub_assembly_items(self, manufacturing_type=None): | ||||||
|  | 		self.sub_assembly_items = [] | ||||||
|  | 		for row in self.po_items: | ||||||
|  | 			bom_data = [] | ||||||
|  | 			get_sub_assembly_items(row.bom_no, bom_data, row.planned_qty) | ||||||
|  | 			self.set_sub_assembly_items_based_on_level(row, bom_data, manufacturing_type) | ||||||
|  | 
 | ||||||
|  | 		self.save() | ||||||
|  | 
 | ||||||
|  | 	def set_sub_assembly_items_based_on_level(self, row, bom_data, manufacturing_type=None): | ||||||
|  | 		bom_data = sorted(bom_data, key = lambda i: i.bom_level) | ||||||
|  | 
 | ||||||
|  | 		for data in bom_data: | ||||||
|  | 			data.qty = data.stock_qty | ||||||
|  | 			data.production_plan_item = row.name | ||||||
|  | 			data.fg_warehouse = row.warehouse | ||||||
|  | 			data.schedule_date = row.planned_start_date | ||||||
|  | 			data.type_of_manufacturing = manufacturing_type or ("Subcontract" if data.is_sub_contracted_item | ||||||
|  | 				else "In House") | ||||||
|  | 
 | ||||||
|  | 			self.append("sub_assembly_items", data) | ||||||
|  | 
 | ||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def download_raw_materials(doc, warehouses=None): | def download_raw_materials(doc, warehouses=None): | ||||||
| 	if isinstance(doc, string_types): | 	if isinstance(doc, str): | ||||||
| 		doc = frappe._dict(json.loads(doc)) | 		doc = frappe._dict(json.loads(doc)) | ||||||
| 
 | 
 | ||||||
| 	item_list = [['Item Code', 'Description', 'Stock UOM', 'Warehouse', 'Required Qty as per BOM', | 	item_list = [['Item Code', 'Description', 'Stock UOM', 'Warehouse', 'Required Qty as per BOM', | ||||||
| @ -660,7 +723,7 @@ def get_sales_orders(self): | |||||||
| 
 | 
 | ||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def get_bin_details(row, company, for_warehouse=None, all_warehouse=False): | def get_bin_details(row, company, for_warehouse=None, all_warehouse=False): | ||||||
| 	if isinstance(row, string_types): | 	if isinstance(row, str): | ||||||
| 		row = frappe._dict(json.loads(row)) | 		row = frappe._dict(json.loads(row)) | ||||||
| 
 | 
 | ||||||
| 	company = frappe.db.escape(company) | 	company = frappe.db.escape(company) | ||||||
| @ -684,8 +747,11 @@ def get_bin_details(row, company, for_warehouse=None, all_warehouse=False): | |||||||
| 		group by item_code, warehouse | 		group by item_code, warehouse | ||||||
| 	""".format(conditions=conditions), { "item_code": row['item_code'] }, as_dict=1) | 	""".format(conditions=conditions), { "item_code": row['item_code'] }, as_dict=1) | ||||||
| 
 | 
 | ||||||
| def get_warehouse_list(warehouses, warehouse_list=[]): | def get_warehouse_list(warehouses, warehouse_list=None): | ||||||
| 	if isinstance(warehouses, string_types): | 	if not warehouse_list: | ||||||
|  | 		warehouse_list = [] | ||||||
|  | 
 | ||||||
|  | 	if isinstance(warehouses, str): | ||||||
| 		warehouses = json.loads(warehouses) | 		warehouses = json.loads(warehouses) | ||||||
| 
 | 
 | ||||||
| 	for row in warehouses: | 	for row in warehouses: | ||||||
| @ -697,7 +763,7 @@ def get_warehouse_list(warehouses, warehouse_list=[]): | |||||||
| 
 | 
 | ||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def get_items_for_material_requests(doc, warehouses=None, get_parent_warehouse_data=None): | def get_items_for_material_requests(doc, warehouses=None, get_parent_warehouse_data=None): | ||||||
| 	if isinstance(doc, string_types): | 	if isinstance(doc, str): | ||||||
| 		doc = frappe._dict(json.loads(doc)) | 		doc = frappe._dict(json.loads(doc)) | ||||||
| 
 | 
 | ||||||
| 	warehouse_list = [] | 	warehouse_list = [] | ||||||
| @ -726,6 +792,9 @@ def get_items_for_material_requests(doc, warehouses=None, get_parent_warehouse_d | |||||||
| 
 | 
 | ||||||
| 	so_item_details = frappe._dict() | 	so_item_details = frappe._dict() | ||||||
| 	for data in po_items: | 	for data in po_items: | ||||||
|  | 		if not data.get("include_exploded_items") and doc.get("sub_assembly_items"): | ||||||
|  | 			data["include_exploded_items"] = 1 | ||||||
|  | 
 | ||||||
| 		planned_qty = data.get('required_qty') or data.get('planned_qty') | 		planned_qty = data.get('required_qty') or data.get('planned_qty') | ||||||
| 		ignore_existing_ordered_qty = data.get('ignore_existing_ordered_qty') or ignore_existing_ordered_qty | 		ignore_existing_ordered_qty = data.get('ignore_existing_ordered_qty') or ignore_existing_ordered_qty | ||||||
| 		warehouse = doc.get('for_warehouse') | 		warehouse = doc.get('for_warehouse') | ||||||
| @ -857,23 +926,28 @@ def get_item_data(item_code): | |||||||
| #		"description": item_details.get("description") | #		"description": item_details.get("description") | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| def get_sub_assembly_items(bom_no, bom_data, to_produce_qty): | def get_sub_assembly_items(bom_no, bom_data, to_produce_qty, indent=0): | ||||||
| 	data = get_children('BOM', parent = bom_no) | 	data = get_children('BOM', parent = bom_no) | ||||||
| 	for d in data: | 	for d in data: | ||||||
| 		if d.expandable: | 		if d.expandable: | ||||||
| 			key = (d.name, d.value) | 			parent_item_code = frappe.get_cached_value("BOM", bom_no, "item") | ||||||
| 			if key not in bom_data: | 			bom_level = (frappe.get_cached_value("BOM", d.value, "bom_level") | ||||||
| 				bom_data.setdefault(key, { | 				if d.value else 0) | ||||||
| 					'stock_qty': 0, |  | ||||||
| 					'description': d.description, |  | ||||||
| 					'production_item': d.item_code, |  | ||||||
| 					'item_name': d.item_name, |  | ||||||
| 					'stock_uom': d.stock_uom, |  | ||||||
| 					'uom': d.stock_uom, |  | ||||||
| 					'bom_no': d.value |  | ||||||
| 				}) |  | ||||||
| 
 | 
 | ||||||
| 			bom_item = bom_data.get(key) | 			stock_qty = (d.stock_qty / d.parent_bom_qty) * flt(to_produce_qty) | ||||||
| 			bom_item["stock_qty"] += (d.stock_qty / d.parent_bom_qty) * flt(to_produce_qty) | 			bom_data.append(frappe._dict({ | ||||||
|  | 				'parent_item_code': parent_item_code, | ||||||
|  | 				'description': d.description, | ||||||
|  | 				'production_item': d.item_code, | ||||||
|  | 				'item_name': d.item_name, | ||||||
|  | 				'stock_uom': d.stock_uom, | ||||||
|  | 				'uom': d.stock_uom, | ||||||
|  | 				'bom_no': d.value, | ||||||
|  | 				'is_sub_contracted_item': d.is_sub_contracted_item, | ||||||
|  | 				'bom_level': bom_level, | ||||||
|  | 				'indent': indent, | ||||||
|  | 				'stock_qty': stock_qty | ||||||
|  | 			})) | ||||||
| 
 | 
 | ||||||
| 			get_sub_assembly_items(bom_item.get("bom_no"), bom_data, bom_item["stock_qty"]) | 			if d.value: | ||||||
|  | 				get_sub_assembly_items(d.value, bom_data, stock_qty, indent=indent+1) | ||||||
|  | |||||||
| @ -9,5 +9,9 @@ def get_data(): | |||||||
| 				'label': _('Transactions'), | 				'label': _('Transactions'), | ||||||
| 				'items': ['Work Order', 'Material Request'] | 				'items': ['Work Order', 'Material Request'] | ||||||
| 			}, | 			}, | ||||||
|  | 			{ | ||||||
|  | 				'label': _('Subcontract'), | ||||||
|  | 				'items': ['Purchase Order'] | ||||||
|  | 			}, | ||||||
| 		] | 		] | ||||||
| 	} | 	} | ||||||
| @ -169,7 +169,7 @@ class TestProductionPlan(unittest.TestCase): | |||||||
| 		pln.get_items() | 		pln.get_items() | ||||||
| 		pln.submit() | 		pln.submit() | ||||||
| 
 | 
 | ||||||
| 		self.assertTrue(pln.po_items[0].planned_qty, 3)	 | 		self.assertTrue(pln.po_items[0].planned_qty, 3) | ||||||
| 
 | 
 | ||||||
| 		pln.make_work_order() | 		pln.make_work_order() | ||||||
| 		work_order = frappe.db.get_value('Work Order', { | 		work_order = frappe.db.get_value('Work Order', { | ||||||
| @ -193,10 +193,10 @@ class TestProductionPlan(unittest.TestCase): | |||||||
| 		for so_item in so_items: | 		for so_item in so_items: | ||||||
| 			so_wo_qty = frappe.db.get_value('Sales Order Item', so_item, 'work_order_qty') | 			so_wo_qty = frappe.db.get_value('Sales Order Item', so_item, 'work_order_qty') | ||||||
| 			self.assertEqual(so_wo_qty, 0.0) | 			self.assertEqual(so_wo_qty, 0.0) | ||||||
| 		 | 
 | ||||||
| 		latest_plan = frappe.get_doc('Production Plan', pln.name) | 		latest_plan = frappe.get_doc('Production Plan', pln.name) | ||||||
| 		latest_plan.cancel() | 		latest_plan.cancel() | ||||||
| 	 | 
 | ||||||
| 	def test_pp_to_mr_customer_provided(self): | 	def test_pp_to_mr_customer_provided(self): | ||||||
| 		#Material Request from Production Plan for Customer Provided | 		#Material Request from Production Plan for Customer Provided | ||||||
| 		create_item('CUST-0987', is_customer_provided_item = 1, customer = '_Test Customer', is_purchase_item = 0) | 		create_item('CUST-0987', is_customer_provided_item = 1, customer = '_Test Customer', is_purchase_item = 0) | ||||||
| @ -236,10 +236,10 @@ class TestProductionPlan(unittest.TestCase): | |||||||
| 		pln.append("po_items", { | 		pln.append("po_items", { | ||||||
| 			"item_code": item_code, | 			"item_code": item_code, | ||||||
| 			"bom_no": frappe.db.get_value('BOM', {'item': "Test BOM 1"}), | 			"bom_no": frappe.db.get_value('BOM', {'item': "Test BOM 1"}), | ||||||
| 			"planned_qty": 3, | 			"planned_qty": 3 | ||||||
| 			"make_work_order_for_sub_assembly_items": 1 |  | ||||||
| 		}) | 		}) | ||||||
| 
 | 
 | ||||||
|  | 		pln.get_sub_assembly_items('In House') | ||||||
| 		pln.submit() | 		pln.submit() | ||||||
| 		pln.make_work_order() | 		pln.make_work_order() | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -9,18 +9,17 @@ | |||||||
|   "include_exploded_items", |   "include_exploded_items", | ||||||
|   "item_code", |   "item_code", | ||||||
|   "bom_no", |   "bom_no", | ||||||
|   "planned_qty", |  | ||||||
|   "column_break_6", |   "column_break_6", | ||||||
|   "make_work_order_for_sub_assembly_items", |   "planned_qty", | ||||||
|   "warehouse", |   "warehouse", | ||||||
|   "planned_start_date", |   "planned_start_date", | ||||||
|   "section_break_9", |   "section_break_9", | ||||||
|   "pending_qty", |   "pending_qty", | ||||||
|   "ordered_qty", |   "ordered_qty", | ||||||
|   "produced_qty", |  | ||||||
|   "column_break_17", |   "column_break_17", | ||||||
|   "description", |   "description", | ||||||
|   "stock_uom", |   "stock_uom", | ||||||
|  |   "produced_qty", | ||||||
|   "reference_section", |   "reference_section", | ||||||
|   "sales_order", |   "sales_order", | ||||||
|   "sales_order_item", |   "sales_order_item", | ||||||
| @ -32,11 +31,10 @@ | |||||||
|  ], |  ], | ||||||
|  "fields": [ |  "fields": [ | ||||||
|   { |   { | ||||||
|    "columns": 2, |    "columns": 1, | ||||||
|    "default": "0", |    "default": "1", | ||||||
|    "fieldname": "include_exploded_items", |    "fieldname": "include_exploded_items", | ||||||
|    "fieldtype": "Check", |    "fieldtype": "Check", | ||||||
|    "in_list_view": 1, |  | ||||||
|    "label": "Include Exploded Items" |    "label": "Include Exploded Items" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
| @ -80,13 +78,6 @@ | |||||||
|    "fieldname": "column_break_6", |    "fieldname": "column_break_6", | ||||||
|    "fieldtype": "Column Break" |    "fieldtype": "Column Break" | ||||||
|   }, |   }, | ||||||
|   { |  | ||||||
|    "default": "0", |  | ||||||
|    "description": "If enabled, system will create the work order for the exploded items against which BOM is available.", |  | ||||||
|    "fieldname": "make_work_order_for_sub_assembly_items", |  | ||||||
|    "fieldtype": "Check", |  | ||||||
|    "label": "Make Work Order for Sub Assembly Items" |  | ||||||
|   }, |  | ||||||
|   { |   { | ||||||
|    "fieldname": "warehouse", |    "fieldname": "warehouse", | ||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
| @ -218,7 +209,7 @@ | |||||||
|  "idx": 1, |  "idx": 1, | ||||||
|  "istable": 1, |  "istable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-04-28 19:14:57.772123", |  "modified": "2021-06-28 18:31:06.822168", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "Production Plan Item", |  "name": "Production Plan Item", | ||||||
|  | |||||||
| @ -0,0 +1,202 @@ | |||||||
|  | { | ||||||
|  |  "actions": [], | ||||||
|  |  "creation": "2020-12-27 16:08:36.127199", | ||||||
|  |  "doctype": "DocType", | ||||||
|  |  "editable_grid": 1, | ||||||
|  |  "engine": "InnoDB", | ||||||
|  |  "field_order": [ | ||||||
|  |   "production_item", | ||||||
|  |   "item_name", | ||||||
|  |   "fg_warehouse", | ||||||
|  |   "parent_item_code", | ||||||
|  |   "schedule_date", | ||||||
|  |   "column_break_3", | ||||||
|  |   "qty", | ||||||
|  |   "bom_no", | ||||||
|  |   "bom_level", | ||||||
|  |   "type_of_manufacturing", | ||||||
|  |   "supplier", | ||||||
|  |   "work_order_details_section", | ||||||
|  |   "work_order", | ||||||
|  |   "purchase_order", | ||||||
|  |   "production_plan_item", | ||||||
|  |   "column_break_7", | ||||||
|  |   "produced_qty", | ||||||
|  |   "received_qty", | ||||||
|  |   "indent", | ||||||
|  |   "section_break_19", | ||||||
|  |   "uom", | ||||||
|  |   "stock_uom", | ||||||
|  |   "column_break_22", | ||||||
|  |   "description" | ||||||
|  |  ], | ||||||
|  |  "fields": [ | ||||||
|  |   { | ||||||
|  |    "fetch_from": "sub_assembly_item_code.item_name", | ||||||
|  |    "fieldname": "item_name", | ||||||
|  |    "fieldtype": "Data", | ||||||
|  |    "label": "Item Name", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "column_break_3", | ||||||
|  |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "depends_on": "eval:doc.type_of_manufacturing == \"In House\"", | ||||||
|  |    "fieldname": "work_order_details_section", | ||||||
|  |    "fieldtype": "Section Break", | ||||||
|  |    "label": "Reference" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "work_order", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Work Order", | ||||||
|  |    "options": "Work Order", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "column_break_7", | ||||||
|  |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "columns": 1, | ||||||
|  |    "fieldname": "qty", | ||||||
|  |    "fieldtype": "Float", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Required Qty", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "purchase_order", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Purchase Order", | ||||||
|  |    "options": "Purchase Order", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "received_qty", | ||||||
|  |    "fieldtype": "Float", | ||||||
|  |    "label": "Received Qty" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "bom_no", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Bom No", | ||||||
|  |    "options": "BOM" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "production_plan_item", | ||||||
|  |    "fieldtype": "Data", | ||||||
|  |    "hidden": 1, | ||||||
|  |    "label": "Production Plan Item", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "parent_item_code", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Finished Good", | ||||||
|  |    "options": "Item", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "columns": 1, | ||||||
|  |    "fetch_from": "bom_no.bom_level", | ||||||
|  |    "fieldname": "bom_level", | ||||||
|  |    "fieldtype": "Int", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Level (BOM)", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "collapsible": 1, | ||||||
|  |    "fieldname": "section_break_19", | ||||||
|  |    "fieldtype": "Section Break", | ||||||
|  |    "label": "Item Details" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "uom", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "UOM", | ||||||
|  |    "options": "UOM", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "stock_uom", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Stock UOM", | ||||||
|  |    "options": "UOM", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "column_break_22", | ||||||
|  |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "description", | ||||||
|  |    "fieldtype": "Small Text", | ||||||
|  |    "label": "description", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "production_item", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Sub Assembly Item Code", | ||||||
|  |    "options": "Item", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "indent", | ||||||
|  |    "fieldtype": "Int", | ||||||
|  |    "label": "Indent" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "fg_warehouse", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Target Warehouse", | ||||||
|  |    "options": "Warehouse" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "produced_qty", | ||||||
|  |    "fieldtype": "Data", | ||||||
|  |    "label": "Produced Quantity", | ||||||
|  |    "read_only": 1 | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "default": "In House", | ||||||
|  |    "fieldname": "type_of_manufacturing", | ||||||
|  |    "fieldtype": "Select", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Manufacturing Type", | ||||||
|  |    "options": "In House\nSubcontract" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "supplier", | ||||||
|  |    "fieldtype": "Link", | ||||||
|  |    "label": "Supplier", | ||||||
|  |    "mandatory_depends_on": "eval:doc.type_of_manufacturing == 'Subcontract'", | ||||||
|  |    "options": "Supplier" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "schedule_date", | ||||||
|  |    "fieldtype": "Datetime", | ||||||
|  |    "in_list_view": 1, | ||||||
|  |    "label": "Schedule Date" | ||||||
|  |   } | ||||||
|  |  ], | ||||||
|  |  "index_web_pages_for_search": 1, | ||||||
|  |  "istable": 1, | ||||||
|  |  "links": [], | ||||||
|  |  "modified": "2021-06-28 20:10:56.296410", | ||||||
|  |  "modified_by": "Administrator", | ||||||
|  |  "module": "Manufacturing", | ||||||
|  |  "name": "Production Plan Sub Assembly Item", | ||||||
|  |  "owner": "Administrator", | ||||||
|  |  "permissions": [], | ||||||
|  |  "quick_entry": 1, | ||||||
|  |  "sort_field": "modified", | ||||||
|  |  "sort_order": "DESC", | ||||||
|  |  "track_changes": 1 | ||||||
|  | } | ||||||
| @ -0,0 +1,10 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors | ||||||
|  | # For license information, please see license.txt | ||||||
|  | 
 | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | # import frappe | ||||||
|  | from frappe.model.document import Document | ||||||
|  | 
 | ||||||
|  | class ProductionPlanSubAssemblyItem(Document): | ||||||
|  | 	pass | ||||||
| @ -64,11 +64,16 @@ | |||||||
|   "description", |   "description", | ||||||
|   "stock_uom", |   "stock_uom", | ||||||
|   "column_break2", |   "column_break2", | ||||||
|  |   "references_section", | ||||||
|   "material_request", |   "material_request", | ||||||
|   "material_request_item", |   "material_request_item", | ||||||
|   "sales_order_item", |   "sales_order_item", | ||||||
|  |   "column_break_61", | ||||||
|   "production_plan", |   "production_plan", | ||||||
|   "production_plan_item", |   "production_plan_item", | ||||||
|  |   "production_plan_sub_assembly_item", | ||||||
|  |   "parent_work_order", | ||||||
|  |   "bom_level", | ||||||
|   "product_bundle_item", |   "product_bundle_item", | ||||||
|   "amended_from" |   "amended_from" | ||||||
|  ], |  ], | ||||||
| @ -546,17 +551,26 @@ | |||||||
|    "no_copy": 1, |    "no_copy": 1, | ||||||
|    "print_hide": 1, |    "print_hide": 1, | ||||||
|    "read_only": 1 |    "read_only": 1 | ||||||
|   } |   }, | ||||||
|  |   { | ||||||
|  |     "fieldname": "production_plan_sub_assembly_item", | ||||||
|  |     "fieldtype": "Data", | ||||||
|  |     "label": "Production Plan Sub-assembly Item", | ||||||
|  |     "no_copy": 1, | ||||||
|  |     "print_hide": 1, | ||||||
|  |     "read_only": 1 | ||||||
|  |    } | ||||||
|  ], |  ], | ||||||
|  "icon": "fa fa-cogs", |  "icon": "fa fa-cogs", | ||||||
|  "idx": 1, |  "idx": 1, | ||||||
|  "image_field": "image", |  "image_field": "image", | ||||||
|  "is_submittable": 1, |  "is_submittable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-06-20 15:19:14.902699", |  "modified": "2021-06-28 16:19:14.902699", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "Work Order", |  "name": "Work Order", | ||||||
|  |  "nsm_parent_field": "parent_work_order", | ||||||
|  "owner": "Administrator", |  "owner": "Administrator", | ||||||
|  "permissions": [ |  "permissions": [ | ||||||
|   { |   { | ||||||
|  | |||||||
| @ -483,7 +483,7 @@ class WorkOrder(Document): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 		self.set('operations', []) | 		self.set('operations', []) | ||||||
| 		if not self.bom_no: | 		if not self.bom_no or not frappe.get_cached_value('BOM', self.bom_no, 'with_operations'): | ||||||
| 			return | 			return | ||||||
| 
 | 
 | ||||||
| 		operations = [] | 		operations = [] | ||||||
| @ -590,6 +590,7 @@ class WorkOrder(Document): | |||||||
| 	def validate_operation_time(self): | 	def validate_operation_time(self): | ||||||
| 		for d in self.operations: | 		for d in self.operations: | ||||||
| 			if not d.time_in_mins > 0: | 			if not d.time_in_mins > 0: | ||||||
|  | 				print(self.bom_no, self.production_item) | ||||||
| 				frappe.throw(_("Operation Time must be greater than 0 for Operation {0}").format(d.operation)) | 				frappe.throw(_("Operation Time must be greater than 0 for Operation {0}").format(d.operation)) | ||||||
| 
 | 
 | ||||||
| 	def update_required_items(self): | 	def update_required_items(self): | ||||||
|  | |||||||
| @ -20,17 +20,20 @@ def get_exploded_items(bom, data, indent=0, qty=1): | |||||||
| 		fields= ['qty','bom_no','qty','scrap','item_code','item_name','description','uom']) | 		fields= ['qty','bom_no','qty','scrap','item_code','item_name','description','uom']) | ||||||
| 
 | 
 | ||||||
| 	for item in exploded_items: | 	for item in exploded_items: | ||||||
|  | 		print(item.bom_no, indent) | ||||||
| 		item["indent"] = indent | 		item["indent"] = indent | ||||||
| 		data.append({ | 		data.append({ | ||||||
| 			'item_code': item.item_code, | 			'item_code': item.item_code, | ||||||
| 			'item_name': item.item_name, | 			'item_name': item.item_name, | ||||||
| 			'indent': indent, | 			'indent': indent, | ||||||
|  | 			'bom_level': (frappe.get_cached_value("BOM", item.bom_no, "bom_level") | ||||||
|  | 				if item.bom_no else ""), | ||||||
| 			'bom': item.bom_no, | 			'bom': item.bom_no, | ||||||
| 			'qty': item.qty * qty, | 			'qty': item.qty * qty, | ||||||
| 			'uom': item.uom, | 			'uom': item.uom, | ||||||
| 			'description': item.description, | 			'description': item.description, | ||||||
| 			'scrap': item.scrap | 			'scrap': item.scrap | ||||||
| 			}) | 		}) | ||||||
| 		if item.bom_no: | 		if item.bom_no: | ||||||
| 			get_exploded_items(item.bom_no, data, indent=indent+1, qty=item.qty) | 			get_exploded_items(item.bom_no, data, indent=indent+1, qty=item.qty) | ||||||
| 
 | 
 | ||||||
| @ -68,6 +71,12 @@ def get_columns(): | |||||||
| 			"fieldname": "uom", | 			"fieldname": "uom", | ||||||
| 			"width": 100 | 			"width": 100 | ||||||
| 		}, | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "BOM Level", | ||||||
|  | 			"fieldtype": "Data", | ||||||
|  | 			"fieldname": "bom_level", | ||||||
|  | 			"width": 100 | ||||||
|  | 		}, | ||||||
| 		{ | 		{ | ||||||
| 			"label": "Standard Description", | 			"label": "Standard Description", | ||||||
| 			"fieldtype": "data", | 			"fieldtype": "data", | ||||||
|  | |||||||
| @ -70,12 +70,12 @@ def get_bom_stock(filters): | |||||||
| 					ON bom_item.item_code = ledger.item_code | 					ON bom_item.item_code = ledger.item_code | ||||||
| 				{conditions} | 				{conditions} | ||||||
| 			WHERE | 			WHERE | ||||||
| 				bom_item.parent = '{bom}' and bom_item.parenttype='BOM' | 				bom_item.parent = {bom} and bom_item.parenttype='BOM' | ||||||
| 
 | 
 | ||||||
| 			GROUP BY bom_item.item_code""".format( | 			GROUP BY bom_item.item_code""".format( | ||||||
| 				qty_field=qty_field, | 				qty_field=qty_field, | ||||||
| 				table=table, | 				table=table, | ||||||
| 				conditions=conditions, | 				conditions=conditions, | ||||||
| 				bom=bom, | 				bom=frappe.db.escape(bom), | ||||||
| 				qty_to_produce=qty_to_produce or 1) | 				qty_to_produce=qty_to_produce or 1) | ||||||
| 			) | 			) | ||||||
|  | |||||||
| @ -68,6 +68,18 @@ frappe.query_reports["Job Card Summary"] = { | |||||||
| 			get_data: function(txt) { | 			get_data: function(txt) { | ||||||
| 				return frappe.db.get_link_options('Item', txt); | 				return frappe.db.get_link_options('Item', txt); | ||||||
| 			} | 			} | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			label: __("Workstation"), | ||||||
|  | 			fieldname: "workstation", | ||||||
|  | 			fieldtype: "Link", | ||||||
|  | 			options: "Workstation" | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			label: __("Operation"), | ||||||
|  | 			fieldname: "operation", | ||||||
|  | 			fieldtype: "Link", | ||||||
|  | 			options: "Operation" | ||||||
| 		} | 		} | ||||||
| 	] | 	] | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -1,14 +1,16 @@ | |||||||
| { | { | ||||||
|  "add_total_row": 0, |  "add_total_row": 1, | ||||||
|  |  "columns": [], | ||||||
|  "creation": "2020-04-20 12:00:21.436619", |  "creation": "2020-04-20 12:00:21.436619", | ||||||
|  "disable_prepared_report": 0, |  "disable_prepared_report": 0, | ||||||
|  "disabled": 0, |  "disabled": 0, | ||||||
|  "docstatus": 0, |  "docstatus": 0, | ||||||
|  "doctype": "Report", |  "doctype": "Report", | ||||||
|  |  "filters": [], | ||||||
|  "idx": 0, |  "idx": 0, | ||||||
|  "is_standard": "Yes", |  "is_standard": "Yes", | ||||||
|  "letter_head": "Gadgets International", |  "letter_head": "", | ||||||
|  "modified": "2020-04-20 12:00:21.436619", |  "modified": "2020-12-30 11:49:21.713561", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Manufacturing", |  "module": "Manufacturing", | ||||||
|  "name": "Job Card Summary", |  "name": "Job Card Summary", | ||||||
|  | |||||||
| @ -0,0 +1,32 @@ | |||||||
|  | // Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
 | ||||||
|  | // For license information, please see license.txt
 | ||||||
|  | /* eslint-disable */ | ||||||
|  | 
 | ||||||
|  | frappe.query_reports["Production Plan Summary"] = { | ||||||
|  | 	"filters": [ | ||||||
|  | 		{ | ||||||
|  | 			fieldname: "production_plan", | ||||||
|  | 			label: __("Production Plan"), | ||||||
|  | 			fieldtype: "Link", | ||||||
|  | 			options: "Production Plan", | ||||||
|  | 			reqd: 1, | ||||||
|  | 			get_query: function() { | ||||||
|  | 				return { | ||||||
|  | 					filters: { | ||||||
|  | 						"docstatus": 1 | ||||||
|  | 					} | ||||||
|  | 				}; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	], | ||||||
|  | 	"formatter": function(value, row, column, data, default_formatter) { | ||||||
|  | 		value = default_formatter(value, row, column, data); | ||||||
|  | 
 | ||||||
|  | 		if (column.fieldname == "document_name") { | ||||||
|  | 			var color = data.pending_qty > 0 ? 'red': 'green'; | ||||||
|  | 			value = `<a style='color:${color}' href="#Form/${data['document_type']}/${data['document_name']}" data-doctype="${data['document_type']}">${data['document_name']}</a>`; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		return value; | ||||||
|  | 	}, | ||||||
|  | }; | ||||||
| @ -0,0 +1,26 @@ | |||||||
|  | { | ||||||
|  |  "add_total_row": 0, | ||||||
|  |  "columns": [], | ||||||
|  |  "creation": "2020-12-27 11:43:39.781793", | ||||||
|  |  "disable_prepared_report": 0, | ||||||
|  |  "disabled": 0, | ||||||
|  |  "docstatus": 0, | ||||||
|  |  "doctype": "Report", | ||||||
|  |  "filters": [], | ||||||
|  |  "idx": 0, | ||||||
|  |  "is_standard": "Yes", | ||||||
|  |  "modified": "2020-12-27 11:43:42.677584", | ||||||
|  |  "modified_by": "Administrator", | ||||||
|  |  "module": "Manufacturing", | ||||||
|  |  "name": "Production Plan Summary", | ||||||
|  |  "owner": "Administrator", | ||||||
|  |  "prepared_report": 0, | ||||||
|  |  "ref_doctype": "Production Plan", | ||||||
|  |  "report_name": "Production Plan Summary", | ||||||
|  |  "report_type": "Script Report", | ||||||
|  |  "roles": [ | ||||||
|  |   { | ||||||
|  |    "role": "Manufacturing User" | ||||||
|  |   } | ||||||
|  |  ] | ||||||
|  | } | ||||||
| @ -0,0 +1,136 @@ | |||||||
|  | # Copyright (c) 2013, Frappe Technologies Pvt. Ltd. and contributors | ||||||
|  | # For license information, please see license.txt | ||||||
|  | 
 | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | import frappe | ||||||
|  | from frappe.utils import flt | ||||||
|  | 
 | ||||||
|  | def execute(filters=None): | ||||||
|  | 	columns, data = [], [] | ||||||
|  | 	data = get_data(filters) | ||||||
|  | 	columns = get_column(filters) | ||||||
|  | 
 | ||||||
|  | 	return columns, data | ||||||
|  | 
 | ||||||
|  | def get_data(filters): | ||||||
|  | 	data = [] | ||||||
|  | 
 | ||||||
|  | 	order_details = {} | ||||||
|  | 	get_work_order_details(filters, order_details) | ||||||
|  | 	get_purchase_order_details(filters, order_details) | ||||||
|  | 	get_production_plan_item_details(filters, data, order_details) | ||||||
|  | 
 | ||||||
|  | 	return data | ||||||
|  | 
 | ||||||
|  | def get_production_plan_item_details(filters, data, order_details): | ||||||
|  | 	itemwise_indent = {} | ||||||
|  | 
 | ||||||
|  | 	production_plan_doc = frappe.get_cached_doc("Production Plan", filters.get("production_plan")) | ||||||
|  | 	for row in production_plan_doc.po_items: | ||||||
|  | 		work_order = frappe.get_cached_value("Work Order", {"production_plan_item": row.name, | ||||||
|  | 			"bom_no": row.bom_no, "production_item": row.item_code}, "name") | ||||||
|  | 
 | ||||||
|  | 		if row.item_code not in itemwise_indent: | ||||||
|  | 			itemwise_indent.setdefault(row.item_code, {}) | ||||||
|  | 
 | ||||||
|  | 		data.append({ | ||||||
|  | 			"indent": 0, | ||||||
|  | 			"item_code": row.item_code, | ||||||
|  | 			"item_name": frappe.get_cached_value("Item", row.item_code, "item_name"), | ||||||
|  | 			"qty": row.planned_qty, | ||||||
|  | 			"document_type": "Work Order", | ||||||
|  | 			"document_name": work_order, | ||||||
|  | 			"bom_level": frappe.get_cached_value("BOM", row.bom_no, "bom_level"), | ||||||
|  | 			"produced_qty": order_details.get((work_order, row.item_code)).get("produced_qty"), | ||||||
|  | 			"pending_qty": flt(row.planned_qty) - flt(order_details.get((work_order, row.item_code)).get("produced_qty")) | ||||||
|  | 		}) | ||||||
|  | 
 | ||||||
|  | 		get_production_plan_sub_assembly_item_details(filters, row, production_plan_doc, data, order_details) | ||||||
|  | 
 | ||||||
|  | def get_production_plan_sub_assembly_item_details(filters, row, production_plan_doc, data, order_details): | ||||||
|  | 	for item in production_plan_doc.sub_assembly_items: | ||||||
|  | 		if row.name == item.production_plan_item: | ||||||
|  | 			subcontracted_item = (item.type_of_manufacturing == 'Subcontract') | ||||||
|  | 
 | ||||||
|  | 			if subcontracted_item: | ||||||
|  | 				docname = frappe.get_cached_value("Purchase Order Item", | ||||||
|  | 					{"production_plan_sub_assembly_item": item.name, "docstatus": ("<", 2)}, "parent") | ||||||
|  | 			else: | ||||||
|  | 				docname = frappe.get_cached_value("Work Order", | ||||||
|  | 					{"production_plan_sub_assembly_item": item.name, "docstatus": ("<", 2)}, "name") | ||||||
|  | 
 | ||||||
|  | 			data.append({ | ||||||
|  | 				"indent": 1, | ||||||
|  | 				"item_code": item.production_item, | ||||||
|  | 				"item_name": item.item_name, | ||||||
|  | 				"qty": item.qty, | ||||||
|  | 				"document_type": "Work Order" if not subcontracted_item else "Purchase Order", | ||||||
|  | 				"document_name": docname, | ||||||
|  | 				"bom_level": item.bom_level, | ||||||
|  | 				"produced_qty": order_details.get((docname, item.production_item)).get("produced_qty"), | ||||||
|  | 				"pending_qty": flt(item.qty) - flt(order_details.get((docname, item.production_item)).get("produced_qty")) | ||||||
|  | 			}) | ||||||
|  | 
 | ||||||
|  | def get_work_order_details(filters, order_details): | ||||||
|  | 	for row in frappe.get_all("Work Order", filters = {"production_plan": filters.get("production_plan")}, | ||||||
|  | 		fields=["name", "produced_qty", "production_plan", "production_item"]): | ||||||
|  | 		order_details.setdefault((row.name, row.production_item), row) | ||||||
|  | 
 | ||||||
|  | def get_purchase_order_details(filters, order_details): | ||||||
|  | 	for row in frappe.get_all("Purchase Order Item", filters = {"production_plan": filters.get("production_plan")}, | ||||||
|  | 		fields=["parent", "received_qty as produced_qty", "item_code"]): | ||||||
|  | 		order_details.setdefault((row.parent, row.item_code), row) | ||||||
|  | 
 | ||||||
|  | def get_column(filters): | ||||||
|  | 	return [ | ||||||
|  | 		{ | ||||||
|  | 			"label": "Finished Good", | ||||||
|  | 			"fieldtype": "Link", | ||||||
|  | 			"fieldname": "item_code", | ||||||
|  | 			"width": 300, | ||||||
|  | 			"options": "Item" | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Item Name", | ||||||
|  | 			"fieldtype": "data", | ||||||
|  | 			"fieldname": "item_name", | ||||||
|  | 			"width": 100 | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Document Type", | ||||||
|  | 			"fieldtype": "Link", | ||||||
|  | 			"fieldname": "document_type", | ||||||
|  | 			"width": 150, | ||||||
|  | 			"options": "DocType" | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Document Name", | ||||||
|  | 			"fieldtype": "Dynamic Link", | ||||||
|  | 			"fieldname": "document_name", | ||||||
|  | 			"width": 150 | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "BOM Level", | ||||||
|  | 			"fieldtype": "Int", | ||||||
|  | 			"fieldname": "bom_level", | ||||||
|  | 			"width": 100 | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Order Qty", | ||||||
|  | 			"fieldtype": "Float", | ||||||
|  | 			"fieldname": "qty", | ||||||
|  | 			"width": 120 | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Received Qty", | ||||||
|  | 			"fieldtype": "Float", | ||||||
|  | 			"fieldname": "produced_qty", | ||||||
|  | 			"width": 160 | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			"label": "Pending Qty", | ||||||
|  | 			"fieldtype": "Float", | ||||||
|  | 			"fieldname": "pending_qty", | ||||||
|  | 			"width": 110 | ||||||
|  | 		} | ||||||
|  | 	] | ||||||
| @ -19,7 +19,7 @@ def execute(filters=None): | |||||||
| 	return columns, data, None, chart_data | 	return columns, data, None, chart_data | ||||||
| 
 | 
 | ||||||
| def get_data(filters): | def get_data(filters): | ||||||
| 	query_filters = {"docstatus": 1} | 	query_filters = {"docstatus": ("<", 2)} | ||||||
| 
 | 
 | ||||||
| 	fields = ["name", "status", "sales_order", "production_item", "qty", "produced_qty", | 	fields = ["name", "status", "sales_order", "production_item", "qty", "produced_qty", | ||||||
| 		"planned_start_date", "planned_end_date", "actual_start_date", "actual_end_date", "lead_time"] | 		"planned_start_date", "planned_end_date", "actual_start_date", "actual_end_date", "lead_time"] | ||||||
| @ -62,7 +62,8 @@ def get_chart_based_on_status(data): | |||||||
| 		"Not Started": 0, | 		"Not Started": 0, | ||||||
| 		"In Process": 0, | 		"In Process": 0, | ||||||
| 		"Stopped": 0, | 		"Stopped": 0, | ||||||
| 		"Completed": 0 | 		"Completed": 0, | ||||||
|  | 		"Draft": 0 | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	for d in data: | 	for d in data: | ||||||
|  | |||||||
| @ -290,3 +290,4 @@ erpnext.patches.v13_0.set_training_event_attendance | |||||||
| erpnext.patches.v13_0.rename_issue_status_hold_to_on_hold | erpnext.patches.v13_0.rename_issue_status_hold_to_on_hold | ||||||
| erpnext.patches.v13_0.bill_for_rejected_quantity_in_purchase_invoice | erpnext.patches.v13_0.bill_for_rejected_quantity_in_purchase_invoice | ||||||
| erpnext.patches.v13_0.update_job_card_details | erpnext.patches.v13_0.update_job_card_details | ||||||
|  | erpnext.patches.v13_0.update_level_in_bom #1234sswef | ||||||
|  | |||||||
							
								
								
									
										30
									
								
								erpnext/patches/v13_0/update_level_in_bom.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								erpnext/patches/v13_0/update_level_in_bom.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | |||||||
|  | # Copyright (c) 2020, Frappe and Contributors | ||||||
|  | # License: GNU General Public License v3. See license.txt | ||||||
|  | 
 | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | import frappe | ||||||
|  | 
 | ||||||
|  | def execute(): | ||||||
|  | 	for document in ["bom", "bom_item", "bom_explosion_item"]: | ||||||
|  | 		frappe.reload_doc('manufacturing', 'doctype', document) | ||||||
|  | 
 | ||||||
|  | 	frappe.db.sql(" update `tabBOM` set bom_level = 0 where docstatus = 1") | ||||||
|  | 
 | ||||||
|  | 	bom_list = frappe.db.sql_list("""select name from `tabBOM` bom | ||||||
|  | 		where docstatus=1 and is_active=1 and not exists(select bom_no from `tabBOM Item` | ||||||
|  | 		where parent=bom.name and ifnull(bom_no, '')!='')""") | ||||||
|  | 
 | ||||||
|  | 	count = 0 | ||||||
|  | 	while(count < len(bom_list)): | ||||||
|  | 		for parent_bom in get_parent_boms(bom_list[count]): | ||||||
|  | 			bom_doc = frappe.get_cached_doc("BOM", parent_bom) | ||||||
|  | 			bom_doc.set_bom_level(update=True) | ||||||
|  | 			bom_list.append(parent_bom) | ||||||
|  | 		count += 1 | ||||||
|  | 
 | ||||||
|  | def get_parent_boms(bom_no): | ||||||
|  | 	return frappe.db.sql_list(""" | ||||||
|  | 		select distinct bom_item.parent from `tabBOM Item` bom_item | ||||||
|  | 		where bom_item.bom_no = %s and bom_item.docstatus=1 and bom_item.parenttype='BOM' | ||||||
|  | 			and exists(select bom.name from `tabBOM` bom where bom.name=bom_item.parent and bom.is_active=1) | ||||||
|  | 	""", bom_no) | ||||||
| @ -117,7 +117,6 @@ class PayrollEntry(Document): | |||||||
| 			Creates salary slip for selected employees if already not created | 			Creates salary slip for selected employees if already not created | ||||||
| 		""" | 		""" | ||||||
| 		self.check_permission('write') | 		self.check_permission('write') | ||||||
| 		self.created = 1 |  | ||||||
| 		employees = [emp.employee for emp in self.employees] | 		employees = [emp.employee for emp in self.employees] | ||||||
| 		if employees: | 		if employees: | ||||||
| 			args = frappe._dict({ | 			args = frappe._dict({ | ||||||
| @ -686,7 +685,7 @@ def employee_query(doctype, txt, searchfield, start, page_len, filters): | |||||||
| 
 | 
 | ||||||
| 	if filters.start_date and filters.end_date: | 	if filters.start_date and filters.end_date: | ||||||
| 		employee_list = get_employee_list(filters) | 		employee_list = get_employee_list(filters) | ||||||
| 		emp = filters.get('employees') | 		emp = filters.get('employees') or [] | ||||||
| 		include_employees = [employee.employee for employee in employee_list if employee.employee not in emp] | 		include_employees = [employee.employee for employee in employee_list if employee.employee not in emp] | ||||||
| 		filters.pop('start_date') | 		filters.pop('start_date') | ||||||
| 		filters.pop('end_date') | 		filters.pop('end_date') | ||||||
|  | |||||||
| @ -147,7 +147,7 @@ erpnext.setup.slides_settings = [ | |||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			// Validate bank name
 | 			// Validate bank name
 | ||||||
| 			if(me.values.bank_account){ | 			if(me.values.bank_account) {  | ||||||
| 				frappe.call({ | 				frappe.call({ | ||||||
| 					async: false, | 					async: false, | ||||||
| 					method: "erpnext.accounts.doctype.account.chart_of_accounts.chart_of_accounts.validate_bank_account", | 					method: "erpnext.accounts.doctype.account.chart_of_accounts.chart_of_accounts.validate_bank_account", | ||||||
|  | |||||||
| @ -35,6 +35,7 @@ frappe.ui.form.on('GST Settings', { | |||||||
| 			return { | 			return { | ||||||
| 				filters: { | 				filters: { | ||||||
| 					company: row.company, | 					company: row.company, | ||||||
|  | 					account_type: "Tax", | ||||||
| 					is_group: 0 | 					is_group: 0 | ||||||
| 				} | 				} | ||||||
| 			}; | 			}; | ||||||
|  | |||||||
| @ -19,6 +19,21 @@ class GSTSettings(Document): | |||||||
| 			from tabAddress where country = "India" and ifnull(gstin, '')!='' ''') | 			from tabAddress where country = "India" and ifnull(gstin, '')!='' ''') | ||||||
| 		self.set_onload('data', data) | 		self.set_onload('data', data) | ||||||
| 
 | 
 | ||||||
|  | 	def validate(self): | ||||||
|  | 		# Validate duplicate accounts | ||||||
|  | 		self.validate_duplicate_accounts() | ||||||
|  | 
 | ||||||
|  | 	def validate_duplicate_accounts(self): | ||||||
|  | 		account_list = [] | ||||||
|  | 		for account in self.get('gst_accounts'): | ||||||
|  | 			for fieldname in ['cgst_account', 'sgst_account', 'igst_account', 'cess_account']: | ||||||
|  | 				if account.get(fieldname) in account_list: | ||||||
|  | 					frappe.throw(_("Account {0} appears multiple times").format( | ||||||
|  | 						frappe.bold(account.get(fieldname)))) | ||||||
|  | 
 | ||||||
|  | 				if account.get(fieldname): | ||||||
|  | 					account_list.append(account.get(fieldname)) | ||||||
|  | 
 | ||||||
| @frappe.whitelist() | @frappe.whitelist() | ||||||
| def send_reminder(): | def send_reminder(): | ||||||
| 	frappe.has_permission('GST Settings', throw=True) | 	frappe.has_permission('GST Settings', throw=True) | ||||||
|  | |||||||
| @ -46,14 +46,14 @@ class TestGSTR3BReport(unittest.TestCase): | |||||||
| 		make_sales_invoice() | 		make_sales_invoice() | ||||||
| 		create_purchase_invoices() | 		create_purchase_invoices() | ||||||
| 
 | 
 | ||||||
| 		if frappe.db.exists("GSTR 3B Report", "GSTR3B-March-2019-_Test Address-Billing"): | 		if frappe.db.exists("GSTR 3B Report", "GSTR3B-March-2019-_Test Address GST-Billing"): | ||||||
| 			report = frappe.get_doc("GSTR 3B Report", "GSTR3B-March-2019-_Test Address-Billing") | 			report = frappe.get_doc("GSTR 3B Report", "GSTR3B-March-2019-_Test Address GST-Billing") | ||||||
| 			report.save() | 			report.save() | ||||||
| 		else: | 		else: | ||||||
| 			report = frappe.get_doc({ | 			report = frappe.get_doc({ | ||||||
| 				"doctype": "GSTR 3B Report", | 				"doctype": "GSTR 3B Report", | ||||||
| 				"company": "_Test Company GST", | 				"company": "_Test Company GST", | ||||||
| 				"company_address": "_Test Address-Billing", | 				"company_address": "_Test Address GST-Billing", | ||||||
| 				"year": getdate().year, | 				"year": getdate().year, | ||||||
| 				"month": month_number_mapping.get(getdate().month) | 				"month": month_number_mapping.get(getdate().month) | ||||||
| 			}).insert() | 			}).insert() | ||||||
| @ -89,7 +89,7 @@ class TestGSTR3BReport(unittest.TestCase): | |||||||
| 
 | 
 | ||||||
| 		si.append("taxes", { | 		si.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "IGST - _GST", | 			"account_head": "Output Tax IGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "IGST @ 18.0", | 			"description": "IGST @ 18.0", | ||||||
| 			"rate": 18 | 			"rate": 18 | ||||||
| @ -117,7 +117,7 @@ def make_sales_invoice(): | |||||||
| 
 | 
 | ||||||
| 	si.append("taxes", { | 	si.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "IGST - _GST", | 			"account_head": "Output Tax IGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "IGST @ 18.0", | 			"description": "IGST @ 18.0", | ||||||
| 			"rate": 18 | 			"rate": 18 | ||||||
| @ -138,7 +138,7 @@ def make_sales_invoice(): | |||||||
| 
 | 
 | ||||||
| 	si1.append("taxes", { | 	si1.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "IGST - _GST", | 			"account_head": "Output Tax IGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "IGST @ 18.0", | 			"description": "IGST @ 18.0", | ||||||
| 			"rate": 18 | 			"rate": 18 | ||||||
| @ -159,7 +159,7 @@ def make_sales_invoice(): | |||||||
| 
 | 
 | ||||||
| 	si2.append("taxes", { | 	si2.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "IGST - _GST", | 			"account_head": "Output Tax IGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "IGST @ 18.0", | 			"description": "IGST @ 18.0", | ||||||
| 			"rate": 18 | 			"rate": 18 | ||||||
| @ -195,7 +195,7 @@ def create_purchase_invoices(): | |||||||
| 
 | 
 | ||||||
| 	pi.append("taxes", { | 	pi.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "CGST - _GST", | 			"account_head": "Input Tax CGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "CGST @ 9.0", | 			"description": "CGST @ 9.0", | ||||||
| 			"rate": 9 | 			"rate": 9 | ||||||
| @ -203,7 +203,7 @@ def create_purchase_invoices(): | |||||||
| 
 | 
 | ||||||
| 	pi.append("taxes", { | 	pi.append("taxes", { | ||||||
| 			"charge_type": "On Net Total", | 			"charge_type": "On Net Total", | ||||||
| 			"account_head": "SGST - _GST", | 			"account_head": "Input Tax SGST - _GST", | ||||||
| 			"cost_center": "Main - _GST", | 			"cost_center": "Main - _GST", | ||||||
| 			"description": "SGST @ 9.0", | 			"description": "SGST @ 9.0", | ||||||
| 			"rate": 9 | 			"rate": 9 | ||||||
| @ -410,10 +410,10 @@ def make_company(): | |||||||
| 	company.country = "India" | 	company.country = "India" | ||||||
| 	company.insert() | 	company.insert() | ||||||
| 
 | 
 | ||||||
| 	if not frappe.db.exists('Address', '_Test Address-Billing'): | 	if not frappe.db.exists('Address', '_Test Address GST-Billing'): | ||||||
| 		address = frappe.get_doc({ | 		address = frappe.get_doc({ | ||||||
|  | 			"address_title": "_Test Address GST", | ||||||
| 			"address_line1": "_Test Address Line 1", | 			"address_line1": "_Test Address Line 1", | ||||||
| 			"address_title": "_Test Address", |  | ||||||
| 			"address_type": "Billing", | 			"address_type": "Billing", | ||||||
| 			"city": "_Test City", | 			"city": "_Test City", | ||||||
| 			"state": "Test State", | 			"state": "Test State", | ||||||
| @ -444,9 +444,9 @@ def set_account_heads(): | |||||||
| 	if not gst_account: | 	if not gst_account: | ||||||
| 		gst_settings.append("gst_accounts", { | 		gst_settings.append("gst_accounts", { | ||||||
| 			"company": "_Test Company GST", | 			"company": "_Test Company GST", | ||||||
| 			"cgst_account": "CGST - _GST", | 			"cgst_account": "Output Tax CGST - _GST", | ||||||
| 			"sgst_account": "SGST - _GST", | 			"sgst_account": "Output Tax SGST - _GST", | ||||||
| 			"igst_account": "IGST - _GST", | 			"igst_account": "Output Tax IGST - _GST" | ||||||
| 		}) | 		}) | ||||||
| 
 | 
 | ||||||
| 		gst_settings.save() | 		gst_settings.save() | ||||||
|  | |||||||
| @ -1,6 +1,8 @@ | |||||||
| erpnext.setup_einvoice_actions = (doctype) => { | erpnext.setup_einvoice_actions = (doctype) => { | ||||||
| 	frappe.ui.form.on(doctype, { | 	frappe.ui.form.on(doctype, { | ||||||
| 		async refresh(frm) { | 		async refresh(frm) { | ||||||
|  | 			if (frm.doc.docstatus == 2) return; | ||||||
|  | 
 | ||||||
| 			const res = await frappe.call({ | 			const res = await frappe.call({ | ||||||
| 				method: 'erpnext.regional.india.e_invoice.utils.validate_eligibility', | 				method: 'erpnext.regional.india.e_invoice.utils.validate_eligibility', | ||||||
| 				args: { doc: frm.doc } | 				args: { doc: frm.doc } | ||||||
| @ -111,7 +113,7 @@ erpnext.setup_einvoice_actions = (doctype) => { | |||||||
| 
 | 
 | ||||||
| 			if (irn && ewaybill && !irn_cancelled && !eway_bill_cancelled) { | 			if (irn && ewaybill && !irn_cancelled && !eway_bill_cancelled) { | ||||||
| 				const action = () => { | 				const action = () => { | ||||||
| 					let message = __('Cancellation of e-way bill is currently not supported. '); | 					let message = __('Cancellation of e-way bill is currently not supported.') + ' '; | ||||||
| 					message += '<br><br>'; | 					message += '<br><br>'; | ||||||
| 					message += __('You must first use the portal to cancel the e-way bill and then update the cancelled status in the ERPNext system.'); | 					message += __('You must first use the portal to cancel the e-way bill and then update the cancelled status in the ERPNext system.'); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -42,7 +42,10 @@ def validate_eligibility(doc): | |||||||
| 	invalid_company = not frappe.db.get_value('E Invoice User', { 'company': doc.get('company') }) | 	invalid_company = not frappe.db.get_value('E Invoice User', { 'company': doc.get('company') }) | ||||||
| 	invalid_supply_type = doc.get('gst_category') not in ['Registered Regular', 'SEZ', 'Overseas', 'Deemed Export'] | 	invalid_supply_type = doc.get('gst_category') not in ['Registered Regular', 'SEZ', 'Overseas', 'Deemed Export'] | ||||||
| 	company_transaction = doc.get('billing_address_gstin') == doc.get('company_gstin') | 	company_transaction = doc.get('billing_address_gstin') == doc.get('company_gstin') | ||||||
| 	no_taxes_applied = not doc.get('taxes') | 
 | ||||||
|  | 	# if export invoice, then taxes can be empty | ||||||
|  | 	# invoice can only be ineligible if no taxes applied and is not an export invoice | ||||||
|  | 	no_taxes_applied = not doc.get('taxes') and not doc.get('gst_category') == 'Overseas' | ||||||
| 	has_non_gst_item = any(d for d in doc.get('items', []) if d.get('is_non_gst')) | 	has_non_gst_item = any(d for d in doc.get('items', []) if d.get('is_non_gst')) | ||||||
| 
 | 
 | ||||||
| 	if invalid_company or invalid_supply_type or company_transaction or no_taxes_applied or has_non_gst_item: | 	if invalid_company or invalid_supply_type or company_transaction or no_taxes_applied or has_non_gst_item: | ||||||
| @ -188,9 +191,10 @@ def get_item_list(invoice): | |||||||
| 
 | 
 | ||||||
| 		item.qty = abs(item.qty) | 		item.qty = abs(item.qty) | ||||||
| 
 | 
 | ||||||
| 		item.unit_rate = abs((abs(item.taxable_value) - item.discount_amount)/ item.qty) | 		item.unit_rate = abs(item.taxable_value / item.qty) | ||||||
| 		item.gross_amount = abs(item.taxable_value) + item.discount_amount | 		item.gross_amount = abs(item.taxable_value) | ||||||
| 		item.taxable_value = abs(item.taxable_value) | 		item.taxable_value = abs(item.taxable_value) | ||||||
|  | 		item.discount_amount = 0 | ||||||
| 
 | 
 | ||||||
| 		item.batch_expiry_date = frappe.db.get_value('Batch', d.batch_no, 'expiry_date') if d.batch_no else None | 		item.batch_expiry_date = frappe.db.get_value('Batch', d.batch_no, 'expiry_date') if d.batch_no else None | ||||||
| 		item.batch_expiry_date = format_date(item.batch_expiry_date, 'dd/mm/yyyy') if item.batch_expiry_date else None | 		item.batch_expiry_date = format_date(item.batch_expiry_date, 'dd/mm/yyyy') if item.batch_expiry_date else None | ||||||
|  | |||||||
| @ -25,6 +25,7 @@ def setup_company_independent_fixtures(patch=False): | |||||||
| 	frappe.enqueue('erpnext.regional.india.setup.add_hsn_sac_codes', now=frappe.flags.in_test) | 	frappe.enqueue('erpnext.regional.india.setup.add_hsn_sac_codes', now=frappe.flags.in_test) | ||||||
| 	create_gratuity_rule() | 	create_gratuity_rule() | ||||||
| 	add_print_formats() | 	add_print_formats() | ||||||
|  | 	update_accounts_settings_for_taxes() | ||||||
| 
 | 
 | ||||||
| def add_hsn_sac_codes(): | def add_hsn_sac_codes(): | ||||||
| 	if frappe.flags.in_test and frappe.flags.created_hsn_codes: | 	if frappe.flags.in_test and frappe.flags.created_hsn_codes: | ||||||
| @ -680,7 +681,7 @@ def make_custom_fields(update=True): | |||||||
| 
 | 
 | ||||||
| def make_fixtures(company=None): | def make_fixtures(company=None): | ||||||
| 	docs = [] | 	docs = [] | ||||||
| 	company = company.name if company else frappe.db.get_value("Global Defaults", None, "default_company") | 	company = company or frappe.db.get_value("Global Defaults", None, "default_company") | ||||||
| 
 | 
 | ||||||
| 	set_salary_components(docs) | 	set_salary_components(docs) | ||||||
| 	set_tds_account(docs, company) | 	set_tds_account(docs, company) | ||||||
| @ -698,6 +699,53 @@ def make_fixtures(company=None): | |||||||
| 	# create records for Tax Withholding Category | 	# create records for Tax Withholding Category | ||||||
| 	set_tax_withholding_category(company) | 	set_tax_withholding_category(company) | ||||||
| 
 | 
 | ||||||
|  | def update_regional_tax_settings(country, company): | ||||||
|  | 	# Will only add default GST accounts if present | ||||||
|  | 	input_account_names = ['Input Tax CGST', 'Input Tax SGST', 'Input Tax IGST'] | ||||||
|  | 	output_account_names = ['Output Tax CGST', 'Output Tax SGST', 'Output Tax IGST'] | ||||||
|  | 	rcm_accounts = ['Input Tax CGST RCM', 'Input Tax SGST RCM', 'Input Tax IGST RCM'] | ||||||
|  | 	gst_settings = frappe.get_single('GST Settings') | ||||||
|  | 	existing_account_list = [] | ||||||
|  | 
 | ||||||
|  | 	for account in gst_settings.get('gst_accounts'): | ||||||
|  | 		for key in ['cgst_account', 'sgst_account', 'igst_account']: | ||||||
|  | 			existing_account_list.append(account.get(key)) | ||||||
|  | 
 | ||||||
|  | 	gst_accounts = frappe._dict(frappe.get_all("Account", | ||||||
|  | 		{'company': company, 'account_name': ('in', input_account_names + | ||||||
|  | 			output_account_names + rcm_accounts)}, ['account_name', 'name'], as_list=1)) | ||||||
|  | 
 | ||||||
|  | 	add_accounts_in_gst_settings(company,  input_account_names, gst_accounts, | ||||||
|  | 		existing_account_list, gst_settings) | ||||||
|  | 	add_accounts_in_gst_settings(company, output_account_names, gst_accounts, | ||||||
|  | 		existing_account_list, gst_settings) | ||||||
|  | 	add_accounts_in_gst_settings(company, rcm_accounts, gst_accounts, | ||||||
|  | 		existing_account_list, gst_settings, is_reverse_charge=1) | ||||||
|  | 
 | ||||||
|  | 	gst_settings.save() | ||||||
|  | 
 | ||||||
|  | def add_accounts_in_gst_settings(company, account_names, gst_accounts, | ||||||
|  | 	existing_account_list, gst_settings, is_reverse_charge=0): | ||||||
|  | 	accounts_not_added = 1 | ||||||
|  | 
 | ||||||
|  | 	for account in account_names: | ||||||
|  | 		# Default Account Added does not exists | ||||||
|  | 		if not gst_accounts.get(account): | ||||||
|  | 			accounts_not_added = 0 | ||||||
|  | 
 | ||||||
|  | 		# Check if already added in GST Settings | ||||||
|  | 		if gst_accounts.get(account) in existing_account_list: | ||||||
|  | 			accounts_not_added = 0 | ||||||
|  | 
 | ||||||
|  | 	if accounts_not_added: | ||||||
|  | 		gst_settings.append('gst_accounts', { | ||||||
|  | 			'company': company, | ||||||
|  | 			'cgst_account': gst_accounts.get(account_names[0]), | ||||||
|  | 			'sgst_account': gst_accounts.get(account_names[1]), | ||||||
|  | 			'igst_account': gst_accounts.get(account_names[2]), | ||||||
|  | 			'is_reverse_charge_account': is_reverse_charge | ||||||
|  | 		}) | ||||||
|  | 
 | ||||||
| def set_salary_components(docs): | def set_salary_components(docs): | ||||||
| 	docs.extend([ | 	docs.extend([ | ||||||
| 		{'doctype': 'Salary Component', 'salary_component': 'Professional Tax', | 		{'doctype': 'Salary Component', 'salary_component': 'Professional Tax', | ||||||
| @ -731,12 +779,13 @@ def set_tax_withholding_category(company): | |||||||
| 	docs = get_tds_details(accounts, fiscal_year) | 	docs = get_tds_details(accounts, fiscal_year) | ||||||
| 
 | 
 | ||||||
| 	for d in docs: | 	for d in docs: | ||||||
| 		try: | 		if not frappe.db.exists("Tax Withholding Category", d.get("name")): | ||||||
| 			doc = frappe.get_doc(d) | 			doc = frappe.get_doc(d) | ||||||
|  | 			doc.flags.ignore_validate = True | ||||||
| 			doc.flags.ignore_permissions = True | 			doc.flags.ignore_permissions = True | ||||||
| 			doc.flags.ignore_mandatory = True | 			doc.flags.ignore_mandatory = True | ||||||
| 			doc.insert() | 			doc.insert() | ||||||
| 		except frappe.DuplicateEntryError: | 		else: | ||||||
| 			doc = frappe.get_doc("Tax Withholding Category", d.get("name")) | 			doc = frappe.get_doc("Tax Withholding Category", d.get("name")) | ||||||
| 
 | 
 | ||||||
| 			if accounts: | 			if accounts: | ||||||
| @ -749,11 +798,12 @@ def set_tax_withholding_category(company): | |||||||
| 					doc.append("rates", d.get('rates')[0]) | 					doc.append("rates", d.get('rates')[0]) | ||||||
| 
 | 
 | ||||||
| 			doc.flags.ignore_permissions = True | 			doc.flags.ignore_permissions = True | ||||||
|  | 			doc.flags.ignore_validate = True | ||||||
| 			doc.flags.ignore_mandatory = True | 			doc.flags.ignore_mandatory = True | ||||||
|  | 			doc.flags.ignore_links = True | ||||||
| 			doc.save() | 			doc.save() | ||||||
| 
 | 
 | ||||||
| def set_tds_account(docs, company): | def set_tds_account(docs, company): | ||||||
| 	abbr = frappe.get_value("Company", company, "abbr") |  | ||||||
| 	parent_account = frappe.db.get_value("Account", filters = {"account_name": "Duties and Taxes", "company": company}) | 	parent_account = frappe.db.get_value("Account", filters = {"account_name": "Duties and Taxes", "company": company}) | ||||||
| 	if parent_account: | 	if parent_account: | ||||||
| 		docs.extend([ | 		docs.extend([ | ||||||
| @ -912,7 +962,6 @@ def get_tds_details(accounts, fiscal_year): | |||||||
| 	] | 	] | ||||||
| 
 | 
 | ||||||
| def create_gratuity_rule(): | def create_gratuity_rule(): | ||||||
| 
 |  | ||||||
| 	# Standard Indain Gratuity Rule | 	# Standard Indain Gratuity Rule | ||||||
| 	if not frappe.db.exists("Gratuity Rule", "Indian Standard Gratuity Rule"): | 	if not frappe.db.exists("Gratuity Rule", "Indian Standard Gratuity Rule"): | ||||||
| 		rule = frappe.new_doc("Gratuity Rule") | 		rule = frappe.new_doc("Gratuity Rule") | ||||||
| @ -930,3 +979,7 @@ def create_gratuity_rule(): | |||||||
| 
 | 
 | ||||||
| 		rule.flags.ignore_mandatory = True | 		rule.flags.ignore_mandatory = True | ||||||
| 		rule.save() | 		rule.save() | ||||||
|  | 
 | ||||||
|  | def update_accounts_settings_for_taxes(): | ||||||
|  | 	if frappe.db.count('Company') == 1: | ||||||
|  | 		frappe.db.set_value('Accounts Settings', None, "add_taxes_from_item_tax_template", 0) | ||||||
| @ -11,7 +11,7 @@ | |||||||
|  "is_standard": "Yes", |  "is_standard": "Yes", | ||||||
|  "json": "{}", |  "json": "{}", | ||||||
|  "letter_head": "Logo", |  "letter_head": "Logo", | ||||||
|  "modified": "2021-03-12 12:36:48.689413", |  "modified": "2021-03-13 12:36:48.689413", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Regional", |  "module": "Regional", | ||||||
|  "name": "E-Invoice Summary", |  "name": "E-Invoice Summary", | ||||||
|  | |||||||
| @ -472,12 +472,7 @@ erpnext.PointOfSale.ItemCart = class { | |||||||
| 		const grand_total = cint(frappe.sys_defaults.disable_rounded_total) ? frm.doc.grand_total : frm.doc.rounded_total; | 		const grand_total = cint(frappe.sys_defaults.disable_rounded_total) ? frm.doc.grand_total : frm.doc.rounded_total; | ||||||
| 		this.render_grand_total(grand_total); | 		this.render_grand_total(grand_total); | ||||||
| 
 | 
 | ||||||
| 		const taxes = frm.doc.taxes.map(t => { | 		this.render_taxes(frm.doc.taxes); | ||||||
| 			return { |  | ||||||
| 				description: t.description, rate: t.rate |  | ||||||
| 			}; |  | ||||||
| 		}); |  | ||||||
| 		this.render_taxes(frm.doc.total_taxes_and_charges, taxes); |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	render_net_total(value) { | 	render_net_total(value) { | ||||||
| @ -502,14 +497,14 @@ erpnext.PointOfSale.ItemCart = class { | |||||||
| 		); | 		); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	render_taxes(value, taxes) { | 	render_taxes(taxes) { | ||||||
| 		if (taxes.length) { | 		if (taxes.length) { | ||||||
| 			const currency = this.events.get_frm().doc.currency; | 			const currency = this.events.get_frm().doc.currency; | ||||||
| 			const taxes_html = taxes.map(t => { | 			const taxes_html = taxes.map(t => { | ||||||
| 				const description = /[0-9]+/.test(t.description) ? t.description : `${t.description} @ ${t.rate}%`; | 				const description = /[0-9]+/.test(t.description) ? t.description : `${t.description} @ ${t.rate}%`; | ||||||
| 				return `<div class="tax-row">
 | 				return `<div class="tax-row">
 | ||||||
| 					<div class="tax-label">${description}</div> | 					<div class="tax-label">${description}</div> | ||||||
| 					<div class="tax-value">${format_currency(value, currency)}</div> | 					<div class="tax-value">${format_currency(t.tax_amount_after_discount_amount, currency)}</div> | ||||||
| 				</div>`; | 				</div>`; | ||||||
| 			}).join(''); | 			}).join(''); | ||||||
| 			this.$totals_section.find('.taxes-container').css('display', 'flex').html(taxes_html); | 			this.$totals_section.find('.taxes-container').css('display', 'flex').html(taxes_html); | ||||||
|  | |||||||
| @ -56,7 +56,7 @@ erpnext.PointOfSale.Payment = class { | |||||||
| 				); | 				); | ||||||
| 				let df_events = { | 				let df_events = { | ||||||
| 					onchange: function() { | 					onchange: function() { | ||||||
| 						frm.set_value(this.df.fieldname, this.value); | 						frm.set_value(this.df.fieldname, this.get_value()); | ||||||
| 					} | 					} | ||||||
| 				}; | 				}; | ||||||
| 				if (df.fieldtype == "Button") { | 				if (df.fieldtype == "Button") { | ||||||
|  | |||||||
| @ -110,7 +110,7 @@ class Company(NestedSet): | |||||||
| 				self.create_default_warehouses() | 				self.create_default_warehouses() | ||||||
| 
 | 
 | ||||||
| 		if frappe.flags.country_change: | 		if frappe.flags.country_change: | ||||||
| 			install_country_fixtures(self.name) | 			install_country_fixtures(self.name, self.country) | ||||||
| 			self.create_default_tax_template() | 			self.create_default_tax_template() | ||||||
| 
 | 
 | ||||||
| 		if not frappe.db.get_value("Department", {"company": self.name}): | 		if not frappe.db.get_value("Department", {"company": self.name}): | ||||||
| @ -440,16 +440,15 @@ def get_name_with_abbr(name, company): | |||||||
| 
 | 
 | ||||||
| 	return " - ".join(parts) | 	return " - ".join(parts) | ||||||
| 
 | 
 | ||||||
| def install_country_fixtures(company): | def install_country_fixtures(company, country): | ||||||
| 	company_doc = frappe.get_doc("Company", company) | 	path = frappe.get_app_path('erpnext', 'regional', frappe.scrub(country)) | ||||||
| 	path = frappe.get_app_path('erpnext', 'regional', frappe.scrub(company_doc.country)) |  | ||||||
| 	if os.path.exists(path.encode("utf-8")): | 	if os.path.exists(path.encode("utf-8")): | ||||||
| 		try: | 		try: | ||||||
| 			module_name = "erpnext.regional.{0}.setup.setup".format(frappe.scrub(company_doc.country)) | 			module_name = "erpnext.regional.{0}.setup.setup".format(frappe.scrub(country)) | ||||||
| 			frappe.get_attr(module_name)(company_doc, False) | 			frappe.get_attr(module_name)(company, False) | ||||||
| 		except Exception as e: | 		except Exception as e: | ||||||
| 			frappe.log_error() | 			frappe.log_error() | ||||||
| 			frappe.throw(_("Failed to setup defaults for country {0}. Please contact support@erpnext.com").format(frappe.bold(company_doc.country))) | 			frappe.throw(_("Failed to setup defaults for country {0}. Please contact support@erpnext.com").format(frappe.bold(country))) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def update_company_current_month_sales(company): | def update_company_current_month_sales(company): | ||||||
|  | |||||||
| @ -1164,33 +1164,292 @@ | |||||||
| 	}, | 	}, | ||||||
| 
 | 
 | ||||||
| 	"India": { | 	"India": { | ||||||
|  | 		"tax_categories": [ | ||||||
|  | 			{ | ||||||
|  | 				"title": "In-State", | ||||||
|  | 				"is_inter_state": 0, | ||||||
|  | 				"gst_state": "" | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				"title": "Out-State", | ||||||
|  | 				"is_inter_state": 1, | ||||||
|  | 				"gst_state": "" | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				"title": "Reverse Charge In-State", | ||||||
|  | 				"is_inter_state": 0, | ||||||
|  | 				"gst_state": "" | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				"title": "Reverse Charge Out-State", | ||||||
|  | 				"is_inter_state": 1, | ||||||
|  | 				"gst_state": "" | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				"title": "Registered Composition", | ||||||
|  | 				"is_inter_state": 0, | ||||||
|  | 				"gst_state": "" | ||||||
|  | 			} | ||||||
|  | 		], | ||||||
| 		"chart_of_accounts": { | 		"chart_of_accounts": { | ||||||
| 			"*": { | 			"*": { | ||||||
| 				"item_tax_templates": [ | 				"item_tax_templates": [ | ||||||
| 					{ | 					{ | ||||||
| 						"title": "In State GST", | 						"title": "GST 9%", | ||||||
| 						"taxes": [ | 						"taxes": [ | ||||||
| 							{ | 							{ | ||||||
| 								"tax_type": { | 								"tax_type": { | ||||||
| 									"account_name": "SGST", | 									"account_name": "Output Tax SGST", | ||||||
| 									"tax_rate": 9.00 | 									"tax_rate": 9.00 | ||||||
| 								} | 								} | ||||||
| 							}, | 							}, | ||||||
| 							{ | 							{ | ||||||
| 								"tax_type": { | 								"tax_type": { | ||||||
| 									"account_name": "CGST", | 									"account_name": "Output Tax CGST", | ||||||
| 									"tax_rate": 9.00 | 									"tax_rate": 9.00 | ||||||
| 								} | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax IGST", | ||||||
|  | 									"tax_rate": 18.00 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST", | ||||||
|  | 									"tax_rate": 18.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST RCM", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST RCM", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST RCM", | ||||||
|  | 									"tax_rate": 18.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
| 							} | 							} | ||||||
| 						] | 						] | ||||||
| 					}, | 					}, | ||||||
| 					{ | 					{ | ||||||
| 						"title": "Out of State GST", | 						"title": "GST 5%", | ||||||
| 						"taxes": [ | 						"taxes": [ | ||||||
| 							{ | 							{ | ||||||
| 								"tax_type": { | 								"tax_type": { | ||||||
| 									"account_name": "IGST", | 									"account_name": "Output Tax SGST", | ||||||
| 									"tax_rate": 18.00 | 									"tax_rate": 2.5 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax CGST", | ||||||
|  | 									"tax_rate": 2.5 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax IGST", | ||||||
|  | 									"tax_rate": 5.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST", | ||||||
|  | 									"tax_rate": 2.5, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST", | ||||||
|  | 									"tax_rate": 2.5, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST", | ||||||
|  | 									"tax_rate": 5.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST RCM", | ||||||
|  | 									"tax_rate": 2.50, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST RCM", | ||||||
|  | 									"tax_rate": 2.50, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST RCM", | ||||||
|  | 									"tax_rate": 5.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						] | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						"title": "GST 12%", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax SGST", | ||||||
|  | 									"tax_rate": 6.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax CGST", | ||||||
|  | 									"tax_rate": 6.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax IGST", | ||||||
|  | 									"tax_rate": 12.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST", | ||||||
|  | 									"tax_rate": 6.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST", | ||||||
|  | 									"tax_rate": 6.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST", | ||||||
|  | 									"tax_rate": 12.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST RCM", | ||||||
|  | 									"tax_rate": 6.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST RCM", | ||||||
|  | 									"tax_rate": 6.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST RCM", | ||||||
|  | 									"tax_rate": 12.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						] | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						"title": "GST 28%", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax SGST", | ||||||
|  | 									"tax_rate": 14.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax CGST", | ||||||
|  | 									"tax_rate": 14.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Output Tax IGST", | ||||||
|  | 									"tax_rate": 28.0 | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST", | ||||||
|  | 									"tax_rate": 14.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST", | ||||||
|  | 									"tax_rate": 14.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST", | ||||||
|  | 									"tax_rate": 28.0, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax SGST RCM", | ||||||
|  | 									"tax_rate": 14.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax CGST RCM", | ||||||
|  | 									"tax_rate": 14.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"tax_type": { | ||||||
|  | 									"account_name": "Input Tax IGST RCM", | ||||||
|  | 									"tax_rate": 28.00, | ||||||
|  | 									"root_type": "Asset" | ||||||
| 								} | 								} | ||||||
| 							} | 							} | ||||||
| 						] | 						] | ||||||
| @ -1229,35 +1488,116 @@ | |||||||
| 						] | 						] | ||||||
| 					} | 					} | ||||||
| 				], | 				], | ||||||
| 				"*": [ | 				"sales_tax_templates": [ | ||||||
| 					{ | 					{ | ||||||
| 						"title": "In State GST", | 						"title": "Output GST In-state", | ||||||
| 						"taxes": [ | 						"taxes": [ | ||||||
| 							{ | 							{ | ||||||
| 								"account_head": { | 								"account_head": { | ||||||
| 									"account_name": "SGST", | 									"account_name": "Output Tax SGST", | ||||||
| 									"tax_rate": 9.00 | 									"tax_rate": 9.00, | ||||||
|  | 									"account_type": "Tax" | ||||||
| 								} | 								} | ||||||
| 							}, | 							}, | ||||||
| 							{ | 							{ | ||||||
| 								"account_head": { | 								"account_head": { | ||||||
| 									"account_name": "CGST", | 									"account_name": "Output Tax CGST", | ||||||
| 									"tax_rate": 9.00 | 									"tax_rate": 9.00, | ||||||
|  | 									"account_type": "Tax" | ||||||
| 								} | 								} | ||||||
| 							} | 							} | ||||||
| 						] | 						], | ||||||
|  | 						"tax_category": "In-State" | ||||||
| 					}, | 					}, | ||||||
| 					{ | 					{ | ||||||
| 						"title": "Out of State GST", | 						"title": "Output GST Out-state", | ||||||
| 						"taxes": [ | 						"taxes": [ | ||||||
| 							{ | 							{ | ||||||
| 								"account_head": { | 								"account_head": { | ||||||
| 									"account_name": "IGST", | 									"account_name": "Output Tax IGST", | ||||||
| 									"tax_rate": 18.00 | 									"tax_rate": 18.00, | ||||||
|  | 									"account_type": "Tax" | ||||||
| 								} | 								} | ||||||
| 							} | 							} | ||||||
| 						] | 						], | ||||||
|  | 						"tax_category": "Out-State" | ||||||
|  | 					} | ||||||
|  | 				], | ||||||
|  | 				"purchase_tax_templates": [ | ||||||
|  | 					{ | ||||||
|  | 						"title": "Input GST In-state", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax SGST", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax CGST", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						], | ||||||
|  | 						"tax_category": "In-State" | ||||||
| 					}, | 					}, | ||||||
|  | 					{ | ||||||
|  | 						"title": "Input GST Out-state", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax IGST", | ||||||
|  | 									"tax_rate": 18.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						], | ||||||
|  | 						"tax_category": "Out-State" | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						"title": "Input GST RCM In-state", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax SGST RCM", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							}, | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax CGST RCM", | ||||||
|  | 									"tax_rate": 9.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						], | ||||||
|  | 						"tax_category": "Reverse Charge In-State" | ||||||
|  | 					}, | ||||||
|  | 					{ | ||||||
|  | 						"title": "Input GST RCM Out-state", | ||||||
|  | 						"taxes": [ | ||||||
|  | 							{ | ||||||
|  | 								"account_head": { | ||||||
|  | 									"account_name": "Input Tax IGST RCM", | ||||||
|  | 									"tax_rate": 18.00, | ||||||
|  | 									"root_type": "Asset", | ||||||
|  | 									"account_type": "Tax" | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						], | ||||||
|  | 						"tax_category": "Reverse Charge Out-State" | ||||||
|  | 					} | ||||||
|  | 				], | ||||||
|  | 				"*": [ | ||||||
| 					{ | 					{ | ||||||
| 						"title": "VAT 5%", | 						"title": "VAT 5%", | ||||||
| 						"taxes": [ | 						"taxes": [ | ||||||
| @ -1349,7 +1689,7 @@ | |||||||
| 		"Italy VAT 4%":{ | 		"Italy VAT 4%":{ | ||||||
| 			"account_name": "IVA 4%", | 			"account_name": "IVA 4%", | ||||||
| 			"tax_rate": 4.00 | 			"tax_rate": 4.00 | ||||||
| 		}		 | 		} | ||||||
| 	}, | 	}, | ||||||
| 
 | 
 | ||||||
| 	"Ivory Coast": { | 	"Ivory Coast": { | ||||||
|  | |||||||
| @ -42,29 +42,6 @@ def enable_shopping_cart(args): | |||||||
| 		'quotation_series': "QTN-", | 		'quotation_series': "QTN-", | ||||||
| 	}).insert() | 	}).insert() | ||||||
| 
 | 
 | ||||||
| def create_bank_account(args): |  | ||||||
| 	if args.get("bank_account"): |  | ||||||
| 		company_name = args.get('company_name') |  | ||||||
| 		bank_account_group =  frappe.db.get_value("Account", |  | ||||||
| 			{"account_type": "Bank", "is_group": 1, "root_type": "Asset", |  | ||||||
| 				"company": company_name}) |  | ||||||
| 		if bank_account_group: |  | ||||||
| 			bank_account = frappe.get_doc({ |  | ||||||
| 				"doctype": "Account", |  | ||||||
| 				'account_name': args.get("bank_account"), |  | ||||||
| 				'parent_account': bank_account_group, |  | ||||||
| 				'is_group':0, |  | ||||||
| 				'company': company_name, |  | ||||||
| 				"account_type": "Bank", |  | ||||||
| 			}) |  | ||||||
| 			try: |  | ||||||
| 				return bank_account.insert() |  | ||||||
| 			except RootNotEditable: |  | ||||||
| 				frappe.throw(_("Bank account cannot be named as {0}").format(args.get("bank_account"))) |  | ||||||
| 			except frappe.DuplicateEntryError: |  | ||||||
| 				# bank account same as a CoA entry |  | ||||||
| 				pass |  | ||||||
| 
 |  | ||||||
| def create_email_digest(): | def create_email_digest(): | ||||||
| 	from frappe.utils.user import get_system_managers | 	from frappe.utils.user import get_system_managers | ||||||
| 	system_managers = get_system_managers(only_name=True) | 	system_managers = get_system_managers(only_name=True) | ||||||
|  | |||||||
| @ -448,6 +448,8 @@ def install_defaults(args=None): | |||||||
| 	set_active_domains(args) | 	set_active_domains(args) | ||||||
| 	update_stock_settings() | 	update_stock_settings() | ||||||
| 	update_shopping_cart_settings(args) | 	update_shopping_cart_settings(args) | ||||||
|  | 
 | ||||||
|  | 	args.update({"set_default": 1}) | ||||||
| 	create_bank_account(args) | 	create_bank_account(args) | ||||||
| 
 | 
 | ||||||
| def set_global_defaults(args): | def set_global_defaults(args): | ||||||
| @ -479,17 +481,17 @@ def update_stock_settings(): | |||||||
| 	stock_settings.save() | 	stock_settings.save() | ||||||
| 
 | 
 | ||||||
| def create_bank_account(args): | def create_bank_account(args): | ||||||
| 	if not args.bank_account: | 	if not args.get('bank_account'): | ||||||
| 		return | 		return | ||||||
| 
 | 
 | ||||||
| 	company_name = args.company_name | 	company_name = args.get('company_name') | ||||||
| 	bank_account_group =  frappe.db.get_value("Account", | 	bank_account_group =  frappe.db.get_value("Account", | ||||||
| 		{"account_type": "Bank", "is_group": 1, "root_type": "Asset", | 		{"account_type": "Bank", "is_group": 1, "root_type": "Asset", | ||||||
| 			"company": company_name}) | 			"company": company_name}) | ||||||
| 	if bank_account_group: | 	if bank_account_group: | ||||||
| 		bank_account = frappe.get_doc({ | 		bank_account = frappe.get_doc({ | ||||||
| 			"doctype": "Account", | 			"doctype": "Account", | ||||||
| 			'account_name': args.bank_account, | 			'account_name': args.get('bank_account'), | ||||||
| 			'parent_account': bank_account_group, | 			'parent_account': bank_account_group, | ||||||
| 			'is_group':0, | 			'is_group':0, | ||||||
| 			'company': company_name, | 			'company': company_name, | ||||||
| @ -498,10 +500,13 @@ def create_bank_account(args): | |||||||
| 		try: | 		try: | ||||||
| 			doc = bank_account.insert() | 			doc = bank_account.insert() | ||||||
| 
 | 
 | ||||||
| 			frappe.db.set_value("Company", args.company_name, "default_bank_account", bank_account.name, update_modified=False) | 			if args.get('set_default'): | ||||||
|  | 				frappe.db.set_value("Company", args.get('company_name'), "default_bank_account", bank_account.name, update_modified=False) | ||||||
|  | 
 | ||||||
|  | 			return doc | ||||||
| 
 | 
 | ||||||
| 		except RootNotEditable: | 		except RootNotEditable: | ||||||
| 			frappe.throw(_("Bank account cannot be named as {0}").format(args.bank_account)) | 			frappe.throw(_("Bank account cannot be named as {0}").format(args.get('bank_account'))) | ||||||
| 		except frappe.DuplicateEntryError: | 		except frappe.DuplicateEntryError: | ||||||
| 			# bank account same as a CoA entry | 			# bank account same as a CoA entry | ||||||
| 			pass | 			pass | ||||||
|  | |||||||
| @ -27,6 +27,7 @@ def setup_taxes_and_charges(company_name: str, country: str): | |||||||
| 		country_wise_tax = simple_to_detailed(country_wise_tax) | 		country_wise_tax = simple_to_detailed(country_wise_tax) | ||||||
| 
 | 
 | ||||||
| 	from_detailed_data(company_name, country_wise_tax) | 	from_detailed_data(company_name, country_wise_tax) | ||||||
|  | 	update_regional_tax_settings(country, company_name) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def simple_to_detailed(templates): | def simple_to_detailed(templates): | ||||||
| @ -101,6 +102,17 @@ def from_detailed_data(company_name, data): | |||||||
| 			make_item_tax_template(company_name, template) | 			make_item_tax_template(company_name, template) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def update_regional_tax_settings(country, company): | ||||||
|  | 	path = frappe.get_app_path('erpnext', 'regional', frappe.scrub(country)) | ||||||
|  | 	if os.path.exists(path.encode("utf-8")): | ||||||
|  | 		try: | ||||||
|  | 			module_name = "erpnext.regional.{0}.setup.update_regional_tax_settings".format(frappe.scrub(country)) | ||||||
|  | 			frappe.get_attr(module_name)(country, company) | ||||||
|  | 		except Exception as e: | ||||||
|  | 			# Log error and ignore if failed to setup regional tax settings | ||||||
|  | 			frappe.log_error() | ||||||
|  | 			pass | ||||||
|  | 
 | ||||||
| def make_taxes_and_charges_template(company_name, doctype, template): | def make_taxes_and_charges_template(company_name, doctype, template): | ||||||
| 	template['company'] = company_name | 	template['company'] = company_name | ||||||
| 	template['doctype'] = doctype | 	template['doctype'] = doctype | ||||||
| @ -130,8 +142,14 @@ def make_taxes_and_charges_template(company_name, doctype, template): | |||||||
| 			if fieldname not in tax_row: | 			if fieldname not in tax_row: | ||||||
| 				tax_row[fieldname] = default_value | 				tax_row[fieldname] = default_value | ||||||
| 
 | 
 | ||||||
| 	return frappe.get_doc(template).insert(ignore_permissions=True) | 	doc = frappe.get_doc(template) | ||||||
| 
 | 
 | ||||||
|  | 	# Data in country wise json is already pre validated, hence validations can be ignored  | ||||||
|  | 	# Ingone validations to make doctypes faster | ||||||
|  | 	doc.flags.ignore_links = True | ||||||
|  | 	doc.flags.ignore_validate = True | ||||||
|  | 	doc.insert(ignore_permissions=True) | ||||||
|  | 	return doc | ||||||
| 
 | 
 | ||||||
| def make_item_tax_template(company_name, template): | def make_item_tax_template(company_name, template): | ||||||
| 	"""Create an Item Tax Template. | 	"""Create an Item Tax Template. | ||||||
| @ -156,8 +174,24 @@ def make_item_tax_template(company_name, template): | |||||||
| 			if 'tax_rate' not in tax_row: | 			if 'tax_rate' not in tax_row: | ||||||
| 				tax_row['tax_rate'] = account_data.get('tax_rate') | 				tax_row['tax_rate'] = account_data.get('tax_rate') | ||||||
| 
 | 
 | ||||||
| 	return frappe.get_doc(template).insert(ignore_permissions=True) | 	doc = frappe.get_doc(template) | ||||||
| 
 | 
 | ||||||
|  | 	# Data in country wise json is already pre validated, hence validations can be ignored  | ||||||
|  | 	# Ingone validations to make doctypes faster | ||||||
|  | 	doc.flags.ignore_links = True | ||||||
|  | 	doc.flags.ignore_validate = True | ||||||
|  | 	doc.insert(ignore_permissions=True) | ||||||
|  | 	return doc | ||||||
|  | 
 | ||||||
|  | def make_tax_category(tax_category): | ||||||
|  | 	""" Make tax category based on title if not already created """ | ||||||
|  | 	doctype = 'Tax Category' | ||||||
|  | 	if not frappe.db.exists(doctype, tax_category['title']): | ||||||
|  | 		tax_category['doctype'] = doctype | ||||||
|  | 		doc = frappe.get_doc(tax_category) | ||||||
|  | 		doc.flags.ignore_links = True | ||||||
|  | 		doc.flags.ignore_validate = True | ||||||
|  | 		doc.insert(ignore_permissions=True) | ||||||
| 
 | 
 | ||||||
| def get_or_create_account(company_name, account): | def get_or_create_account(company_name, account): | ||||||
| 	""" | 	""" | ||||||
| @ -175,8 +209,7 @@ def get_or_create_account(company_name, account): | |||||||
| 		or_filters={ | 		or_filters={ | ||||||
| 			'account_name': account.get('account_name'), | 			'account_name': account.get('account_name'), | ||||||
| 			'account_number': account.get('account_number') | 			'account_number': account.get('account_number') | ||||||
| 		} | 		}) | ||||||
| 	) |  | ||||||
| 
 | 
 | ||||||
| 	if existing_accounts: | 	if existing_accounts: | ||||||
| 		return frappe.get_doc('Account', existing_accounts[0].name) | 		return frappe.get_doc('Account', existing_accounts[0].name) | ||||||
| @ -191,8 +224,11 @@ def get_or_create_account(company_name, account): | |||||||
| 	account['root_type'] = root_type | 	account['root_type'] = root_type | ||||||
| 	account['is_group'] = 0 | 	account['is_group'] = 0 | ||||||
| 
 | 
 | ||||||
| 	return frappe.get_doc(account).insert(ignore_permissions=True, ignore_mandatory=True) | 	doc = frappe.get_doc(account) | ||||||
| 
 | 	doc.flags.ignore_links = True | ||||||
|  | 	doc.flags.ignore_validate = True | ||||||
|  | 	doc.insert(ignore_permissions=True, ignore_mandatory=True) | ||||||
|  | 	return doc | ||||||
| 
 | 
 | ||||||
| def get_or_create_tax_group(company_name, root_type): | def get_or_create_tax_group(company_name, root_type): | ||||||
| 	# Look for a group account of type 'Tax' | 	# Look for a group account of type 'Tax' | ||||||
| @ -237,7 +273,11 @@ def get_or_create_tax_group(company_name, root_type): | |||||||
| 		'account_type': 'Tax', | 		'account_type': 'Tax', | ||||||
| 		'account_name': account_name, | 		'account_name': account_name, | ||||||
| 		'parent_account': root_account.name | 		'parent_account': root_account.name | ||||||
| 	}).insert(ignore_permissions=True) | 	}) | ||||||
|  | 
 | ||||||
|  | 	tax_group_account.flags.ignore_links = True | ||||||
|  | 	tax_group_account.flags.ignore_validate = True | ||||||
|  | 	tax_group_account.insert(ignore_permissions=True) | ||||||
| 
 | 
 | ||||||
| 	tax_group_name = tax_group_account.name | 	tax_group_name = tax_group_account.name | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -11,10 +11,11 @@ | |||||||
|  "hide_custom": 0, |  "hide_custom": 0, | ||||||
|  "icon": "settings", |  "icon": "settings", | ||||||
|  "idx": 0, |  "idx": 0, | ||||||
|  |  "is_default": 0, | ||||||
|  "is_standard": 1, |  "is_standard": 1, | ||||||
|  "label": "ERPNext Settings", |  "label": "ERPNext Settings", | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2020-12-01 13:38:37.759596", |  "modified": "2021-06-12 01:58:11.399566", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Setup", |  "module": "Setup", | ||||||
|  "name": "ERPNext Settings", |  "name": "ERPNext Settings", | ||||||
| @ -109,6 +110,13 @@ | |||||||
|    "label": "Domain Settings", |    "label": "Domain Settings", | ||||||
|    "link_to": "Domain Settings", |    "link_to": "Domain Settings", | ||||||
|    "type": "DocType" |    "type": "DocType" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "doc_view": "", | ||||||
|  |    "icon": "retail", | ||||||
|  |    "label": "Products Settings", | ||||||
|  |    "link_to": "Products Settings", | ||||||
|  |    "type": "DocType" | ||||||
|   } |   } | ||||||
|  ] |  ] | ||||||
| } | } | ||||||
|  | |||||||
| @ -71,7 +71,8 @@ class ProductQuery: | |||||||
| 					], | 					], | ||||||
| 					or_filters=self.or_filters, | 					or_filters=self.or_filters, | ||||||
| 					start=start, | 					start=start, | ||||||
| 					limit=self.page_length | 					limit=self.page_length, | ||||||
|  | 					order_by="weightage desc" | ||||||
| 				) | 				) | ||||||
| 
 | 
 | ||||||
| 				items_dict = {item.name: item for item in items} | 				items_dict = {item.name: item for item in items} | ||||||
| @ -86,7 +87,8 @@ class ProductQuery: | |||||||
| 				filters=self.filters, | 				filters=self.filters, | ||||||
| 				or_filters=self.or_filters, | 				or_filters=self.or_filters, | ||||||
| 				start=start, | 				start=start, | ||||||
| 				limit=self.page_length | 				limit=self.page_length, | ||||||
|  | 				order_by="weightage desc" | ||||||
| 			) | 			) | ||||||
| 
 | 
 | ||||||
| 		# Combine results having context of website item groups into item results | 		# Combine results having context of website item groups into item results | ||||||
|  | |||||||
| @ -193,7 +193,7 @@ | |||||||
|  "image_field": "image", |  "image_field": "image", | ||||||
|  "links": [], |  "links": [], | ||||||
|  "max_attachments": 5, |  "max_attachments": 5, | ||||||
|  "modified": "2021-01-07 11:10:09.149170", |  "modified": "2021-07-08 16:22:01.343105", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Stock", |  "module": "Stock", | ||||||
|  "name": "Batch", |  "name": "Batch", | ||||||
| @ -217,5 +217,6 @@ | |||||||
|  "quick_entry": 1, |  "quick_entry": 1, | ||||||
|  "sort_field": "modified", |  "sort_field": "modified", | ||||||
|  "sort_order": "DESC", |  "sort_order": "DESC", | ||||||
|  "title_field": "batch_id" |  "title_field": "batch_id", | ||||||
|  |  "track_changes": 1 | ||||||
| } | } | ||||||
| @ -230,9 +230,8 @@ def set_batch_nos(doc, warehouse_field, throw=False, child_table="items"): | |||||||
| 	"""Automatically select `batch_no` for outgoing items in item table""" | 	"""Automatically select `batch_no` for outgoing items in item table""" | ||||||
| 	for d in doc.get(child_table): | 	for d in doc.get(child_table): | ||||||
| 		qty = d.get('stock_qty') or d.get('transfer_qty') or d.get('qty') or 0 | 		qty = d.get('stock_qty') or d.get('transfer_qty') or d.get('qty') or 0 | ||||||
| 		has_batch_no = frappe.db.get_value('Item', d.item_code, 'has_batch_no') |  | ||||||
| 		warehouse = d.get(warehouse_field, None) | 		warehouse = d.get(warehouse_field, None) | ||||||
| 		if has_batch_no and warehouse and qty > 0: | 		if warehouse and qty > 0 and frappe.db.get_value('Item', d.item_code, 'has_batch_no'): | ||||||
| 			if not d.batch_no: | 			if not d.batch_no: | ||||||
| 				d.batch_no = get_batch_no(d.item_code, warehouse, qty, throw, d.serial_no) | 				d.batch_no = get_batch_no(d.item_code, warehouse, qty, throw, d.serial_no) | ||||||
| 			else: | 			else: | ||||||
| @ -313,4 +312,4 @@ def validate_serial_no_with_batch(serial_nos, item_code): | |||||||
| def make_batch(args): | def make_batch(args): | ||||||
| 	if frappe.db.get_value("Item", args.item, "has_batch_no"): | 	if frappe.db.get_value("Item", args.item, "has_batch_no"): | ||||||
| 		args.doctype = "Batch" | 		args.doctype = "Batch" | ||||||
| 		frappe.get_doc(args).insert().name | 		frappe.get_doc(args).insert().name | ||||||
|  | |||||||
| @ -182,9 +182,8 @@ class DeliveryNote(SellingController): | |||||||
| 		super(DeliveryNote, self).validate_warehouse() | 		super(DeliveryNote, self).validate_warehouse() | ||||||
| 
 | 
 | ||||||
| 		for d in self.get_item_list(): | 		for d in self.get_item_list(): | ||||||
| 			if frappe.db.get_value("Item", d['item_code'], "is_stock_item") == 1: | 			if not d['warehouse'] and frappe.db.get_value("Item", d['item_code'], "is_stock_item") == 1: | ||||||
| 				if not d['warehouse']: | 				frappe.throw(_("Warehouse required for stock Item {0}").format(d["item_code"])) | ||||||
| 					frappe.throw(_("Warehouse required for stock Item {0}").format(d["item_code"])) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	def update_current_stock(self): | 	def update_current_stock(self): | ||||||
|  | |||||||
| @ -93,7 +93,7 @@ frappe.ui.form.on("Item", { | |||||||
| 
 | 
 | ||||||
| 		erpnext.item.edit_prices_button(frm); | 		erpnext.item.edit_prices_button(frm); | ||||||
| 		erpnext.item.toggle_attributes(frm); | 		erpnext.item.toggle_attributes(frm); | ||||||
| 		 | 
 | ||||||
| 		if (!frm.doc.is_fixed_asset) { | 		if (!frm.doc.is_fixed_asset) { | ||||||
| 			erpnext.item.make_dashboard(frm); | 			erpnext.item.make_dashboard(frm); | ||||||
| 		} | 		} | ||||||
| @ -381,7 +381,8 @@ $.extend(erpnext.item, { | |||||||
| 		// Show Stock Levels only if is_stock_item
 | 		// Show Stock Levels only if is_stock_item
 | ||||||
| 		if (frm.doc.is_stock_item) { | 		if (frm.doc.is_stock_item) { | ||||||
| 			frappe.require('assets/js/item-dashboard.min.js', function() { | 			frappe.require('assets/js/item-dashboard.min.js', function() { | ||||||
| 				const section = frm.dashboard.add_section('', __("Stock Levels")); | 				frm.dashboard.parent.find('.stock-levels').remove(); | ||||||
|  | 				const section = frm.dashboard.add_section('', __("Stock Levels"), 'stock-levels'); | ||||||
| 				erpnext.item.item_dashboard = new erpnext.stock.ItemDashboard({ | 				erpnext.item.item_dashboard = new erpnext.stock.ItemDashboard({ | ||||||
| 					parent: section, | 					parent: section, | ||||||
| 					item_code: frm.doc.name, | 					item_code: frm.doc.name, | ||||||
|  | |||||||
| @ -41,7 +41,7 @@ class LandedCostVoucher(Document): | |||||||
| 
 | 
 | ||||||
| 	def validate(self): | 	def validate(self): | ||||||
| 		self.check_mandatory() | 		self.check_mandatory() | ||||||
| 		self.validate_purchase_receipts() | 		self.validate_receipt_documents() | ||||||
| 		init_landed_taxes_and_totals(self) | 		init_landed_taxes_and_totals(self) | ||||||
| 		self.set_total_taxes_and_charges() | 		self.set_total_taxes_and_charges() | ||||||
| 		if not self.get("items"): | 		if not self.get("items"): | ||||||
| @ -56,14 +56,23 @@ class LandedCostVoucher(Document): | |||||||
| 			frappe.throw(_("Please enter Receipt Document")) | 			frappe.throw(_("Please enter Receipt Document")) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	def validate_purchase_receipts(self): | 	def validate_receipt_documents(self): | ||||||
| 		receipt_documents = [] | 		receipt_documents = [] | ||||||
| 
 | 
 | ||||||
| 		for d in self.get("purchase_receipts"): | 		for d in self.get("purchase_receipts"): | ||||||
| 			if frappe.db.get_value(d.receipt_document_type, d.receipt_document, "docstatus") != 1: | 			docstatus = frappe.db.get_value(d.receipt_document_type, d.receipt_document, "docstatus") | ||||||
| 				frappe.throw(_("Receipt document must be submitted")) | 			if docstatus != 1: | ||||||
| 			else: | 				msg = f"Row {d.idx}: {d.receipt_document_type} {frappe.bold(d.receipt_document)} must be submitted" | ||||||
| 				receipt_documents.append(d.receipt_document) | 				frappe.throw(_(msg), title=_("Invalid Document")) | ||||||
|  | 
 | ||||||
|  | 			if d.receipt_document_type == "Purchase Invoice": | ||||||
|  | 				update_stock = frappe.db.get_value(d.receipt_document_type, d.receipt_document, "update_stock") | ||||||
|  | 				if not update_stock: | ||||||
|  | 					msg = _("Row {0}: Purchase Invoice {1} has no stock impact.").format(d.idx, frappe.bold(d.receipt_document)) | ||||||
|  | 					msg += "<br>" + _("Please create Landed Cost Vouchers against Invoices that have 'Update Stock' enabled.") | ||||||
|  | 					frappe.throw(msg, title=_("Incorrect Invoice")) | ||||||
|  | 
 | ||||||
|  | 			receipt_documents.append(d.receipt_document) | ||||||
| 
 | 
 | ||||||
| 		for item in self.get("items"): | 		for item in self.get("items"): | ||||||
| 			if not item.receipt_document: | 			if not item.receipt_document: | ||||||
|  | |||||||
| @ -189,7 +189,7 @@ class MaterialRequest(BuyingController): | |||||||
| 		item_wh_list = [] | 		item_wh_list = [] | ||||||
| 		for d in self.get("items"): | 		for d in self.get("items"): | ||||||
| 			if (not mr_item_rows or d.name in mr_item_rows) and [d.item_code, d.warehouse] not in item_wh_list \ | 			if (not mr_item_rows or d.name in mr_item_rows) and [d.item_code, d.warehouse] not in item_wh_list \ | ||||||
| 					and frappe.db.get_value("Item", d.item_code, "is_stock_item") == 1 and d.warehouse: | 					and d.warehouse and frappe.db.get_value("Item", d.item_code, "is_stock_item") == 1 : | ||||||
| 				item_wh_list.append([d.item_code, d.warehouse]) | 				item_wh_list.append([d.item_code, d.warehouse]) | ||||||
| 
 | 
 | ||||||
| 		for item_code, warehouse in item_wh_list: | 		for item_code, warehouse in item_wh_list: | ||||||
|  | |||||||
| @ -184,4 +184,4 @@ | |||||||
|  "sort_field": "modified", |  "sort_field": "modified", | ||||||
|  "sort_order": "DESC", |  "sort_order": "DESC", | ||||||
|  "track_changes": 1 |  "track_changes": 1 | ||||||
| } | } | ||||||
|  | |||||||
| @ -17,6 +17,9 @@ from erpnext.selling.doctype.sales_order.sales_order import make_delivery_note a | |||||||
| # TODO: Prioritize SO or WO group warehouse | # TODO: Prioritize SO or WO group warehouse | ||||||
| 
 | 
 | ||||||
| class PickList(Document): | class PickList(Document): | ||||||
|  | 	def validate(self): | ||||||
|  | 		self.validate_for_qty() | ||||||
|  | 
 | ||||||
| 	def before_save(self): | 	def before_save(self): | ||||||
| 		self.set_item_locations() | 		self.set_item_locations() | ||||||
| 
 | 
 | ||||||
| @ -35,6 +38,7 @@ class PickList(Document): | |||||||
| 
 | 
 | ||||||
| 	@frappe.whitelist() | 	@frappe.whitelist() | ||||||
| 	def set_item_locations(self, save=False): | 	def set_item_locations(self, save=False): | ||||||
|  | 		self.validate_for_qty() | ||||||
| 		items = self.aggregate_item_qty() | 		items = self.aggregate_item_qty() | ||||||
| 		self.item_location_map = frappe._dict() | 		self.item_location_map = frappe._dict() | ||||||
| 
 | 
 | ||||||
| @ -107,6 +111,11 @@ class PickList(Document): | |||||||
| 
 | 
 | ||||||
| 		return item_map.values() | 		return item_map.values() | ||||||
| 
 | 
 | ||||||
|  | 	def validate_for_qty(self): | ||||||
|  | 		if self.purpose == "Material Transfer for Manufacture" \ | ||||||
|  | 				and (self.for_qty is None or self.for_qty == 0): | ||||||
|  | 			frappe.throw(_("Qty of Finished Goods Item should be greater than 0.")) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| def validate_item_locations(pick_list): | def validate_item_locations(pick_list): | ||||||
| 	if not pick_list.locations: | 	if not pick_list.locations: | ||||||
|  | |||||||
| @ -37,6 +37,7 @@ class TestPickList(unittest.TestCase): | |||||||
| 			'company': '_Test Company', | 			'company': '_Test Company', | ||||||
| 			'customer': '_Test Customer', | 			'customer': '_Test Customer', | ||||||
| 			'items_based_on': 'Sales Order', | 			'items_based_on': 'Sales Order', | ||||||
|  | 			'purpose': 'Delivery', | ||||||
| 			'locations': [{ | 			'locations': [{ | ||||||
| 				'item_code': '_Test Item', | 				'item_code': '_Test Item', | ||||||
| 				'qty': 5, | 				'qty': 5, | ||||||
| @ -90,6 +91,7 @@ class TestPickList(unittest.TestCase): | |||||||
| 			'company': '_Test Company', | 			'company': '_Test Company', | ||||||
| 			'customer': '_Test Customer', | 			'customer': '_Test Customer', | ||||||
| 			'items_based_on': 'Sales Order', | 			'items_based_on': 'Sales Order', | ||||||
|  | 			'purpose': 'Delivery', | ||||||
| 			'locations': [{ | 			'locations': [{ | ||||||
| 				'item_code': '_Test Item Warehouse Group Wise Reorder', | 				'item_code': '_Test Item Warehouse Group Wise Reorder', | ||||||
| 				'qty': 1000, | 				'qty': 1000, | ||||||
| @ -135,6 +137,7 @@ class TestPickList(unittest.TestCase): | |||||||
| 			'company': '_Test Company', | 			'company': '_Test Company', | ||||||
| 			'customer': '_Test Customer', | 			'customer': '_Test Customer', | ||||||
| 			'items_based_on': 'Sales Order', | 			'items_based_on': 'Sales Order', | ||||||
|  | 			'purpose': 'Delivery', | ||||||
| 			'locations': [{ | 			'locations': [{ | ||||||
| 				'item_code': '_Test Serialized Item', | 				'item_code': '_Test Serialized Item', | ||||||
| 				'qty': 1000, | 				'qty': 1000, | ||||||
| @ -264,6 +267,7 @@ class TestPickList(unittest.TestCase): | |||||||
| 			'company': '_Test Company', | 			'company': '_Test Company', | ||||||
| 			'customer': '_Test Customer', | 			'customer': '_Test Customer', | ||||||
| 			'items_based_on': 'Sales Order', | 			'items_based_on': 'Sales Order', | ||||||
|  | 			'purpose': 'Delivery', | ||||||
| 			'locations': [{ | 			'locations': [{ | ||||||
| 				'item_code': '_Test Item', | 				'item_code': '_Test Item', | ||||||
| 				'qty': 5, | 				'qty': 5, | ||||||
| @ -319,6 +323,7 @@ class TestPickList(unittest.TestCase): | |||||||
| 			'company': '_Test Company', | 			'company': '_Test Company', | ||||||
| 			'customer': '_Test Customer', | 			'customer': '_Test Customer', | ||||||
| 			'items_based_on': 'Sales Order', | 			'items_based_on': 'Sales Order', | ||||||
|  | 			'purpose': 'Delivery', | ||||||
| 			'locations': [{ | 			'locations': [{ | ||||||
| 				'item_code': '_Test Item', | 				'item_code': '_Test Item', | ||||||
| 				'qty': 1, | 				'qty': 1, | ||||||
|  | |||||||
| @ -97,7 +97,7 @@ def apply_putaway_rule(doctype, items, company, sync=None, purpose=None): | |||||||
| 		at_capacity, rules = get_ordered_putaway_rules(item_code, company, source_warehouse=source_warehouse) | 		at_capacity, rules = get_ordered_putaway_rules(item_code, company, source_warehouse=source_warehouse) | ||||||
| 
 | 
 | ||||||
| 		if not rules: | 		if not rules: | ||||||
| 			warehouse = source_warehouse or item.warehouse | 			warehouse = source_warehouse or item.get('warehouse') | ||||||
| 			if at_capacity: | 			if at_capacity: | ||||||
| 				# rules available, but no free space | 				# rules available, but no free space | ||||||
| 				items_not_accomodated.append([item_code, pending_qty]) | 				items_not_accomodated.append([item_code, pending_qty]) | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ from erpnext.controllers.stock_controller import ( | |||||||
| ) | ) | ||||||
| from erpnext.stock.doctype.delivery_note.test_delivery_note import create_delivery_note | from erpnext.stock.doctype.delivery_note.test_delivery_note import create_delivery_note | ||||||
| from erpnext.stock.doctype.item.test_item import create_item | from erpnext.stock.doctype.item.test_item import create_item | ||||||
| from erpnext.stock.doctype.stock_entry.test_stock_entry import make_stock_entry | from erpnext.stock.doctype.stock_entry.stock_entry_utils import make_stock_entry | ||||||
| 
 | 
 | ||||||
| # test_records = frappe.get_test_records('Quality Inspection') | # test_records = frappe.get_test_records('Quality Inspection') | ||||||
| 
 | 
 | ||||||
| @ -159,6 +159,47 @@ class TestQualityInspection(unittest.TestCase): | |||||||
| 			frappe.delete_doc("Quality Inspection", qi) | 			frappe.delete_doc("Quality Inspection", qi) | ||||||
| 		dn.delete() | 		dn.delete() | ||||||
| 
 | 
 | ||||||
|  | 	def test_rejected_qi_validation(self): | ||||||
|  | 		"""Test if rejected QI blocks Stock Entry as per Stock Settings.""" | ||||||
|  | 		se = make_stock_entry( | ||||||
|  | 			item_code="_Test Item with QA", | ||||||
|  | 			target="_Test Warehouse - _TC", | ||||||
|  | 			qty=1, | ||||||
|  | 			basic_rate=100, | ||||||
|  | 			inspection_required=True, | ||||||
|  | 			do_not_submit=True | ||||||
|  | 		) | ||||||
|  | 
 | ||||||
|  | 		readings = [ | ||||||
|  | 			{ | ||||||
|  | 				"specification": "Iron Content", | ||||||
|  | 				"min_value": 0.1, | ||||||
|  | 				"max_value": 0.9, | ||||||
|  | 				"reading_1": "0.4" | ||||||
|  | 			} | ||||||
|  | 		] | ||||||
|  | 
 | ||||||
|  | 		qa = create_quality_inspection( | ||||||
|  | 			reference_type="Stock Entry", | ||||||
|  | 			reference_name=se.name, | ||||||
|  | 			readings=readings, | ||||||
|  | 			status="Rejected" | ||||||
|  | 		) | ||||||
|  | 
 | ||||||
|  | 		frappe.db.set_value("Stock Settings", None, "action_if_quality_inspection_is_rejected", "Stop") | ||||||
|  | 		se.reload() | ||||||
|  | 		self.assertRaises(QualityInspectionRejectedError, se.submit) # when blocked in Stock settings, block rejected QI | ||||||
|  | 
 | ||||||
|  | 		frappe.db.set_value("Stock Settings", None, "action_if_quality_inspection_is_rejected", "Warn") | ||||||
|  | 		se.reload() | ||||||
|  | 		se.submit() # when allowed in Stock settings, allow rejected QI | ||||||
|  | 
 | ||||||
|  | 		# teardown | ||||||
|  | 		qa.reload() | ||||||
|  | 		qa.cancel() | ||||||
|  | 		se.reload() | ||||||
|  | 		se.cancel() | ||||||
|  | 		frappe.db.set_value("Stock Settings", None, "action_if_quality_inspection_is_rejected", "Stop") | ||||||
| 
 | 
 | ||||||
| def create_quality_inspection(**args): | def create_quality_inspection(**args): | ||||||
| 	args = frappe._dict(args) | 	args = frappe._dict(args) | ||||||
| @ -175,12 +216,11 @@ def create_quality_inspection(**args): | |||||||
| 	if not args.readings: | 	if not args.readings: | ||||||
| 		create_quality_inspection_parameter("Size") | 		create_quality_inspection_parameter("Size") | ||||||
| 		readings = {"specification": "Size", "min_value": 0, "max_value": 10} | 		readings = {"specification": "Size", "min_value": 0, "max_value": 10} | ||||||
|  | 		if args.status == "Rejected": | ||||||
|  | 			readings["reading_1"] = "12"  # status is auto set in child on save | ||||||
| 	else: | 	else: | ||||||
| 		readings = args.readings | 		readings = args.readings | ||||||
| 
 | 
 | ||||||
| 	if args.status == "Rejected": |  | ||||||
| 		readings["reading_1"] = "12"  # status is auto set in child on save |  | ||||||
| 
 |  | ||||||
| 	if isinstance(readings, list): | 	if isinstance(readings, list): | ||||||
| 		for entry in readings: | 		for entry in readings: | ||||||
| 			create_quality_inspection_parameter(entry["specification"]) | 			create_quality_inspection_parameter(entry["specification"]) | ||||||
|  | |||||||
| @ -45,6 +45,8 @@ def make_stock_entry(**args): | |||||||
| 		s.posting_date = args.posting_date | 		s.posting_date = args.posting_date | ||||||
| 	if args.posting_time: | 	if args.posting_time: | ||||||
| 		s.posting_time = args.posting_time | 		s.posting_time = args.posting_time | ||||||
|  | 	if args.inspection_required: | ||||||
|  | 		s.inspection_required = args.inspection_required | ||||||
| 
 | 
 | ||||||
| 	# map names | 	# map names | ||||||
| 	if args.from_warehouse: | 	if args.from_warehouse: | ||||||
|  | |||||||
| @ -307,6 +307,7 @@ | |||||||
|    "fieldname": "quality_inspection", |    "fieldname": "quality_inspection", | ||||||
|    "fieldtype": "Link", |    "fieldtype": "Link", | ||||||
|    "label": "Quality Inspection", |    "label": "Quality Inspection", | ||||||
|  |    "no_copy": 1, | ||||||
|    "options": "Quality Inspection" |    "options": "Quality Inspection" | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
| @ -548,7 +549,7 @@ | |||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "istable": 1, |  "istable": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-04-22 20:08:23.799715", |  "modified": "2021-06-21 16:03:18.834880", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Stock", |  "module": "Stock", | ||||||
|  "name": "Stock Entry Detail", |  "name": "Stock Entry Detail", | ||||||
|  | |||||||
| @ -54,7 +54,7 @@ class TestStockLedgerEntry(unittest.TestCase): | |||||||
| 		) | 		) | ||||||
| 
 | 
 | ||||||
| 		# _Test Item for Reposting transferred from Stores to FG warehouse on 30-04-2020 | 		# _Test Item for Reposting transferred from Stores to FG warehouse on 30-04-2020 | ||||||
| 		make_stock_entry( | 		se = make_stock_entry( | ||||||
| 			item_code="_Test Item for Reposting", | 			item_code="_Test Item for Reposting", | ||||||
| 			source="Stores - _TC", | 			source="Stores - _TC", | ||||||
| 			target="Finished Goods - _TC", | 			target="Finished Goods - _TC", | ||||||
| @ -64,29 +64,29 @@ class TestStockLedgerEntry(unittest.TestCase): | |||||||
| 			posting_date='2020-04-30', | 			posting_date='2020-04-30', | ||||||
| 			posting_time='14:00' | 			posting_time='14:00' | ||||||
| 		) | 		) | ||||||
| 		target_wh_sle = get_previous_sle({ | 		target_wh_sle = frappe.db.get_value('Stock Ledger Entry', { | ||||||
| 			"item_code": "_Test Item for Reposting", | 			"item_code": "_Test Item for Reposting", | ||||||
| 			"warehouse": "Finished Goods - _TC", | 			"warehouse": "Finished Goods - _TC", | ||||||
| 			"posting_date": '2020-04-30', | 			"voucher_type": "Stock Entry", | ||||||
| 			"posting_time": '14:00' | 			"voucher_no": se.name | ||||||
| 		}) | 		}, ["valuation_rate"], as_dict=1) | ||||||
| 
 | 
 | ||||||
| 		self.assertEqual(target_wh_sle.get("valuation_rate"), 150) | 		self.assertEqual(target_wh_sle.get("valuation_rate"), 150) | ||||||
| 
 | 
 | ||||||
| 		# Repack entry on 5-5-2020 | 		# Repack entry on 5-5-2020 | ||||||
| 		repack = create_repack_entry(company=company, posting_date='2020-05-05', posting_time='14:00') | 		repack = create_repack_entry(company=company, posting_date='2020-05-05', posting_time='14:00') | ||||||
| 
 | 
 | ||||||
| 		finished_item_sle = get_previous_sle({ | 		finished_item_sle = frappe.db.get_value('Stock Ledger Entry', { | ||||||
| 			"item_code": "_Test Finished Item for Reposting", | 			"item_code": "_Test Finished Item for Reposting", | ||||||
| 			"warehouse": "Finished Goods - _TC", | 			"warehouse": "Finished Goods - _TC", | ||||||
| 			"posting_date": '2020-05-05', | 			"voucher_type": "Stock Entry", | ||||||
| 			"posting_time": '14:00' | 			"voucher_no": repack.name | ||||||
| 		}) | 		}, ["incoming_rate", "valuation_rate"], as_dict=1) | ||||||
| 		self.assertEqual(finished_item_sle.get("incoming_rate"), 540) | 		self.assertEqual(finished_item_sle.get("incoming_rate"), 540) | ||||||
| 		self.assertEqual(finished_item_sle.get("valuation_rate"), 540) | 		self.assertEqual(finished_item_sle.get("valuation_rate"), 540) | ||||||
| 
 | 
 | ||||||
| 		# Reconciliation for _Test Item for Reposting at Stores on 12-04-2020: Qty = 50, Rate = 150 | 		# Reconciliation for _Test Item for Reposting at Stores on 12-04-2020: Qty = 50, Rate = 150 | ||||||
| 		create_stock_reconciliation( | 		sr = create_stock_reconciliation( | ||||||
| 			item_code="_Test Item for Reposting", | 			item_code="_Test Item for Reposting", | ||||||
| 			warehouse="Stores - _TC", | 			warehouse="Stores - _TC", | ||||||
| 			qty=50, | 			qty=50, | ||||||
| @ -109,12 +109,12 @@ class TestStockLedgerEntry(unittest.TestCase): | |||||||
| 		self.assertEqual(target_wh_sle.get("valuation_rate"), 175) | 		self.assertEqual(target_wh_sle.get("valuation_rate"), 175) | ||||||
| 
 | 
 | ||||||
| 		# Check valuation rate of repacked item after back-dated entry at Stores | 		# Check valuation rate of repacked item after back-dated entry at Stores | ||||||
| 		finished_item_sle = get_previous_sle({ | 		finished_item_sle = frappe.db.get_value('Stock Ledger Entry', { | ||||||
| 			"item_code": "_Test Finished Item for Reposting", | 			"item_code": "_Test Finished Item for Reposting", | ||||||
| 			"warehouse": "Finished Goods - _TC", | 			"warehouse": "Finished Goods - _TC", | ||||||
| 			"posting_date": '2020-05-05', | 			"voucher_type": "Stock Entry", | ||||||
| 			"posting_time": '14:00' | 			"voucher_no": repack.name | ||||||
| 		}) | 		}, ["incoming_rate", "valuation_rate"], as_dict=1) | ||||||
| 		self.assertEqual(finished_item_sle.get("incoming_rate"), 790) | 		self.assertEqual(finished_item_sle.get("incoming_rate"), 790) | ||||||
| 		self.assertEqual(finished_item_sle.get("valuation_rate"), 790) | 		self.assertEqual(finished_item_sle.get("valuation_rate"), 790) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -357,6 +357,7 @@ class StockReconciliation(StockController): | |||||||
| 			if row.current_qty: | 			if row.current_qty: | ||||||
| 				data.actual_qty = -1 * row.current_qty | 				data.actual_qty = -1 * row.current_qty | ||||||
| 				data.qty_after_transaction = flt(row.current_qty) | 				data.qty_after_transaction = flt(row.current_qty) | ||||||
|  | 				data.previous_qty_after_transaction = flt(row.qty) | ||||||
| 				data.valuation_rate = flt(row.current_valuation_rate) | 				data.valuation_rate = flt(row.current_valuation_rate) | ||||||
| 				data.stock_value = data.qty_after_transaction * data.valuation_rate | 				data.stock_value = data.qty_after_transaction * data.valuation_rate | ||||||
| 				data.stock_value_difference = -1 * flt(row.amount_difference) | 				data.stock_value_difference = -1 * flt(row.amount_difference) | ||||||
| @ -404,17 +405,18 @@ class StockReconciliation(StockController): | |||||||
| 
 | 
 | ||||||
| 			key = (d.item_code, d.warehouse) | 			key = (d.item_code, d.warehouse) | ||||||
| 			if key not in merge_similar_entries: | 			if key not in merge_similar_entries: | ||||||
|  | 				d.total_amount = (d.actual_qty * d.valuation_rate) | ||||||
| 				merge_similar_entries[key] = d | 				merge_similar_entries[key] = d | ||||||
| 			elif d.serial_no: | 			elif d.serial_no: | ||||||
| 				data = merge_similar_entries[key] | 				data = merge_similar_entries[key] | ||||||
| 				data.actual_qty += d.actual_qty | 				data.actual_qty += d.actual_qty | ||||||
| 				data.qty_after_transaction += d.qty_after_transaction | 				data.qty_after_transaction += d.qty_after_transaction | ||||||
| 
 | 
 | ||||||
| 				data.valuation_rate = (data.valuation_rate + d.valuation_rate) / data.actual_qty | 				data.total_amount += (d.actual_qty * d.valuation_rate) | ||||||
|  | 				data.valuation_rate = (data.total_amount) / data.actual_qty | ||||||
| 				data.serial_no += '\n' + d.serial_no | 				data.serial_no += '\n' + d.serial_no | ||||||
| 
 | 
 | ||||||
| 				if data.incoming_rate: | 				data.incoming_rate = (data.total_amount) / data.actual_qty | ||||||
| 					data.incoming_rate = (data.incoming_rate + d.incoming_rate) / data.actual_qty |  | ||||||
| 
 | 
 | ||||||
| 		for key, value in merge_similar_entries.items(): | 		for key, value in merge_similar_entries.items(): | ||||||
| 			new_sl_entries.append(value) | 			new_sl_entries.append(value) | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ | |||||||
| 
 | 
 | ||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
| import frappe, unittest | import frappe, unittest | ||||||
| from frappe.utils import flt, nowdate, nowtime | from frappe.utils import flt, nowdate, nowtime, random_string, add_days | ||||||
| from erpnext.accounts.utils import get_stock_and_account_balance | from erpnext.accounts.utils import get_stock_and_account_balance | ||||||
| from erpnext.stock.stock_ledger import get_previous_sle, update_entries_after | from erpnext.stock.stock_ledger import get_previous_sle, update_entries_after | ||||||
| from erpnext.stock.doctype.stock_reconciliation.stock_reconciliation import EmptyStockReconciliationItemsError, get_items | from erpnext.stock.doctype.stock_reconciliation.stock_reconciliation import EmptyStockReconciliationItemsError, get_items | ||||||
| @ -14,6 +14,7 @@ from erpnext.stock.doctype.warehouse.test_warehouse import create_warehouse | |||||||
| from erpnext.stock.doctype.item.test_item import create_item | from erpnext.stock.doctype.item.test_item import create_item | ||||||
| from erpnext.stock.utils import get_incoming_rate, get_stock_value_on, get_valuation_method | from erpnext.stock.utils import get_incoming_rate, get_stock_value_on, get_valuation_method | ||||||
| from erpnext.stock.doctype.serial_no.serial_no import get_serial_nos | from erpnext.stock.doctype.serial_no.serial_no import get_serial_nos | ||||||
|  | from erpnext.stock.doctype.purchase_receipt.test_purchase_receipt import make_purchase_receipt | ||||||
| 
 | 
 | ||||||
| class TestStockReconciliation(unittest.TestCase): | class TestStockReconciliation(unittest.TestCase): | ||||||
| 	@classmethod | 	@classmethod | ||||||
| @ -150,6 +151,42 @@ class TestStockReconciliation(unittest.TestCase): | |||||||
| 			stock_doc = frappe.get_doc("Stock Reconciliation", d) | 			stock_doc = frappe.get_doc("Stock Reconciliation", d) | ||||||
| 			stock_doc.cancel() | 			stock_doc.cancel() | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 	def test_stock_reco_for_merge_serialized_item(self): | ||||||
|  | 		to_delete_records = [] | ||||||
|  | 
 | ||||||
|  | 		# Add new serial nos | ||||||
|  | 		serial_item_code = "Stock-Reco-Serial-Item-2" | ||||||
|  | 		serial_warehouse = "_Test Warehouse for Stock Reco1 - _TC" | ||||||
|  | 
 | ||||||
|  | 		sr = create_stock_reconciliation(item_code=serial_item_code, serial_no=random_string(6), | ||||||
|  | 			warehouse = serial_warehouse, qty=1, rate=100, do_not_submit=True, purpose='Opening Stock') | ||||||
|  | 
 | ||||||
|  | 		for i in range(3): | ||||||
|  | 			sr.append('items', { | ||||||
|  | 				'item_code': serial_item_code, | ||||||
|  | 				'warehouse': serial_warehouse, | ||||||
|  | 				'qty': 1, | ||||||
|  | 				'valuation_rate': 100, | ||||||
|  | 				'serial_no': random_string(6) | ||||||
|  | 			}) | ||||||
|  | 
 | ||||||
|  | 		sr.save() | ||||||
|  | 		sr.submit() | ||||||
|  | 
 | ||||||
|  | 		sle_entries = frappe.get_all('Stock Ledger Entry', filters= {'voucher_no': sr.name}, | ||||||
|  | 			fields = ['name', 'incoming_rate']) | ||||||
|  | 
 | ||||||
|  | 		self.assertEqual(len(sle_entries), 1) | ||||||
|  | 		self.assertEqual(sle_entries[0].incoming_rate, 100) | ||||||
|  | 
 | ||||||
|  | 		to_delete_records.append(sr.name) | ||||||
|  | 		to_delete_records.reverse() | ||||||
|  | 
 | ||||||
|  | 		for d in to_delete_records: | ||||||
|  | 			stock_doc = frappe.get_doc("Stock Reconciliation", d) | ||||||
|  | 			stock_doc.cancel() | ||||||
|  | 
 | ||||||
| 	def test_stock_reco_for_batch_item(self): | 	def test_stock_reco_for_batch_item(self): | ||||||
| 		to_delete_records = [] | 		to_delete_records = [] | ||||||
| 		to_delete_serial_nos = [] | 		to_delete_serial_nos = [] | ||||||
| @ -204,6 +241,117 @@ class TestStockReconciliation(unittest.TestCase): | |||||||
| 		self.assertEqual(sr.get("items")[0].valuation_rate, 0) | 		self.assertEqual(sr.get("items")[0].valuation_rate, 0) | ||||||
| 		self.assertEqual(sr.get("items")[0].amount, 0) | 		self.assertEqual(sr.get("items")[0].amount, 0) | ||||||
| 
 | 
 | ||||||
|  | 	def test_backdated_stock_reco_qty_reposting(self): | ||||||
|  | 		""" | ||||||
|  | 			Test if a backdated stock reco recalculates future qty until next reco. | ||||||
|  | 			------------------------------------------- | ||||||
|  | 			Var		| Doc	|	Qty	| Balance | ||||||
|  | 			------------------------------------------- | ||||||
|  | 			SR5		| Reco	|	0	|	8	(posting date: today-4) [backdated] | ||||||
|  | 			PR1		| PR	|	10	|	18	(posting date: today-3) | ||||||
|  | 			PR2		| PR	|	1	|	19	(posting date: today-2) | ||||||
|  | 			SR4		| Reco	|	0	|	6	(posting date: today-1) [backdated] | ||||||
|  | 			PR3		| PR	|	1	|	7	(posting date: today) # can't post future PR | ||||||
|  | 		""" | ||||||
|  | 		item_code = "Backdated-Reco-Item" | ||||||
|  | 		warehouse = "_Test Warehouse - _TC" | ||||||
|  | 		create_item(item_code) | ||||||
|  | 
 | ||||||
|  | 		pr1 = make_purchase_receipt(item_code=item_code, warehouse=warehouse, qty=10, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -3)) | ||||||
|  | 		pr2 = make_purchase_receipt(item_code=item_code, warehouse=warehouse, qty=1, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -2)) | ||||||
|  | 		pr3 = make_purchase_receipt(item_code=item_code, warehouse=warehouse, qty=1, rate=100, | ||||||
|  | 			posting_date=nowdate()) | ||||||
|  | 
 | ||||||
|  | 		pr1_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr1.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		pr3_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr3.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		self.assertEqual(pr1_balance, 10) | ||||||
|  | 		self.assertEqual(pr3_balance, 12) | ||||||
|  | 
 | ||||||
|  | 		# post backdated stock reco in between | ||||||
|  | 		sr4 = create_stock_reconciliation(item_code=item_code, warehouse=warehouse, qty=6, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -1)) | ||||||
|  | 		pr3_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr3.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		self.assertEqual(pr3_balance, 7) | ||||||
|  | 
 | ||||||
|  | 		# post backdated stock reco at the start | ||||||
|  | 		sr5 = create_stock_reconciliation(item_code=item_code, warehouse=warehouse, qty=8, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -4)) | ||||||
|  | 		pr1_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr1.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		pr2_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr2.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		sr4_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": sr4.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		self.assertEqual(pr1_balance, 18) | ||||||
|  | 		self.assertEqual(pr2_balance, 19) | ||||||
|  | 		self.assertEqual(sr4_balance, 6) # check if future stock reco is unaffected | ||||||
|  | 
 | ||||||
|  | 		# cancel backdated stock reco and check future impact | ||||||
|  | 		sr5.cancel() | ||||||
|  | 		pr1_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr1.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		pr2_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr2.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		sr4_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": sr4.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		self.assertEqual(pr1_balance, 10) | ||||||
|  | 		self.assertEqual(pr2_balance, 11) | ||||||
|  | 		self.assertEqual(sr4_balance, 6) # check if future stock reco is unaffected | ||||||
|  | 
 | ||||||
|  | 		# teardown | ||||||
|  | 		sr4.cancel() | ||||||
|  | 		pr3.cancel() | ||||||
|  | 		pr2.cancel() | ||||||
|  | 		pr1.cancel() | ||||||
|  | 
 | ||||||
|  | 	def test_backdated_stock_reco_future_negative_stock(self): | ||||||
|  | 		""" | ||||||
|  | 			Test if a backdated stock reco causes future negative stock and is blocked. | ||||||
|  | 			------------------------------------------- | ||||||
|  | 			Var		| Doc	|	Qty	| Balance | ||||||
|  | 			------------------------------------------- | ||||||
|  | 			PR1		| PR	|	10	|	10		(posting date: today-2) | ||||||
|  | 			SR3		| Reco	|	0	|	1		(posting date: today-1) [backdated & blocked] | ||||||
|  | 			DN2		| DN	|	-2	|	8(-1)	(posting date: today) | ||||||
|  | 		""" | ||||||
|  | 		from erpnext.stock.stock_ledger import NegativeStockError | ||||||
|  | 		from erpnext.stock.doctype.delivery_note.test_delivery_note import create_delivery_note | ||||||
|  | 
 | ||||||
|  | 		item_code = "Backdated-Reco-Item" | ||||||
|  | 		warehouse = "_Test Warehouse - _TC" | ||||||
|  | 		create_item(item_code) | ||||||
|  | 
 | ||||||
|  | 		negative_stock_setting = frappe.db.get_single_value("Stock Settings", "allow_negative_stock") | ||||||
|  | 		frappe.db.set_value("Stock Settings", None, "allow_negative_stock", 0) | ||||||
|  | 
 | ||||||
|  | 		pr1 = make_purchase_receipt(item_code=item_code, warehouse=warehouse, qty=10, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -2)) | ||||||
|  | 		dn2 = create_delivery_note(item_code=item_code, warehouse=warehouse, qty=2, rate=120, | ||||||
|  | 			posting_date=nowdate()) | ||||||
|  | 
 | ||||||
|  | 		pr1_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": pr1.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		dn2_balance = frappe.db.get_value("Stock Ledger Entry", {"voucher_no": dn2.name, "is_cancelled": 0}, | ||||||
|  | 			"qty_after_transaction") | ||||||
|  | 		self.assertEqual(pr1_balance, 10) | ||||||
|  | 		self.assertEqual(dn2_balance, 8) | ||||||
|  | 
 | ||||||
|  | 		# check if stock reco is blocked | ||||||
|  | 		sr3 = create_stock_reconciliation(item_code=item_code, warehouse=warehouse, qty=1, rate=100, | ||||||
|  | 			posting_date=add_days(nowdate(), -1), do_not_submit=True) | ||||||
|  | 		self.assertRaises(NegativeStockError, sr3.submit) | ||||||
|  | 
 | ||||||
|  | 		# teardown | ||||||
|  | 		frappe.db.set_value("Stock Settings", None, "allow_negative_stock", negative_stock_setting) | ||||||
|  | 		sr3.cancel() | ||||||
|  | 		dn2.cancel() | ||||||
|  | 		pr1.cancel() | ||||||
|  | 
 | ||||||
| def insert_existing_sle(warehouse): | def insert_existing_sle(warehouse): | ||||||
| 	from erpnext.stock.doctype.stock_entry.test_stock_entry import make_stock_entry | 	from erpnext.stock.doctype.stock_entry.test_stock_entry import make_stock_entry | ||||||
| 
 | 
 | ||||||
| @ -231,6 +379,12 @@ def create_batch_or_serial_no_items(): | |||||||
| 		serial_item_doc.serial_no_series = "SRSI.####" | 		serial_item_doc.serial_no_series = "SRSI.####" | ||||||
| 		serial_item_doc.save(ignore_permissions=True) | 		serial_item_doc.save(ignore_permissions=True) | ||||||
| 
 | 
 | ||||||
|  | 	serial_item_doc = create_item("Stock-Reco-Serial-Item-2", is_stock_item=1) | ||||||
|  | 	if not serial_item_doc.has_serial_no: | ||||||
|  | 		serial_item_doc.has_serial_no = 1 | ||||||
|  | 		serial_item_doc.serial_no_series = "SRSII.####" | ||||||
|  | 		serial_item_doc.save(ignore_permissions=True) | ||||||
|  | 
 | ||||||
| 	batch_item_doc = create_item("Stock-Reco-batch-Item-1", is_stock_item=1) | 	batch_item_doc = create_item("Stock-Reco-batch-Item-1", is_stock_item=1) | ||||||
| 	if not batch_item_doc.has_batch_no: | 	if not batch_item_doc.has_batch_no: | ||||||
| 		batch_item_doc.has_batch_no = 1 | 		batch_item_doc.has_batch_no = 1 | ||||||
|  | |||||||
| @ -23,7 +23,10 @@ | |||||||
|   "allow_negative_stock", |   "allow_negative_stock", | ||||||
|   "show_barcode_field", |   "show_barcode_field", | ||||||
|   "clean_description_html", |   "clean_description_html", | ||||||
|  |   "quality_inspection_settings_section", | ||||||
|   "action_if_quality_inspection_is_not_submitted", |   "action_if_quality_inspection_is_not_submitted", | ||||||
|  |   "column_break_21", | ||||||
|  |   "action_if_quality_inspection_is_rejected", | ||||||
|   "section_break_7", |   "section_break_7", | ||||||
|   "automatically_set_serial_nos_based_on_fifo", |   "automatically_set_serial_nos_based_on_fifo", | ||||||
|   "set_qty_in_transactions_based_on_serial_no_input", |   "set_qty_in_transactions_based_on_serial_no_input", | ||||||
| @ -264,6 +267,22 @@ | |||||||
|   { |   { | ||||||
|    "fieldname": "column_break_31", |    "fieldname": "column_break_31", | ||||||
|    "fieldtype": "Column Break" |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "quality_inspection_settings_section", | ||||||
|  |    "fieldtype": "Section Break", | ||||||
|  |    "label": "Quality Inspection Settings" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "fieldname": "column_break_21", | ||||||
|  |    "fieldtype": "Column Break" | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "default": "Stop", | ||||||
|  |    "fieldname": "action_if_quality_inspection_is_rejected", | ||||||
|  |    "fieldtype": "Select", | ||||||
|  |    "label": "Action If Quality Inspection Is Rejected", | ||||||
|  |    "options": "Stop\nWarn" | ||||||
|   } |   } | ||||||
|  ], |  ], | ||||||
|  "icon": "icon-cog", |  "icon": "icon-cog", | ||||||
| @ -271,7 +290,7 @@ | |||||||
|  "index_web_pages_for_search": 1, |  "index_web_pages_for_search": 1, | ||||||
|  "issingle": 1, |  "issingle": 1, | ||||||
|  "links": [], |  "links": [], | ||||||
|  "modified": "2021-04-30 17:27:42.709231", |  "modified": "2021-07-10 16:17:42.159829", | ||||||
|  "modified_by": "Administrator", |  "modified_by": "Administrator", | ||||||
|  "module": "Stock", |  "module": "Stock", | ||||||
|  "name": "Stock Settings", |  "name": "Stock Settings", | ||||||
|  | |||||||
| @ -55,6 +55,11 @@ def make_sl_entries(sl_entries, allow_negative_stock=False, via_landed_cost_vouc | |||||||
| 				sle_doc = make_entry(sle, allow_negative_stock, via_landed_cost_voucher) | 				sle_doc = make_entry(sle, allow_negative_stock, via_landed_cost_voucher) | ||||||
| 
 | 
 | ||||||
| 			args = sle_doc.as_dict() | 			args = sle_doc.as_dict() | ||||||
|  | 
 | ||||||
|  | 			if sle.get("voucher_type") == "Stock Reconciliation": | ||||||
|  | 				# preserve previous_qty_after_transaction for qty reposting | ||||||
|  | 				args.previous_qty_after_transaction = sle.get("previous_qty_after_transaction") | ||||||
|  | 
 | ||||||
| 			update_bin(args, allow_negative_stock, via_landed_cost_voucher) | 			update_bin(args, allow_negative_stock, via_landed_cost_voucher) | ||||||
| 
 | 
 | ||||||
| def get_args_for_future_sle(row): | def get_args_for_future_sle(row): | ||||||
| @ -215,7 +220,7 @@ class update_entries_after(object): | |||||||
| 		""" | 		""" | ||||||
| 		self.data.setdefault(args.warehouse, frappe._dict()) | 		self.data.setdefault(args.warehouse, frappe._dict()) | ||||||
| 		warehouse_dict = self.data[args.warehouse] | 		warehouse_dict = self.data[args.warehouse] | ||||||
| 		previous_sle = self.get_previous_sle_of_current_voucher(args) | 		previous_sle = get_previous_sle_of_current_voucher(args) | ||||||
| 		warehouse_dict.previous_sle = previous_sle | 		warehouse_dict.previous_sle = previous_sle | ||||||
| 
 | 
 | ||||||
| 		for key in ("qty_after_transaction", "valuation_rate", "stock_value"): | 		for key in ("qty_after_transaction", "valuation_rate", "stock_value"): | ||||||
| @ -227,29 +232,6 @@ class update_entries_after(object): | |||||||
| 			"stock_value_difference": 0.0 | 			"stock_value_difference": 0.0 | ||||||
| 		}) | 		}) | ||||||
| 
 | 
 | ||||||
| 	def get_previous_sle_of_current_voucher(self, args): |  | ||||||
| 		"""get stock ledger entries filtered by specific posting datetime conditions""" |  | ||||||
| 
 |  | ||||||
| 		args['time_format'] = '%H:%i:%s' |  | ||||||
| 		if not args.get("posting_date"): |  | ||||||
| 			args["posting_date"] = "1900-01-01" |  | ||||||
| 		if not args.get("posting_time"): |  | ||||||
| 			args["posting_time"] = "00:00" |  | ||||||
| 
 |  | ||||||
| 		sle = frappe.db.sql(""" |  | ||||||
| 			select *, timestamp(posting_date, posting_time) as "timestamp" |  | ||||||
| 			from `tabStock Ledger Entry` |  | ||||||
| 			where item_code = %(item_code)s |  | ||||||
| 				and warehouse = %(warehouse)s |  | ||||||
| 				and is_cancelled = 0 |  | ||||||
| 				and timestamp(posting_date, time_format(posting_time, %(time_format)s)) < timestamp(%(posting_date)s, time_format(%(posting_time)s, %(time_format)s)) |  | ||||||
| 			order by timestamp(posting_date, posting_time) desc, creation desc |  | ||||||
| 			limit 1 |  | ||||||
| 			for update""", args, as_dict=1) |  | ||||||
| 
 |  | ||||||
| 		return sle[0] if sle else frappe._dict() |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 	def build(self): | 	def build(self): | ||||||
| 		from erpnext.controllers.stock_controller import future_sle_exists | 		from erpnext.controllers.stock_controller import future_sle_exists | ||||||
| 
 | 
 | ||||||
| @ -734,6 +716,35 @@ class update_entries_after(object): | |||||||
| 			bin_doc.flags.via_stock_ledger_entry = True | 			bin_doc.flags.via_stock_ledger_entry = True | ||||||
| 			bin_doc.save(ignore_permissions=True) | 			bin_doc.save(ignore_permissions=True) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | def get_previous_sle_of_current_voucher(args, exclude_current_voucher=False): | ||||||
|  | 	"""get stock ledger entries filtered by specific posting datetime conditions""" | ||||||
|  | 
 | ||||||
|  | 	args['time_format'] = '%H:%i:%s' | ||||||
|  | 	if not args.get("posting_date"): | ||||||
|  | 		args["posting_date"] = "1900-01-01" | ||||||
|  | 	if not args.get("posting_time"): | ||||||
|  | 		args["posting_time"] = "00:00" | ||||||
|  | 
 | ||||||
|  | 	voucher_condition = "" | ||||||
|  | 	if exclude_current_voucher: | ||||||
|  | 		voucher_no = args.get("voucher_no") | ||||||
|  | 		voucher_condition = f"and voucher_no != '{voucher_no}'" | ||||||
|  | 
 | ||||||
|  | 	sle = frappe.db.sql(""" | ||||||
|  | 		select *, timestamp(posting_date, posting_time) as "timestamp" | ||||||
|  | 		from `tabStock Ledger Entry` | ||||||
|  | 		where item_code = %(item_code)s | ||||||
|  | 			and warehouse = %(warehouse)s | ||||||
|  | 			and is_cancelled = 0 | ||||||
|  | 			{voucher_condition} | ||||||
|  | 			and timestamp(posting_date, time_format(posting_time, %(time_format)s)) < timestamp(%(posting_date)s, time_format(%(posting_time)s, %(time_format)s)) | ||||||
|  | 		order by timestamp(posting_date, posting_time) desc, creation desc | ||||||
|  | 		limit 1 | ||||||
|  | 		for update""".format(voucher_condition=voucher_condition), args, as_dict=1) | ||||||
|  | 
 | ||||||
|  | 	return sle[0] if sle else frappe._dict() | ||||||
|  | 
 | ||||||
| def get_previous_sle(args, for_update=False): | def get_previous_sle(args, for_update=False): | ||||||
| 	""" | 	""" | ||||||
| 		get the last sle on or before the current time-bucket, | 		get the last sle on or before the current time-bucket, | ||||||
| @ -862,9 +873,24 @@ def get_valuation_rate(item_code, warehouse, voucher_type, voucher_no, | |||||||
| 	return valuation_rate | 	return valuation_rate | ||||||
| 
 | 
 | ||||||
| def update_qty_in_future_sle(args, allow_negative_stock=None): | def update_qty_in_future_sle(args, allow_negative_stock=None): | ||||||
|  | 	"""Recalculate Qty after Transaction in future SLEs based on current SLE.""" | ||||||
|  | 	datetime_limit_condition = "" | ||||||
|  | 	qty_shift = args.actual_qty | ||||||
|  | 
 | ||||||
|  | 	# find difference/shift in qty caused by stock reconciliation | ||||||
|  | 	if args.voucher_type == "Stock Reconciliation": | ||||||
|  | 		qty_shift = get_stock_reco_qty_shift(args) | ||||||
|  | 
 | ||||||
|  | 	# find the next nearest stock reco so that we only recalculate SLEs till that point | ||||||
|  | 	next_stock_reco_detail = get_next_stock_reco(args) | ||||||
|  | 	if next_stock_reco_detail: | ||||||
|  | 		detail = next_stock_reco_detail[0] | ||||||
|  | 		# add condition to update SLEs before this date & time | ||||||
|  | 		datetime_limit_condition = get_datetime_limit_condition(detail) | ||||||
|  | 
 | ||||||
| 	frappe.db.sql(""" | 	frappe.db.sql(""" | ||||||
| 		update `tabStock Ledger Entry` | 		update `tabStock Ledger Entry` | ||||||
| 		set qty_after_transaction = qty_after_transaction + {qty} | 		set qty_after_transaction = qty_after_transaction + {qty_shift} | ||||||
| 		where | 		where | ||||||
| 			item_code = %(item_code)s | 			item_code = %(item_code)s | ||||||
| 			and warehouse = %(warehouse)s | 			and warehouse = %(warehouse)s | ||||||
| @ -876,15 +902,70 @@ def update_qty_in_future_sle(args, allow_negative_stock=None): | |||||||
| 					and creation > %(creation)s | 					and creation > %(creation)s | ||||||
| 				) | 				) | ||||||
| 			) | 			) | ||||||
| 	""".format(qty=args.actual_qty), args) | 		{datetime_limit_condition} | ||||||
|  | 		""".format(qty_shift=qty_shift, datetime_limit_condition=datetime_limit_condition), args) | ||||||
| 
 | 
 | ||||||
| 	validate_negative_qty_in_future_sle(args, allow_negative_stock) | 	validate_negative_qty_in_future_sle(args, allow_negative_stock) | ||||||
| 
 | 
 | ||||||
|  | def get_stock_reco_qty_shift(args): | ||||||
|  | 	stock_reco_qty_shift = 0 | ||||||
|  | 	if args.get("is_cancelled"): | ||||||
|  | 		if args.get("previous_qty_after_transaction"): | ||||||
|  | 			# get qty (balance) that was set at submission | ||||||
|  | 			last_balance = args.get("previous_qty_after_transaction") | ||||||
|  | 			stock_reco_qty_shift = flt(args.qty_after_transaction) - flt(last_balance) | ||||||
|  | 		else: | ||||||
|  | 			stock_reco_qty_shift = flt(args.actual_qty) | ||||||
|  | 	else: | ||||||
|  | 		# reco is being submitted | ||||||
|  | 		last_balance = get_previous_sle_of_current_voucher(args, | ||||||
|  | 			exclude_current_voucher=True).get("qty_after_transaction") | ||||||
|  | 
 | ||||||
|  | 		if last_balance is not None: | ||||||
|  | 			stock_reco_qty_shift = flt(args.qty_after_transaction) - flt(last_balance) | ||||||
|  | 		else: | ||||||
|  | 			stock_reco_qty_shift = args.qty_after_transaction | ||||||
|  | 
 | ||||||
|  | 	return stock_reco_qty_shift | ||||||
|  | 
 | ||||||
|  | def get_next_stock_reco(args): | ||||||
|  | 	"""Returns next nearest stock reconciliaton's details.""" | ||||||
|  | 
 | ||||||
|  | 	return frappe.db.sql(""" | ||||||
|  | 		select | ||||||
|  | 			name, posting_date, posting_time, creation, voucher_no | ||||||
|  | 		from | ||||||
|  | 			`tabStock Ledger Entry` | ||||||
|  | 		where | ||||||
|  | 			item_code = %(item_code)s | ||||||
|  | 			and warehouse = %(warehouse)s | ||||||
|  | 			and voucher_type = 'Stock Reconciliation' | ||||||
|  | 			and voucher_no != %(voucher_no)s | ||||||
|  | 			and is_cancelled = 0 | ||||||
|  | 			and (timestamp(posting_date, posting_time) > timestamp(%(posting_date)s, %(posting_time)s) | ||||||
|  | 				or ( | ||||||
|  | 					timestamp(posting_date, posting_time) = timestamp(%(posting_date)s, %(posting_time)s) | ||||||
|  | 					and creation > %(creation)s | ||||||
|  | 				) | ||||||
|  | 			) | ||||||
|  | 		limit 1 | ||||||
|  | 	""", args, as_dict=1) | ||||||
|  | 
 | ||||||
|  | def get_datetime_limit_condition(detail): | ||||||
|  | 	return f""" | ||||||
|  | 		and | ||||||
|  | 		(timestamp(posting_date, posting_time) < timestamp('{detail.posting_date}', '{detail.posting_time}') | ||||||
|  | 			or ( | ||||||
|  | 				timestamp(posting_date, posting_time) = timestamp('{detail.posting_date}', '{detail.posting_time}') | ||||||
|  | 				and creation < '{detail.creation}' | ||||||
|  | 			) | ||||||
|  | 		)""" | ||||||
|  | 
 | ||||||
| def validate_negative_qty_in_future_sle(args, allow_negative_stock=None): | def validate_negative_qty_in_future_sle(args, allow_negative_stock=None): | ||||||
| 	allow_negative_stock = allow_negative_stock \ | 	allow_negative_stock = allow_negative_stock \ | ||||||
| 		or cint(frappe.db.get_single_value("Stock Settings", "allow_negative_stock")) | 		or cint(frappe.db.get_single_value("Stock Settings", "allow_negative_stock")) | ||||||
| 
 | 
 | ||||||
| 	if args.actual_qty < 0 and not allow_negative_stock: | 	if (args.actual_qty < 0 or args.voucher_type == "Stock Reconciliation") and not allow_negative_stock: | ||||||
| 		sle = get_future_sle_with_negative_qty(args) | 		sle = get_future_sle_with_negative_qty(args) | ||||||
| 		if sle: | 		if sle: | ||||||
| 			message = _("{0} units of {1} needed in {2} on {3} {4} for {5} to complete this transaction.").format( | 			message = _("{0} units of {1} needed in {2} on {3} {4} for {5} to complete this transaction.").format( | ||||||
|  | |||||||
| @ -1,28 +1,54 @@ | |||||||
| {% if doc.status=="Open" %} | {% if doc.status == "Open" %} | ||||||
| <div class="web-list-item"> |   <div class="web-list-item transaction-list-item"> | ||||||
| 	<a class="no-decoration" href="/projects?project={{ doc.name | urlencode }}"> |     <div class="row"> | ||||||
| 		<div class="row"> |       <div class="col-xs-2"> | ||||||
| 			<div class="col-xs-6"> |         <a class="transaction-item-link" href="/projects?project={{ doc.name | urlencode }}">Link</a> | ||||||
| 
 |         {{ doc.name }} | ||||||
| 				{{ doc.name }} |       </div> | ||||||
| 			</div> |       <div class="col-xs-2"> | ||||||
| 			<div class="col-xs-3"> |         {{ doc.project_name }} | ||||||
| 				{% if doc.percent_complete %} |       </div> | ||||||
| 					<div class="progress" style="margin-bottom: 0!important; margin-top: 10px!important; height:5px;"> |       <div class="col-xs-3 text-center"> | ||||||
| 					  <div class="progress-bar progress-bar-{{ "warning" if doc.percent_complete|round < 100 else "success"}}" role="progressbar" |         {% if doc.percent_complete %} | ||||||
| 					  	aria-valuenow="{{ doc.percent_complete|round|int }}" |           {% set pill_class = "green" if doc.percent_complete | round == 100 else | ||||||
| 					  	aria-valuemin="0" aria-valuemax="100" style="width:{{ doc.percent_complete|round|int }}%;"> |             "orange" %} | ||||||
| 					  </div> |           <div class="ellipsis"> | ||||||
| 					</div> |             <span class="indicator-pill {{ pill_class }} filterable ellipsis"> | ||||||
| 				{% else %} |               <span>{{ frappe.utils.cint(doc.percent_complete) }} | ||||||
| 					<span class="indicator {{ "red" if doc.status=="Open" else "gray"  }}"> |                 %</span> | ||||||
| 						{{ doc.status }}</span> |             </span> | ||||||
| 				{% endif %} |           </div> | ||||||
| 			</div> |         {% else %} | ||||||
| 			<div class="col-xs-3 text-right small text-muted"> |           <span class="indicator-pill {{ " red" if doc.status=="Open" else " darkgrey" }}"> | ||||||
| 				{{ frappe.utils.pretty_date(doc.modified) }} |             {{ doc.status }}</span> | ||||||
| 			</div> |         {% endif %} | ||||||
| 		</div> |       </div> | ||||||
| 	</a> |       {% if doc["_assign"] %} | ||||||
| </div> |         {% set assigned_users = json.loads(doc["_assign"])%} | ||||||
|  |         <div class="col-xs-2"> | ||||||
|  |           {% for user in assigned_users %} | ||||||
|  |             {% set user_details = frappe | ||||||
|  |               .db | ||||||
|  |               .get_value("User", user, [ | ||||||
|  |                 "full_name", "user_image" | ||||||
|  |               ], as_dict = True) %} | ||||||
|  |             {% if user_details.user_image %} | ||||||
|  |               <span class="avatar avatar-small" style="width:32px; height:32px;" title="{{ user_details.full_name }}"> | ||||||
|  |                 <img src="{{ user_details.user_image }}"> | ||||||
|  |               </span> | ||||||
|  |             {% else %} | ||||||
|  |               <span class="avatar avatar-small" style="width:32px; height:32px;" title="{{ user_details.full_name }}"> | ||||||
|  |                 <div class='standard-image' style="background-color: #F5F4F4; color: #000;"> | ||||||
|  |                   {{ frappe.utils.get_abbr(user_details.full_name) }} | ||||||
|  |                 </div> | ||||||
|  |               </span> | ||||||
|  |             {% endif %} | ||||||
|  |           {% endfor %} | ||||||
|  |         </div> | ||||||
|  |       {% endif %} | ||||||
|  |       <div class="col-xs-3 text-right small text-muted"> | ||||||
|  |         {{ frappe.utils.pretty_date(doc.modified) }} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
| {% endif %} | {% endif %} | ||||||
|  | |||||||
| @ -1,32 +1,5 @@ | |||||||
| {% for task in doc.tasks %} | {% for task in doc.tasks %} | ||||||
| 	<div class='task'> |   <div class="web-list-item transaction-list-item"> | ||||||
| 		<a class="no-decoration task-link {{ task.css_seen }}" href="/tasks?name={{ task.name }}"> |     {{ task_row(task, 0) }} | ||||||
| 		<div class='row project-item'> |   </div> | ||||||
| 			<div class='col-xs-9'> |  | ||||||
| 				<span class="indicator {{ "red" if task.status=="Open" else "green" if task.status=="Closed" else "gray" }}" title="{{ task.status }}"  > {{ task.subject }}</span> |  | ||||||
| 	 				<div class="small text-muted item-timestamp" |  | ||||||
| 	 					title="{{ frappe.utils.pretty_date(task.modified) }}"> |  | ||||||
| 						{{ _("modified") }} {{ frappe.utils.pretty_date(task.modified) }} |  | ||||||
| 	 				</div> |  | ||||||
| 			</div> |  | ||||||
| 			<div class='col-xs-1'>{% if task.todo %} |  | ||||||
| 					{% if task.todo.user_image %} |  | ||||||
| 						<span class="avatar avatar-small" title="{{ task.todo.owner }}"> |  | ||||||
| 							<img src="{{ task.todo.user_image }}"> |  | ||||||
| 						</span> |  | ||||||
| 					{% else %} |  | ||||||
| 						<span class="avatar avatar-small standard-image" title="Assigned to {{ task.todo.owner }}"> |  | ||||||
| 
 |  | ||||||
| 						</span> |  | ||||||
| 					{% endif %} |  | ||||||
| 				{% endif %}	 </div> |  | ||||||
| 			<div class='col-xs-2'> |  | ||||||
| 				<span class="pull-right list-comment-count small {{ "text-extra-muted" if task.comment_count==0 else "text-muted" }}"> |  | ||||||
| 					<i class="octicon octicon-comment-discussion"></i> |  | ||||||
| 						{{ task.comment_count }} |  | ||||||
| 				</span> |  | ||||||
| 			</div> |  | ||||||
| 		</div> |  | ||||||
| 		</a> |  | ||||||
| 	</div> |  | ||||||
| {% endfor %} | {% endfor %} | ||||||
|  | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user