Odoo18-Base/addons/mrp/tests/test_bom.py
2025-03-10 11:12:23 +07:00

1725 lines
78 KiB
Python

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo import exceptions, Command, fields
from odoo.tests import Form
from odoo.addons.mrp.tests.common import TestMrpCommon
from odoo.tools import float_compare, float_round, float_repr
from freezegun import freeze_time
@freeze_time(fields.Date.today())
class TestBoM(TestMrpCommon):
@classmethod
def setUpClass(cls):
super().setUpClass()
cls.env.ref('base.group_user').write({'implied_ids': [(4, cls.env.ref('product.group_product_variant').id)]})
def test_01_explode(self):
boms, lines = self.bom_1.explode(self.product_4, 3)
self.assertEqual(set([bom[0].id for bom in boms]), set(self.bom_1.ids))
self.assertEqual(set([line[0].id for line in lines]), set(self.bom_1.bom_line_ids.ids))
boms, lines = self.bom_3.explode(self.product_6, 3)
self.assertEqual(set([bom[0].id for bom in boms]), set((self.bom_2 | self.bom_3).ids))
self.assertEqual(
set([line[0].id for line in lines]),
set((self.bom_2 | self.bom_3).mapped('bom_line_ids').filtered(lambda line: not line.child_bom_id or line.child_bom_id.type != 'phantom').ids))
def test_10_variants(self):
test_bom = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 4.0,
'type': 'normal',
'operation_ids': [
Command.create({
'name': 'Cutting Machine',
'workcenter_id': self.workcenter_1.id,
'time_cycle': 12,
'sequence': 1
}),
Command.create({
'name': 'Weld Machine',
'workcenter_id': self.workcenter_1.id,
'time_cycle': 18,
'sequence': 2,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v1.id)]
}),
Command.create({
'name': 'Taking a coffee',
'workcenter_id': self.workcenter_1.id,
'time_cycle': 5,
'sequence': 3,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v2.id)]
})
],
'byproduct_ids': [
Command.create({
'product_id': self.product_1.id,
'product_uom_id': self.product_1.uom_id.id,
'product_qty': 1,
}),
Command.create({
'product_id': self.product_2.id,
'product_uom_id': self.product_2.uom_id.id,
'product_qty': 1,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v1.id)]
}),
Command.create({
'product_id': self.product_3.id,
'product_uom_id': self.product_3.uom_id.id,
'product_qty': 1,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v2.id)]
}),
],
'bom_line_ids': [
Command.create({
'product_id': self.product_2.id,
'product_qty': 2,
}),
Command.create({
'product_id': self.product_3.id,
'product_qty': 2,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v1.id)],
}),
Command.create({
'product_id': self.product_4.id,
'product_qty': 2,
'bom_product_template_attribute_value_ids': [Command.link(self.product_7_attr1_v2.id)],
}),
]
})
test_bom_l1, test_bom_l2, test_bom_l3 = test_bom.bom_line_ids
boms, lines = test_bom.explode(self.product_7_3, 4)
self.assertIn(test_bom, [b[0]for b in boms])
self.assertIn(test_bom_l1, [l[0] for l in lines])
self.assertNotIn(test_bom_l2, [l[0] for l in lines])
self.assertNotIn(test_bom_l3, [l[0] for l in lines])
boms, lines = test_bom.explode(self.product_7_1, 4)
self.assertIn(test_bom, [b[0]for b in boms])
self.assertIn(test_bom_l1, [l[0] for l in lines])
self.assertIn(test_bom_l2, [l[0] for l in lines])
self.assertNotIn(test_bom_l3, [l[0] for l in lines])
boms, lines = test_bom.explode(self.product_7_2, 4)
self.assertIn(test_bom, [b[0]for b in boms])
self.assertIn(test_bom_l1, [l[0] for l in lines])
self.assertNotIn(test_bom_l2, [l[0] for l in lines])
self.assertIn(test_bom_l3, [l[0] for l in lines])
mrp_order_form = Form(self.env['mrp.production'])
mrp_order_form.product_id = self.product_7_3
mrp_order = mrp_order_form.save()
self.assertEqual(mrp_order.bom_id, test_bom)
self.assertEqual(len(mrp_order.workorder_ids), 1)
self.assertEqual(mrp_order.workorder_ids.operation_id, test_bom.operation_ids[0])
self.assertEqual(len(mrp_order.move_byproduct_ids), 1)
self.assertEqual(mrp_order.move_byproduct_ids.product_id, self.product_1)
mrp_order_form = Form(self.env['mrp.production'])
mrp_order_form.product_id = self.product_7_1
mrp_order_form.product_id = self.env['product.product'] # Check form
mrp_order_form.product_id = self.product_7_1
mrp_order_form.bom_id = self.env['mrp.bom'] # Check form
mrp_order_form.bom_id = test_bom
mrp_order = mrp_order_form.save()
self.assertEqual(mrp_order.bom_id, test_bom)
self.assertEqual(len(mrp_order.workorder_ids), 2)
self.assertEqual(mrp_order.workorder_ids.operation_id, test_bom.operation_ids[:2])
self.assertEqual(len(mrp_order.move_byproduct_ids), 2)
self.assertEqual(mrp_order.move_byproduct_ids.product_id, self.product_1 | self.product_2)
mrp_order_form = Form(self.env['mrp.production'])
mrp_order_form.product_id = self.product_7_2
mrp_order = mrp_order_form.save()
self.assertEqual(mrp_order.bom_id, test_bom)
self.assertEqual(len(mrp_order.workorder_ids), 2)
self.assertEqual(mrp_order.workorder_ids.operation_id, test_bom.operation_ids[0] | test_bom.operation_ids[2])
self.assertEqual(len(mrp_order.move_byproduct_ids), 2)
self.assertEqual(mrp_order.move_byproduct_ids.product_id, self.product_1 | self.product_3)
def test_11_multi_level_variants(self):
tmp_picking_type = self.env['stock.picking.type'].create({
'name': 'Manufacturing',
'code': 'mrp_operation',
'sequence_code': 'TMP',
'sequence_id': self.env['ir.sequence'].create({
'code': 'mrp.production',
'name': 'tmp_production_sequence',
}).id,
})
test_bom_1 = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_5.product_tmpl_id.id,
'product_uom_id': self.product_5.uom_id.id,
'product_qty': 1.0,
'type': 'phantom'
})
test_bom_1.write({
'operation_ids': [
(0, 0, {'name': 'Gift Wrap Maching', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 15, 'sequence': 1}),
],
})
test_bom_1.bom_line_ids = [(0, 0, {
'product_id': self.product_3.id,
'product_qty': 3,
})]
test_bom_2 = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 4.0,
'type': 'normal',
})
test_bom_2.write({
'operation_ids': [
(0, 0, {'name': 'Cutting Machine', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 12, 'sequence': 1}),
(0, 0, {'name': 'Weld Machine', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 18, 'sequence': 2}),
]
})
test_bom_2.bom_line_ids = [(0, 0, {
'product_id': self.product_2.id,
'product_qty': 2,
})]
test_bom_2.bom_line_ids = [(0, 0, {
'product_id': self.product_5.id,
'product_qty': 2,
'bom_product_template_attribute_value_ids': [(4, self.product_7_attr1_v1.id)],
})]
test_bom_2.bom_line_ids = [(0, 0, {
'product_id': self.product_5.id,
'product_qty': 2,
'bom_product_template_attribute_value_ids': [(4, self.product_7_attr1_v2.id)],
})]
test_bom_2.bom_line_ids = [(0, 0, {
'product_id': self.product_4.id,
'product_qty': 2,
})]
test_bom_2_l1, _test_bom_2_l2, _test_bom_2_l3, test_bom_2_l4 = test_bom_2.bom_line_ids
# check product > product_tmpl
boms, lines = test_bom_2.explode(self.product_7_1, 4)
self.assertEqual(set((test_bom_2 | self.bom_2).ids), set([b[0].id for b in boms]))
self.assertEqual(set((test_bom_2_l1 | test_bom_2_l4 | self.bom_2.bom_line_ids).ids), set([l[0].id for l in lines]))
# check sequence priority
test_bom_1.write({'sequence': 1})
boms, lines = test_bom_2.explode(self.product_7_1, 4)
self.assertEqual(set((test_bom_2 | test_bom_1).ids), set([b[0].id for b in boms]))
self.assertEqual(set((test_bom_2_l1 | test_bom_2_l4 | test_bom_1.bom_line_ids).ids), set([l[0].id for l in lines]))
# check with another picking_type
test_bom_1.write({'picking_type_id': self.warehouse_1.manu_type_id.id})
self.bom_2.write({'picking_type_id': tmp_picking_type.id})
test_bom_2.write({'picking_type_id': tmp_picking_type.id})
boms, lines = test_bom_2.explode(self.product_7_1, 4)
self.assertEqual(set((test_bom_2 | self.bom_2).ids), set([b[0].id for b in boms]))
self.assertEqual(set((test_bom_2_l1 | test_bom_2_l4 | self.bom_2.bom_line_ids).ids), set([l[0].id for l in lines]))
self.product_9, self.product_10 = self.env['product.product'].create([{
'name': 'Paper', # product_9
}, {
'name': 'Stone', # product_10
}])
#check recursion
test_bom_3 = self.env['mrp.bom'].create({
'product_id': self.product_9.id,
'product_tmpl_id': self.product_9.product_tmpl_id.id,
'product_uom_id': self.product_9.uom_id.id,
'product_qty': 1.0,
'consumption': 'flexible',
'type': 'normal'
})
test_bom_4 = self.env['mrp.bom'].create({
'product_id': self.product_10.id,
'product_tmpl_id': self.product_10.product_tmpl_id.id,
'product_uom_id': self.product_10.uom_id.id,
'product_qty': 1.0,
'consumption': 'flexible',
'type': 'phantom'
})
test_bom_3.bom_line_ids = [(0, 0, {
'product_id': self.product_10.id,
'product_qty': 1.0,
})]
with self.assertRaises(exceptions.UserError):
test_bom_4.bom_line_ids = [(0, 0, {
'product_id': self.product_9.id,
'product_qty': 1.0,
})]
def test_12_multi_level_variants2(self):
"""Test skip bom line with same attribute values in bom lines."""
Product = self.env['product.product']
ProductAttribute = self.env['product.attribute']
ProductAttributeValue = self.env['product.attribute.value']
# Product Attribute
att_color = ProductAttribute.create({'name': 'Color', 'sequence': 1})
att_size = ProductAttribute.create({'name': 'size', 'sequence': 2})
# Product Attribute color Value
att_color_red = ProductAttributeValue.create({'name': 'red', 'attribute_id': att_color.id, 'sequence': 1})
att_color_blue = ProductAttributeValue.create({'name': 'blue', 'attribute_id': att_color.id, 'sequence': 2})
# Product Attribute size Value
att_size_big = ProductAttributeValue.create({'name': 'big', 'attribute_id': att_size.id, 'sequence': 1})
att_size_medium = ProductAttributeValue.create({'name': 'medium', 'attribute_id': att_size.id, 'sequence': 2})
# Create Template Product
product_template = self.env['product.template'].create({
'name': 'Sofa',
'attribute_line_ids': [
(0, 0, {
'attribute_id': att_color.id,
'value_ids': [(6, 0, [att_color_red.id, att_color_blue.id])]
}),
(0, 0, {
'attribute_id': att_size.id,
'value_ids': [(6, 0, [att_size_big.id, att_size_medium.id])]
})
]
})
sofa_red = product_template.attribute_line_ids[0].product_template_value_ids[0]
sofa_blue = product_template.attribute_line_ids[0].product_template_value_ids[1]
sofa_big = product_template.attribute_line_ids[1].product_template_value_ids[0]
sofa_medium = product_template.attribute_line_ids[1].product_template_value_ids[1]
# Create components Of BOM
product_A = Product.create({
'name': 'Wood'})
product_B = Product.create({
'name': 'Clothes'})
# Create BOM
self.env['mrp.bom'].create({
'product_tmpl_id': product_template.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {
'product_id': product_A.id,
'product_qty': 1,
'bom_product_template_attribute_value_ids': [(4, sofa_red.id), (4, sofa_blue.id), (4, sofa_big.id)],
}),
(0, 0, {
'product_id': product_B.id,
'product_qty': 1,
'bom_product_template_attribute_value_ids': [(4, sofa_red.id), (4, sofa_blue.id)]
})
]
})
dict_consumed_products = {
sofa_red + sofa_big: product_A + product_B,
sofa_red + sofa_medium: product_B,
sofa_blue + sofa_big: product_A + product_B,
sofa_blue + sofa_medium: product_B,
}
# Create production order for all variants.
for combination, consumed_products in dict_consumed_products.items():
product = product_template.product_variant_ids.filtered(lambda p: p.product_template_attribute_value_ids == combination)
mrp_order_form = Form(self.env['mrp.production'])
mrp_order_form.product_id = product
mrp_order = mrp_order_form.save()
# Check consumed materials in production order.
self.assertEqual(mrp_order.move_raw_ids.product_id, consumed_products)
def test_13_bom_kit_qty(self):
self.env['mrp.bom'].create({
'product_id': self.product_7_3.id,
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 4.0,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': self.product_2.id,
'product_qty': 2,
}),
(0, 0, {
'product_id': self.product_3.id,
'product_qty': 2,
})
]
})
location = self.env.ref('stock.stock_location_stock')
self.env['stock.quant']._update_available_quantity(self.product_2, location, 4.0)
self.env['stock.quant']._update_available_quantity(self.product_3, location, 8.0)
# Force the kit product available qty to be computed at the same time than its component quantities
# Because `qty_available` of a bom kit "recurse" on `qty_available` of its component,
# and this is a tricky thing for the ORM:
# `qty_available` gets called for `product_7_3`, `product_2` and `product_3`
# which then recurse on calling `qty_available` for `product_2` and `product_3` to compute the quantity of
# the kit `product_7_3`. `product_2` and `product_3` gets protected at the first call of the compute method,
# ending the recurse call to not call the compute method and just left the Falsy value `0.0`
# for the components available qty.
kit_product_qty, _, _ = (self.product_7_3 + self.product_2 + self.product_3).mapped("qty_available")
self.assertEqual(kit_product_qty, 8)
def test_14_bom_kit_qty_multi_uom(self):
uom_dozens = self.env.ref('uom.product_uom_dozen')
uom_unit = self.env.ref('uom.product_uom_unit')
product_unit = self.env['product.product'].create({
'name': 'Test units',
'type': 'product',
'uom_id': uom_unit.id,
})
product_dozens = self.env['product.product'].create({
'name': 'Test dozens',
'type': 'product',
'uom_id': uom_dozens.id,
})
self.env['mrp.bom'].create({
'product_tmpl_id': product_unit.product_tmpl_id.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 1.0,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': product_dozens.id,
'product_qty': 1,
'product_uom_id': uom_unit.id,
})
]
})
location = self.env.ref('stock.stock_location_stock')
self.env['stock.quant']._update_available_quantity(product_dozens, location, 1.0)
self.assertEqual(product_unit.qty_available, 12.0)
def test_13_negative_on_hand_qty(self):
# We set the Product Unit of Measure digits to 5.
# Because float_round(-384.0, 5) = -384.00000000000006
# And float_round(-384.0, 2) = -384.0
precision = self.env.ref('product.decimal_product_uom')
precision.digits = 5
# We set the Unit(s) rounding to 0.0001 (digit = 4)
uom_unit = self.env.ref('uom.product_uom_unit')
uom_unit.rounding = 0.0001
_ = self.env['mrp.bom'].create({
'product_id': self.product_2.id,
'product_tmpl_id': self.product_2.product_tmpl_id.id,
'product_uom_id': uom_unit.id,
'product_qty': 1.00,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': self.product_3.id,
'product_qty': 1.000,
}),
]
})
self.env['stock.quant']._update_available_quantity(self.product_3, self.env.ref('stock.stock_location_stock'), -384.0)
kit_product_qty = self.product_2.qty_available # Without product_3 in the prefetch
# Use the float_repr to remove extra small decimal (and represent the front-end behavior)
self.assertEqual(float_repr(float_round(kit_product_qty, precision_digits=precision.digits), precision_digits=precision.digits), '-384.00000')
self.product_2.invalidate_recordset(['qty_available'])
kit_product_qty, _ = (self.product_2 + self.product_3).mapped("qty_available") # With product_3 in the prefetch
self.assertEqual(float_repr(float_round(kit_product_qty, precision_digits=precision.digits), precision_digits=precision.digits), '-384.00000')
def test_13_bom_kit_qty_multi_uom(self):
uom_dozens = self.env.ref('uom.product_uom_dozen')
uom_unit = self.env.ref('uom.product_uom_unit')
product_unit = self.env['product.product'].create({
'name': 'Test units',
'type': 'product',
'uom_id': uom_unit.id,
})
product_dozens = self.env['product.product'].create({
'name': 'Test dozens',
'type': 'product',
'uom_id': uom_dozens.id,
})
self.env['mrp.bom'].create({
'product_tmpl_id': product_unit.product_tmpl_id.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 1.0,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': product_dozens.id,
'product_qty': 1,
'product_uom_id': uom_unit.id,
})
]
})
location = self.env.ref('stock.stock_location_stock')
self.env['stock.quant']._update_available_quantity(product_dozens, location, 1.0)
self.assertEqual(product_unit.qty_available, 12.0)
def test_19_bom_kit_field_is_kits_bom_with_product_id(self):
kit_products = self.env['product.product'].create({
'name': 'No Kit',
'type': 'product',
'uom_id': self.uom_unit.id,
})
self.env['mrp.bom'].create({
'product_id': kit_products.id,
'product_tmpl_id': kit_products.product_tmpl_id.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 4.0,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': self.product_2.id,
'product_qty': 2,
}),
(0, 0, {
'product_id': self.product_3.id,
'product_qty': 2,
})
]
})
self.assertTrue(kit_products.is_kits)
self.assertTrue(kit_products.product_tmpl_id.is_kits)
no_kit_products = self.env['product.product'].create({
'name': 'No Kit',
'type': 'product',
'uom_id': self.uom_unit.id,
})
self.assertFalse(no_kit_products.is_kits)
self.assertFalse(no_kit_products.product_tmpl_id.is_kits)
product_with_kit = self.env['product.product'].search(
[('is_kits', '=', True)])
product_tmpl_with_kit = self.env['product.template'].search(
[('is_kits', '=', True)])
self.assertIn(kit_products, product_with_kit)
self.assertIn(kit_products.product_tmpl_id, product_tmpl_with_kit)
self.assertNotIn(no_kit_products, product_with_kit)
self.assertNotIn(no_kit_products.product_tmpl_id,
product_tmpl_with_kit)
product_without_kit = self.env['product.product'].search(
[('is_kits', '=', False)])
product_tmpl_without_kit = self.env['product.template'].search(
[('is_kits', '=', False)])
self.assertIn(no_kit_products, product_without_kit)
self.assertIn(no_kit_products.product_tmpl_id,
product_tmpl_without_kit)
self.assertNotIn(kit_products, product_without_kit)
self.assertNotIn(kit_products.product_tmpl_id,
product_tmpl_without_kit)
def test_19_bom_kit_field_is_kits_bom_without_product_id(self):
kit_products = self.env['product.product'].create({
'name': 'No Kit',
'type': 'product',
'uom_id': self.uom_unit.id,
})
self.env['mrp.bom'].create({
'product_tmpl_id': kit_products.product_tmpl_id.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 4.0,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {
'product_id': self.product_2.id,
'product_qty': 2,
}),
(0, 0, {
'product_id': self.product_3.id,
'product_qty': 2,
})
]
})
self.assertTrue(kit_products.is_kits)
self.assertTrue(kit_products.product_tmpl_id.is_kits)
no_kit_products = self.env['product.product'].create({
'name': 'No Kit',
'type': 'product',
'uom_id': self.uom_unit.id,
})
self.assertFalse(no_kit_products.is_kits)
self.assertFalse(no_kit_products.product_tmpl_id.is_kits)
product_with_kit = self.env['product.product'].search(
[('is_kits', '=', True)])
product_tmpl_with_kit = self.env['product.template'].search(
[('is_kits', '=', True)])
self.assertIn(kit_products, product_with_kit)
self.assertIn(kit_products.product_tmpl_id, product_tmpl_with_kit)
self.assertNotIn(no_kit_products, product_with_kit)
self.assertNotIn(no_kit_products.product_tmpl_id,
product_tmpl_with_kit)
product_without_kit = self.env['product.product'].search(
[('is_kits', '=', False)])
product_tmpl_without_kit = self.env['product.template'].search(
[('is_kits', '=', False)])
self.assertIn(no_kit_products, product_without_kit)
self.assertIn(no_kit_products.product_tmpl_id,
product_tmpl_without_kit)
self.assertNotIn(kit_products, product_without_kit)
self.assertNotIn(kit_products.product_tmpl_id,
product_tmpl_without_kit)
def test_20_bom_report(self):
""" Simulate a crumble receipt with mrp and open the bom structure
report and check that data insde are correct.
"""
uom_kg = self.env.ref('uom.product_uom_kgm')
uom_litre = self.env.ref('uom.product_uom_litre')
crumble = self.env['product.product'].create({
'name': 'Crumble',
'type': 'product',
'uom_id': uom_kg.id,
'uom_po_id': uom_kg.id,
})
butter = self.env['product.product'].create({
'name': 'Butter',
'type': 'product',
'uom_id': uom_kg.id,
'uom_po_id': uom_kg.id,
'standard_price': 7.01
})
biscuit = self.env['product.product'].create({
'name': 'Biscuit',
'type': 'product',
'uom_id': uom_kg.id,
'uom_po_id': uom_kg.id,
'standard_price': 1.5
})
bom_form_crumble = Form(self.env['mrp.bom'])
bom_form_crumble.product_tmpl_id = crumble.product_tmpl_id
bom_form_crumble.product_qty = 11
bom_form_crumble.product_uom_id = uom_kg
bom_crumble = bom_form_crumble.save()
workcenter = self.env['mrp.workcenter'].create({
'costs_hour': 10,
'name': 'Deserts Table'
})
# Required to display `operation_ids` in the form view
self.env.user.groups_id += self.env.ref("mrp.group_mrp_routings")
with Form(bom_crumble) as bom:
with bom.bom_line_ids.new() as line:
line.product_id = butter
line.product_uom_id = uom_kg
line.product_qty = 5
with bom.bom_line_ids.new() as line:
line.product_id = biscuit
line.product_uom_id = uom_kg
line.product_qty = 6
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter
operation.name = 'Prepare biscuits'
operation.time_cycle_manual = 5
operation.bom_id = bom_crumble # Can't handle by the testing env
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter
operation.name = 'Prepare butter'
operation.time_cycle_manual = 3
operation.bom_id = bom_crumble
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter
operation.name = 'Mix manually'
operation.time_cycle_manual = 5
operation.bom_id = bom_crumble
# TEST BOM STRUCTURE VALUE WITH BOM QUANTITY
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_crumble.id, searchQty=11, searchVariant=False)
# 5 min 'Prepare biscuits' + 3 min 'Prepare butter' + 5 min 'Mix manually' = 13 minutes for 1 biscuits so 13 * 11 = 143 minutes
self.assertEqual(report_values['lines']['operations_time'], 143.0, 'Operation time should be the same for 1 unit or for the batch')
# Operation cost is the sum of operation line.
self.assertEqual(float_compare(report_values['lines']['operations_cost'], 23.84, precision_digits=2), 0, '143 minute for 10$/hours -> 23.84')
for component_line in report_values['lines']['components']:
# standard price * bom line quantity * current quantity / bom finished product quantity
if component_line['product'].id == butter.id:
# 5 kg of butter at 7.01$ for 11kg of crumble -> 35.05$
self.assertEqual(float_compare(component_line['bom_cost'], (7.01 * 5), precision_digits=2), 0)
if component_line['product'].id == biscuit.id:
# 6 kg of biscuits at 1.50$ for 11kg of crumble -> 9$
self.assertEqual(float_compare(component_line['bom_cost'], (1.5 * 6), precision_digits=2), 0)
# total price = 35.05 + 9 + operation_cost(23.84) = 67.89
self.assertEqual(float_compare(report_values['lines']['bom_cost'], 67.89, precision_digits=2), 0, 'Product Bom Price is not correct')
self.assertEqual(float_compare(report_values['lines']['bom_cost'] / 11.0, 6.17, precision_digits=2), 0, 'Product Unit Bom Price is not correct')
# TEST BOM STRUCTURE VALUE BY UNIT
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_crumble.id, searchQty=1, searchVariant=False)
# 5 min 'Prepare biscuits' + 3 min 'Prepare butter' + 5 min 'Mix manually' = 13 minutes
self.assertEqual(report_values['lines']['operations_time'], 13.0, 'Operation time should be the same for 1 unit or for the batch')
# Operation cost is the sum of operation line.
operation_cost = float_round(5 / 60 * 10, precision_digits=2) * 2 + float_round(3 / 60 * 10, precision_digits=2)
self.assertEqual(float_compare(report_values['lines']['operations_cost'], operation_cost, precision_digits=2), 0, '13 minute for 10$/hours -> 2.16')
for component_line in report_values['lines']['components']:
# standard price * bom line quantity * current quantity / bom finished product quantity
if component_line['product'].id == butter.id:
# 5 kg of butter at 7.01$ for 11kg of crumble -> / 11 for price per unit (3.19)
self.assertEqual(float_compare(component_line['bom_cost'], (7.01 * 5) * (1 / 11), precision_digits=2), 0)
if component_line['product'].id == biscuit.id:
# 6 kg of biscuits at 1.50$ for 11kg of crumble -> / 11 for price per unit (0.82)
self.assertEqual(float_compare(component_line['bom_cost'], (1.5 * 6) * (1 / 11), precision_digits=2), 0)
# total price = 3.19 + 0.82 + operation_cost(0.83 + 0.83 + 0.5 = 2.16) = 6,17
self.assertEqual(float_compare(report_values['lines']['bom_cost'], 6.17, precision_digits=2), 0, 'Product Unit Bom Price is not correct')
# TEST OPERATION COST WHEN PRODUCED QTY > BOM QUANTITY
report_values_12 = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_crumble.id, searchQty=12, searchVariant=False)
report_values_22 = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_crumble.id, searchQty=22, searchVariant=False)
#Operation cost = 47.66 € = 256 (min) * 10€/h
self.assertEqual(float_compare(report_values_22['lines']['operations_cost'], 47.66, precision_digits=2), 0, 'Operation cost is not correct')
# Create a more complex BoM with a sub product
cheese_cake = self.env['product.product'].create({
'name': 'Cheese Cake 300g',
'type': 'product',
})
cream = self.env['product.product'].create({
'name': 'cream',
'type': 'product',
'uom_id': uom_litre.id,
'uom_po_id': uom_litre.id,
'standard_price': 5.17,
})
bom_form_cheese_cake = Form(self.env['mrp.bom'])
bom_form_cheese_cake.product_tmpl_id = cheese_cake.product_tmpl_id
bom_form_cheese_cake.product_qty = 60
bom_form_cheese_cake.product_uom_id = self.uom_unit
bom_cheese_cake = bom_form_cheese_cake.save()
workcenter_2 = self.env['mrp.workcenter'].create({
'name': 'cake mounting',
'costs_hour': 20,
'time_start': 10,
'time_stop': 15
})
self.env['mrp.workcenter.capacity'].create({
'product_id': cheese_cake.id,
'workcenter_id': workcenter_2.id,
'time_start': 2,
'time_stop': 1,
})
with Form(bom_cheese_cake) as bom:
with bom.bom_line_ids.new() as line:
line.product_id = cream
line.product_uom_id = uom_litre
line.product_qty = 3
with bom.bom_line_ids.new() as line:
line.product_id = crumble
line.product_uom_id = uom_kg
line.product_qty = 5.4
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter
operation.name = 'Mix cheese and crumble'
operation.time_cycle_manual = 10
operation.bom_id = bom_cheese_cake
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter_2
operation.name = 'Cake mounting'
operation.time_cycle_manual = 5
operation.bom_id = bom_cheese_cake
# TEST CHEESE BOM STRUCTURE VALUE WITH BOM QUANTITY
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_cheese_cake.id, searchQty=60, searchVariant=False)
#Operation time = 15 min * 60 + time_start + time_stop + capacity_time_start + capacity_time_stop= 928
self.assertEqual(report_values['lines']['operations_time'], 928.0, 'Operation time should be the same for 1 unit or for the batch')
# Operation cost is the sum of operation line : (60 * 10)/60 * 10€ + (10 + 15 + 60 * 5)/60 * 20€ + (1 + 2)/60 * 20€ = 209,33€
self.assertEqual(float_compare(report_values['lines']['operations_cost'], 209.33, precision_digits=2), 0)
for component_line in report_values['lines']['components']:
# standard price * bom line quantity * current quantity / bom finished product quantity
if component_line['product'].id == cream.id:
# 3 liter of cream at 5.17$ for 60 unit of cheese cake -> 15.51$
self.assertEqual(float_compare(component_line['bom_cost'], (3 * 5.17), precision_digits=2), 0)
if component_line['product'].id == crumble.id:
# 5.4 kg of crumble at the cost of a batch.
crumble_cost = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_crumble.id, searchQty=5.4, searchVariant=False)['lines']['bom_cost']
self.assertEqual(float_compare(component_line['bom_cost'], crumble_cost, precision_digits=2), 0)
# total price = Cream (15.51€) + crumble_cost (34.63 €) + operation_cost(209,33) = 259.47€
self.assertEqual(float_compare(report_values['lines']['bom_cost'], 259.47, precision_digits=2), 0, 'Product Bom Price is not correct')
def test_bom_report_dozens(self):
""" Simulate a drawer bom with dozens as bom units
"""
uom_dozen = self.env.ref('uom.product_uom_dozen')
uom_unit = self.env.ref('uom.product_uom_unit')
drawer = self.env['product.product'].create({
'name': 'drawer',
'type': 'product',
'uom_id': uom_unit.id,
'uom_po_id': uom_unit.id,
})
screw = self.env['product.product'].create({
'name': 'screw',
'type': 'product',
'uom_id': uom_unit.id,
'uom_po_id': uom_unit.id,
'standard_price': 7.01
})
bom_form_drawer = Form(self.env['mrp.bom'])
bom_form_drawer.product_tmpl_id = drawer.product_tmpl_id
bom_form_drawer.product_qty = 11
bom_form_drawer.product_uom_id = uom_dozen
bom_drawer = bom_form_drawer.save()
workcenter = self.env['mrp.workcenter'].create({
'costs_hour': 10,
'name': 'Deserts Table'
})
# Required to display `operation_ids` in the form view
self.env.user.groups_id += self.env.ref("mrp.group_mrp_routings")
with Form(bom_drawer) as bom:
with bom.bom_line_ids.new() as line:
line.product_id = screw
line.product_uom_id = uom_unit
line.product_qty = 5
with bom.operation_ids.new() as operation:
operation.workcenter_id = workcenter
operation.name = 'Screw drawer'
operation.time_cycle_manual = 5
operation.bom_id = bom_drawer
# TEST BOM STRUCTURE VALUE WITH BOM QUANTITY
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_drawer.id, searchQty=11, searchVariant=False)
# 5 min 'Prepare biscuits' + 3 min 'Prepare butter' + 5 min 'Mix manually' = 13 minutes
self.assertEqual(report_values['lines']['operations_time'], 660.0, 'Operation time should be the same for 1 unit or for the batch')
def test_21_bom_report_variant(self):
""" Test a sub BoM process with multiple variants.
BOM 1:
product template = car
quantity = 5 units
- red paint 50l -> red car (product.product)
- blue paint 50l -> blue car
- red dashboard with gps -> red car with GPS
- red dashboard w/h gps -> red w/h GPS
- blue dashboard with gps -> blue car with GPS
- blue dashboard w/h gps -> blue w/h GPS
BOM 2:
product_tmpl = dashboard
quantity = 2
- red paint 1l -> red dashboard (product.product)
- blue paint 1l -> blue dashboard
- gps -> dashboard with gps
Check the Price for a Blue Car with GPS -> 910$:
10l of blue paint -> 200$
1 blue dashboard GPS -> 710$:
- 0.5l of blue paint -> 10$
- GPS -> 700$
Check the price for a red car -> 10.5l of red paint -> 210$
"""
# Create a product template car with attributes gps(yes, no), color(red, blue)
self.car = self.env['product.template'].create({
'name': 'Car',
})
self.gps_attribute = self.env['product.attribute'].create({'name': 'GPS', 'sequence': 1})
self.gps_yes = self.env['product.attribute.value'].create({
'name': 'Yes',
'attribute_id': self.gps_attribute.id,
'sequence': 1,
})
self.gps_no = self.env['product.attribute.value'].create({
'name': 'No',
'attribute_id': self.gps_attribute.id,
'sequence': 2,
})
self.car_gps_attribute_line = self.env['product.template.attribute.line'].create({
'product_tmpl_id': self.car.id,
'attribute_id': self.gps_attribute.id,
'value_ids': [(6, 0, [self.gps_yes.id, self.gps_no.id])],
})
self.car_gps_yes = self.car_gps_attribute_line.product_template_value_ids[0]
self.car_gps_no = self.car_gps_attribute_line.product_template_value_ids[1]
self.color_attribute = self.env['product.attribute'].create({'name': 'Color', 'sequence': 1})
self.color_red = self.env['product.attribute.value'].create({
'name': 'Red',
'attribute_id': self.color_attribute.id,
'sequence': 1,
})
self.color_blue = self.env['product.attribute.value'].create({
'name': 'Blue',
'attribute_id': self.color_attribute.id,
'sequence': 2,
})
self.car_color_attribute_line = self.env['product.template.attribute.line'].create({
'product_tmpl_id': self.car.id,
'attribute_id': self.color_attribute.id,
'value_ids': [(6, 0, [self.color_red.id, self.color_blue.id])],
})
self.car_color_red = self.car_color_attribute_line.product_template_value_ids[0]
self.car_color_blue = self.car_color_attribute_line.product_template_value_ids[1]
# Blue and red paint
uom_litre = self.env.ref('uom.product_uom_litre')
self.paint = self.env['product.template'].create({
'name': 'Paint',
'uom_id': uom_litre.id,
'uom_po_id': uom_litre.id
})
self.paint_color_attribute_line = self.env['product.template.attribute.line'].create({
'product_tmpl_id': self.paint.id,
'attribute_id': self.color_attribute.id,
'value_ids': [(6, 0, [self.color_red.id, self.color_blue.id])],
})
self.paint_color_red = self.paint_color_attribute_line.product_template_value_ids[0]
self.paint_color_blue = self.paint_color_attribute_line.product_template_value_ids[1]
self.paint.product_variant_ids.write({'standard_price': 20})
self.dashboard = self.env['product.template'].create({
'name': 'Dashboard',
'standard_price': 1000,
})
self.dashboard_gps_attribute_line = self.env['product.template.attribute.line'].create({
'product_tmpl_id': self.dashboard.id,
'attribute_id': self.gps_attribute.id,
'value_ids': [(6, 0, [self.gps_yes.id, self.gps_no.id])],
})
self.dashboard_gps_yes = self.dashboard_gps_attribute_line.product_template_value_ids[0]
self.dashboard_gps_no = self.dashboard_gps_attribute_line.product_template_value_ids[1]
self.dashboard_color_attribute_line = self.env['product.template.attribute.line'].create({
'product_tmpl_id': self.dashboard.id,
'attribute_id': self.color_attribute.id,
'value_ids': [(6, 0, [self.color_red.id, self.color_blue.id])],
})
self.dashboard_color_red = self.dashboard_color_attribute_line.product_template_value_ids[0]
self.dashboard_color_blue = self.dashboard_color_attribute_line.product_template_value_ids[1]
self.gps = self.env['product.product'].create({
'name': 'GPS',
'standard_price': 700,
})
bom_form_car = Form(self.env['mrp.bom'])
bom_form_car.product_tmpl_id = self.car
bom_form_car.product_qty = 5
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.paint._get_variant_for_combination(self.paint_color_red)
line.product_uom_id = uom_litre
line.product_qty = 50
line.bom_product_template_attribute_value_ids.add(self.car_color_red)
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.paint._get_variant_for_combination(self.paint_color_blue)
line.product_uom_id = uom_litre
line.product_qty = 50
line.bom_product_template_attribute_value_ids.add(self.car_color_blue)
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.dashboard._get_variant_for_combination(self.dashboard_gps_yes + self.dashboard_color_red)
line.product_qty = 5
line.bom_product_template_attribute_value_ids.add(self.car_gps_yes)
line.bom_product_template_attribute_value_ids.add(self.car_color_red)
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.dashboard._get_variant_for_combination(self.dashboard_gps_yes + self.dashboard_color_blue)
line.product_qty = 5
line.bom_product_template_attribute_value_ids.add(self.car_gps_yes)
line.bom_product_template_attribute_value_ids.add(self.car_color_blue)
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.dashboard._get_variant_for_combination(self.dashboard_gps_no + self.dashboard_color_red)
line.product_qty = 5
line.bom_product_template_attribute_value_ids.add(self.car_gps_no)
line.bom_product_template_attribute_value_ids.add(self.car_color_red)
with bom_form_car.bom_line_ids.new() as line:
line.product_id = self.dashboard._get_variant_for_combination(self.dashboard_gps_no + self.dashboard_color_blue)
line.product_qty = 5
line.bom_product_template_attribute_value_ids.add(self.car_gps_no)
line.bom_product_template_attribute_value_ids.add(self.car_color_blue)
bom_car = bom_form_car.save()
bom_dashboard = Form(self.env['mrp.bom'])
bom_dashboard.product_tmpl_id = self.dashboard
bom_dashboard.product_qty = 2
with bom_dashboard.bom_line_ids.new() as line:
line.product_id = self.paint._get_variant_for_combination(self.paint_color_red)
line.product_uom_id = uom_litre
line.product_qty = 1
line.bom_product_template_attribute_value_ids.add(self.dashboard_color_red)
with bom_dashboard.bom_line_ids.new() as line:
line.product_id = self.paint._get_variant_for_combination(self.paint_color_blue)
line.product_uom_id = uom_litre
line.product_qty = 1
line.bom_product_template_attribute_value_ids.add(self.dashboard_color_blue)
with bom_dashboard.bom_line_ids.new() as line:
line.product_id = self.gps
line.product_qty = 2
line.bom_product_template_attribute_value_ids.add(self.dashboard_gps_yes)
bom_dashboard = bom_dashboard.save()
blue_car_with_gps = self.car._get_variant_for_combination(self.car_color_blue + self.car_gps_yes)
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_car.id, searchQty=1, searchVariant=blue_car_with_gps.id)
# Two lines. blue dashboard with gps and blue paint.
self.assertEqual(len(report_values['lines']['components']), 2)
# 10l of blue paint
blue_paint = self.paint._get_variant_for_combination(self.paint_color_blue)
self.assertEqual(blue_paint.id, report_values['lines']['components'][0]['product'].id)
self.assertEqual(report_values['lines']['components'][0]['quantity'], 10)
# 1 blue dashboard with GPS
blue_dashboard_gps = self.dashboard._get_variant_for_combination(self.dashboard_color_blue + self.dashboard_gps_yes)
self.assertEqual(blue_dashboard_gps.id, report_values['lines']['components'][1]['product'].id)
self.assertEqual(report_values['lines']['components'][1]['quantity'], 1)
report_values_dashboad = report_values['lines']['components'][1]
self.assertEqual(len(report_values_dashboad['components']), 2)
self.assertEqual(blue_paint.id, report_values_dashboad['components'][0]['product'].id)
self.assertEqual(self.gps.id, report_values_dashboad['components'][1]['product'].id)
# 0.5l of paint at price of 20$/litre -> 10$
self.assertEqual(report_values_dashboad['components'][0]['bom_cost'], 10)
# GPS 700$
self.assertEqual(report_values_dashboad['components'][1]['bom_cost'], 700)
# Dashboard blue with GPS should have a BoM cost of 710$
self.assertEqual(report_values['lines']['components'][1]['bom_cost'], 710)
# 10l of paint at price of 20$/litre -> 200$
self.assertEqual(report_values['lines']['components'][0]['bom_cost'], 200)
# Total cost of blue car with GPS: 10 + 700 + 200 = 910
self.assertEqual(report_values['lines']['bom_cost'], 910)
red_car_without_gps = self.car._get_variant_for_combination(self.car_color_red + self.car_gps_no)
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_car.id, searchQty=1, searchVariant=red_car_without_gps.id)
# Same math than before but without GPS
self.assertEqual(report_values['lines']['bom_cost'], 210)
def test_22_bom_report_recursive_bom(self):
""" Test report with recursive BoM and different quantities.
BoM 1:
product = Finished (units)
quantity = 100 units
- Semi-Finished 5 kg
BoM 2:
product = Semi-Finished (kg)
quantity = 11 kg
- Assembly 2 dozens
BoM 3:
product = Assembly (dozens)
quantity = 5 dozens
- Raw Material 4 litres (product.product 5$/litre)
Check the Price for 80 units of Finished -> 2.92$:
"""
# Create a products templates
uom_unit = self.env.ref('uom.product_uom_unit')
uom_kg = self.env.ref('uom.product_uom_kgm')
uom_dozen = self.env.ref('uom.product_uom_dozen')
uom_litre = self.env.ref('uom.product_uom_litre')
finished = self.env['product.product'].create({
'name': 'Finished',
'type': 'product',
'uom_id': uom_unit.id,
'uom_po_id': uom_unit.id,
})
semi_finished = self.env['product.product'].create({
'name': 'Semi-Finished',
'type': 'product',
'uom_id': uom_kg.id,
'uom_po_id': uom_kg.id,
})
assembly = self.env['product.product'].create({
'name': 'Assembly',
'type': 'product',
'uom_id': uom_dozen.id,
'uom_po_id': uom_dozen.id,
})
raw_material = self.env['product.product'].create({
'name': 'Raw Material',
'type': 'product',
'uom_id': uom_litre.id,
'uom_po_id': uom_litre.id,
'standard_price': 5,
})
#Create bom
bom_finished = Form(self.env['mrp.bom'])
bom_finished.product_tmpl_id = finished.product_tmpl_id
bom_finished.product_qty = 100
with bom_finished.bom_line_ids.new() as line:
line.product_id = semi_finished
line.product_uom_id = uom_kg
line.product_qty = 5
bom_finished = bom_finished.save()
bom_semi_finished = Form(self.env['mrp.bom'])
bom_semi_finished.product_tmpl_id = semi_finished.product_tmpl_id
bom_semi_finished.product_qty = 11
with bom_semi_finished.bom_line_ids.new() as line:
line.product_id = assembly
line.product_uom_id = uom_dozen
line.product_qty = 2
bom_semi_finished = bom_semi_finished.save()
bom_assembly = Form(self.env['mrp.bom'])
bom_assembly.product_tmpl_id = assembly.product_tmpl_id
bom_assembly.product_qty = 5
with bom_assembly.bom_line_ids.new() as line:
line.product_id = raw_material
line.product_uom_id = uom_litre
line.product_qty = 4
bom_assembly = bom_assembly.save()
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom_finished.id, searchQty=80)
self.assertAlmostEqual(report_values['lines']['bom_cost'], 2.92)
def test_bom_report_capacity_with_quantity_of_0(self):
uom_unit = self.env.ref('uom.product_uom_unit')
location = self.env.ref('stock.stock_location_stock')
target = self.env['product.product'].create({
'name': 'Target',
'type': 'product',
})
product_one = self.env['product.product'].create({
'name': 'Component one',
'type': 'product',
})
self.env['stock.quant']._update_available_quantity(product_one, location, 3.0)
product_two = self.env['product.product'].create({
'name': 'Component two',
'type': 'product',
})
self.env['stock.quant']._update_available_quantity(product_two, location, 4.0)
bom = self.env['mrp.bom'].create({
'product_tmpl_id': target.product_tmpl_id.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 1.0,
'type': 'phantom',
'bom_line_ids': [
Command.create({
'product_id': product_one.id,
'product_qty': 0,
'product_uom_id': uom_unit.id,
}),
Command.create({
'product_id': product_two.id,
'product_qty': 0.1,
'product_uom_id': uom_unit.id,
})
]
})
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom.id)
# The first product shouldn't affect the producible quantity because the target needs none of it
# So with 4 of the second product available, we can produce 40 items
self.assertEqual(report_values["lines"]["producible_qty"], 40)
def test_bom_report_capacity_with_duplicate_components(self):
location = self.env.ref('stock.stock_location_stock')
self.env['stock.quant']._update_available_quantity(self.product_2, location, 2.0)
bom = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_3.product_tmpl_id.id,
'product_qty': 1,
'bom_line_ids': [
Command.create({
'product_id': self.product_2.id,
'product_qty': 2,
}),
Command.create({
'product_id': self.product_2.id,
'product_qty': 2,
})
]
})
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom.id)
# Total quantity of components is 4, so shouldn't be able to produce a single one.
self.assertEqual(report_values['lines']['producible_qty'], 0)
def test_bom_report_same_component(self):
""" Test report bom structure with duplicated components.
"""
location = self.env.ref('stock.stock_location_stock')
uom_unit = self.env.ref('uom.product_uom_unit')
final_product_tmpl = self.env['product.template'].create({'name': 'Final Product', 'type': 'product'})
component_product = self.env['product.product'].create({'name': 'Compo 1', 'type': 'product'})
self.env['stock.quant']._update_available_quantity(component_product, location, 3.0)
bom = self.env['mrp.bom'].create({
'product_tmpl_id': final_product_tmpl.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
Command.create({
'product_id': component_product.id,
'product_qty': 3,
'product_uom_id': uom_unit.id,
}),
Command.create({
'product_id': component_product.id,
'product_qty': 3,
'product_uom_id': uom_unit.id,
})
]
})
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=bom.id)
line1_values = report_values['lines']['components'][0]
line2_values = report_values['lines']['components'][1]
self.assertEqual(line1_values['availability_state'], 'available', 'The first component should be available.')
self.assertEqual(line2_values['availability_state'], 'unavailable', 'The second component should be marked as unavailable')
def test_report_data_bom_with_0_qty(self):
"""
Test that a bom with a child-bom set with a zero qty will still have have 0 qty for the child-bom on the report.
"""
self.bom_4.bom_line_ids = [(0, 0, {
'product_id': self.bom_2.product_id.id,
'product_qty': 1.0,
})]
self.bom_4.bom_line_ids.product_qty = 0
report_values = self.env['report.mrp.report_bom_structure']._get_report_data(bom_id=self.bom_4.id, searchQty=1, searchVariant=False)
self.assertEqual(sum([value['quantity'] for value in report_values['lines']['components'][:2]]), 0, 'The quantity should be set to 0 for all components of the bom.')
def test_validate_no_bom_line_with_same_product(self):
"""
Cannot set a BOM line on a BOM with the same product as the BOM itself
"""
uom_unit = self.env.ref('uom.product_uom_unit')
finished = self.env['product.product'].create({
'name': 'Finished',
'type': 'product',
'uom_id': uom_unit.id,
'uom_po_id': uom_unit.id,
})
bom_finished = Form(self.env['mrp.bom'])
bom_finished.product_tmpl_id = finished.product_tmpl_id
bom_finished.product_qty = 100
with bom_finished.bom_line_ids.new() as line:
line.product_id = finished
line.product_uom_id = uom_unit
line.product_qty = 5
with self.assertRaises(exceptions.ValidationError), self.cr.savepoint():
bom_finished = bom_finished.save()
def test_validate_no_bom_line_with_same_product_variant(self):
"""
Cannot set a BOM line on a BOM with the same product variant as the BOM itself
"""
uom_unit = self.env.ref('uom.product_uom_unit')
bom_finished = Form(self.env['mrp.bom'])
bom_finished.product_tmpl_id = self.product_7_template
bom_finished.product_id = self.product_7_3
bom_finished.product_qty = 100
with bom_finished.bom_line_ids.new() as line:
line.product_id = self.product_7_3
line.product_uom_id = uom_unit
line.product_qty = 5
with self.assertRaises(exceptions.ValidationError), self.cr.savepoint():
bom_finished = bom_finished.save()
def test_validate_bom_line_with_different_product_variant(self):
"""
Can set a BOM line on a BOM with a different product variant as the BOM itself (same product)
Usecase for example A black T-shirt made from a white T-shirt and
black color.
"""
uom_unit = self.env.ref('uom.product_uom_unit')
bom_finished = Form(self.env['mrp.bom'])
bom_finished.product_tmpl_id = self.product_7_template
bom_finished.product_id = self.product_7_3
bom_finished.product_qty = 100
with bom_finished.bom_line_ids.new() as line:
line.product_id = self.product_7_2
line.product_uom_id = uom_unit
line.product_qty = 5
bom_finished = bom_finished.save()
def test_validate_bom_line_with_variant_of_bom_product(self):
"""
Can set a BOM line on a BOM with a product variant when the BOM has no variant selected
"""
uom_unit = self.env.ref('uom.product_uom_unit')
bom_finished = Form(self.env['mrp.bom'])
bom_finished.product_tmpl_id = self.product_6.product_tmpl_id
# no product_id
bom_finished.product_qty = 100
with bom_finished.bom_line_ids.new() as line:
line.product_id = self.product_7_2
line.product_uom_id = uom_unit
line.product_qty = 5
bom_finished = bom_finished.save()
def test_replenishment(self):
""" Tests the auto generation of manual orderpoints.
The multiple quantity of the orderpoint should be the
quantity of the BoM in the UoM of the product.
"""
uom_kg = self.env.ref('uom.product_uom_kgm')
uom_gram = self.env.ref('uom.product_uom_gram')
manufacturing_route_id = self.ref('mrp.route_warehouse0_manufacture')
product_gram = self.env['product.product'].create({
'name': 'Product sold in grams',
'type': 'product',
'uom_id': uom_gram.id,
'uom_po_id': uom_gram.id,
'route_ids': [(4, manufacturing_route_id)],
})
# We create a BoM that manufactures 2kg of product
self.env['mrp.bom'].create({
'product_id': product_gram.id,
'product_tmpl_id': product_gram.product_tmpl_id.id,
'product_uom_id': uom_kg.id,
'product_qty': 2.0,
'type': 'normal',
})
# We create a delivery order of 2300 grams
picking_form = Form(self.env['stock.picking'])
picking_form.picking_type_id = self.env.ref('stock.picking_type_out')
with picking_form.move_ids_without_package.new() as move:
move.product_id = product_gram
move.product_uom_qty = 2300.0
customer_picking = picking_form.save()
customer_picking.action_confirm()
# We check the created orderpoint
self.env.flush_all()
self.env['stock.warehouse.orderpoint']._get_orderpoint_action()
orderpoint = self.env['stock.warehouse.orderpoint'].search([('product_id', '=', product_gram.id)])
self.assertEqual(orderpoint.route_id.id, manufacturing_route_id)
self.assertEqual(orderpoint.qty_multiple, 2000.0)
self.assertEqual(orderpoint.qty_to_order, 4000.0)
def test_bom_kit_with_sub_kit(self):
p1, p2, p3, p4, p5, p6 = self.make_prods(6)
prod1, prod2 = self.make_prods(2)
self.make_bom(p1, p2, p3)
self.make_bom(p2, p3, p4)
bom = self.make_bom(prod1, prod2)
bom.product_qty = 100
bom = self.make_bom(p5, p6)
bom.bom_line_ids[0].product_qty = 0.1
loc = self.env.ref("stock.stock_location_stock")
self.env["stock.quant"]._update_available_quantity(p3, loc, 10)
self.env["stock.quant"]._update_available_quantity(p4, loc, 10)
self.env["stock.quant"]._update_available_quantity(prod2, loc, 5.57)
self.env["stock.quant"]._update_available_quantity(prod2, loc, -5)
self.env["stock.quant"]._update_available_quantity(p6, loc, 5.5)
self.env["stock.quant"]._update_available_quantity(p6, loc, -5.2)
self.assertEqual(p1.qty_available, 5.0)
self.assertEqual(p2.qty_available, 10.0)
self.assertEqual(p3.qty_available, 10.0)
self.assertEqual(prod1.qty_available, 57.0)
self.assertEqual(p5.qty_available, 3.0)
def test_operation_blocked_by_another_operation(self):
""" Test that an operation is not blocked by another operation if the variant is different
Product with 4 variants (red big, red medium, blue big, blue medium)
BoM:
- OP1 (apply on Red)
- OP2 (blocked by OP1)
Create a MO for Red big, OP1 is started, OP2 should be blocked
Create a Mo for Blue big, OP1 is not applied, OP2 should not be blocked
"""
ProductAttribute = self.env['product.attribute']
ProductAttributeValue = self.env['product.attribute.value']
# Product Attribute
att_color = ProductAttribute.create({'name': 'Color', 'sequence': 1})
att_size = ProductAttribute.create({'name': 'size', 'sequence': 2})
# Product Attribute color Value
att_color_red = ProductAttributeValue.create({'name': 'red', 'attribute_id': att_color.id, 'sequence': 1})
att_color_blue = ProductAttributeValue.create({'name': 'blue', 'attribute_id': att_color.id, 'sequence': 2})
# Product Attribute size Value
att_size_big = ProductAttributeValue.create({'name': 'big', 'attribute_id': att_size.id, 'sequence': 1})
att_size_medium = ProductAttributeValue.create({'name': 'medium', 'attribute_id': att_size.id, 'sequence': 2})
# Create create a product with 4 variants
product_template = self.env['product.template'].create({
'name': 'Sofa',
'attribute_line_ids': [
(0, 0, {
'attribute_id': att_color.id,
'value_ids': [(6, 0, [att_color_red.id, att_color_blue.id])]
}),
(0, 0, {
'attribute_id': att_size.id,
'value_ids': [(6, 0, [att_size_big.id, att_size_medium.id])]
})
]
})
bom = self.env['mrp.bom'].create({
'product_tmpl_id': product_template.id,
'product_uom_id': self.uom_unit.id,
'product_qty': 1.0,
'allow_operation_dependencies': True,
'operation_ids': [(0, 0, {'name': 'op1', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 1.0, 'bom_product_template_attribute_value_ids': [(4, att_color_blue.pav_attribute_line_ids.product_template_value_ids[0].id)]}),
(0, 0, {'name': 'op2', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 1.0})],
})
# Make 1st workorder depend on 2nd
bom.operation_ids[1].blocked_by_operation_ids = [Command.link(bom.operation_ids[0].id)]
# Make MO for red big
mo_form = Form(self.env['mrp.production'])
mo_form.product_id = product_template.product_variant_ids[0]
mo_form.bom_id = bom
mo_form.product_qty = 1.0
mo = mo_form.save()
mo.action_confirm()
self.assertEqual(mo.state, 'confirmed')
# Make MO for blue big
mo_form = Form(self.env['mrp.production'])
mo_form.product_id = product_template.product_variant_ids[2]
mo_form.bom_id = bom
mo_form.product_qty = 1.0
mo = mo_form.save()
mo.action_confirm()
self.assertEqual(mo.state, 'confirmed')
mo.qty_producing = 1.0
mo.action_assign()
mo.button_plan()
mo.button_mark_done()
self.assertEqual(mo.state, 'done')
def test_cycle_on_line_creation(self):
bom_1_finished_product = self.bom_1.product_id
bom_2_finished_product = self.bom_2.product_id
with self.assertRaises(exceptions.ValidationError):
# finished product is one of the components:
self.bom_1.bom_line_ids = [(0, 0, {'product_id': bom_1_finished_product.id, 'product_qty': 1.0},)]
with self.assertRaises(exceptions.ValidationError):
# cycle:
self.bom_1.bom_line_ids = [(0, 0, {'product_id': bom_2_finished_product.id, 'product_qty': 1.0},)]
def test_cycle_on_line_update(self):
lines = self.bom_1.bom_line_ids
bom_2_finished_product = self.bom_2.product_id
with self.assertRaises(exceptions.ValidationError):
self.bom_1.bom_line_ids = [(1, lines[0].id, {'product_id': bom_2_finished_product.id})]
def test_cycle_on_bom_unarchive(self):
finished_product = self.bom_1.product_id
component = self.bom_1.bom_line_ids.product_id[0]
self.bom_1.active = False
self.env['mrp.bom'].create({
'product_id': component.id,
'product_tmpl_id': component.product_tmpl_id.id,
'product_uom_id': component.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': finished_product.id, 'product_qty': 1.0}),
],
})
with self.assertRaises(exceptions.ValidationError):
self.bom_1.active = True
def test_cycle_on_bom_creation(self):
finished_product = self.bom_4.product_id
component = self.bom_4.bom_line_ids.product_id
with self.assertRaises(exceptions.ValidationError):
self.env['mrp.bom'].create({
'product_id': component.id,
'product_tmpl_id': component.product_tmpl_id.id,
'product_uom_id': component.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': finished_product.id, 'product_qty': 1.0}),
],
})
def test_indirect_cycle_on_bom_creation(self):
"""
Three BoMs:
A -> D
A -> B
B -> C
Create a new BoM C -> A. At first glance, this new BoM is ok because it
does nat have a cycle (C -> A -> D). But there is an indirect cycle:
A -> B -> C -> A
Hence this new BoM should raise an error.
"""
product_A, product_B, product_C, product_D = self.env['product.product'].create([{
'name': '%s' % i
} for i in range(4)])
self.env['mrp.bom'].create([{
'product_id': finished.id,
'product_tmpl_id': finished.product_tmpl_id.id,
'product_uom_id': finished.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': compo.id, 'product_qty': 1.0}),
],
} for finished, compo, in [
(product_A, product_D),
(product_A, product_B),
(product_B, product_C),
]])
with self.assertRaises(exceptions.ValidationError):
self.env['mrp.bom'].create({
'product_id': product_C.id,
'product_tmpl_id': product_C.product_tmpl_id.id,
'product_uom_id': product_C.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': product_A.id, 'product_qty': 1.0}),
],
})
def test_cycle_on_bom_sequencing(self):
"""
Six BoMs:
A -> D
A -> B
C -> D
C -> E
B -> C
C -> A
First new sequence: we reverse C->D and C->E, this is ok as it does not
create any cycle. Change the sequence again and set C->A before C->D: it
should raise an error because C->A becomes the main BoM of C, and this
will create a cycle: A -> B -> C -> A
"""
product_A, product_B, product_C, product_D, product_E = self.env['product.product'].create([{
'name': '%s' % i
} for i in range(5)])
boms = self.env['mrp.bom'].create([{
'product_id': finished.id,
'product_tmpl_id': finished.product_tmpl_id.id,
'product_uom_id': finished.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': compo.id, 'product_qty': 1.0}),
],
} for finished, compo, in [
(product_A, product_D),
(product_A, product_B),
(product_C, product_D),
(product_C, product_E),
(product_B, product_C),
(product_C, product_A),
]])
# simulate resequence from UI (reverse C->D and C->E)
# (see odoo/addons/web/controllers/main.py:1352)
boms.invalidate_recordset()
for i, record in enumerate(boms[0] | boms[1] | boms[3] | boms[2] | boms[4] | boms[5]):
record.write({'sequence': i})
# simulate a second resequencing (set C->A before C->D)
with self.assertRaises(exceptions.ValidationError):
for i, record in enumerate(boms[0] | boms[1] | boms[5] | boms[3] | boms[2] | boms[4]):
record.write({'sequence': i})
def test_cycle_on_legit_apply_variants(self):
""" Should not raise anything """
self.env['mrp.bom'].create({
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.product_7_template.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {
'product_id': self.product_1.id,
'product_qty': 1.0
}),
(0, 0, {
'product_id': self.product_2.id,
'product_qty': 1.0,
'bom_product_template_attribute_value_ids': [(4, self.product_7_attr1_v2.id)]
}),
],
})
self.env['mrp.bom'].create({
'product_tmpl_id': self.product_2.product_tmpl_id.id,
'product_uom_id': self.product_2.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [
(0, 0, {'product_id': self.product_7_1.id, 'product_qty': 1.0}),
],
})
def test_component_when_bom_change(self):
"""
Checks that the component of the previous BoM is removed when another BoM is set on the MO:
- Create a product with 2 BoMs:
BoM 1: compoennt 1
BoM 2: component 2
- Create a MO for the product with BoM 1
- check that the component 1 is set
- change the BoM on the MO to BoM 2
- come back to BoM 1
- check that the component 2 is removed and replaced by the component 1
"""
# Create BoM 1 with component 1
bom_1 = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.product_7_template.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [Command.create({
'product_id': self.product_1.id,
'product_qty': 1.0,
})],
})
# Create BoM 2 with component 2
bom_2 = self.env['mrp.bom'].create({
'product_tmpl_id': self.product_7_template.id,
'product_uom_id': self.product_7_template.uom_id.id,
'product_qty': 1.0,
'type': 'normal',
'bom_line_ids': [Command.create({
'product_id': self.product_2.id,
'product_qty': 1.0,
})],
})
# Create a MO with BoM 1
mo = self.env['mrp.production'].create({
'product_qty': 1.0,
'bom_id': bom_1.id,
})
# Check that component 1 is set
self.assertEqual(mo.move_raw_ids.product_id, self.product_1)
# Change BoM in the MO to BoM 2
mo_form = Form(mo)
mo_form.bom_id = bom_2
# Check that component 2 is set
self.assertEqual(mo_form.move_raw_ids._records[0]['product_id'], self.product_2.id)
self.assertEqual(len(mo_form.move_raw_ids._records), 1)
# Revert back to BoM 1
mo_form.bom_id = bom_1
# Check that component 1 is set again and component 2 is removed
self.assertEqual(mo_form.move_raw_ids._records[0]['product_id'], self.product_1.id)
self.assertEqual(len(mo_form.move_raw_ids._records), 1)
def test_update_bom_in_routing_workcenter(self):
"""
This test checks the behaviour of updating the BoM associated with a routing workcenter,
It verifies that the link between the BOM lines and the operation is correctly deleted.
"""
resource_calendar_std_id = self.env.ref('resource.resource_calendar_std').id
mrp_workcenter_1 = self.env['mrp.workcenter'].create({
'name': 'Drill Station 1',
'resource_calendar_id': resource_calendar_std_id,
})
p1, c1, c2, byproduct = self.make_prods(4)
bom = self.env['mrp.bom'].create({
'product_tmpl_id': p1.product_tmpl_id.id,
'product_qty': 1.0,
'bom_line_ids': [
Command.create({'product_id': c1.id, 'product_qty': 1.0}),
Command.create({'product_id': c2.id, 'product_qty': 1.0})
],
'byproduct_ids': [
Command.create({
'product_id': byproduct.id, 'product_uom_id': byproduct.uom_id.id, 'product_qty': 1.0,
})]
})
operation_1, operation_2 = self.env['mrp.routing.workcenter'].create([
{
'name': 'Operation 1',
'workcenter_id': mrp_workcenter_1.id,
'bom_id': bom.id,
},
{
'name': 'Operation 2',
'workcenter_id': mrp_workcenter_1.id,
'bom_id': bom.id,
}
])
bom.bom_line_ids.operation_id = operation_1
bom.byproduct_ids.operation_id = operation_1
operation_2.blocked_by_operation_ids = operation_1
self.assertEqual(operation_1.bom_id, bom)
operation_1.bom_id = self.bom_1
self.assertEqual(operation_1.bom_id, self.bom_1)
self.assertFalse(bom.bom_line_ids.operation_id)
self.assertFalse(bom.byproduct_ids.operation_id)
self.assertFalse(operation_2.blocked_by_operation_ids)
def test_compute_days_to_prepare_from_mo_if_unavailable(self):
"""
Checks that a notification is sent when at least one component can not be resupplied.
"""
product = self.bom_1.product_id
manufacturing_route_id = self.ref('mrp.route_warehouse0_manufacture')
product.route_ids = [Command.set([manufacturing_route_id])]
notification = product.product_tmpl_id.action_compute_bom_days()
self.assertEqual(product.days_to_prepare_mo, 0.0)
self.assertEqual((notification['type'], notification['tag']), ('ir.actions.client', 'display_notification'))
def test_archive_operation(self):
""" Checks that archiving an operation having both a bom line and a byproduct line linked to it properly unlinks them.
"""
final, comp1, comp2, bp1, bp2 = self.make_prods(5)
bom = self.env['mrp.bom'].create({
'product_tmpl_id': final.product_tmpl_id.id,
'product_qty': 1.0,
'bom_line_ids': [
Command.create({'product_id': comp1.id, 'product_qty': 1.0}),
Command.create({'product_id': comp2.id, 'product_qty': 1.0}),
],
'byproduct_ids': [
Command.create({'product_id': bp1.id, 'product_qty': 1.0}),
Command.create({'product_id': bp2.id, 'product_qty': 1.0}),
],
'operation_ids': [
Command.create({'name': 'OPE_1', 'workcenter_id': self.workcenter_1.id}),
Command.create({'name': 'OPE_2', 'workcenter_id': self.workcenter_1.id}),
],
})
# Assign operation to each bom/byproduct line
ope_1, ope_2 = bom.operation_ids
bom.bom_line_ids[0].operation_id = ope_1
bom.byproduct_ids[0].operation_id = ope_1
bom.bom_line_ids[1].operation_id = ope_2
bom.byproduct_ids[1].operation_id = ope_2
# Archive first operation
ope_1.action_archive()
self.assertFalse(bom.bom_line_ids[0].operation_id)
self.assertFalse(bom.byproduct_ids[0].operation_id)
self.assertEqual(bom.bom_line_ids[1].operation_id, ope_2)
self.assertEqual(bom.byproduct_ids[1].operation_id, ope_2)