sale_purchase/tests/test_sale_purchase.py

308 lines
18 KiB
Python

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo import Command
from odoo.exceptions import UserError, AccessError
from odoo.tests import tagged
from odoo.addons.sale_purchase.tests.common import TestCommonSalePurchaseNoChart
@tagged('-at_install', 'post_install')
class TestSalePurchase(TestCommonSalePurchaseNoChart):
@classmethod
def setUpClass(cls, chart_template_ref=None):
super().setUpClass(chart_template_ref=chart_template_ref)
# create a generic Sale Order with 2 classical products and a purchase service
SaleOrder = cls.env['sale.order'].with_context(tracking_disable=True)
cls.sale_order_1 = SaleOrder.create({
'partner_id': cls.partner_a.id,
'partner_invoice_id': cls.partner_a.id,
'partner_shipping_id': cls.partner_a.id,
'pricelist_id': cls.company_data['default_pricelist'].id,
})
cls.sol1_service_deliver = cls.env['sale.order.line'].create({
'product_id': cls.company_data['product_service_delivery'].id,
'product_uom_qty': 1,
'order_id': cls.sale_order_1.id,
'tax_id': False,
})
cls.sol1_product_order = cls.env['sale.order.line'].create({
'product_id': cls.company_data['product_order_no'].id,
'product_uom_qty': 2,
'order_id': cls.sale_order_1.id,
'tax_id': False,
})
cls.sol1_service_purchase_1 = cls.env['sale.order.line'].create({
'product_id': cls.service_purchase_1.id,
'product_uom_qty': 4,
'order_id': cls.sale_order_1.id,
'tax_id': False,
})
cls.sale_order_2 = SaleOrder.create({
'partner_id': cls.partner_a.id,
'partner_invoice_id': cls.partner_a.id,
'partner_shipping_id': cls.partner_a.id,
'pricelist_id': cls.company_data['default_pricelist'].id,
})
cls.sol2_product_deliver = cls.env['sale.order.line'].create({
'product_id': cls.company_data['product_delivery_no'].id,
'product_uom_qty': 5,
'order_id': cls.sale_order_2.id,
'tax_id': False,
})
cls.sol2_service_order = cls.env['sale.order.line'].create({
'product_id': cls.company_data['product_service_order'].id,
'product_uom_qty': 6,
'order_id': cls.sale_order_2.id,
'tax_id': False,
})
cls.sol2_service_purchase_2 = cls.env['sale.order.line'].create({
'product_id': cls.service_purchase_2.id,
'product_uom_qty': 7,
'order_id': cls.sale_order_2.id,
'tax_id': False,
})
def test_sale_create_purchase(self):
""" Confirming 2 sales orders with a service that should create a PO, then cancelling the PO should shedule 1 next activity per SO """
self.sale_order_1.action_confirm()
self.sale_order_2.action_confirm()
purchase_order = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines_so1 = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_line1 = purchase_lines_so1[0]
purchase_lines_so2 = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_2.order_line.ids)])
purchase_line2 = purchase_lines_so2[0]
self.assertEqual(len(purchase_order), 1, "Only one PO should have been created, from the 2 Sales orders")
self.assertEqual(len(purchase_order.order_line), 2, "The purchase order should have 2 lines")
self.assertEqual(len(purchase_lines_so1), 1, "Only one SO line from SO 1 should have create a PO line")
self.assertEqual(len(purchase_lines_so2), 1, "Only one SO line from SO 2 should have create a PO line")
self.assertEqual(len(purchase_order.activity_ids), 0, "No activity should be scheduled on the PO")
self.assertEqual(purchase_order.state, 'draft', "The created PO should be in draft state")
self.assertNotEqual(purchase_line1.product_id, purchase_line2.product_id, "The 2 PO line should have different products")
self.assertEqual(purchase_line1.product_id, self.sol1_service_purchase_1.product_id, "The create PO line must have the same product as its mother SO line")
self.assertEqual(purchase_line2.product_id, self.sol2_service_purchase_2.product_id, "The create PO line must have the same product as its mother SO line")
purchase_order.button_cancel()
self.assertEqual(len(self.sale_order_1.activity_ids), 1, "One activity should be scheduled on the SO 1 since the PO has been cancelled")
self.assertEqual(self.sale_order_1.user_id, self.sale_order_1.activity_ids[0].user_id, "The activity should be assigned to the SO responsible")
self.assertEqual(len(self.sale_order_2.activity_ids), 1, "One activity should be scheduled on the SO 2 since the PO has been cancelled")
self.assertEqual(self.sale_order_2.user_id, self.sale_order_2.activity_ids[0].user_id, "The activity should be assigned to the SO responsible")
def test_uom_conversion(self):
""" Test generated PO use the right UoM according to product configuration """
self.sale_order_2.action_confirm()
purchase_line = self.env['purchase.order.line'].search([('sale_line_id', '=', self.sol2_service_purchase_2.id)]) # only one line
self.assertTrue(purchase_line, "The SO line should generate a PO line")
self.assertEqual(purchase_line.product_uom, self.service_purchase_2.uom_po_id, "The UoM on the purchase line should be the one from the product configuration")
self.assertNotEqual(purchase_line.product_uom, self.sol2_service_purchase_2.product_uom, "As the product configuration, the UoM on the SO line should still be different from the one on the PO line")
self.assertEqual(purchase_line.product_qty, self.sol2_service_purchase_2.product_uom_qty * 12, "The quantity from the SO should be converted with th UoM factor on the PO line")
def test_no_supplier(self):
""" Test confirming SO with product with no supplier raise Error """
# delete the suppliers
self.supplierinfo1.unlink()
# confirm the SO should raise UserError
with self.assertRaises(UserError):
self.sale_order_1.action_confirm()
def test_reconfirm_sale_order(self):
""" Confirm SO, cancel it, then re-confirm it should not regenerate a purchase line """
self.sale_order_1.action_confirm()
purchase_order = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_line = purchase_lines[0]
self.assertEqual(len(purchase_lines), 1, "Only one purchase line should be created on SO confirmation")
self.assertEqual(len(purchase_order), 1, "One purchase order should have been created on SO confirmation")
self.assertEqual(len(purchase_order.order_line), 1, "Only one line on PO, after SO confirmation")
self.assertEqual(purchase_order, purchase_lines.order_id, "The generated purchase line should be in the generated purchase order")
self.assertEqual(purchase_order.state, 'draft', "Generated purchase should be in draft state")
self.assertEqual(purchase_line.price_unit, self.supplierinfo1.price, "Purchase line price is the one from the supplier")
self.assertEqual(purchase_line.product_qty, self.sol1_service_purchase_1.product_uom_qty, "Quantity on SO line is not the same on the purchase line (same UoM)")
self.sale_order_1._action_cancel()
self.assertEqual(len(purchase_order.activity_ids), 1, "One activity should be scheduled on the PO since a SO has been cancelled")
purchase_order = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_line = purchase_lines[0]
self.assertEqual(len(purchase_lines), 1, "Always one purchase line even after SO cancellation")
self.assertTrue(purchase_order, "Always one purchase order even after SO cancellation")
self.assertEqual(len(purchase_order.order_line), 1, "Still one line on PO, even after SO cancellation")
self.assertEqual(purchase_order, purchase_lines.order_id, "The generated purchase line should still be in the generated purchase order")
self.assertEqual(purchase_order.state, 'draft', "Generated purchase should still be in draft state")
self.assertEqual(purchase_line.price_unit, self.supplierinfo1.price, "Purchase line price is still the one from the supplier")
self.assertEqual(purchase_line.product_qty, self.sol1_service_purchase_1.product_uom_qty, "Quantity on SO line should still be the same on the purchase line (same UoM)")
self.sale_order_1.action_draft()
self.sale_order_1.action_confirm()
purchase_order = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_line = purchase_lines[0]
self.assertEqual(len(purchase_lines), 1, "Still only one purchase line should be created even after SO reconfirmation")
self.assertEqual(len(purchase_order), 1, "Still one purchase order should be after SO reconfirmation")
self.assertEqual(len(purchase_order.order_line), 1, "Only one line on PO, even after SO reconfirmation")
self.assertEqual(purchase_order, purchase_lines.order_id, "The generated purchase line should be in the generated purchase order")
self.assertEqual(purchase_order.state, 'draft', "Generated purchase should be in draft state")
self.assertEqual(purchase_line.price_unit, self.supplierinfo1.price, "Purchase line price is the one from the supplier")
self.assertEqual(purchase_line.product_qty, self.sol1_service_purchase_1.product_uom_qty, "Quantity on SO line is not the same on the purchase line (same UoM)")
def test_update_ordered_sale_quantity(self):
""" Test the purchase order behovior when changing the ordered quantity on the sale order line.
Increase of qty on the SO
- If PO is draft ['draft', 'sent', 'to approve'] : increase the quantity on the PO
- If PO is confirmed ['purchase', 'done', 'cancel'] : create a new PO
Decrease of qty on the SO
- If PO is draft ['draft', 'sent', 'to approve'] : next activity on the PO
- If PO is confirmed ['purchase', 'done', 'cancel'] : next activity on the PO
"""
self.sale_order_1.action_confirm()
purchase_order = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_line = purchase_lines[0]
self.assertEqual(purchase_order.state, 'draft', "The created purchase should be in draft state")
self.assertFalse(purchase_order.activity_ids, "There is no activities on the PO")
self.assertEqual(purchase_line.product_qty, self.sol1_service_purchase_1.product_uom_qty, "Quantity on SO line is not the same on the purchase line (same UoM)")
# increase the ordered quantity on sale line
self.sol1_service_purchase_1.write({'product_uom_qty': self.sol1_service_purchase_1.product_uom_qty + 12}) # product_uom_qty = 16
self.assertEqual(purchase_line.product_qty, self.sol1_service_purchase_1.product_uom_qty, "The quantity of draft PO line should be increased as the one from the sale line changed")
sale_line_old_quantity = self.sol1_service_purchase_1.product_uom_qty
# decrease the ordered quantity on sale line
self.sol1_service_purchase_1.write({'product_uom_qty': self.sol1_service_purchase_1.product_uom_qty - 3}) # product_uom_qty = 13
self.assertEqual(len(purchase_order.activity_ids), 1, "One activity should have been created on the PO")
self.assertEqual(purchase_order.activity_ids.user_id, purchase_order.user_id, "Activity assigned to PO responsible")
self.assertEqual(purchase_order.activity_ids.state, 'today', "Activity is for today, as it is urgent")
# confirm the PO
purchase_order.button_confirm()
# decrease the ordered quantity on sale line
self.sol1_service_purchase_1.write({'product_uom_qty': self.sol1_service_purchase_1.product_uom_qty - 5}) # product_uom_qty = 8
self.env.invalidate_all() # Note: creating a second activity will not refresh the cache
self.assertEqual(purchase_line.product_qty, sale_line_old_quantity, "The quantity on the PO line should not have changed.")
self.assertEqual(len(purchase_order.activity_ids), 2, "a second activity should have been created on the PO")
self.assertEqual(purchase_order.activity_ids.mapped('user_id'), purchase_order.user_id, "Activities assigned to PO responsible")
self.assertEqual(purchase_order.activity_ids.mapped('state'), ['today', 'today'], "Activities are for today, as it is urgent")
# increase the ordered quantity on sale line
delta = 8
self.sol1_service_purchase_1.write({'product_uom_qty': self.sol1_service_purchase_1.product_uom_qty + delta}) # product_uom_qty = 16
self.assertEqual(purchase_line.product_qty, sale_line_old_quantity, "The quantity on the PO line should not have changed.")
self.assertEqual(len(purchase_order.activity_ids), 2, "Always 2 activity on confirmed the PO")
purchase_order2 = self.env['purchase.order'].search([('partner_id', '=', self.supplierinfo1.partner_id.id), ('state', '=', 'draft')])
purchase_lines = self.env['purchase.order.line'].search([('sale_line_id', 'in', self.sale_order_1.order_line.ids)])
purchase_lines2 = purchase_lines.filtered(lambda pol: pol.order_id == purchase_order2)
purchase_line2 = purchase_lines2[0]
self.assertTrue(purchase_order2, "A second PO is created by increasing sale quantity when first PO is confirmed")
self.assertEqual(purchase_order2.state, 'draft', "The second PO is in draft state")
self.assertNotEqual(purchase_order, purchase_order2, "The 2 PO are different")
self.assertEqual(len(purchase_lines), 2, "The same Sale Line has created 2 purchase lines")
self.assertEqual(len(purchase_order2.order_line), 1, "The 2nd PO has only one line")
self.assertEqual(purchase_line2.sale_line_id, self.sol1_service_purchase_1, "The 2nd PO line came from the SO line sol1_service_purchase_1")
self.assertEqual(purchase_line2.product_qty, delta, "The quantity of the new PO line is the quantity added on the Sale Line, after first PO confirmation")
def test_pol_description(self):
"""
test cases when product names are different from how the vendor refers to, which is allowed
"""
service = self.env['product.product'].create({
'name': 'Super Product',
'type': 'service',
'service_to_purchase': True,
'seller_ids': [(0, 0, {
'partner_id': self.partner_vendor_service.id,
'min_qty': 1,
'price': 10,
'product_code': 'C01',
'product_name': 'Name01',
'sequence': 1,
})]
})
so = self.env['sale.order'].create({
'partner_id': self.partner_a.id,
'order_line': [
(0, 0, {
'name': service.name,
'product_id': service.id,
'product_uom_qty': 1,
})
],
})
so.action_confirm()
po = self.env['purchase.order'].search([('partner_id', '=', self.partner_vendor_service.id)], order='id desc', limit=1)
self.assertEqual(po.order_line.name, "[C01] Name01")
def test_pol_custom_attribute(self):
"""
test that custom atributes are passed from the SO the PO for service products
"""
# Setup service product variants
product_attribute = self.env['product.attribute'].create({
'name': 'product attribute',
'display_type': 'radio',
'create_variant': 'always'
})
product_attribute_value = self.env['product.attribute.value'].create({
'name': 'single product attribute value',
'is_custom': True,
'attribute_id': product_attribute.id
})
product_attribute_line = self.env['product.template.attribute.line'].create({
'attribute_id': product_attribute.id,
'product_tmpl_id': self.service_purchase_1.product_tmpl_id.id,
'value_ids': [Command.link(product_attribute_value.id)]
})
custom_value = "test"
# create and confirm SO
sale_order = self.env['sale.order'].create({
'partner_id': self.partner_a.id,
'order_line': [
Command.create({
'name': self.service_purchase_1.name,
'product_id': self.service_purchase_1.id,
'product_uom_qty': 1,
'product_custom_attribute_value_ids': [
Command.create({
'custom_product_template_attribute_value_id': product_attribute_line.product_template_value_ids.id,
'custom_value': custom_value,
})
],
})
],
})
sale_order.action_confirm()
pol = sale_order._get_purchase_orders().order_line
self.assertEqual(pol.name, f"{self.service_purchase_1.display_name}\n\n{product_attribute.name}: {product_attribute_value.name}: {custom_value}")