diff options
| author | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
|---|---|---|
| committer | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
| commit | 3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch) | |
| tree | a44932296ef4a9b71d5f010906253d8c53727726 /addons/purchase/tests | |
| parent | 0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff) | |
initial commit 2
Diffstat (limited to 'addons/purchase/tests')
| -rw-r--r-- | addons/purchase/tests/__init__.py | 7 | ||||
| -rw-r--r-- | addons/purchase/tests/test_access_rights.py | 160 | ||||
| -rw-r--r-- | addons/purchase/tests/test_purchase.py | 205 | ||||
| -rw-r--r-- | addons/purchase/tests/test_purchase_invoice.py | 301 | ||||
| -rw-r--r-- | addons/purchase/tests/test_purchase_order_report.py | 68 |
5 files changed, 741 insertions, 0 deletions
diff --git a/addons/purchase/tests/__init__.py b/addons/purchase/tests/__init__.py new file mode 100644 index 00000000..b97052ae --- /dev/null +++ b/addons/purchase/tests/__init__.py @@ -0,0 +1,7 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +from . import test_purchase +from . import test_purchase_order_report +from . import test_purchase_invoice +from . import test_access_rights diff --git a/addons/purchase/tests/test_access_rights.py b/addons/purchase/tests/test_access_rights.py new file mode 100644 index 00000000..855b0270 --- /dev/null +++ b/addons/purchase/tests/test_access_rights.py @@ -0,0 +1,160 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +from odoo.addons.account.tests.common import AccountTestInvoicingCommon +from odoo.tests import Form, tagged +from odoo.exceptions import AccessError + + +@tagged('post_install', '-at_install') +class TestPurchaseInvoice(AccountTestInvoicingCommon): + + @classmethod + def setUpClass(cls, chart_template_ref=None): + super().setUpClass(chart_template_ref=chart_template_ref) + + # Create a users + group_purchase_user = cls.env.ref('purchase.group_purchase_user') + group_employee = cls.env.ref('base.group_user') + group_partner_manager = cls.env.ref('base.group_partner_manager') + + cls.purchase_user = cls.env['res.users'].with_context( + no_reset_password=True + ).create({ + 'name': 'Purchase user', + 'login': 'purchaseUser', + 'email': 'pu@odoo.com', + 'groups_id': [(6, 0, [group_purchase_user.id, group_employee.id, group_partner_manager.id])], + }) + + cls.vendor = cls.env['res.partner'].create({ + 'name': 'Supplier', + 'email': 'supplier.serv@supercompany.com', + }) + + user_type_expense = cls.env.ref('account.data_account_type_expenses') + cls.account_expense_product = cls.env['account.account'].create({ + 'code': 'EXPENSE_PROD111', + 'name': 'Expense - Test Account', + 'user_type_id': user_type_expense.id, + }) + # Create category + cls.product_category = cls.env['product.category'].create({ + 'name': 'Product Category with Expense account', + 'property_account_expense_categ_id': cls.account_expense_product.id + }) + cls.product = cls.env['product.product'].create({ + 'name': "Product", + 'standard_price': 200.0, + 'list_price': 180.0, + 'type': 'service', + }) + + def test_create_purchase_order(self): + """Check a purchase user can create a vendor bill from a purchase order but not post it""" + purchase_order_form = Form(self.env['purchase.order'].with_user(self.purchase_user)) + purchase_order_form.partner_id = self.vendor + with purchase_order_form.order_line.new() as line: + line.name = self.product.name + line.product_id = self.product + line.product_qty = 4 + line.price_unit = 5 + + purchase_order = purchase_order_form.save() + purchase_order.button_confirm() + + purchase_order.order_line.qty_received = 4 + purchase_order.action_create_invoice() + invoice = purchase_order.invoice_ids + with self.assertRaises(AccessError): + invoice.action_post() + + def test_read_purchase_order(self): + """ Check that a purchase user can read all purchase order and 'in' invoices""" + purchase_user_2 = self.purchase_user.copy({ + 'name': 'Purchase user 2', + 'login': 'purchaseUser2', + 'email': 'pu2@odoo.com', + }) + + purchase_order_form = Form(self.env['purchase.order'].with_user(purchase_user_2)) + purchase_order_form.partner_id = self.vendor + with purchase_order_form.order_line.new() as line: + line.name = self.product.name + line.product_id = self.product + line.product_qty = 4 + line.price_unit = 5 + + purchase_order_user2 = purchase_order_form.save() + purchase_order_user2.button_confirm() + + purchase_order_user2.order_line.qty_received = 4 + purchase_order_user2.action_create_invoice() + vendor_bill_user2 = purchase_order_user2.invoice_ids + + # open purchase_order_user2 and vendor_bill_user2 with `self.purchase_user` + purchase_order_user1 = Form(purchase_order_user2.with_user(self.purchase_user)) + purchase_order_user1 = purchase_order_user1.save() + vendor_bill_user1 = Form(vendor_bill_user2.with_user(self.purchase_user)) + vendor_bill_user1 = vendor_bill_user1.save() + + def test_read_purchase_order_2(self): + """ Check that a 2 purchase users with open the vendor bill the same + way even with a 'own documents only' record rule. """ + + # edit the account.move record rule for purchase user in order to ensure + # a user can only see his own invoices + rule = self.env.ref('purchase.purchase_user_account_move_rule') + rule.domain_force = "['&', ('move_type', 'in', ('in_invoice', 'in_refund', 'in_receipt')), ('invoice_user_id', '=', user.id)]" + + # create a purchase and make a vendor bill from it as purchase user 2 + purchase_user_2 = self.purchase_user.copy({ + 'name': 'Purchase user 2', + 'login': 'purchaseUser2', + 'email': 'pu2@odoo.com', + }) + + purchase_order_form = Form(self.env['purchase.order'].with_user(purchase_user_2)) + purchase_order_form.partner_id = self.vendor + with purchase_order_form.order_line.new() as line: + line.name = self.product.name + line.product_id = self.product + line.product_qty = 4 + line.price_unit = 5 + + purchase_order_user2 = purchase_order_form.save() + purchase_order_user2.button_confirm() + + purchase_order_user2.order_line.qty_received = 4 + purchase_order_user2.action_create_invoice() + vendor_bill_user2 = purchase_order_user2.invoice_ids + + # check user 1 cannot read the invoice + with self.assertRaises(AccessError): + Form(vendor_bill_user2.with_user(self.purchase_user)) + + # Check that calling 'action_view_invoice' return the same action despite the record rule + action_user_1 = purchase_order_user2.with_user(self.purchase_user).action_view_invoice() + purchase_order_user2.invalidate_cache() + action_user_2 = purchase_order_user2.with_user(purchase_user_2).action_view_invoice() + self.assertEqual(action_user_1, action_user_2) + + def test_double_validation(self): + """Only purchase managers can approve a purchase order when double + validation is enabled""" + group_purchase_manager = self.env.ref('purchase.group_purchase_manager') + order = self.env.ref("purchase.purchase_order_1") + company = order.sudo().company_id + company.po_double_validation = 'two_step' + company.po_double_validation_amount = 0 + self.purchase_user.write({ + 'company_ids': [(4, company.id)], + 'company_id': company.id, + 'groups_id': [(3, group_purchase_manager.id)], + }) + order.with_user(self.purchase_user).button_confirm() + self.assertEqual(order.state, 'to approve') + order.with_user(self.purchase_user).button_approve() + self.assertEqual(order.state, 'to approve') + self.purchase_user.groups_id += group_purchase_manager + order.with_user(self.purchase_user).button_approve() + self.assertEqual(order.state, 'purchase') diff --git a/addons/purchase/tests/test_purchase.py b/addons/purchase/tests/test_purchase.py new file mode 100644 index 00000000..8571f351 --- /dev/null +++ b/addons/purchase/tests/test_purchase.py @@ -0,0 +1,205 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +from odoo.addons.account.tests.common import AccountTestInvoicingCommon +from odoo.tests import tagged, Form +from odoo import fields + + +from datetime import timedelta + + +@tagged('-at_install', 'post_install') +class TestPurchase(AccountTestInvoicingCommon): + + def test_date_planned(self): + """Set a date planned on 2 PO lines. Check that the PO date_planned is the earliest PO line date + planned. Change one of the dates so it is even earlier and check that the date_planned is set to + this earlier date. + """ + po = Form(self.env['purchase.order']) + po.partner_id = self.partner_a + with po.order_line.new() as po_line: + po_line.product_id = self.product_a + po_line.product_qty = 1 + po_line.price_unit = 100 + with po.order_line.new() as po_line: + po_line.product_id = self.product_b + po_line.product_qty = 10 + po_line.price_unit = 200 + po = po.save() + + # Check that the same date is planned on both PO lines. + self.assertNotEqual(po.order_line[0].date_planned, False) + self.assertAlmostEqual(po.order_line[0].date_planned, po.order_line[1].date_planned, delta=timedelta(seconds=10)) + self.assertAlmostEqual(po.order_line[0].date_planned, po.date_planned, delta=timedelta(seconds=10)) + + orig_date_planned = po.order_line[0].date_planned + + # Set an earlier date planned on a PO line and check that the PO expected date matches it. + new_date_planned = orig_date_planned - timedelta(hours=1) + po.order_line[0].date_planned = new_date_planned + self.assertAlmostEqual(po.order_line[0].date_planned, po.date_planned, delta=timedelta(seconds=10)) + + # Set an even earlier date planned on the other PO line and check that the PO expected date matches it. + new_date_planned = orig_date_planned - timedelta(hours=72) + po.order_line[1].date_planned = new_date_planned + self.assertAlmostEqual(po.order_line[1].date_planned, po.date_planned, delta=timedelta(seconds=10)) + + def test_purchase_order_sequence(self): + PurchaseOrder = self.env['purchase.order'].with_context(tracking_disable=True) + company = self.env.user.company_id + self.env['ir.sequence'].search([ + ('code', '=', 'purchase.order'), + ]).write({ + 'use_date_range': True, 'prefix': 'PO/%(range_year)s/', + }) + vals = { + 'partner_id': self.partner_a.id, + 'company_id': company.id, + 'currency_id': company.currency_id.id, + 'date_order': '2019-01-01', + } + purchase_order = PurchaseOrder.create(vals.copy()) + self.assertTrue(purchase_order.name.startswith('PO/2019/')) + vals['date_order'] = '2020-01-01' + purchase_order = PurchaseOrder.create(vals.copy()) + self.assertTrue(purchase_order.name.startswith('PO/2020/')) + # In EU/BXL tz, this is actually already 01/01/2020 + vals['date_order'] = '2019-12-31 23:30:00' + purchase_order = PurchaseOrder.with_context(tz='Europe/Brussels').create(vals.copy()) + self.assertTrue(purchase_order.name.startswith('PO/2020/')) + + def test_reminder_1(self): + """Set to send reminder today, check if a reminder can be send to the + partner. + """ + po = Form(self.env['purchase.order']) + po.partner_id = self.partner_a + with po.order_line.new() as po_line: + po_line.product_id = self.product_a + po_line.product_qty = 1 + po_line.price_unit = 100 + with po.order_line.new() as po_line: + po_line.product_id = self.product_b + po_line.product_qty = 10 + po_line.price_unit = 200 + # set to send reminder today + po.date_planned = fields.Datetime.now() + timedelta(days=1) + po.receipt_reminder_email = True + po.reminder_date_before_receipt = 1 + po = po.save() + po.button_confirm() + + # check vendor is a message recipient + self.assertTrue(po.partner_id in po.message_partner_ids) + + old_messages = po.message_ids + po._send_reminder_mail() + messages_send = po.message_ids - old_messages + # check reminder send + self.assertTrue(messages_send) + self.assertTrue(po.partner_id in messages_send.mapped('partner_ids')) + + # check confirm button + po.confirm_reminder_mail() + self.assertTrue(po.mail_reminder_confirmed) + + def test_reminder_2(self): + """Set to send reminder tomorrow, check if no reminder can be send. + """ + po = Form(self.env['purchase.order']) + po.partner_id = self.partner_a + with po.order_line.new() as po_line: + po_line.product_id = self.product_a + po_line.product_qty = 1 + po_line.price_unit = 100 + with po.order_line.new() as po_line: + po_line.product_id = self.product_b + po_line.product_qty = 10 + po_line.price_unit = 200 + # set to send reminder tomorrow + po.date_planned = fields.Datetime.now() + timedelta(days=2) + po.receipt_reminder_email = True + po.reminder_date_before_receipt = 1 + po = po.save() + po.button_confirm() + + # check vendor is a message recipient + self.assertTrue(po.partner_id in po.message_partner_ids) + + old_messages = po.message_ids + po._send_reminder_mail() + messages_send = po.message_ids - old_messages + # check no reminder send + self.assertFalse(messages_send) + + def test_update_date_planned(self): + po = Form(self.env['purchase.order']) + po.partner_id = self.partner_a + with po.order_line.new() as po_line: + po_line.product_id = self.product_a + po_line.product_qty = 1 + po_line.price_unit = 100 + po_line.date_planned = '2020-06-06 00:00:00' + with po.order_line.new() as po_line: + po_line.product_id = self.product_b + po_line.product_qty = 10 + po_line.price_unit = 200 + po_line.date_planned = '2020-06-06 00:00:00' + po = po.save() + po.button_confirm() + + # update first line + po._update_date_planned_for_lines([(po.order_line[0], fields.Datetime.today())]) + self.assertEqual(po.order_line[0].date_planned, fields.Datetime.today()) + activity = self.env['mail.activity'].search([ + ('summary', '=', 'Date Updated'), + ('res_model_id', '=', 'purchase.order'), + ('res_id', '=', po.id), + ]) + self.assertTrue(activity) + self.assertIn( + '<p> partner_a modified receipt dates for the following products:</p><p> \xa0 - product_a from 2020-06-06 to %s </p>' % fields.Date.today(), + activity.note, + ) + + # update second line + po._update_date_planned_for_lines([(po.order_line[1], fields.Datetime.today())]) + self.assertEqual(po.order_line[1].date_planned, fields.Datetime.today()) + self.assertIn( + '<p> partner_a modified receipt dates for the following products:</p><p> \xa0 - product_a from 2020-06-06 to %s </p><p> \xa0 - product_b from 2020-06-06 to %s </p>' % (fields.Date.today(), fields.Date.today()), + activity.note, + ) + + def test_with_different_uom(self): + """ This test ensures that the unit price is correctly computed""" + uom_units = self.env['ir.model.data'].xmlid_to_object('uom.product_uom_unit') + uom_dozens = self.env['ir.model.data'].xmlid_to_object('uom.product_uom_dozen') + uom_pairs = self.env['uom.uom'].create({ + 'name': 'Pairs', + 'category_id': uom_units.category_id.id, + 'uom_type': 'bigger', + 'factor_inv': 2, + 'rounding': 1, + }) + product_data = { + 'name': 'SuperProduct', + 'type': 'consu', + 'uom_id': uom_units.id, + 'uom_po_id': uom_pairs.id, + 'standard_price': 100 + } + product_01 = self.env['product.product'].create(product_data) + product_02 = self.env['product.product'].create(product_data) + + po_form = Form(self.env['purchase.order']) + po_form.partner_id = self.partner_a + with po_form.order_line.new() as po_line: + po_line.product_id = product_01 + with po_form.order_line.new() as po_line: + po_line.product_id = product_02 + po_line.product_uom = uom_dozens + po = po_form.save() + + self.assertEqual(po.order_line[0].price_unit, 200) + self.assertEqual(po.order_line[1].price_unit, 1200) diff --git a/addons/purchase/tests/test_purchase_invoice.py b/addons/purchase/tests/test_purchase_invoice.py new file mode 100644 index 00000000..3c67d2eb --- /dev/null +++ b/addons/purchase/tests/test_purchase_invoice.py @@ -0,0 +1,301 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +from odoo.addons.account.tests.common import AccountTestInvoicingCommon +from odoo.tests import tagged +from odoo.tests.common import Form + + +@tagged('post_install', '-at_install') +class TestPurchaseToInvoice(AccountTestInvoicingCommon): + + @classmethod + def setUpClass(cls): + super(TestPurchaseToInvoice, cls).setUpClass() + uom_unit = cls.env.ref('uom.product_uom_unit') + uom_hour = cls.env.ref('uom.product_uom_hour') + cls.product_order = cls.env['product.product'].create({ + 'name': "Zed+ Antivirus", + 'standard_price': 235.0, + 'list_price': 280.0, + 'type': 'consu', + 'uom_id': uom_unit.id, + 'uom_po_id': uom_unit.id, + 'purchase_method': 'purchase', + 'default_code': 'PROD_ORDER', + 'taxes_id': False, + }) + cls.service_deliver = cls.env['product.product'].create({ + 'name': "Cost-plus Contract", + 'standard_price': 200.0, + 'list_price': 180.0, + 'type': 'service', + 'uom_id': uom_unit.id, + 'uom_po_id': uom_unit.id, + 'purchase_method': 'receive', + 'default_code': 'SERV_DEL', + 'taxes_id': False, + }) + cls.service_order = cls.env['product.product'].create({ + 'name': "Prepaid Consulting", + 'standard_price': 40.0, + 'list_price': 90.0, + 'type': 'service', + 'uom_id': uom_hour.id, + 'uom_po_id': uom_hour.id, + 'purchase_method': 'purchase', + 'default_code': 'PRE-PAID', + 'taxes_id': False, + }) + cls.product_deliver = cls.env['product.product'].create({ + 'name': "Switch, 24 ports", + 'standard_price': 55.0, + 'list_price': 70.0, + 'type': 'consu', + 'uom_id': uom_unit.id, + 'uom_po_id': uom_unit.id, + 'purchase_method': 'receive', + 'default_code': 'PROD_DEL', + 'taxes_id': False, + }) + + def test_vendor_bill_delivered(self): + """Test if a order of product invoiced by delivered quantity can be + correctly invoiced.""" + purchase_order = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + }) + PurchaseOrderLine = self.env['purchase.order.line'].with_context(tracking_disable=True) + pol_prod_deliver = PurchaseOrderLine.create({ + 'name': self.product_deliver.name, + 'product_id': self.product_deliver.id, + 'product_qty': 10.0, + 'product_uom': self.product_deliver.uom_id.id, + 'price_unit': self.product_deliver.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + pol_serv_deliver = PurchaseOrderLine.create({ + 'name': self.service_deliver.name, + 'product_id': self.service_deliver.id, + 'product_qty': 10.0, + 'product_uom': self.service_deliver.uom_id.id, + 'price_unit': self.service_deliver.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + purchase_order.button_confirm() + + self.assertEqual(purchase_order.invoice_status, "no") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 0.0) + + purchase_order.order_line.qty_received = 5 + self.assertEqual(purchase_order.invoice_status, "to invoice") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 5) + self.assertEqual(line.qty_invoiced, 0.0) + + purchase_order.action_create_invoice() + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 5) + + def test_vendor_bill_ordered(self): + """Test if a order of product invoiced by ordered quantity can be + correctly invoiced.""" + purchase_order = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + }) + PurchaseOrderLine = self.env['purchase.order.line'].with_context(tracking_disable=True) + pol_prod_order = PurchaseOrderLine.create({ + 'name': self.product_order.name, + 'product_id': self.product_order.id, + 'product_qty': 10.0, + 'product_uom': self.product_order.uom_id.id, + 'price_unit': self.product_order.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + pol_serv_order = PurchaseOrderLine.create({ + 'name': self.service_order.name, + 'product_id': self.service_order.id, + 'product_qty': 10.0, + 'product_uom': self.service_order.uom_id.id, + 'price_unit': self.service_order.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + purchase_order.button_confirm() + + self.assertEqual(purchase_order.invoice_status, "to invoice") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 10) + self.assertEqual(line.qty_invoiced, 0.0) + + purchase_order.order_line.qty_received = 5 + self.assertEqual(purchase_order.invoice_status, "to invoice") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 10) + self.assertEqual(line.qty_invoiced, 0.0) + + purchase_order.action_create_invoice() + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 10) + + def test_vendor_bill_delivered_return(self): + """Test when return product, a order of product invoiced by delivered + quantity can be correctly invoiced.""" + purchase_order = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + }) + PurchaseOrderLine = self.env['purchase.order.line'].with_context(tracking_disable=True) + pol_prod_deliver = PurchaseOrderLine.create({ + 'name': self.product_deliver.name, + 'product_id': self.product_deliver.id, + 'product_qty': 10.0, + 'product_uom': self.product_deliver.uom_id.id, + 'price_unit': self.product_deliver.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + pol_serv_deliver = PurchaseOrderLine.create({ + 'name': self.service_deliver.name, + 'product_id': self.service_deliver.id, + 'product_qty': 10.0, + 'product_uom': self.service_deliver.uom_id.id, + 'price_unit': self.service_deliver.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + purchase_order.button_confirm() + + purchase_order.order_line.qty_received = 10 + purchase_order.action_create_invoice() + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 10) + + purchase_order.order_line.qty_received = 5 + self.assertEqual(purchase_order.invoice_status, "to invoice") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, -5) + self.assertEqual(line.qty_invoiced, 10) + purchase_order.action_create_invoice() + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 5) + + def test_vendor_bill_ordered_return(self): + """Test when return product, a order of product invoiced by ordered + quantity can be correctly invoiced.""" + purchase_order = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + }) + PurchaseOrderLine = self.env['purchase.order.line'].with_context(tracking_disable=True) + pol_prod_order = PurchaseOrderLine.create({ + 'name': self.product_order.name, + 'product_id': self.product_order.id, + 'product_qty': 10.0, + 'product_uom': self.product_order.uom_id.id, + 'price_unit': self.product_order.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + pol_serv_order = PurchaseOrderLine.create({ + 'name': self.service_order.name, + 'product_id': self.service_order.id, + 'product_qty': 10.0, + 'product_uom': self.service_order.uom_id.id, + 'price_unit': self.service_order.list_price, + 'order_id': purchase_order.id, + 'taxes_id': False, + }) + purchase_order.button_confirm() + + purchase_order.order_line.qty_received = 10 + purchase_order.action_create_invoice() + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 10) + + purchase_order.order_line.qty_received = 5 + self.assertEqual(purchase_order.invoice_status, "invoiced") + for line in purchase_order.order_line: + self.assertEqual(line.qty_to_invoice, 0.0) + self.assertEqual(line.qty_invoiced, 10) + + def test_vendor_severals_bills_and_multicurrency(self): + """ + This test ensures that, when adding several PO to a bill, if they are expressed with different + currency, the amount of each AML is converted to the bill's currency + """ + PurchaseOrderLine = self.env['purchase.order.line'] + PurchaseBillUnion = self.env['purchase.bill.union'] + ResCurrencyRate = self.env['res.currency.rate'] + usd = self.env.ref('base.USD') + eur = self.env.ref('base.EUR') + purchase_orders = [] + + ResCurrencyRate.create({'currency_id': usd.id, 'rate': 1}) + ResCurrencyRate.create({'currency_id': eur.id, 'rate': 2}) + + for currency in [usd, eur]: + po = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + 'currency_id': currency.id, + }) + pol_prod_order = PurchaseOrderLine.create({ + 'name': self.product_order.name, + 'product_id': self.product_order.id, + 'product_qty': 1, + 'product_uom': self.product_order.uom_id.id, + 'price_unit': 1000, + 'order_id': po.id, + 'taxes_id': False, + }) + po.button_confirm() + pol_prod_order.write({'qty_received': 1}) + purchase_orders.append(po) + + move_form = Form(self.env['account.move'].with_context(default_move_type='in_invoice')) + move_form.purchase_vendor_bill_id = PurchaseBillUnion.browse(-purchase_orders[0].id) + move_form.purchase_vendor_bill_id = PurchaseBillUnion.browse(-purchase_orders[1].id) + move = move_form.save() + amls = move.line_ids.filtered(lambda l: l.account_internal_group == 'expense') + + self.assertEqual(move.amount_total, 1500) + self.assertEqual(move.currency_id, usd) + self.assertEqual(len(amls), 2) + self.assertEqual(amls[0].amount_currency, 1000) + self.assertEqual(amls[1].amount_currency, 500) + + def test_product_price_decimal_accuracy(self): + self.env['ir.model.data'].xmlid_to_object('product.decimal_price').digits = 3 + self.env.company.currency_id.rounding = 0.01 + + po = self.env['purchase.order'].with_context(tracking_disable=True).create({ + 'partner_id': self.partner_a.id, + 'order_line': [(0, 0, { + 'name': self.product_a.name, + 'product_id': self.product_a.id, + 'product_qty': 12, + 'product_uom': self.product_a.uom_id.id, + 'price_unit': 0.001, + 'taxes_id': False, + })] + }) + po.button_confirm() + po.order_line.qty_received = 12 + + move_form = Form(self.env['account.move'].with_context(default_move_type='in_invoice')) + move_form.purchase_vendor_bill_id = self.env['purchase.bill.union'].browse(-po.id) + move = move_form.save() + + self.assertEqual(move.amount_total, 0.01) diff --git a/addons/purchase/tests/test_purchase_order_report.py b/addons/purchase/tests/test_purchase_order_report.py new file mode 100644 index 00000000..8b386113 --- /dev/null +++ b/addons/purchase/tests/test_purchase_order_report.py @@ -0,0 +1,68 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. +from odoo.addons.account.tests.common import AccountTestInvoicingCommon +from odoo.tests import Form, tagged + +from datetime import datetime + + +@tagged('post_install', '-at_install') +class TestPurchaseOrderReport(AccountTestInvoicingCommon): + + def test_00_purchase_order_report(self): + uom_dozen = self.env.ref('uom.product_uom_dozen') + + po = self.env['purchase.order'].create({ + 'partner_id': self.partner_a.id, + 'currency_id': self.currency_data['currency'].id, + 'order_line': [ + (0, 0, { + 'name': self.product_a.name, + 'product_id': self.product_a.id, + 'product_qty': 1.0, + 'product_uom': uom_dozen.id, + 'price_unit': 100.0, + 'date_planned': datetime.today(), + 'taxes_id': False, + }), + (0, 0, { + 'name': self.product_b.name, + 'product_id': self.product_b.id, + 'product_qty': 1.0, + 'product_uom': uom_dozen.id, + 'price_unit': 200.0, + 'date_planned': datetime.today(), + 'taxes_id': False, + }), + ], + }) + po.button_confirm() + + f = Form(self.env['account.move'].with_context(default_move_type='in_invoice')) + f.invoice_date = f.date + f.partner_id = po.partner_id + f.purchase_id = po + invoice = f.save() + invoice.action_post() + po.flush() + + res_product1 = self.env['purchase.report'].search([ + ('order_id', '=', po.id), + ('product_id', '=', self.product_a.id), + ('company_id', '=', self.company_data['company'].id), + ]) + + # check that report will convert dozen to unit or not + self.assertEqual(res_product1.qty_ordered, 12.0, 'UoM conversion is not working') + # report should show in company currency (amount/rate) = (100/2) + self.assertEqual(res_product1.price_total, 50.0, 'Currency conversion is not working') + + res_product2 = self.env['purchase.report'].search([ + ('order_id', '=', po.id), + ('product_id', '=', self.product_b.id), + ('company_id', '=', self.company_data['company'].id), + ]) + + self.assertEqual(res_product2.qty_ordered, 1.0, 'No conversion needed since product_b is already a dozen') + # report should show in company currency (amount/rate) = (200/2) + self.assertEqual(res_product2.price_total, 100.0, 'Currency conversion is not working') |
