# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. from datetime import datetime, timedelta from freezegun import freeze_time from odoo import Command, fields from odoo.exceptions import UserError from odoo.tests import Form from odoo.tools.misc import format_date from odoo.addons.mrp.tests.common import TestMrpCommon class TestMrpOrder(TestMrpCommon): @classmethod def setUpClass(cls): super().setUpClass() cls.env.ref('base.group_user').write({'implied_ids': [(4, cls.env.ref('stock.group_production_lot').id)]}) def test_access_rights_manager(self): """ Checks an MRP manager can create, confirm and cancel a manufacturing order. """ man_order_form = Form(self.env['mrp.production'].with_user(self.user_mrp_manager)) man_order_form.product_id = self.product_4 man_order_form.product_qty = 5.0 man_order_form.bom_id = self.bom_1 man_order_form.location_src_id = self.location_1 man_order_form.location_dest_id = self.warehouse_1.wh_output_stock_loc_id man_order = man_order_form.save() man_order.action_confirm() man_order.action_cancel() self.assertEqual(man_order.state, 'cancel', "Production order should be in cancel state.") man_order.unlink() def test_access_rights_user(self): """ Checks an MRP user can create, confirm and cancel a manufacturing order. """ man_order_form = Form(self.env['mrp.production'].with_user(self.user_mrp_user)) man_order_form.product_id = self.product_4 man_order_form.product_qty = 5.0 man_order_form.bom_id = self.bom_1 man_order_form.location_src_id = self.location_1 man_order_form.location_dest_id = self.warehouse_1.wh_output_stock_loc_id man_order = man_order_form.save() man_order.action_confirm() man_order.action_cancel() self.assertEqual(man_order.state, 'cancel', "Production order should be in cancel state.") man_order.unlink() def test_basic(self): """ Checks a basic manufacturing order: no routing (thus no workorders), no lot and consume strictly what's needed. """ self.product_1.type = 'product' self.product_2.type = 'product' self.env['stock.quant'].create({ 'location_id': self.warehouse_1.lot_stock_id.id, 'product_id': self.product_1.id, 'inventory_quantity': 500 }).action_apply_inventory() self.env['stock.quant'].create({ 'location_id': self.warehouse_1.lot_stock_id.id, 'product_id': self.product_2.id, 'inventory_quantity': 500 }).action_apply_inventory() date_start = fields.Datetime.now() - timedelta(days=1) test_quantity = 3.0 man_order_form = Form(self.env['mrp.production'].with_user(self.user_mrp_user)) man_order_form.product_id = self.product_4 man_order_form.bom_id = self.bom_1 man_order_form.product_uom_id = self.product_4.uom_id man_order_form.product_qty = test_quantity man_order_form.date_start = date_start man_order_form.location_src_id = self.location_1 man_order_form.location_dest_id = self.warehouse_1.wh_output_stock_loc_id man_order = man_order_form.save() self.assertEqual(man_order.state, 'draft', "Production order should be in draft state.") man_order.action_confirm() self.assertEqual(man_order.state, 'confirmed', "Production order should be in confirmed state.") # check production move production_move = man_order.move_finished_ids self.assertAlmostEqual(production_move.date, date_start + timedelta(hours=1), delta=timedelta(seconds=10)) self.assertEqual(production_move.product_id, self.product_4) self.assertEqual(production_move.product_uom, man_order.product_uom_id) self.assertEqual(production_move.product_qty, man_order.product_qty) self.assertEqual(production_move.location_id, self.product_4.property_stock_production) self.assertEqual(production_move.location_dest_id, man_order.location_dest_id) # check consumption moves for move in man_order.move_raw_ids: self.assertEqual(move.date, date_start) first_move = man_order.move_raw_ids.filtered(lambda move: move.product_id == self.product_2) self.assertEqual(first_move.product_qty, test_quantity / self.bom_1.product_qty * self.product_4.uom_id.factor_inv * 2) first_move = man_order.move_raw_ids.filtered(lambda move: move.product_id == self.product_1) self.assertEqual(first_move.product_qty, test_quantity / self.bom_1.product_qty * self.product_4.uom_id.factor_inv * 4) # produce product mo_form = Form(man_order) mo_form.qty_producing = 2.0 man_order = mo_form.save() action = man_order.button_mark_done() self.assertEqual(man_order.state, 'progress', "Production order should be open a backorder wizard, then not done yet.") quantity_issues = man_order._get_consumption_issues() action = man_order._action_generate_consumption_wizard(quantity_issues) backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_close_mo() self.assertEqual(man_order.state, 'done', "Production order should be done.") # check that copy handles moves correctly mo_copy = man_order.copy() self.assertEqual(mo_copy.state, 'draft', "Copied production order should be draft.") self.assertEqual(len(mo_copy.move_raw_ids), 2, "Incorrect number of component moves [i.e. all non-0 (even cancelled) moves should be copied].") self.assertEqual(len(mo_copy.move_finished_ids), 1, "Incorrect number of moves for products to produce [i.e. cancelled moves should not be copied") self.assertEqual(mo_copy.move_finished_ids.product_uom_qty, 3, "Incorrect qty of products to produce") # check that a cancelled MO is copied correctly mo_copy.action_cancel() self.assertEqual(mo_copy.state, 'cancel') mo_copy_2 = mo_copy.copy() self.assertEqual(mo_copy_2.state, 'draft', "Copied production order should be draft.") self.assertEqual(len(mo_copy_2.move_raw_ids), 2, "Incorrect number of component moves.") self.assertEqual(len(mo_copy_2.move_finished_ids), 1, "Incorrect number of moves for products to produce [i.e. copying a cancelled MO should copy its cancelled moves]") self.assertEqual(mo_copy_2.move_finished_ids.product_uom_qty, 3, "Incorrect qty of products to produce") def test_production_availability(self): """ Checks the availability of a production order through mutliple calls to `action_assign`. """ self.bom_3.bom_line_ids.filtered(lambda x: x.product_id == self.product_5).unlink() self.bom_3.bom_line_ids.filtered(lambda x: x.product_id == self.product_4).unlink() self.bom_3.ready_to_produce = 'all_available' production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = self.bom_3 production_form.product_qty = 5.0 production_form.product_uom_id = self.product_6.uom_id production_2 = production_form.save() production_2.action_confirm() production_2.action_assign() # check sub product availability state is waiting self.assertEqual(production_2.reservation_state, 'confirmed', 'Production order should be availability for waiting state') # Update Inventory self.env['stock.quant'].with_context(inventory_mode=True).create({ 'product_id': self.product_2.id, 'inventory_quantity': 2.0, 'location_id': self.stock_location_14.id }).action_apply_inventory() production_2.action_assign() # check sub product availability state is partially available self.assertEqual(production_2.reservation_state, 'confirmed', 'Production order should be availability for partially available state') # Update Inventory self.env['stock.quant'].with_context(inventory_mode=True).create({ 'product_id': self.product_2.id, 'inventory_quantity': 5.0, 'location_id': self.stock_location_14.id }).action_apply_inventory() production_2.action_assign() # check sub product availability state is assigned self.assertEqual(production_2.reservation_state, 'assigned', 'Production order should be availability for assigned state') @freeze_time('2022-06-28 08:00') def test_end_date(self): """ End date must be the day the MO is done (regardless of lead times)""" mo, bom_id, _p_final, _p1, _p2 = self.generate_mo(qty_base_1=10, qty_final=1, qty_base_2=1) bom_id.produce_delay = 5 mo.button_mark_done() self.assertEqual(mo.date_finished.day, 28) def test_over_consumption(self): """ Consume more component quantity than the initial demand. No split on moves. """ mo, _bom, _p_final, _p1, _p2 = self.generate_mo(qty_base_1=10, qty_final=1, qty_base_2=1) mo.action_assign() # check is_quantity_done_editable mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 2 details_operation_form.save() details_operation_form = Form(mo.move_raw_ids[1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 11 details_operation_form.save() self.assertEqual(len(mo.move_raw_ids), 2) self.assertEqual(len(mo.move_raw_ids.mapped('move_line_ids')), 2) self.assertEqual(mo.move_raw_ids[0].move_line_ids.mapped('quantity'), [2]) self.assertEqual(mo.move_raw_ids[1].move_line_ids.mapped('quantity'), [11]) self.assertEqual(mo.move_raw_ids[0].quantity, 2) self.assertEqual(mo.move_raw_ids[1].quantity, 11) mo.button_mark_done() self.assertEqual(len(mo.move_raw_ids), 2) self.assertEqual(len(mo.move_raw_ids.mapped('move_line_ids')), 2) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [2, 11]) self.assertEqual(mo.move_raw_ids.mapped('move_line_ids.quantity'), [2, 11]) def test_under_consumption(self): """ Consume less component quantity than the initial demand. Before done: p1, to consume = 1, consumed = 0 p2, to consume = 10, consumed = 5 After done: p1, to consume = 1, consumed = 0, state = cancel p2, to consume = 10, consumed = 5, state = done """ mo, _bom, _p_final, _p1, _p2 = self.generate_mo(qty_base_1=10, qty_final=1, qty_base_2=1) mo.action_assign() # check is_quantity_done_editable mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 0 details_operation_form.save() details_operation_form = Form(mo.move_raw_ids[1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 5 details_operation_form.save() self.assertEqual(len(mo.move_raw_ids), 2) self.assertEqual(len(mo.move_raw_ids.mapped('move_line_ids')), 2) self.assertEqual(mo.move_raw_ids[0].move_line_ids.mapped('quantity'), [0]) self.assertEqual(mo.move_raw_ids[1].move_line_ids.mapped('quantity'), [5]) self.assertEqual(mo.move_raw_ids[0].quantity, 0) self.assertEqual(mo.move_raw_ids[1].quantity, 5) mo.button_mark_done() self.assertEqual(len(mo.move_raw_ids), 2) self.assertEqual(len(mo.move_raw_ids.mapped('move_line_ids')), 1) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [0, 5]) self.assertEqual(mo.move_raw_ids.mapped('product_uom_qty'), [1, 10]) self.assertEqual(mo.move_raw_ids.mapped('state'), ['cancel', 'done']) self.assertEqual(mo.move_raw_ids.mapped('move_line_ids.quantity'), [5]) def test_update_quantity_1(self): """ Build 5 final products with different consumed lots, then edit the finished quantity and update the Manufacturing order quantity. Then check if the produced quantity do not change and it is possible to close the MO. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(tracking_base_1='lot') self.assertEqual(len(mo), 1, 'MO should have been created') lot_1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': p1.id, 'company_id': self.env.company.id, }) lot_2 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': p1.id, 'company_id': self.env.company.id, }) self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 10, lot_id=lot_1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 10, lot_id=lot_2) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = lot_1 ml.quantity = 20 details_operation_form.save() mo.move_raw_ids[1].picked = True update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': mo.id, 'product_qty': 4, }) update_quantity_wizard.change_prod_qty() self.assertEqual(mo.move_raw_ids.filtered(lambda m: m.product_id == p1).quantity, 20, 'Update the produce quantity should not impact already produced quantity.') self.assertEqual(mo.move_finished_ids.product_uom_qty, 4) mo.button_mark_done() def test_update_quantity_2(self): """ Build 5 final products with different consumed lots, then edit the finished quantity and update the Manufacturing order quantity. Then check if the produced quantity do not change and it is possible to close the MO. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=3) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 20) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) mo_form.qty_producing = 2 mo = mo_form.save() # Produce & backorder action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] self.assertEqual(mo_backorder.product_qty, 1) update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': mo_backorder.id, 'product_qty': 3, }) update_quantity_wizard.change_prod_qty() mo_back_form = Form(mo_backorder) mo_back_form.qty_producing = 3 mo_backorder = mo_back_form.save() mo_backorder.button_mark_done() productions = mo | mo_backorder self.assertEqual(sum(productions.move_raw_ids.filtered(lambda m: m.product_id == p1).mapped('quantity')), 20) self.assertEqual(sum(productions.move_finished_ids.mapped('quantity')), 5) def test_update_quantity_3(self): bom = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 2.03}), (0, 0, {'product_id': self.product_8.id, 'product_qty': 4.16}) ], 'operation_ids': [ (0, 0, {'name': 'Gift Wrap Maching', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 15, 'sequence': 1}), ] }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = bom production_form.product_qty = 1 production_form.product_uom_id = self.product_6.uom_id production = production_form.save() self.assertEqual(production.workorder_ids.duration_expected, 90) mo_form = Form(production) mo_form.product_qty = 3 production = mo_form.save() self.assertEqual(production.workorder_ids.duration_expected, 165) # The same test than above but without form production = self.env['mrp.production'].create({ 'product_id': self.product_6.id, 'bom_id': bom.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, }) self.assertEqual(production.workorder_ids.duration_expected, 90) production.product_qty = 3 self.assertEqual(production.workorder_ids.duration_expected, 165) def test_update_quantity_4(self): """ Workcenter 1 has 10' start time and 5' stop time """ # Required for `workerorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') bom = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 2.03}), (0, 0, {'product_id': self.product_8.id, 'product_qty': 4.16}) ], }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = bom production_form.product_qty = 1 production_form.product_uom_id = self.product_6.uom_id production = production_form.save() production_form = Form(production) with production_form.workorder_ids.new() as wo: wo.name = 'OP1' wo.workcenter_id = self.workcenter_1 wo.duration_expected = 40 production = production_form.save() self.assertEqual(production.workorder_ids.duration_expected, 40) mo_form = Form(production) mo_form.product_qty = 3 production = mo_form.save() self.assertEqual(production.workorder_ids.duration_expected, 40) production.action_confirm() update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': production.id, 'product_qty': 9, }) update_quantity_wizard.change_prod_qty() self.assertEqual(production.workorder_ids.duration_expected, 90) # The same test than above but without form production = self.env['mrp.production'].create({ 'product_id': self.product_6.id, 'bom_id': bom.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'workorder_ids': [Command.create({ 'name': 'OP1', 'product_uom_id': self.product_6.uom_id.id, 'workcenter_id': self.workcenter_1.id, 'duration_expected': 40, })], }) self.assertEqual(production.workorder_ids.duration_expected, 40) production.product_qty = 3 self.assertEqual(production.workorder_ids.duration_expected, 40) production.action_confirm() update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': production.id, 'product_qty': 9, }) update_quantity_wizard.change_prod_qty() self.assertEqual(production.workorder_ids.duration_expected, 90) def test_qty_producing(self): """Qty producing should be the qty remain to produce, instead of 0""" # Required for `workerorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') bom = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 2.00}), ], }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = bom production_form.product_qty = 5 production_form.product_uom_id = self.product_6.uom_id production = production_form.save() production_form = Form(production) with production_form.workorder_ids.new() as wo: wo.name = 'OP1' wo.workcenter_id = self.workcenter_1 wo.duration_expected = 40 production = production_form.save() production.action_confirm() production.button_plan() wo = production.workorder_ids[0] wo.button_start() self.assertEqual(wo.qty_producing, 5, "Wrong quantity is suggested to produce.") # Simulate changing the qty_producing in the frontend wo.qty_producing = 4 wo.button_pending() wo.button_start() self.assertEqual(wo.qty_producing, 4, "Changing the qty_producing in the frontend is not persisted") def test_recursive_work_orders(self): """ When planning more than 322 work orders, there is a recursion error (with the default getrecursionlimit of 1000) """ product_uom_id = self.env.ref('uom.product_uom_unit').id mo_no_company = self.env['mrp.production'].create({ 'product_id': self.product.id, 'product_uom_id': product_uom_id, }) values = [ { 'name': f'Work order {n}', 'workcenter_id': self.workcenter_1.id, 'product_uom_id': product_uom_id, 'production_id': mo_no_company.id, 'duration': 60, } for n in range(300) ] self.env['mrp.workorder'].create(values) mo_no_company.action_confirm() mo_no_company.button_plan() def test_update_quantity_5(self): bom = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 3}), ], }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = bom production_form.product_qty = 1 production_form.product_uom_id = self.product_6.uom_id production = production_form.save() production.action_confirm() production.action_assign() production.is_locked = False production_form = Form(production) # change the quantity producing and the initial demand # in the same transaction production_form.qty_producing = 10 with production_form.move_raw_ids.edit(0) as move: move.product_uom_qty = 2 production = production_form.save() production.button_mark_done() def test_update_plan_date(self): """Editing the scheduled date after planning the MO should unplan the MO, and adjust the date on the stock moves""" date_start = datetime(2023, 5, 15, 9, 0) mo_form = Form(self.env['mrp.production']) mo_form.product_id = self.product_4 mo_form.bom_id = self.bom_1 mo_form.product_qty = 1 mo_form.date_start = date_start mo = mo_form.save() self.assertEqual(mo.move_finished_ids[0].date, datetime(2023, 5, 15, 10, 0)) mo.action_confirm() mo.button_plan() mo.date_start = datetime(2024, 5, 15, 9, 0) self.assertEqual(mo.move_finished_ids[0].date, datetime(2024, 5, 15, 10, 0)) def test_rounding(self): """ Checks we round up when bringing goods to produce and round half-up when producing. This implementation allows to implement an efficiency notion (see rev 347f140fe63612ee05e). """ self.product_6.uom_id.rounding = 1.0 bom_eff = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_qty': 1, 'product_uom_id': self.product_6.uom_id.id, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 2.03}), (0, 0, {'product_id': self.product_8.id, 'product_qty': 4.16}) ] }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_6 production_form.bom_id = bom_eff production_form.product_qty = 20 production_form.product_uom_id = self.product_6.uom_id production = production_form.save() production.action_confirm() #Check the production order has the right quantities self.assertEqual(production.move_raw_ids[0].product_qty, 41, 'The quantity should be rounded up') self.assertEqual(production.move_raw_ids[1].product_qty, 84, 'The quantity should be rounded up') # produce product mo_form = Form(production) mo_form.qty_producing = 8 production = mo_form.save() self.assertEqual(production.move_raw_ids[0].quantity, 16, 'Should use half-up rounding when producing') self.assertEqual(production.move_raw_ids[1].quantity, 34, 'Should use half-up rounding when producing') def test_product_produce_1(self): """ Checks the production wizard contains lines even for untracked products. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo() self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() # change the quantity done in one line details_operation_form = Form(mo.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 1 details_operation_form.save() # change the quantity producing mo_form = Form(mo) mo_form.qty_producing = 3 # check than all quantities are update correctly self.assertEqual(mo_form.move_raw_ids._records[0]['product_uom_qty'], 5, "Wrong quantity to consume") self.assertEqual(mo_form.move_raw_ids._records[0]['quantity'], 3, "Wrong quantity done") self.assertEqual(mo_form.move_raw_ids._records[1]['product_uom_qty'], 20, "Wrong quantity to consume") self.assertEqual(mo_form.move_raw_ids._records[1]['quantity'], 12, "Wrong quantity done") def test_product_produce_2(self): """ Checks that, for a BOM where one of the components is tracked by serial number and the other is not tracked, when creating a manufacturing order for two finished products and reserving, the produce wizards proposes the corrects lines when producing one at a time. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(tracking_base_1='serial', qty_base_1=1, qty_final=2) self.assertEqual(len(mo), 1, 'MO should have been created') lot_p1_1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': p1.id, 'company_id': self.env.company.id, }) lot_p1_2 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': p1.id, 'company_id': self.env.company.id, }) self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 1, lot_id=lot_p1_1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 1, lot_id=lot_p1_2) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() self.assertEqual(len(mo.move_raw_ids.move_line_ids), 3, 'You should have 3 stock move lines. One for each serial to consume and for the untracked product.') mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() # get the proposed lot details_operation_form = Form(mo.move_raw_ids.filtered(lambda move: move.product_id == p1), view=self.env.ref('stock.view_stock_move_operations')) self.assertEqual(len(details_operation_form.move_line_ids), 1) with details_operation_form.move_line_ids.edit(0) as ml: consumed_lots = ml.lot_id ml.quantity = 1 details_operation_form.save() remaining_lot = (lot_p1_1 | lot_p1_2) - consumed_lots remaining_lot.ensure_one() action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() # Check MO backorder mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] mo_form = Form(mo_backorder) mo_form.qty_producing = 1 mo_backorder = mo_form.save() details_operation_form = Form(mo_backorder.move_raw_ids.filtered(lambda move: move.product_id == p1), view=self.env.ref('stock.view_stock_move_operations')) self.assertEqual(len(details_operation_form.move_line_ids), 1) with details_operation_form.move_line_ids.edit(0) as ml: self.assertEqual(ml.lot_id, remaining_lot) def test_product_produce_3(self): """ Checks that, for a BOM where one of the components is tracked by lot and the other is not tracked, when creating a manufacturing order for 1 finished product and reserving, the reserved lines are displayed. Then, over-consume by creating new line. """ self.stock_location = self.env.ref('stock.stock_location_stock') self.stock_shelf_1 = self.stock_location_components self.stock_shelf_2 = self.stock_location_14 mo, _, p_final, p1, p2 = self.generate_mo(tracking_base_1='lot', qty_base_1=10, qty_final=1) # Required for `lot_producing_id` to be visible in the view # p_final.tracking = 'lot' self.assertEqual(len(mo), 1, 'MO should have been created') first_lot_for_p1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': p1.id, 'company_id': self.env.company.id, }) second_lot_for_p1 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': p1.id, 'company_id': self.env.company.id, }) final_product_lot = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': p_final.id, 'company_id': self.env.company.id, }) self.env['stock.quant']._update_available_quantity(p1, self.stock_shelf_1, 3, lot_id=first_lot_for_p1) self.env['stock.quant']._update_available_quantity(p1, self.stock_shelf_2, 3, lot_id=first_lot_for_p1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 8, lot_id=second_lot_for_p1) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) mo_form.qty_producing = 1.0 mo_form.lot_producing_id = final_product_lot mo = mo_form.save() # p2 details_operation_form = Form(mo.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as line: line.quantity = 1 details_operation_form.save() # p1 details_operation_form = Form(mo.move_raw_ids[1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as line: line.quantity = 2 line.lot_id = first_lot_for_p1 with details_operation_form.move_line_ids.new() as line: line.quantity = 1 line.lot_id = second_lot_for_p1 details_operation_form.save() move_1 = mo.move_raw_ids.filtered(lambda m: m.product_id == p1) # quantity/reserved_uom_qty lot # 3/3 lot 1 shelf 1 # 1/1 lot 1 shelf 2 # 2/2 lot 1 shelf 2 # 2/0 lot 1 other # 5/4 lot 2 ml_to_shelf_1 = move_1.move_line_ids.filtered(lambda ml: ml.lot_id == first_lot_for_p1 and ml.location_id == self.stock_shelf_1) ml_to_shelf_2 = move_1.move_line_ids.filtered(lambda ml: ml.lot_id == first_lot_for_p1 and ml.location_id == self.stock_shelf_2) self.assertEqual(sum(ml_to_shelf_1.mapped('quantity')), 3.0, '3 units should be took from shelf1 as reserved.') self.assertEqual(sum(ml_to_shelf_2.mapped('quantity')), 3.0, '3 units should be took from shelf2 as reserved.') self.assertEqual(move_1.quantity, 13, 'You should have used the tem units.') mo.button_mark_done() self.assertEqual(mo.state, 'done', "Production order should be in done state.") def test_product_produce_4(self): """ Possibility to produce with a given raw material in multiple locations. """ # FIXME sle: how is it possible to consume before producing in the interface? self.stock_location = self.env.ref('stock.stock_location_stock') self.stock_shelf_1 = self.stock_location_components self.stock_shelf_2 = self.stock_location_14 mo, _, p_final, p1, p2 = self.generate_mo(qty_final=1, qty_base_1=5) self.env['stock.quant']._update_available_quantity(p1, self.stock_shelf_1, 2) self.env['stock.quant']._update_available_quantity(p1, self.stock_shelf_2, 3) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 1) mo.action_assign() ml_p1 = mo.move_raw_ids.filtered(lambda x: x.product_id == p1).mapped('move_line_ids') ml_p2 = mo.move_raw_ids.filtered(lambda x: x.product_id == p2).mapped('move_line_ids') self.assertEqual(len(ml_p1), 2) self.assertEqual(len(ml_p2), 1) # Produce baby! mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() m_p1 = mo.move_raw_ids.filtered(lambda x: x.product_id == p1) ml_p1 = m_p1.mapped('move_line_ids') self.assertEqual(len(ml_p1), 2) self.assertEqual(sorted(ml_p1.mapped('quantity')), [2.0, 3.0], 'Quantity should be 2.0 and 3.0') self.assertEqual(m_p1.quantity, 5.0, 'Total qty done should be 5.0') mo.button_mark_done() self.assertEqual(mo.state, 'done', "Production order should be in done state.") def test_product_produce_6(self): """ Plan 5 finished products, reserve and produce 3. Post the current production. Simulate an unlock and edit and, on the opened moves, set the consumed quantity to 3. Now, try to update the quantity to mo2 to 3. It should fail since there are consumed quantities. Unlock and edit, remove the consumed quantities and update the quantity to produce to 3.""" self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo() self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 20) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) mo_form.qty_producing = 3 mo = mo_form.save() mo._post_inventory() self.assertEqual(len(mo.move_raw_ids), 4) mo.move_raw_ids.filtered(lambda m: m.state != 'done')[0].quantity = 3 update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': mo.id, 'product_qty': 3, }) mo.move_raw_ids.filtered(lambda m: m.state != 'done')[0].quantity = 0 update_quantity_wizard.change_prod_qty() self.assertEqual(len(mo.move_raw_ids), 4) mo.move_raw_ids.picked = True mo.button_mark_done() self.assertTrue(all(s in ['done', 'cancel'] for s in mo.move_raw_ids.mapped('state'))) def test_consumption_strict_1(self): """ Checks the constraints of a strict BOM without tracking when playing around quantities to consume.""" self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(consumption='strict', qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) # try adding another line for a bom product to increase the quantity mo_form.qty_producing = 1 with mo_form.move_raw_ids.new() as line: line.product_id = p1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[-1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.quantity = 1 details_operation_form.save() # Won't accept to be done, instead return a wizard mo.button_mark_done() self.assertEqual(mo.state, 'to_close') consumption_issues = mo._get_consumption_issues() action = mo._action_generate_consumption_wizard(consumption_issues) warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])) warning = warning.save() self.assertEqual(len(warning.mrp_consumption_warning_line_ids), 1) self.assertEqual(warning.mrp_consumption_warning_line_ids[0].product_consumed_qty_uom, 5) self.assertEqual(warning.mrp_consumption_warning_line_ids[0].product_expected_qty_uom, 4) # Force the warning (as a manager) warning.action_confirm() self.assertEqual(mo.state, 'done') def test_consumption_warning_1(self): """ Checks the constraints of a strict BOM without tracking when playing around quantities to consume.""" self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(consumption='warning', qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) # try adding another line for a bom product to increase the quantity mo_form.qty_producing = 1 with mo_form.move_raw_ids.new() as line: line.product_id = p1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[-1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.quantity = 1 details_operation_form.save() # Won't accept to be done, instead return a wizard mo.button_mark_done() self.assertEqual(mo.state, 'to_close') consumption_issues = mo._get_consumption_issues() action = mo._action_generate_consumption_wizard(consumption_issues) warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])) warning = warning.save() self.assertEqual(len(warning.mrp_consumption_warning_line_ids), 1) self.assertEqual(warning.mrp_consumption_warning_line_ids[0].product_consumed_qty_uom, 5) self.assertEqual(warning.mrp_consumption_warning_line_ids[0].product_expected_qty_uom, 4) # Force the warning (as a manager or employee) warning.action_confirm() self.assertEqual(mo.state, 'done') def test_consumption_flexible_1(self): """ Checks the constraints of a strict BOM without tracking when playing around quantities to consume.""" self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(consumption='flexible', qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) # try adding another line for a bom product to increase the quantity mo_form.qty_producing = 1 with mo_form.move_raw_ids.new() as line: line.product_id = p1 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[-1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.quantity = 1 details_operation_form.save() # Won't accept to be done, instead return a wizard mo.button_mark_done() self.assertEqual(mo.state, 'done') def test_consumption_flexible_2(self): """ Checks the constraints of a strict BOM only apply to the product of the BoM. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(consumption='flexible', qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) add_product = self.env['product.product'].create({ 'name': 'additional', 'type': 'product', }) mo.action_assign() mo_form = Form(mo) # try adding another line for a bom product to increase the quantity mo_form.qty_producing = 1 with mo_form.move_raw_ids.new() as line: line.product_id = p1 with mo_form.move_raw_ids.new() as line: line.product_id = add_product mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[-1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.quantity = 1 details_operation_form.save() # Won't accept to be done, instead return a wizard mo.button_mark_done() self.assertEqual(mo.state, 'done') def test_product_produce_9(self): """ Checks the production wizard contains lines even for untracked products. """ serial = self.env['product.product'].create({ 'name': 'S1', 'tracking': 'serial', }) self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo() self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 100) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5) mo.action_assign() mo_form = Form(mo) # change the quantity done in one line with self.assertRaises(AssertionError): with mo_form.move_raw_ids.new() as move: move.product_id = serial move.quantity = 2 mo_form.save() def test_product_produce_10(self): """ Produce byproduct with serial, lot and not tracked. byproduct1 serial 1.0 byproduct2 lot 2.0 byproduct3 none 1.0 dozen Check qty producing update and moves finished values. """ # Required for `byproduct_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_byproducts') dozen = self.env.ref('uom.product_uom_dozen') self.byproduct1 = self.env['product.product'].create({ 'name': 'Byproduct 1', 'type': 'product', 'tracking': 'serial' }) self.serial_1 = self.env['stock.lot'].create({ 'product_id': self.byproduct1.id, 'name': 'serial 1', 'company_id': self.env.company.id, }) self.serial_2 = self.env['stock.lot'].create({ 'product_id': self.byproduct1.id, 'name': 'serial 2', 'company_id': self.env.company.id, }) self.byproduct2 = self.env['product.product'].create({ 'name': 'Byproduct 2', 'type': 'product', 'tracking': 'lot', }) self.lot_1 = self.env['stock.lot'].create({ 'product_id': self.byproduct2.id, 'name': 'Lot 1', 'company_id': self.env.company.id, }) self.lot_2 = self.env['stock.lot'].create({ 'product_id': self.byproduct2.id, 'name': 'Lot 2', 'company_id': self.env.company.id, }) self.byproduct3 = self.env['product.product'].create({ 'name': 'Byproduct 3', 'type': 'product', 'tracking': 'none', }) with Form(self.bom_1) as bom: bom.product_qty = 1.0 with bom.byproduct_ids.new() as bp: bp.product_id = self.byproduct1 bp.product_qty = 1.0 with bom.byproduct_ids.new() as bp: bp.product_id = self.byproduct2 bp.product_qty = 2.0 with bom.byproduct_ids.new() as bp: bp.product_id = self.byproduct3 bp.product_qty = 2.0 bp.product_uom_id = dozen mo_form = Form(self.env['mrp.production']) mo_form.product_id = self.product_4 mo_form.bom_id = self.bom_1 mo_form.product_qty = 2 mo = mo_form.save() mo.action_confirm() move_byproduct_1 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct1) self.assertEqual(len(move_byproduct_1), 1) self.assertEqual(move_byproduct_1.product_uom_qty, 2.0) self.assertEqual(move_byproduct_1.quantity, 2) self.assertEqual(len(move_byproduct_1.move_line_ids), 2) move_byproduct_2 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct2) self.assertEqual(len(move_byproduct_2), 1) self.assertEqual(move_byproduct_2.product_uom_qty, 4.0) self.assertEqual(move_byproduct_2.quantity, 4) self.assertEqual(len(move_byproduct_2.move_line_ids), 1) move_byproduct_3 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct3) self.assertEqual(move_byproduct_3.product_uom_qty, 4.0) self.assertEqual(move_byproduct_3.quantity, 4) self.assertEqual(move_byproduct_3.product_uom, dozen) self.assertEqual(len(move_byproduct_3.move_line_ids), 1) mo_form = Form(mo) mo_form.qty_producing = 1.0 mo = mo_form.save() move_byproduct_1 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct1) self.assertEqual(len(move_byproduct_1), 1) self.assertEqual(move_byproduct_1.product_uom_qty, 2.0) self.assertEqual(move_byproduct_1.quantity, 1) self.assertFalse(move_byproduct_1.picked) move_byproduct_2 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct2) self.assertEqual(len(move_byproduct_2), 1) self.assertEqual(move_byproduct_2.product_uom_qty, 4.0) self.assertEqual(move_byproduct_2.quantity, 2) self.assertFalse(move_byproduct_2.picked) move_byproduct_3 = mo.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct3) self.assertEqual(move_byproduct_3.product_uom_qty, 4.0) self.assertEqual(move_byproduct_3.quantity, 2.0) self.assertTrue(move_byproduct_3.picked) self.assertEqual(move_byproduct_3.product_uom, dozen) details_operation_form = Form(move_byproduct_1, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = self.serial_1 details_operation_form.save() details_operation_form = Form(move_byproduct_2, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = self.lot_1 details_operation_form.save() action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() mo2 = mo.procurement_group_id.mrp_production_ids[-1] mo_form = Form(mo2) mo_form.qty_producing = 1 mo2 = mo_form.save() move_byproduct_1 = mo2.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct1) self.assertEqual(len(move_byproduct_1), 1) self.assertEqual(move_byproduct_1.product_uom_qty, 1.0) self.assertEqual(move_byproduct_1.quantity, 1) self.assertFalse(move_byproduct_1.picked) move_byproduct_2 = mo2.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct2) self.assertEqual(len(move_byproduct_2), 1) self.assertEqual(move_byproduct_2.product_uom_qty, 2.0) self.assertEqual(move_byproduct_2.quantity, 2) self.assertFalse(move_byproduct_2.picked) move_byproduct_3 = mo2.move_finished_ids.filtered(lambda l: l.product_id == self.byproduct3) self.assertEqual(move_byproduct_3.product_uom_qty, 2.0) self.assertEqual(move_byproduct_3.quantity, 2.0) self.assertTrue(move_byproduct_3.picked) self.assertEqual(move_byproduct_3.product_uom, dozen) details_operation_form = Form(move_byproduct_1, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = self.serial_2 ml.quantity = 1 details_operation_form.save() details_operation_form = Form(move_byproduct_2, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = self.lot_2 details_operation_form.save() details_operation_form = Form(move_byproduct_3, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.quantity = 3 details_operation_form.save() mo2.button_mark_done() move_lines_byproduct_1 = (mo | mo2).move_finished_ids.filtered(lambda l: l.product_id == self.byproduct1).mapped('move_line_ids') move_lines_byproduct_2 = (mo | mo2).move_finished_ids.filtered(lambda l: l.product_id == self.byproduct2).mapped('move_line_ids') move_lines_byproduct_3 = (mo | mo2).move_finished_ids.filtered(lambda l: l.product_id == self.byproduct3).mapped('move_line_ids') self.assertEqual(move_lines_byproduct_1.filtered(lambda ml: ml.lot_id == self.serial_1).quantity, 1.0) self.assertEqual(move_lines_byproduct_1.filtered(lambda ml: ml.lot_id == self.serial_2).quantity, 1.0) self.assertEqual(move_lines_byproduct_2.filtered(lambda ml: ml.lot_id == self.lot_1).quantity, 2.0) self.assertEqual(move_lines_byproduct_2.filtered(lambda ml: ml.lot_id == self.lot_2).quantity, 2.0) self.assertEqual(sum(move_lines_byproduct_3.mapped('quantity')), 5.0) self.assertEqual(move_lines_byproduct_3.mapped('product_uom_id'), dozen) def test_product_produce_11(self): """ Checks that, for a BOM with two components, when creating a manufacturing order for one finished products and without reserving, the produce wizards proposes the corrects lines even if we change the quantity to produce multiple times. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 4) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 1) mo.bom_id.consumption = 'flexible' # Because we'll over-consume with a product not defined in the BOM mo.action_assign() mo.is_locked = False mo_form = Form(mo) mo_form.qty_producing = 3 self.assertEqual(sum([x['quantity'] for x in mo_form.move_raw_ids._records]), 15, 'Update the produce quantity should change the components quantity.') mo = mo_form.save() mo_form = Form(mo) mo_form.qty_producing = 4 self.assertEqual(sum([x['quantity'] for x in mo_form.move_raw_ids._records]), 20, 'Update the produce quantity should change the components quantity.') mo = mo_form.save() mo_form = Form(mo) mo_form.qty_producing = 1 self.assertEqual(sum([x['quantity'] for x in mo_form.move_raw_ids._records]), 5, 'Update the produce quantity should change the components quantity.') mo = mo_form.save() # try adding another product that doesn't belong to the BoM with mo_form.move_raw_ids.new() as move: move.product_id = self.product_4 mo = mo_form.save() details_operation_form = Form(mo.move_raw_ids[-1], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.quantity = 10 details_operation_form.save() # Check that this new product is not updated by qty_producing mo_form = Form(mo) mo_form.qty_producing = 2 for move in mo_form.move_raw_ids._records: if move['product_id'] == self.product_4.id: self.assertEqual(move['quantity'], 10) break mo = mo_form.save() mo.button_mark_done() def test_product_produce_duplicate_1(self): """ produce a finished product tracked by serial number 2 times with the same SN. Check that an error is raised the second time""" mo1, bom, p_final, p1, p2 = self.generate_mo(tracking_final='serial', qty_final=1, qty_base_1=1,) mo_form = Form(mo1) mo_form.qty_producing = 1 mo1 = mo_form.save() mo1.action_generate_serial() sn = mo1.lot_producing_id mo1.button_mark_done() mo_form = Form(self.env['mrp.production']) mo_form.product_id = p_final mo_form.bom_id = bom mo_form.product_qty = 1 mo2 = mo_form.save() mo2.action_confirm() mo_form = Form(mo2) with self.assertLogs(level="WARNING"): mo_form.lot_producing_id = sn mo2 = mo_form.save() with self.assertRaises(UserError): mo2.button_mark_done() def test_product_produce_duplicate_2(self): """ produce a finished product with component tracked by serial number 2 times with the same SN. Check that an error is raised the second time""" mo1, bom, p_final, p1, p2 = self.generate_mo(tracking_base_2='serial', qty_final=1, qty_base_1=1,) sn = self.env['stock.lot'].create({ 'name': 'sn used twice', 'product_id': p2.id, 'company_id': self.env.company.id, }) mo_form = Form(mo1) mo_form.qty_producing = 1 mo1 = mo_form.save() details_operation_form = Form(mo1.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = sn details_operation_form.save() mo1.move_raw_ids.picked = True mo1.button_mark_done() mo_form = Form(self.env['mrp.production']) mo_form.product_id = p_final mo_form.bom_id = bom mo_form.product_qty = 1 mo2 = mo_form.save() mo2.action_confirm() mo_form = Form(mo2) mo_form.qty_producing = 1 mo2 = mo_form.save() details_operation_form = Form(mo2.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = sn details_operation_form.save() mo2.move_raw_ids.picked = True with self.assertRaises(UserError): mo2.button_mark_done() def test_product_produce_duplicate_3(self): """ produce a finished product with by-product tracked by serial number 2 times with the same SN. Check that an error is raised the second time""" finished_product = self.env['product.product'].create({'name': 'finished product'}) byproduct = self.env['product.product'].create({'name': 'byproduct', 'tracking': 'serial'}) component = self.env['product.product'].create({'name': 'component'}) bom = self.env['mrp.bom'].create({ 'product_id': finished_product.id, 'product_tmpl_id': finished_product.product_tmpl_id.id, 'product_uom_id': finished_product.uom_id.id, 'product_qty': 1.0, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': component.id, 'product_qty': 1}), ], 'byproduct_ids': [ (0, 0, {'product_id': byproduct.id, 'product_qty': 1, 'product_uom_id': byproduct.uom_id.id}) ]}) mo_form = Form(self.env['mrp.production']) mo_form.product_id = finished_product mo_form.bom_id = bom mo_form.product_qty = 1 mo = mo_form.save() mo.action_confirm() sn = self.env['stock.lot'].create({ 'name': 'sn used twice', 'product_id': byproduct.id, 'company_id': self.env.company.id, }) mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() move_byproduct = mo.move_finished_ids.filtered(lambda m: m.product_id != mo.product_id) details_operation_form = Form(move_byproduct, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = sn details_operation_form.save() mo.button_mark_done() mo_form = Form(self.env['mrp.production']) mo_form.product_id = finished_product mo_form.bom_id = bom mo_form.product_qty = 1 mo2 = mo_form.save() mo2.action_confirm() mo_form = Form(mo2) mo_form.qty_producing = 1 mo2 = mo_form.save() move_byproduct = mo2.move_finished_ids.filtered(lambda m: m.product_id != mo.product_id) details_operation_form = Form(move_byproduct, view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.new() as ml: ml.lot_id = sn details_operation_form.save() with self.assertRaises(UserError): mo2.button_mark_done() def test_product_produce_duplicate_4(self): """ Consuming the same serial number two times should not give an error if a repair order of the first production has been made before the second one""" mo1, bom, p_final, p1, p2 = self.generate_mo(tracking_base_2='serial', qty_final=1, qty_base_1=1,) sn = self.env['stock.lot'].create({ 'name': 'sn used twice', 'product_id': p2.id, 'company_id': self.env.company.id, }) mo_form = Form(mo1) mo_form.qty_producing = 1 mo1 = mo_form.save() details_operation_form = Form(mo1.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = sn details_operation_form.save() mo1.move_raw_ids.picked = True mo1.button_mark_done() unbuild_form = Form(self.env['mrp.unbuild']) unbuild_form.product_id = p_final unbuild_form.bom_id = bom unbuild_form.product_qty = 1 unbuild_form.mo_id = mo1 unbuild_order = unbuild_form.save() unbuild_order.action_unbuild() mo_form = Form(self.env['mrp.production']) mo_form.product_id = p_final mo_form.bom_id = bom mo_form.product_qty = 1 mo2 = mo_form.save() mo2.action_confirm() mo_form = Form(mo2) mo_form.qty_producing = 1 mo2 = mo_form.save() details_operation_form = Form(mo2.move_raw_ids[0], view=self.env.ref('stock.view_stock_move_operations')) with details_operation_form.move_line_ids.edit(0) as ml: ml.lot_id = sn details_operation_form.save() mo2.move_raw_ids.picked = True mo2.button_mark_done() def test_product_produce_duplicate_5(self): """Produce a subassembly for the second time with the same serial after having unbuilt both the subassembly and finished good it was part of""" subassembly_product = self.env["product.product"].create( { "name": "Subassembly", "type": "product", "tracking": "serial", } ) subassembly_sn = self.env["stock.lot"].create( { "name": "SN", "product_id": subassembly_product.id, "company_id": self.env.company.id, } ) subassembly_mo1_form = Form(self.env["mrp.production"]) subassembly_mo1_form.product_id = subassembly_product subassembly_mo1 = subassembly_mo1_form.save() subassembly_mo1.action_confirm() with Form(subassembly_mo1) as mo: mo.qty_producing = 1 subassembly_mo1.lot_producing_id = subassembly_sn subassembly_mo1.button_mark_done() finished_good_product = self.env["product.product"].create( { "name": "Finished Good", "type": "product", "tracking": "serial", } ) finished_good_product_bom = self.env["mrp.bom"].create( { "product_tmpl_id": finished_good_product.product_tmpl_id.id, "product_qty": 1, "type": "normal", "bom_line_ids": [ (0, 0, {"product_id": subassembly_product.id, "product_qty": 1}), ], } ) finished_good_mo_form = Form(self.env["mrp.production"]) finished_good_mo_form.product_id = finished_good_product finished_good_mo_form.bom_id = finished_good_product_bom finished_good_mo = finished_good_mo_form.save() finished_good_mo.action_confirm() with Form(finished_good_mo) as mo: mo.qty_producing = 1 finished_good_mo.action_generate_serial() finished_good_detailed_operations_form = Form( finished_good_mo.move_raw_ids[0], view=self.env.ref("stock.view_stock_move_operations"), ) with finished_good_detailed_operations_form.move_line_ids.edit(0) as ml: ml.quantity = 1 ml.lot_id = subassembly_sn finished_good_detailed_operations_form.save() finished_good_mo.move_raw_ids.picked = True finished_good_mo.button_mark_done() finished_good_ub_form = Form(self.env["mrp.unbuild"]) finished_good_ub_form.mo_id = finished_good_mo finished_good_ub = finished_good_ub_form.save() finished_good_ub.action_unbuild() subassembly_ub_form = Form(self.env["mrp.unbuild"]) subassembly_ub_form.mo_id = subassembly_mo1 subassembly_ub = subassembly_ub_form.save() subassembly_ub.action_unbuild() subassembly_mo2_form = Form(self.env["mrp.production"]) subassembly_mo2_form.product_id = subassembly_product subassembly_mo2 = subassembly_mo2_form.save() subassembly_mo2.action_confirm() with Form(subassembly_mo2) as mo: mo.qty_producing = 1 subassembly_mo2.lot_producing_id = subassembly_sn subassembly_mo2.button_mark_done() def test_product_produce_duplicate_6(self): """Produce a product for the second time with the same serial after having unbuilt, scrapped and unscrapped the product""" product = self.env["product.product"].create( { "name": "Product", "type": "product", "tracking": "serial", } ) sn = self.env["stock.lot"].create( { "name": "SN", "product_id": product.id, "company_id": self.env.company.id, } ) mo1_form = Form(self.env["mrp.production"]) mo1_form.product_id = product mo1 = mo1_form.save() mo1.action_confirm() with Form(mo1) as mo: mo.qty_producing = 1 mo1.lot_producing_id = sn mo1.button_mark_done() ub_form = Form(self.env["mrp.unbuild"]) ub_form.mo_id = mo1 ub = ub_form.save() ub.action_unbuild() scrap = self.env['stock.scrap'].create({ 'product_id': product.id, 'product_uom_id': product.uom_id.id, 'lot_id': sn.id, }) scrap.do_scrap() unscrap_picking = self.env['stock.picking'].create({ 'picking_type_id': self.env.ref('stock.picking_type_internal').id, 'location_id': scrap.scrap_location_id.id, 'location_dest_id': scrap.location_id.id, }) unscrap_move = self.env['stock.move'].create({ 'name': 'unscrap', 'location_id': scrap.scrap_location_id.id, 'location_dest_id': scrap.location_id.id, 'product_id': product.id, 'product_uom': product.uom_id.id, 'picking_id': unscrap_picking.id, }) unscrap_picking.action_confirm() self.env['stock.move.line'].create({ 'move_id': unscrap_move.id, 'product_id': unscrap_move.product_id.id, 'lot_id': sn.id, 'quantity': 1, 'product_uom_id': unscrap_move.product_uom.id, 'picking_id': unscrap_move.picking_id.id, }) unscrap_picking.button_validate() mo2_form = Form(self.env["mrp.production"]) mo2_form.product_id = product mo2 = mo2_form.save() mo2.action_confirm() with Form(mo2) as mo: mo.qty_producing = 1 mo2.lot_producing_id = sn mo2.button_mark_done() def test_product_produce_12(self): """ Checks that, the production is robust against deletion of finished move.""" self.stock_location = self.env.ref('stock.stock_location_stock') mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() # remove the finished move from the available to be updated mo.move_finished_ids._action_done() mo.button_mark_done() def test_product_produce_13(self): """ Check that the production can be completed without any consumption.""" product = self.env['product.product'].create({ 'name': 'Product no BoM', 'type': 'product', }) mo_form = Form(self.env['mrp.production']) mo_form.product_id = product mo = mo_form.save() move = self.env['stock.move'].create({ 'product_id': self.product_2.id, 'product_uom': self.ref('uom.product_uom_unit'), 'production_id': mo.id, 'location_dest_id': self.ref('stock.stock_location_output'), }) self.assertEqual(move.name, mo.name) self.assertEqual(move.origin, mo._get_origin()) self.assertEqual(move.group_id, mo.procurement_group_id) self.assertEqual(move.propagate_cancel, mo.propagate_cancel) self.assertFalse(move.raw_material_production_id) self.assertEqual(move.location_id, mo.production_location_id) self.assertEqual(move.date, mo.date_finished) self.assertEqual(move.date_deadline, mo.date_deadline) mo.move_raw_ids |= move mo.action_confirm() mo.qty_producing = 1 mo.button_mark_done() self.assertEqual(mo.state, 'done') self.assertEqual(mo.qty_produced, 1) self.assertEqual(mo.move_raw_ids.state, 'cancel') def test_product_produce_14(self): """ Check two component move with the same product are not merged.""" product = self.env['product.product'].create({ 'name': 'Product no BoM', 'type': 'product', }) mo_form = Form(self.env['mrp.production']) mo_form.product_id = product mo = mo_form.save() for i in range(2): move = self.env['stock.move'].create({ 'name': 'mrp_move_' + str(i), 'product_id': self.product_2.id, 'product_uom': self.ref('uom.product_uom_unit'), 'production_id': mo.id, 'location_id': self.ref('stock.stock_location_stock'), 'location_dest_id': self.ref('stock.stock_location_output'), }) mo.move_raw_ids |= move mo.action_confirm() self.assertEqual(len(mo.move_raw_ids), 2) def test_product_produce_uom(self): """ Produce a finished product tracked by serial number. Set another UoM on the bom. The produce wizard should keep the UoM of the product (unit) and quantity = 1.""" dozen = self.env.ref('uom.product_uom_dozen') unit = self.env.ref('uom.product_uom_unit') plastic_laminate = self.env['product.product'].create({ 'name': 'Plastic Laminate', 'type': 'product', 'uom_id': unit.id, 'uom_po_id': unit.id, 'tracking': 'serial', }) ply_veneer = self.env['product.product'].create({ 'name': 'Ply Veneer', 'type': 'product', 'uom_id': unit.id, 'uom_po_id': unit.id, }) bom = self.env['mrp.bom'].create({ 'product_tmpl_id': plastic_laminate.product_tmpl_id.id, 'product_uom_id': unit.id, 'sequence': 1, 'bom_line_ids': [(0, 0, { 'product_id': ply_veneer.id, 'product_qty': 1, 'product_uom_id': unit.id, 'sequence': 1, })] }) mo_form = Form(self.env['mrp.production']) mo_form.product_id = plastic_laminate mo_form.bom_id = bom mo_form.product_uom_id = dozen mo_form.product_qty = 1 mo = mo_form.save() final_product_lot = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': plastic_laminate.id, 'company_id': self.env.company.id, }) mo.action_confirm() mo.action_assign() self.assertEqual(mo.move_raw_ids.product_qty, 12, '12 units should be reserved.') # produce product mo_form = Form(mo) mo_form.qty_producing = 1/12.0 mo_form.lot_producing_id = final_product_lot mo = mo_form.save() move_line_raw = mo.move_raw_ids.mapped('move_line_ids').filtered(lambda m: m.quantity) self.assertEqual(move_line_raw.quantity, 1) self.assertEqual(move_line_raw.product_uom_id, unit, 'Should be 1 unit since the tracking is serial.') mo._post_inventory() move_line_finished = mo.move_finished_ids.move_line_ids.filtered(lambda m: m.state == 'done' and m.quantity) self.assertEqual(move_line_finished.quantity, 1) self.assertEqual(move_line_finished.product_uom_id, unit, 'Should be 1 unit since the tracking is serial.') def test_product_type_service_1(self): # Create finished product finished_product = self.env['product.product'].create({ 'name': 'Geyser', 'type': 'product', }) # Create service type product product_raw = self.env['product.product'].create({ 'name': 'raw Geyser', 'type': 'service', }) # Create bom for finish product bom = self.env['mrp.bom'].create({ 'product_id': finished_product.id, 'product_tmpl_id': finished_product.product_tmpl_id.id, 'product_uom_id': self.env.ref('uom.product_uom_unit').id, 'product_qty': 1.0, 'type': 'normal', 'bom_line_ids': [(5, 0), (0, 0, {'product_id': product_raw.id})] }) mo_form = Form(self.env['mrp.production']) mo_form.product_id = finished_product mo_form.bom_id = bom mo_form.product_uom_id = self.env.ref('uom.product_uom_unit') mo_form.product_qty = 1 mo = mo_form.save() # Check Mo is created or not self.assertTrue(mo, "Mo is created") def test_immediate_validate_1(self): """ In a production with a single available move raw, clicking on mark as done without filling any quantities should open a wizard asking to process all the reservation (so, the whole move). """ mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=1, qty_base_1=1, qty_base_2=1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location_components, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location_components, 5.0) mo.action_assign() res_dict = mo.button_mark_done() self.assertEqual(mo.move_raw_ids.mapped('state'), ['done', 'done']) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [1, 1]) self.assertEqual(mo.move_finished_ids.state, 'done') self.assertEqual(mo.move_finished_ids.quantity, 1) def test_immediate_validate_3(self): """ In a production with a serial number tracked product. Check that the immediate production only creates one unit of finished product. Test with reservation.""" mo, bom, p_final, p1, p2 = self.generate_mo(tracking_final='serial', qty_final=2, qty_base_1=1, qty_base_2=1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location_components, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location_components, 5.0) mo.action_assign() action = mo.button_mark_done() self.assertEqual(action.get('res_model'), 'mrp.production.backorder') wizard = Form(self.env[action['res_model']].with_context(action['context'])).save() action = wizard.action_backorder() self.assertEqual(mo.qty_producing, 1) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [1, 1]) self.assertEqual(len(mo.procurement_group_id.mrp_production_ids), 2) mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] self.assertEqual(mo_backorder.product_qty, 1) self.assertEqual(mo_backorder.move_raw_ids.mapped('product_uom_qty'), [1, 1]) def test_immediate_validate_4(self): """ In a production with a serial number tracked product. Check that the immediate production only creates one unit of finished product. Test without reservation.""" mo, bom, p_final, p1, p2 = self.generate_mo(tracking_final='serial', qty_final=2, qty_base_1=1, qty_base_2=1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location_components, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location_components, 5.0) action = mo.button_mark_done() self.assertEqual(action.get('res_model'), 'mrp.production.backorder') wizard = Form(self.env[action['res_model']].with_context(action['context'])).save() action = wizard.action_backorder() self.assertEqual(mo.qty_producing, 1) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [1, 1]) self.assertEqual(len(mo.procurement_group_id.mrp_production_ids), 2) mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] self.assertEqual(mo_backorder.product_qty, 1) self.assertEqual(mo_backorder.move_raw_ids.mapped('product_uom_qty'), [1, 1]) def test_immediate_validate_5(self): """Validate three productions at once.""" mo1, bom, p_final, p1, p2 = self.generate_mo(qty_final=1, qty_base_1=1, qty_base_2=1) self.env['stock.quant']._update_available_quantity(p1, self.stock_location_components, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location_components, 5.0) mo1.action_assign() mo2_form = Form(self.env['mrp.production']) mo2_form.product_id = p_final mo2_form.bom_id = bom mo2_form.product_qty = 1 mo2 = mo2_form.save() mo2.action_confirm() mo2.action_assign() mo3_form = Form(self.env['mrp.production']) mo3_form.product_id = p_final mo3_form.bom_id = bom mo3_form.product_qty = 1 mo3 = mo3_form.save() mo3.action_confirm() mo3.action_assign() mos = mo1 | mo2 | mo3 mos.button_mark_done() self.assertEqual(mos.move_raw_ids.mapped('state'), ['done'] * 6) self.assertEqual(mos.move_raw_ids.mapped('quantity'), [1] * 6) self.assertEqual(mos.move_finished_ids.mapped('state'), ['done'] * 3) self.assertEqual(mos.move_finished_ids.mapped('quantity'), [1] * 3) def test_components_availability(self): self.bom_2.unlink() # remove the kit bom of product_5 now = fields.Datetime.now() mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_3 # product_5 (2), product_4 (8), product_2 (12) mo_form.date_start = now mo = mo_form.save() self.assertEqual(mo.components_availability, False) # no compute for draft mo.action_confirm() self.assertEqual(mo.components_availability, 'Not Available') tommorrow = fields.Datetime.now() + timedelta(days=1) after_tommorrow = fields.Datetime.now() + timedelta(days=2) warehouse = self.env.ref('stock.warehouse0') move1 = self._create_move( self.product_5, self.env.ref('stock.stock_location_suppliers'), warehouse.lot_stock_id, product_uom_qty=2, date=tommorrow ) move2 = self._create_move( self.product_4, self.env.ref('stock.stock_location_suppliers'), warehouse.lot_stock_id, product_uom_qty=8, date=tommorrow ) move3 = self._create_move( self.product_2, self.env.ref('stock.stock_location_suppliers'), warehouse.lot_stock_id, product_uom_qty=12, date=tommorrow ) (move1 | move2 | move3)._action_confirm() mo.invalidate_recordset(['components_availability', 'components_availability_state']) self.assertEqual(mo.components_availability, f'Exp {format_date(self.env, tommorrow)}') self.assertEqual(mo.components_availability_state, 'late') mo.date_start = after_tommorrow self.assertEqual(mo.components_availability, f'Exp {format_date(self.env, tommorrow)}') self.assertEqual(mo.components_availability_state, 'expected') (move1 | move2 | move3).picked = True (move1 | move2 | move3)._action_done() mo.invalidate_recordset(['components_availability', 'components_availability_state']) self.assertEqual(mo.components_availability, 'Available') self.assertEqual(mo.components_availability_state, 'available') mo.action_assign() self.assertEqual(mo.reservation_state, 'assigned') self.assertEqual(mo.components_availability, 'Available') self.assertEqual(mo.components_availability_state, 'available') def test_immediate_validate_6(self): """In a production for a tracked product, clicking on mark as done without filling any quantities should pop up the immediate transfer wizard. Processing should choose a new lot for the finished product. """ mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=1, qty_base_1=1, qty_base_2=1, tracking_final='lot') self.env['stock.quant']._update_available_quantity(p1, self.stock_location_components, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location_components, 5.0) mo.action_assign() mo.button_mark_done() self.assertEqual(mo.move_raw_ids.mapped('state'), ['done'] * 2) self.assertEqual(mo.move_raw_ids.mapped('quantity'), [1] * 2) self.assertEqual(mo.move_finished_ids.state, 'done') self.assertEqual(mo.move_finished_ids.quantity, 1) self.assertTrue(mo.move_finished_ids.move_line_ids.lot_id != False) def test_immediate_validate_uom(self): """In a production with a different uom than the finished product one, the immediate production wizard should fill the correct quantities. """ p_final = self.env['product.product'].create({ 'name': 'final', 'type': 'product', }) component = self.env['product.product'].create({ 'name': 'component', 'type': 'product', }) bom = self.env['mrp.bom'].create({ 'product_id': p_final.id, 'product_tmpl_id': p_final.product_tmpl_id.id, 'product_uom_id': self.uom_unit.id, 'product_qty': 1.0, 'type': 'normal', 'consumption': 'flexible', 'bom_line_ids': [(0, 0, {'product_id': component.id, 'product_qty': 1})] }) self.env['stock.quant']._update_available_quantity(component, self.stock_location_components, 25.0) mo_form = Form(self.env['mrp.production']) mo_form.bom_id = bom mo_form.product_uom_id = self.uom_dozen mo_form.product_qty = 1 mo = mo_form.save() mo.action_confirm() mo.action_assign() mo.button_mark_done() self.assertEqual(mo.move_raw_ids.state, 'done') self.assertEqual(mo.move_raw_ids.quantity, 12) self.assertEqual(mo.move_finished_ids.state, 'done') self.assertEqual(mo.move_finished_ids.quantity, 1) self.assertEqual(component.qty_available, 13) def test_immediate_validate_uom_2(self): """The rounding precision of a component should be based on the UoM used in the MO for this component, not on the produced product's UoM nor the default UoM of the component""" uom_units = self.env.ref('uom.product_uom_unit') uom_L = self.env.ref('uom.product_uom_litre') uom_cL = self.env['uom.uom'].create({ 'name': 'cL', 'category_id': uom_L.category_id.id, 'uom_type': 'smaller', 'factor': 100, 'rounding': 1, }) uom_units.rounding = 1 uom_L.rounding = 0.01 product = self.env['product.product'].create({ 'name': 'SuperProduct', 'uom_id': uom_units.id, }) consumable_component = self.env['product.product'].create({ 'name': 'Consumable Component', 'type': 'consu', 'uom_id': uom_cL.id, 'uom_po_id': uom_cL.id, }) storable_component = self.env['product.product'].create({ 'name': 'Storable Component', 'type': 'product', 'uom_id': uom_cL.id, 'uom_po_id': uom_cL.id, }) self.env['stock.quant']._update_available_quantity(storable_component, self.env.ref('stock.stock_location_stock'), 100) # Despite the purpose of this test is to use multi uom # tests the production choose the right uoms on all models without # having the uom fields in the interface views self.env.user.groups_id -= self.env.ref('uom.group_uom') for component in [consumable_component, storable_component]: bom = self.env['mrp.bom'].create({ 'product_tmpl_id': product.product_tmpl_id.id, 'bom_line_ids': [(0, 0, { 'product_id': component.id, 'product_qty': 0.2, 'product_uom_id': uom_L.id, })], }) mo_form = Form(self.env['mrp.production']) mo_form.bom_id = bom mo = mo_form.save() mo.action_confirm() mo.button_mark_done() self.assertEqual(mo.move_raw_ids.product_uom_qty, 0.2) self.assertEqual(mo.move_raw_ids.quantity, 0.2) def test_copy(self): """ Check that copying a done production, create all the stock moves""" mo, bom, p_final, p1, p2 = self.generate_mo(qty_final=1, qty_base_1=1, qty_base_2=1) mo.action_confirm() mo_form = Form(mo) mo_form.qty_producing = 1 mo = mo_form.save() mo.button_mark_done() self.assertEqual(mo.state, 'done') mo_copy = mo.copy() self.assertTrue(mo_copy.move_raw_ids) self.assertTrue(mo_copy.move_finished_ids) mo_copy.action_confirm() mo_form = Form(mo_copy) mo_form.qty_producing = 1 mo_copy = mo_form.save() mo_copy.button_mark_done() self.assertEqual(mo_copy.state, 'done') def test_product_produce_different_uom(self): """ Check that for products tracked by lots, with component product UOM different from UOM used in the BOM, we do not create a new move line due to extra reserved quantity caused by decimal rounding conversions. """ picking_type = self.env['stock.picking.type'].search([('code', '=', 'mrp_operation')])[0] picking_type.use_auto_consume_components_lots = True # the overall decimal accuracy is set to 3 digits precision = self.env.ref('product.decimal_product_uom') precision.digits = 3 # define L and ml, L has rounding .001 but ml has rounding .01 # when producing e.g. 187.5ml, it will be rounded to .188L categ_test = self.env['uom.category'].create({'name': 'Volume Test'}) uom_L = self.env['uom.uom'].create({ 'name': 'Test Liters', 'category_id': categ_test.id, 'uom_type': 'reference', 'rounding': 0.001 }) uom_ml = self.env['uom.uom'].create({ 'name': 'Test ml', 'category_id': categ_test.id, 'uom_type': 'smaller', 'rounding': 0.01, 'factor_inv': 0.001, }) # create a product component and the final product using the component product_comp = self.env['product.product'].create({ 'name': 'Product Component', 'type': 'product', 'tracking': 'lot', 'categ_id': self.env.ref('product.product_category_all').id, 'uom_id': uom_L.id, 'uom_po_id': uom_L.id, }) product_final = self.env['product.product'].create({ 'name': 'Product Final', 'type': 'product', 'tracking': 'lot', 'categ_id': self.env.ref('product.product_category_all').id, 'uom_id': uom_L.id, 'uom_po_id': uom_L.id, }) # the products are tracked by lot, so we go through _generate_consumed_move_line self.env['stock.lot'].create({ 'name': 'Lot Final', 'product_id': product_final.id, 'company_id': self.env.company.id, }) lot_comp = self.env['stock.lot'].create({ 'name': 'Lot Component', 'product_id': product_comp.id, 'company_id': self.env.company.id, }) # update the quantity on hand for Component, in a lot self.stock_location = self.env.ref('stock.stock_location_stock') self.env['stock.quant']._update_available_quantity(product_comp, self.stock_location, 1, lot_id=lot_comp) # create a BOM for Final, using Component test_bom = self.env['mrp.bom'].create({ 'product_id': product_final.id, 'product_tmpl_id': product_final.product_tmpl_id.id, 'product_uom_id': uom_L.id, 'product_qty': 1.0, 'type': 'normal', 'bom_line_ids': [(0, 0, { 'product_id': product_comp.id, 'product_qty': 375.00, 'product_uom_id': uom_ml.id })], }) # create a MO for this BOM mo_product_final_form = Form(self.env['mrp.production']) mo_product_final_form.product_id = product_final mo_product_final_form.product_uom_id = uom_L mo_product_final_form.bom_id = test_bom mo_product_final_form.product_qty = 0.5 mo_product_final_form = mo_product_final_form.save() mo_product_final_form.action_confirm() mo_product_final_form.action_assign() self.assertEqual(mo_product_final_form.reservation_state, 'assigned') # produce mo_product_final_form.button_mark_done() # check that in _generate_consumed_move_line, # we do not create an extra move line because # of a conversion 187.5ml = 0.188L # thus creating an extra line with 'product_uom_qty': 0.5 self.assertEqual(len(mo_product_final_form.move_raw_ids.move_line_ids), 1, 'One move line should exist for the MO.') def test_mo_sn_warning(self): """ Checks that when a MO where the final product is tracked by serial, a warning pops up if the `lot_producting_id` has previously been used already (i.e. dupe SN). Also checks if a scrap linked to a MO has its sn warning correctly pop up. """ self.stock_location = self.env.ref('stock.stock_location_stock') mo, _, p_final, _, _ = self.generate_mo(tracking_final='serial', qty_base_1=1, qty_final=1) self.assertEqual(len(mo), 1, 'MO should have been created') sn1 = self.env['stock.lot'].create({ 'name': 'serial1', 'product_id': p_final.id, 'company_id': self.env.company.id, }) self.env['stock.quant']._update_available_quantity(p_final, self.stock_location, 1, lot_id=sn1) mo.lot_producing_id = sn1 warning = False warning = mo._onchange_lot_producing() self.assertTrue(warning, 'Reuse of existing serial number not detected') self.assertEqual(list(warning.keys())[0], 'warning', 'Warning message was not returned') mo.action_generate_serial() sn2 = mo.lot_producing_id mo.button_mark_done() # scrap linked to MO but with wrong SN location scrap = self.env['stock.scrap'].create({ 'product_id': p_final.id, 'product_uom_id': self.uom_unit.id, 'production_id': mo.id, 'location_id': self.stock_location_14.id, 'lot_id': sn2.id }) warning = False warning = scrap._onchange_serial_number() self.assertTrue(warning, 'Use of wrong serial number location not detected') self.assertEqual(list(warning.keys())[0], 'warning', 'Warning message was not returned') self.assertEqual(scrap.location_id, mo.location_dest_id, 'Location was not auto-corrected') def test_a_multi_button_plan(self): """ Test batch methods (confirm/validate) of the MO with the same bom """ self.bom_2.type = "normal" # avoid to get the operation of the kit bom mo_3 = Form(self.env['mrp.production']) mo_3.bom_id = self.bom_3 mo_3 = mo_3.save() self.assertEqual(len(mo_3.workorder_ids), 2) mo_3.button_plan() self.assertEqual(mo_3.state, 'confirmed') self.assertEqual(mo_3.workorder_ids[0].state, 'waiting') mo_1 = Form(self.env['mrp.production']) mo_1.bom_id = self.bom_3 mo_1 = mo_1.save() mo_2 = Form(self.env['mrp.production']) mo_2.bom_id = self.bom_3 mo_2 = mo_2.save() self.assertEqual(mo_1.product_id, self.product_6) self.assertEqual(mo_2.product_id, self.product_6) self.assertEqual(len(self.bom_3.operation_ids), 2) self.assertEqual(len(mo_1.workorder_ids), 2) self.assertEqual(len(mo_2.workorder_ids), 2) (mo_1 | mo_2).button_plan() # Confirm and plan in the same "request" self.assertEqual(mo_1.state, 'confirmed') self.assertEqual(mo_2.state, 'confirmed') self.assertEqual(mo_1.workorder_ids[0].state, 'waiting') self.assertEqual(mo_2.workorder_ids[0].state, 'waiting') # produce (mo_1 | mo_2).button_mark_done() self.assertEqual(mo_1.state, 'done') self.assertEqual(mo_2.state, 'done') def test_workcenter_timezone(self): # Workcenter is based in Bangkok # Possible working hours are Monday to Friday, from 8:00 to 12:00 and from 13:00 to 17:00 (UTC+7) workcenter = self.workcenter_1 workcenter.resource_calendar_id.tz = 'Asia/Bangkok' # The test will try to plan some WO on next Monday. We need to unlink all # useless times off to ensure that nothing will disturb the slot reservation (workcenter.resource_calendar_id.global_leave_ids | workcenter.resource_calendar_id.leave_ids).unlink() bom = self.env['mrp.bom'].create({ 'product_tmpl_id': self.product_1.product_tmpl_id.id, 'bom_line_ids': [(0, 0, { 'product_id': self.product_2.id, })], 'operation_ids': [(0, 0, { 'name': 'SuperOperation01', 'workcenter_id': workcenter.id, }), (0, 0, { 'name': 'SuperOperation01', 'workcenter_id': workcenter.id, })], }) # Next Monday at 6:00 am UTC date_start = (fields.Datetime.now() + timedelta(days=7 - fields.Datetime.now().weekday())).replace(hour=6, minute=0, second=0) mo_form = Form(self.env['mrp.production']) mo_form.bom_id = bom mo_form.date_start = date_start mo = mo_form.save() mo.workorder_ids[0].duration_expected = 240 mo.workorder_ids[1].duration_expected = 60 mo.action_confirm() mo.button_plan() # Asia/Bangkok is UTC+7 and the start date is on Monday at 06:00 UTC (i.e., 13:00 UTC+7). # So, in Bangkok, the first workorder uses the entire Monday afternoon slot 13:00 - 17:00 UTC+7 (i.e., 06:00 - 10:00 UTC) # The second job uses the beginning of the Tuesday morning slot: 08:00 - 09:00 UTC+7 (i.e., 01:00 - 02:00 UTC) self.assertEqual(mo.workorder_ids[0].date_start, date_start) self.assertEqual(mo.workorder_ids[0].date_finished, date_start + timedelta(hours=4)) tuesday = date_start + timedelta(days=1) self.assertEqual(mo.workorder_ids[1].date_start, tuesday.replace(hour=1)) self.assertEqual(mo.workorder_ids[1].date_finished, tuesday.replace(hour=2)) def test_backorder_with_overconsumption(self): """ Check that the components of the backorder have the correct quantities when there is overconsumption in the initial MO """ mo, _, _, _, _ = self.generate_mo(qty_final=30, qty_base_1=2, qty_base_2=3) mo.action_confirm() mo_form = Form(mo) mo_form.qty_producing = 10 mo = mo_form.save() mo.move_raw_ids[0].quantity = 90 mo.move_raw_ids[1].quantity = 70 action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] # Check quantities of the original MO self.assertEqual(mo.product_uom_qty, 10.0) self.assertEqual(mo.qty_produced, 10.0) move_prod_1 = self.env['stock.move'].search([ ('product_id', '=', mo.bom_id.bom_line_ids[0].product_id.id), ('raw_material_production_id', '=', mo.id)]) move_prod_2 = self.env['stock.move'].search([ ('product_id', '=', mo.bom_id.bom_line_ids[1].product_id.id), ('raw_material_production_id', '=', mo.id)]) self.assertEqual(sum(move_prod_1.mapped('quantity')), 90.0) self.assertEqual(sum(move_prod_1.mapped('product_uom_qty')), 30.0) self.assertEqual(sum(move_prod_2.mapped('quantity')), 70.0) self.assertEqual(sum(move_prod_2.mapped('product_uom_qty')), 20.0) # Check quantities of the backorder MO self.assertEqual(mo_backorder.product_uom_qty, 20.0) move_prod_1_bo = self.env['stock.move'].search([ ('product_id', '=', mo.bom_id.bom_line_ids[0].product_id.id), ('raw_material_production_id', '=', mo_backorder.id)]) move_prod_2_bo = self.env['stock.move'].search([ ('product_id', '=', mo.bom_id.bom_line_ids[1].product_id.id), ('raw_material_production_id', '=', mo_backorder.id)]) self.assertEqual(sum(move_prod_1_bo.mapped('product_uom_qty')), 60.0) self.assertEqual(sum(move_prod_2_bo.mapped('product_uom_qty')), 40.0) def test_backorder_with_underconsumption(self): """ Check that the components of the backorder have the correct quantities when there is underconsumption in the initial MO """ mo, _, _, p1, p2 = self.generate_mo(qty_final=20, qty_base_1=1, qty_base_2=1) mo.action_confirm() mo_form = Form(mo) mo_form.qty_producing = 10 mo = mo_form.save() mo.move_raw_ids.filtered(lambda m: m.product_id == p1).quantity = 5 mo.move_raw_ids.filtered(lambda m: m.product_id == p2).quantity = 10 action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] # Check quantities of the original MO self.assertEqual(mo.product_uom_qty, 10.0) self.assertEqual(mo.qty_produced, 10.0) move_prod_1_done = mo.move_raw_ids.filtered(lambda m: m.product_id == p1) self.assertEqual(sum(move_prod_1_done.mapped('quantity')), 5) self.assertEqual(sum(move_prod_1_done.mapped('product_uom_qty')), 10) move_prod_2 = mo.move_raw_ids.filtered(lambda m: m.product_id == p2) self.assertEqual(sum(move_prod_2.mapped('quantity')), 10) self.assertEqual(sum(move_prod_2.mapped('product_uom_qty')), 10) # Check quantities of the backorder MO self.assertEqual(mo_backorder.product_uom_qty, 10.0) move_prod_1_bo = mo_backorder.move_raw_ids.filtered(lambda m: m.product_id == p1) move_prod_2_bo = mo_backorder.move_raw_ids.filtered(lambda m: m.product_id == p2) self.assertEqual(sum(move_prod_1_bo.mapped('product_uom_qty')), 10.0) self.assertEqual(sum(move_prod_2_bo.mapped('product_uom_qty')), 10.0) def test_state_workorders(self): bom = self.env['mrp.bom'].create({ 'product_id': self.product_4.id, 'product_tmpl_id': self.product_4.product_tmpl_id.id, 'product_uom_id': self.uom_unit.id, 'product_qty': 1.0, 'consumption': 'flexible', 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 1}) ], 'operation_ids': [ (0, 0, {'name': 'amUgbidhaW1lIHBhcyBsZSBKUw==', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 15, 'sequence': 1}), (0, 0, {'name': '137 Python', 'workcenter_id': self.workcenter_1.id, 'time_cycle': 1, 'sequence': 2}), ], }) self.env['stock.quant'].create({ 'location_id': self.stock_location_components.id, 'product_id': self.product_2.id, 'inventory_quantity': 10 }).action_apply_inventory() mo = Form(self.env['mrp.production']) mo.bom_id = bom mo = mo.save() self.assertEqual(list(mo.workorder_ids.mapped("state")), ["waiting", "waiting"]) mo.action_confirm() mo.action_assign() self.assertEqual(mo.move_raw_ids.state, "assigned") self.assertEqual(list(mo.workorder_ids.mapped("state")), ["ready", "pending"]) mo.do_unreserve() self.assertEqual(list(mo.workorder_ids.mapped("state")), ["waiting", "pending"]) mo.workorder_ids[0].unlink() self.assertEqual(list(mo.workorder_ids.mapped("state")), ["waiting"]) mo.action_assign() self.assertEqual(list(mo.workorder_ids.mapped("state")), ["ready"]) mo.button_mark_done() self.assertEqual(list(mo.workorder_ids.mapped("state")), ["done"]) def test_products_with_variants(self): """Check for product with different variants with same bom""" attribute = self.env['product.attribute'].create({ 'name': 'Test Attribute', }) attribute_values = self.env['product.attribute.value'].create([{ 'name': 'Value 1', 'attribute_id': attribute.id, 'sequence': 1, }, { 'name': 'Value 2', 'attribute_id': attribute.id, 'sequence': 2, }]) product = self.env['product.template'].create({ "attribute_line_ids": [ [0, 0, {"attribute_id": attribute.id, "value_ids": [[6, 0, attribute_values.ids]]}] ], "name": "Product with variants", }) variant_1 = product.product_variant_ids[0] variant_2 = product.product_variant_ids[1] component = self.env['product.template'].create({ "name": "Component", }) self.env['mrp.bom'].create({ 'product_id': False, 'product_tmpl_id': product.id, 'bom_line_ids': [ (0, 0, {'product_id': component.product_variant_id.id, 'product_qty': 1}) ] }) # First behavior to check, is changing the product (same product but another variant) after saving the MO a first time. mo_form_1 = Form(self.env['mrp.production']) mo_form_1.product_id = variant_1 mo_1 = mo_form_1.save() mo_form_1 = Form(self.env['mrp.production'].browse(mo_1.id)) mo_form_1.product_id = variant_2 mo_1 = mo_form_1.save() mo_1.action_confirm() mo_1.action_assign() mo_form_1 = Form(self.env['mrp.production'].browse(mo_1.id)) mo_form_1.qty_producing = 1 mo_1 = mo_form_1.save() mo_1.button_mark_done() move_lines_1 = self.env['stock.move.line'].search([("reference", "=", mo_1.name)]) move_finished_ids_1 = self.env['stock.move'].search([("production_id", "=", mo_1.id)]) self.assertEqual(len(move_lines_1), 2, "There should only be 2 move lines: the component line and produced product line") self.assertEqual(len(move_finished_ids_1), 1, "There should only be 1 produced product for this MO") self.assertEqual(move_finished_ids_1.product_id, variant_2, "Incorrect variant produced") # Second behavior is changing the product before saving the MO mo_form_2 = Form(self.env['mrp.production']) mo_form_2.product_id = variant_1 mo_form_2.product_id = variant_2 mo_2 = mo_form_2.save() mo_2.action_confirm() mo_2.action_assign() mo_form_2 = Form(self.env['mrp.production'].browse(mo_2.id)) mo_form_2.qty_producing = 1 mo_2 = mo_form_2.save() mo_2.button_mark_done() move_lines_2 = self.env['stock.move.line'].search([("reference", "=", mo_2.name)]) move_finished_ids_2 = self.env['stock.move'].search([("production_id", "=", mo_2.id)]) self.assertEqual(len(move_lines_2), 2, "There should only be 2 move lines: the component line and produced product line") self.assertEqual(len(move_finished_ids_2), 1, "There should only be 1 produced product for this MO") self.assertEqual(move_finished_ids_2.product_id, variant_2, "Incorrect variant produced") # Third behavior is changing the product before saving the MO, then another time after mo_form_3 = Form(self.env['mrp.production']) mo_form_3.product_id = variant_1 mo_form_3.product_id = variant_2 mo_3 = mo_form_3.save() mo_form_3 = Form(self.env['mrp.production'].browse(mo_3.id)) mo_form_3.product_id = variant_1 mo_3 = mo_form_3.save() mo_3.action_confirm() mo_3.action_assign() mo_form_3 = Form(self.env['mrp.production'].browse(mo_3.id)) mo_form_3.qty_producing = 1 mo_3 = mo_form_3.save() mo_3.button_mark_done() move_lines_3 = self.env['stock.move.line'].search([("reference", "=", mo_3.name)]) move_finished_ids_3 = self.env['stock.move'].search([("production_id", "=", mo_3.id)]) self.assertEqual(len(move_lines_3), 2, "There should only be 2 move lines: the component line and produced product line") self.assertEqual(len(move_finished_ids_3), 1, "There should only be 1 produced product for this MO") self.assertEqual(move_finished_ids_3.product_id, variant_1, "Incorrect variant produced") def test_move_finished_onchanges(self): """ Test that move_finished_ids (i.e. produced products) are still correct even after multiple onchanges have changed the moves """ product1 = self.env['product.product'].create({ 'name': 'Oatmeal Cookie', }) product2 = self.env['product.product'].create({ 'name': 'Chocolate Chip Cookie', }) # ===== product_id onchange checks ===== # # check product_id onchange without saving mo_form = Form(self.env['mrp.production']) mo_form.product_id = product1 mo_form.product_id = product2 mo = mo_form.save() self.assertEqual(len(mo.move_finished_ids), 1, 'Wrong number of finished product moves created') self.assertEqual(mo.move_finished_ids.product_id, product2, 'Wrong product to produce in finished product move') # check product_id onchange after saving mo_form = Form(self.env['mrp.production'].browse(mo.id)) mo_form.product_id = product1 mo = mo_form.save() self.assertEqual(len(mo.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo.move_finished_ids.product_id, product1, 'Wrong product to produce in finished product move') # check product_id onchange when mo._origin.product_id is unchanged mo_form = Form(self.env['mrp.production'].browse(mo.id)) mo_form.product_id = product2 mo_form.product_id = product1 mo = mo_form.save() self.assertEqual(len(mo.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo.move_finished_ids.product_id, product1, 'Wrong product to produce in finished product move') # ===== product_qty onchange checks ===== # # check product_qty onchange without saving mo_form = Form(self.env['mrp.production']) mo_form.product_id = product1 mo_form.product_qty = 5 mo_form.product_qty = 10 mo2 = mo_form.save() self.assertEqual(len(mo2.move_finished_ids), 1, 'Wrong number of finished product moves created') self.assertEqual(mo2.move_finished_ids.product_qty, 10, 'Wrong qty to produce for the finished product move') # check product_qty onchange after saving mo_form = Form(self.env['mrp.production'].browse(mo2.id)) mo_form.product_qty = 5 mo2 = mo_form.save() self.assertEqual(len(mo2.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo2.move_finished_ids.product_qty, 5, 'Wrong qty to produce for the finished product move') # check product_qty onchange when mo._origin.product_id is unchanged mo_form = Form(self.env['mrp.production'].browse(mo2.id)) mo_form.product_qty = 10 mo_form.product_qty = 5 mo2 = mo_form.save() self.assertEqual(len(mo2.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo2.move_finished_ids.product_qty, 5, 'Wrong qty to produce for the finished product move') # ===== product_uom_id onchange checks ===== # mo_form = Form(self.env['mrp.production']) mo_form.product_id = product1 mo_form.product_qty = 1 mo_form.product_uom_id = self.env['uom.uom'].browse(self.ref('uom.product_uom_dozen')) mo3 = mo_form.save() self.assertEqual(len(mo3.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo3.move_finished_ids.product_qty, 12, 'Wrong qty to produce for the finished product move') # ===== bom_id onchange checks ===== # component = self.env['product.product'].create({ "name": "Sugar", }) bom1 = self.env['mrp.bom'].create({ 'product_id': False, 'product_tmpl_id': product1.product_tmpl_id.id, 'bom_line_ids': [ (0, 0, {'product_id': component.id, 'product_qty': 1}) ] }) bom2 = self.env['mrp.bom'].create({ 'product_id': False, 'product_tmpl_id': product1.product_tmpl_id.id, 'bom_line_ids': [ (0, 0, {'product_id': component.id, 'product_qty': 10}) ] }) # check bom_id onchange before product change mo_form = Form(self.env['mrp.production']) mo_form.bom_id = bom1 mo_form.bom_id = bom2 mo_form.product_id = product2 mo4 = mo_form.save() self.assertFalse(mo4.bom_id, 'BoM should have been removed') self.assertEqual(len(mo4.move_finished_ids), 1, 'Wrong number of finished product moves created') self.assertEqual(mo4.move_finished_ids.product_id, product2, 'Wrong product to produce in finished product move') # check bom_id onchange after product change mo_form = Form(self.env['mrp.production'].browse(mo4.id)) mo_form.product_id = product1 mo_form.bom_id = bom1 mo_form.bom_id = bom2 mo4 = mo_form.save() self.assertEqual(len(mo4.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo4.move_finished_ids.product_id, product1, 'Wrong product to produce in finished product move') # check product_id onchange when mo._origin.product_id is unchanged mo_form = Form(self.env['mrp.production'].browse(mo4.id)) mo_form.bom_id = bom2 mo_form.bom_id = bom1 mo4 = mo_form.save() self.assertEqual(len(mo4.move_finished_ids), 1, 'Wrong number of finish product moves created') self.assertEqual(mo4.move_finished_ids.product_id, product1, 'Wrong product to produce in finished product move') def test_compute_tracked_time_1(self): """ Checks that the Duration Computation (`time_mode` of mrp.routing.workcenter) with value `auto` with Based On (`time_mode_batch`) set to 1 actually compute the time based on the last 1 operation, and not more. Create a first production in 15 minutes (expected should go from 60 to 15 Create a second one in 10 minutes (expected should NOT go from 15 to 12.5, it should go from 15 to 10) """ # First production, the default is 60 and there is 0 productions of that operation # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_4 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 60.0, "Default duration is 0+0+1*60.0") production.action_confirm() production.button_plan() # Production planned, time to start, I produce all the 1 product # 'invisible': [('state', '=', 'draft')] production_form = Form(production) production_form.qty_producing = 1 with production_form.workorder_ids.edit(0) as wo: wo.duration = 15 # in 15 minutes production = production_form.save() production.button_mark_done() # It is saved and done, registered in the db. There are now 1 productions of that operation # Same production, let's see what the duration_expected is, last prod was 15 minutes for 1 item production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_4 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 15.0, "Duration is now 0+0+1*15") production.action_confirm() production.button_plan() # Production planned, time to start, I produce all the 1 product # 'invisible': [('state', '=', 'draft')] production_form = Form(production) production_form.qty_producing = 1 with production_form.workorder_ids.edit(0) as wo: wo.duration = 10 # In 10 minutes this time production = production_form.save() production.button_mark_done() # It is saved and done, registered in the db. There are now 2 productions of that operation # Same production, let's see what the duration_expected is, last prod was 10 minutes for 1 item # Total average time would be 12.5 but we compute the duration based on the last 1 item production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_4 production = production_form.save() self.assertNotEqual(production.workorder_ids[0].duration_expected, 12.5, "Duration expected is based on the last 1 production, not last 2") self.assertEqual(production.workorder_ids[0].duration_expected, 10.0, "Duration is now 0+0+1*10") def test_compute_tracked_time_2_under_capacity(self): """ Test that when tracking the 2 last production, if we make one with under capacity, and one with normal capacity, the two are equivalent (1 done with capacity 2 in 10mn = 2 done with capacity 2 in 10mn) """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_5 production = production_form.save() production.action_confirm() production.button_plan() # Production planned, time to start, I produce all the 1 product # 'invisible': [('state', '=', 'draft')] production_form = Form(production) production_form.qty_producing = 1 with production_form.workorder_ids.edit(0) as wo: wo.duration = 10 # in 10 minutes production = production_form.save() production.button_mark_done() # It is saved and done, registered in the db. There are now 1 productions of that operation # Same production, let's see what the duration_expected is, last prod was 10 minutes for 1 item production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_5 production_form.product_qty = 2 # We want to produce 2 items (the capacity) now production = production_form.save() self.assertNotEqual(production.workorder_ids[0].duration_expected, 20.0, "We made 1 item with capacity 2 in 10mn -> so 2 items shouldn't be double that") self.assertEqual(production.workorder_ids[0].duration_expected, 10.0, "Producing 1 or 2 items with capacity 2 is the same duration") production.action_confirm() production.button_plan() # Production planned, time to start, I produce all the 2 product # 'invisible': [('state', '=', 'draft')] production_form = Form(production) production_form.qty_producing = 2 with production_form.workorder_ids.edit(0) as wo: wo.duration = 10 # In 10 minutes this time production = production_form.save() production.button_mark_done() # It is saved and done, registered in the db. There are now 2 productions of that operation but they have the same duration production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_5 production = production_form.save() self.assertNotEqual(production.workorder_ids[0].duration_expected, 15, "Producing 1 or 2 in 10mn with capacity 2 take the same amount of time : 10mn") self.assertEqual(production.workorder_ids[0].duration_expected, 10.0, "Duration is indeed (10+10)/2") def test_capacity_duration_expected(self): """ Test that the duration expected is correctly computed when dealing with below or above capacity 1 -> 10mn 2 -> 10mn 3 -> 20mn 4 -> 20mn 5 -> 30mn ... """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_6 production = production_form.save() production.action_confirm() production.button_plan() # Production planned, time to start, I produce all the 1 product # 'invisible': [('state', '=', 'draft')] production_form = Form(production) production_form.qty_producing = 1 with production_form.workorder_ids.edit(0) as wo: wo.duration = 10 # in 10 minutes production = production_form.save() production.button_mark_done() production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_6 production = production_form.save() # production_form.product_qty = 1 [BY DEFAULT] self.assertEqual(production.workorder_ids[0].duration_expected, 10.0, "Produce 1 with capacity 2, expected is 10mn for each run -> 10mn") production_form.product_qty = 2 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 10.0, "Produce 2 with capacity 2, expected is 10mn for each run -> 10mn") production_form.product_qty = 3 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 20.0, "Produce 3 with capacity 2, expected is 10mn for each run -> 20mn") production_form.product_qty = 4 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 20.0, "Produce 4 with capacity 2, expected is 10mn for each run -> 20mn") production_form.product_qty = 5 production = production_form.save() self.assertEqual(production.workorder_ids[0].duration_expected, 30.0, "Produce 5 with capacity 2, expected is 10mn for each run -> 30mn") def test_workorder_set_duration(self): """ Test that manually setting duration correctly creates/updates time_ids """ mo = Form(self.env['mrp.production']) mo.bom_id = self.bom_4 mo = mo.save() mo.action_confirm() workorder = mo.workorder_ids[0] expected_duration = workorder.duration_expected real_duration_under_expected = expected_duration / 2 # first value real_duration_increased_above_expected = 2 * expected_duration # to create next 2 values real_duration_decreased = expected_duration * .75 # reduced amount that overlaps the last 2 values # update real duration to amount under expected duration workorder.duration = real_duration_under_expected self.assertEqual(len(workorder.time_ids), 1, "A time tracking value should have been created") self.assertEqual(workorder.time_ids[0].loss_type, 'productive', "Total duration < Expected duration => should be productive time") self.assertEqual(workorder.time_ids[0].duration, real_duration_under_expected, "Incorrect duration for time tracking value") # update real duration to amount above expected duration workorder.duration = real_duration_increased_above_expected self.assertEqual(len(workorder.time_ids), 3, "Two more time tracking values should have been created") self.assertEqual(workorder.time_ids[1].loss_type, 'productive', "Duration amount added under the expected duration should be productive time") self.assertEqual(workorder.time_ids[2].loss_type, 'performance', "Duration amount added above expected duration should be performance (i.e. reduced) time") self.assertEqual(workorder.time_ids[1].duration, expected_duration - real_duration_under_expected, "Added (productive) time should be expected duration - already existing duration") self.assertEqual(workorder.time_ids[2].duration, real_duration_increased_above_expected - expected_duration, "Added (reduced) time should be total duration - expected duration") # update real duration to amount below expected duration workorder.duration = real_duration_decreased # reducing time reverses the time_id order... so we reverse the check self.assertEqual(len(workorder.time_ids), 2, "One time tracking values should have been deleted") self.assertEqual(workorder.time_ids[1].loss_type, 'productive', "Original time tracking should be unchanged") self.assertEqual(workorder.time_ids[1].duration, real_duration_under_expected, "Original time tracking should be unchanged") self.assertEqual(workorder.time_ids[0].loss_type, 'productive', "Remaining time tracking should be productive") self.assertEqual(workorder.time_ids[0].duration, real_duration_decreased - real_duration_under_expected, "Time tracking duration should have been reduced to reflect new shorter duration") def test_propagate_quantity_on_backorders(self): """Create a MO for a product with several work orders. Produce different quantities to test quantity propagation and workorder cancellation. """ # setup test work_center_1 = self.env['mrp.workcenter'].create({"name": "WorkCenter 1", "time_start": 11}) work_center_2 = self.env['mrp.workcenter'].create({"name": "WorkCenter 2", "time_start": 12}) work_center_3 = self.env['mrp.workcenter'].create({"name": "WorkCenter 3", "time_start": 13}) product = self.env['product.template'].create({"name": "Finished Product"}) component_1 = self.env['product.template'].create({"name": "Component 1", "type": "product"}) component_2 = self.env['product.template'].create({"name": "Component 2", "type": "product"}) component_3 = self.env['product.template'].create({"name": "Component 3", "type": "product"}) self.env['stock.quant'].create({ "product_id": component_1.product_variant_id.id, "location_id": 8, "quantity": 100 }) self.env['stock.quant'].create({ "product_id": component_2.product_variant_id.id, "location_id": 8, "quantity": 100 }) self.env['stock.quant'].create({ "product_id": component_3.product_variant_id.id, "location_id": 8, "quantity": 100 }) self.env['mrp.bom'].create({ "product_tmpl_id": product.id, "product_id": False, "product_qty": 1, "bom_line_ids": [ [0, 0, {"product_id": component_1.product_variant_id.id, "product_qty": 1}], [0, 0, {"product_id": component_2.product_variant_id.id, "product_qty": 1}], [0, 0, {"product_id": component_3.product_variant_id.id, "product_qty": 1}] ], "operation_ids": [ [0, 0, {"name": "Operation 1", "workcenter_id": work_center_1.id}], [0, 0, {"name": "Operation 2", "workcenter_id": work_center_2.id}], [0, 0, {"name": "Operation 3", "workcenter_id": work_center_3.id}] ] }) # create a manufacturing order for 20 products mo_form = Form(self.env['mrp.production']) mo_form.product_id = product.product_variant_id mo_form.product_qty = 20 mo = mo_form.save() self.assertEqual(mo.state, 'draft') mo.action_confirm() wo_1, wo_2, wo_3 = mo.workorder_ids self.assertEqual(mo.state, 'confirmed') self.assertEqual(wo_1.state, 'ready') self.assertEqual(wo_1.duration_expected, 11 + 20 * 60) # produce 20 / 10 / 5 on workorders, create backorder duration_expected = wo_1.duration_expected wo_1.button_start() wo_1.qty_producing = 20 self.assertEqual(mo.state, 'progress') wo_1.button_finish() self.assertEqual(duration_expected, wo_1.duration_expected) duration_expected = wo_2.duration_expected wo_2.button_start() wo_2.qty_producing = 10 wo_2.button_finish() self.assertEqual(duration_expected, wo_2.duration_expected) duration_expected = wo_3.duration_expected wo_3.button_start() wo_3.qty_producing = 5 wo_3.button_finish() self.assertEqual(duration_expected, wo_3.duration_expected) self.assertEqual(mo.state, 'to_close') mo.button_mark_done() bo = self.env['mrp.production.backorder'].create({ "mrp_production_backorder_line_ids": [ [0, 0, {"mrp_production_id": mo.id, "to_backorder": True}] ] }) bo.action_backorder() self.assertEqual(mo.state, 'done') mo_2 = mo.procurement_group_id.mrp_production_ids - mo wo_4, wo_5, wo_6 = mo_2.workorder_ids self.assertEqual(wo_4.state, 'cancel') self.assertEqual(wo_5.duration_expected, 12 + 15 * 60) # produce 10 / 5, create backorder wo_5.button_start() wo_5.qty_producing = 10 self.assertEqual(mo_2.state, 'progress') wo_5.button_finish() wo_6.button_start() wo_6.qty_producing = 5 wo_6.button_finish() self.assertEqual(mo_2.state, 'to_close') mo_2.button_mark_done() bo = self.env['mrp.production.backorder'].create({ "mrp_production_backorder_line_ids": [ [0, 0, {"mrp_production_id": mo_2.id, "to_backorder": True}] ] }) bo.action_backorder() self.assertEqual(mo_2.state, 'done') mo_3 = mo.procurement_group_id.mrp_production_ids - (mo | mo_2) wo_7, wo_8, wo_9 = mo_3.workorder_ids self.assertEqual(wo_7.state, 'cancel') self.assertEqual(wo_8.state, 'cancel') self.assertEqual(wo_9.duration_expected, 13 + 10 * 60) # produce 10 and finish work wo_9.button_start() wo_9.qty_producing = 10 self.assertEqual(mo_3.state, 'progress') wo_9.button_finish() self.assertEqual(mo_3.state, 'to_close') mo_3.button_mark_done() self.assertEqual(mo_3.state, 'done') def test_planning_workorder(self): """ Check that the fastest work center is used when planning the workorder. - create two work centers with similar production capacity but the work_center_2 with a longer start and stop time. 1:/ produce 2 units > work_center_1 faster because it does not need much time to start and to finish the production. 2/ - update the production capacity of the work_center_2 to 4 - produce 4 units > work_center_2 faster because it must do a single cycle while the work_center_1 have to do two cycles. """ workcenter_1 = self.env['mrp.workcenter'].create({ 'name': 'wc1', 'default_capacity': 2, 'time_start': 1, 'time_stop': 1, 'time_efficiency': 100, }) workcenter_2 = self.env['mrp.workcenter'].create({ 'name': 'wc2', 'default_capacity': 2, 'time_start': 10, 'time_stop': 5, 'time_efficiency': 100, 'alternative_workcenter_ids': [workcenter_1.id] }) product_to_build = self.env['product.product'].create({ 'name': 'final product', 'type': 'product', }) product_to_use = self.env['product.product'].create({ 'name': 'component', 'type': 'product', }) bom = self.env['mrp.bom'].create({ 'product_id': product_to_build.id, 'product_tmpl_id': product_to_build.product_tmpl_id.id, 'product_uom_id': self.uom_unit.id, 'product_qty': 1.0, 'type': 'normal', 'consumption': 'flexible', 'operation_ids': [ (0, 0, {'name': 'Test', 'workcenter_id': workcenter_2.id, 'time_cycle': 60, 'sequence': 1}), ], 'bom_line_ids': [ (0, 0, {'product_id': product_to_use.id, 'product_qty': 1}), ]}) #MO_1 mo_form = Form(self.env['mrp.production']) mo_form.product_id = product_to_build mo_form.bom_id = bom mo_form.product_qty = 2 mo = mo_form.save() mo.action_confirm() mo.button_plan() self.assertEqual(mo.workorder_ids[0].workcenter_id.id, workcenter_1.id, 'workcenter_1 is faster than workcenter_2 to manufacture 2 units') # Unplan the mo to prevent the first workcenter from being busy mo.button_unplan() # Update the production capcity workcenter_2.default_capacity = 4 #MO_2 mo_form = Form(self.env['mrp.production']) mo_form.product_id = product_to_build mo_form.bom_id = bom mo_form.product_qty = 4 mo_2 = mo_form.save() mo_2.action_confirm() mo_2.button_plan() self.assertEqual(mo_2.workorder_ids[0].workcenter_id.id, workcenter_2.id, 'workcenter_2 is faster than workcenter_1 to manufacture 4 units') def test_timers_after_cancelling_mo(self): """ Check that the timers in the workorders are stopped after the cancellation of the MO """ mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_2 mo_form.product_qty = 1 mo = mo_form.save() mo.action_confirm() mo.button_plan() wo = mo.workorder_ids wo.button_start() mo.action_cancel() self.assertEqual(mo.state, 'cancel', 'Manufacturing order should be cancelled.') self.assertEqual(wo.state, 'cancel', 'Workorders should be cancelled.') self.assertTrue(mo.workorder_ids.time_ids.date_end, 'The timers must stop after the cancellation of the MO') def test_starting_wo_twice(self): """ Check that the work order is started only once when clicking the start button several times. """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') production_form = Form(self.env['mrp.production']) production_form.bom_id = self.bom_2 production_form.product_qty = 1 production = production_form.save() production_form = Form(production) with production_form.workorder_ids.new() as wo: wo.name = 'OP1' wo.workcenter_id = self.workcenter_1 wo.duration_expected = 40 production = production_form.save() production.action_confirm() production.button_plan() production.workorder_ids[0].button_start() production.workorder_ids[0].button_start() self.assertEqual(len(production.workorder_ids[0].time_ids.filtered(lambda t: t.date_start and not t.date_end)), 1) def test_qty_update_and_method_reservation(self): """ When the reservation method of Manufacturing is 'manual', updating the quantity of a confirmed MO shouldn't trigger the reservation of the components """ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], order='id', limit=1) warehouse.manu_type_id.reservation_method = 'manual' for product in self.product_1 + self.product_2: product.type = 'product' self.env['stock.quant']._update_available_quantity(product, warehouse.lot_stock_id, 10) mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_1 mo = mo_form.save() mo.action_confirm() self.assertFalse(mo.move_raw_ids.move_line_ids) wizard = self.env['change.production.qty'].create({ 'mo_id': mo.id, 'product_qty': 5, }) wizard.change_prod_qty() self.assertFalse(mo.move_raw_ids.move_line_ids) def test_source_and_child_mo(self): """ Suppose three manufactured products A, B and C. C is a component of B and B is a component of A. If B and C have the routes MTO + Manufacture, when producing one A, it should generate a MO for B and C. Moreover, starting from one of the MOs, we should be able to find the source/child MO. (The test checks the flow in 1-step, 2-steps and 3-steps manufacturing) """ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1) mto_route = warehouse.mto_pull_id.route_id manufacture_route = warehouse.manufacture_pull_id.route_id mto_route.active = True grandparent, parent, child = self.env['product.product'].create([{ 'name': n, 'type': 'product', 'route_ids': [(6, 0, mto_route.ids + manufacture_route.ids)], } for n in ['grandparent', 'parent', 'child']]) component = self.env['product.product'].create({ 'name': 'component', 'type': 'consu', }) self.env['mrp.bom'].create([{ 'product_tmpl_id': finished_product.product_tmpl_id.id, 'product_qty': 1, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': compo.id, 'product_qty': 1}), ], } for finished_product, compo in [(grandparent, parent), (parent, child), (child, component)]]) none_production = self.env['mrp.production'] for steps, case_description, in [('mrp_one_step', '1-step Manufacturing'), ('pbm', '2-steps Manufacturing'), ('pbm_sam', '3-steps Manufacturing')]: warehouse.manufacture_steps = steps grandparent_production_form = Form(self.env['mrp.production']) grandparent_production_form.product_id = grandparent grandparent_production = grandparent_production_form.save() grandparent_production.action_confirm() child_production, parent_production = self.env['mrp.production'].search([('product_id', 'in', (parent + child).ids)], order='id desc', limit=2) for source_mo, mo, product, child_mo in [(none_production, grandparent_production, grandparent, parent_production), (grandparent_production, parent_production, parent, child_production), (parent_production, child_production, child, none_production)]: self.assertEqual(mo.product_id, product, '[%s] There should be a MO for product %s' % (case_description, product.display_name)) self.assertEqual(mo.mrp_production_source_count, len(source_mo), '[%s] Incorrect value for product %s' % (case_description, product.display_name)) self.assertEqual(mo.mrp_production_child_count, len(child_mo), '[%s] Incorrect value for product %s' % (case_description, product.display_name)) source_action = mo.action_view_mrp_production_sources() child_action = mo.action_view_mrp_production_childs() self.assertEqual(source_action.get('res_id', False), source_mo.id, '[%s] Incorrect value for product %s' % (case_description, product.display_name)) self.assertEqual(child_action.get('res_id', False), child_mo.id, '[%s] Incorrect value for product %s' % (case_description, product.display_name)) @freeze_time('2022-06-28 08:00') def test_replan_workorders01(self): """ Create two MO, each one with one WO. Set the same scheduled start date to each WO during the creation of the MO. A warning will be displayed. -> The user replans one of the WO: the warnings should disappear and the WO should be postponed. """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') mos = self.env['mrp.production'] for _ in range(2): mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_4 with mo_form.workorder_ids.edit(0) as wo_line: wo_line.date_start = datetime.now() mos += mo_form.save() mos.action_confirm() mo_01, mo_02 = mos wo_01 = mo_01.workorder_ids wo_02 = mo_02.workorder_ids self.assertTrue(wo_01.show_json_popover) self.assertTrue(wo_02.show_json_popover) wo_02.action_replan() self.assertFalse(wo_01.show_json_popover) self.assertFalse(wo_02.show_json_popover) self.assertEqual(wo_01.date_finished, wo_02.date_start) @freeze_time('2022-06-28 08:00') def test_replan_workorders02(self): """ Create two MO, each one with one WO. Set the same scheduled start date to each WO after the creation of the MO. A warning will be displayed. -> The user replans one of the WO: the warnings should disappear and the WO should be postponed. """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') mos = self.env['mrp.production'] for _ in range(2): mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_4 mos += mo_form.save() mos.action_confirm() mo_01, mo_02 = mos for mo in mos: with Form(mo) as mo_form: with mo_form.workorder_ids.edit(0) as wo_line: wo_line.date_start = datetime.now() wo_01 = mo_01.workorder_ids wo_02 = mo_02.workorder_ids self.assertTrue(wo_01.show_json_popover) self.assertTrue(wo_02.show_json_popover) wo_02.action_replan() self.assertFalse(wo_01.show_json_popover) self.assertFalse(wo_02.show_json_popover) self.assertEqual(wo_01.date_finished, wo_02.date_start) @freeze_time('2022-10-05 12:00') def test_replan_mo_without_bom(self): """ Create 2 MOs without BoM just set the product and a component For first MO : Add 2 WO (with different WC) Don't schedule first WO Schedule second WO Confirm => MO is Confirmed and Planned Schedule first WO before second WO Confirm => MO should Replan without Error For second MO : Add 1 Scheduled WO Confirm => MO is Confirmed and Planned Add a second WO scheduled before the other one (with different WC) Confirm => MO should Replan without Error """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') mos = self.env['mrp.production'] for _ in range(2): mo_form = Form(self.env['mrp.production']) mo_form.product_id = self.product_8 with mo_form.move_raw_ids.new() as component: component.product_id = self.product_6 mos += mo_form.save() mo_01, mo_02 = mos #First MO with Form(mo_01) as mo_01_form: with mo_01_form.workorder_ids.new() as workorder: workorder.name = "OP1" workorder.workcenter_id = self.workcenter_2 with mo_01_form.workorder_ids.new() as workorder: workorder.name = "OP2" workorder.workcenter_id = self.workcenter_3 workorder.date_start = datetime(2022, 10, 23, 12) mo_01 = mo_01_form.save() mo_01.action_confirm() op_1, op_2 = mo_01.workorder_ids.sorted('id') self.assertEqual(op_2.date_start, datetime(2022, 10, 23, 12)) with Form(mo_01) as mo_01_form: with mo_01_form.workorder_ids.edit(1) as workorder: workorder.date_start = datetime(2022, 10, 18, 12) mo_01 = mo_01_form.save() self.assertEqual(op_1.date_start, datetime(2022, 10, 18, 12)) # no auto replan self.assertEqual(op_2.date_start, datetime(2022, 10, 23, 12)) self.assertNotEqual(op_1.date_finished, op_2.date_start) #Second MO with Form(mo_02) as mo_02_form: with mo_02_form.workorder_ids.new() as workorder: workorder.name = "OP1" workorder.workcenter_id = self.workcenter_2 workorder.date_start = datetime(2022, 10, 20, 12) mo_02 = mo_02_form.save() mo_02.action_confirm() with Form(mo_02) as mo_02_form: with mo_02_form.workorder_ids.new() as workorder: workorder.name = "OP2" workorder.workcenter_id = self.workcenter_3 workorder.date_start = datetime(2022, 10, 18, 12) mo_02 = mo_02_form.save() op_1, op_2 = mo_02.workorder_ids.sorted('id') self.assertEqual(op_1.date_start, datetime(2022, 10, 20, 12)) self.assertTrue(op_2.show_json_popover) @freeze_time('2023-03-01 12:00') def test_planning_cancelled_workorder(self): """Test when plan start time for workorders, cancelled workorders won't be taken into account. """ self.env.company.resource_calendar_id.tz = 'Europe/Brussels' workcenter_1 = self.env['mrp.workcenter'].create({ 'name': 'wc1', 'default_capacity': 1, 'time_start': 10, 'time_stop': 5, 'time_efficiency': 100, }) workcenter_2 = self.env['mrp.workcenter'].create({ 'name': 'wc2', 'default_capacity': 1, 'time_start': 10, 'time_stop': 5, 'time_efficiency': 100, }) workcenter_3 = self.env['mrp.workcenter'].create({ 'name': 'wc3', 'default_capacity': 1, 'time_start': 10, 'time_stop': 5, 'time_efficiency': 100, }) bom = self.env['mrp.bom'].create({ 'product_id': self.product_6.id, 'product_tmpl_id': self.product_6.product_tmpl_id.id, 'product_uom_id': self.uom_unit.id, 'ready_to_produce': 'asap', 'consumption': 'flexible', 'product_qty': 1.0, 'operation_ids': [ (0, 0, {'name': 'Cutting Machine', 'workcenter_id': workcenter_1.id, 'time_cycle_manual': 30, 'sequence': 1}), (0, 0, {'name': 'Weld Machine', 'workcenter_id': workcenter_2.id, 'time_cycle_manual': 30, 'sequence': 2}), (0, 0, {'name': 'Gift Wrap Machine', 'workcenter_id': workcenter_3.id, 'time_cycle_manual': 30, 'sequence': 3}), ], 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 1}) ]}) mo_form = Form(self.env['mrp.production']) mo_form.product_id = self.product_6 mo_form.bom_id = bom mo_form.product_qty = 2 mo = mo_form.save() mo.action_confirm() mo.button_plan() self.assertEqual(mo.workorder_ids[0].date_start, datetime(2023, 3, 1, 12, 0)) self.assertEqual(mo.workorder_ids[1].date_start, datetime(2023, 3, 1, 13, 15)) self.assertEqual(mo.workorder_ids[2].date_start, datetime(2023, 3, 1, 14, 30)) # wo_1 completely finished mo_form = Form(mo) mo_form.qty_producing = 2 mo = mo_form.save() mo.workorder_ids[0].button_start() mo.workorder_ids[0].button_finish() # wo_2, wo_3 partially finished mo_form.qty_producing = 1 mo = mo_form.save() mo.workorder_ids[1].button_start() mo.workorder_ids[1].button_finish() mo.workorder_ids[2].button_start() mo.workorder_ids[2].button_finish() action = mo.button_mark_done() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] mo_backorder.button_plan() self.assertEqual(mo_backorder.workorder_ids[0].state, 'cancel') self.assertEqual(mo_backorder.workorder_ids[1].state, 'waiting') self.assertEqual(mo_backorder.workorder_ids[2].state, 'pending') self.assertFalse(mo_backorder.workorder_ids[0].date_start) self.assertEqual(mo_backorder.workorder_ids[1].date_start, datetime(2023, 3, 1, 12, 0)) self.assertEqual(mo_backorder.workorder_ids[2].date_start, datetime(2023, 3, 1, 12, 45)) def test_compute_product_id(self): """ Tests the creation of a production order automatically sets the product when the bom is provided, without the need to put it in the vals of the create nor to call onchanges. """ order = self.env['mrp.production'].create({ 'bom_id': self.bom_1.id, }) self.assertEqual(order.product_id, self.bom_1.product_id) def test_compute_product_uom_id(self): """ Tests the creation of a production order automatically sets the uom when the bom is provided, without the need to put it in the vals of the create nor to call onchanges. """ order = self.env['mrp.production'].create({ 'bom_id': self.bom_1.id, }) self.assertEqual(order.product_uom_id, self.bom_1.product_uom_id) def test_compute_bom_id(self): """ Tests the creation of a production order automatically sets the bom when the product is provided, without the need to put it in the vals of the create nor to call onchanges. """ order = self.env['mrp.production'].create({ 'product_id': self.bom_1.product_id.id, }) self.assertEqual(order.bom_id, self.bom_1) def test_move_raw_uom_rounding(self): """Test that the correct rouding is applied on move_raw in manufacturing orders""" self.box250 = self.env['uom.uom'].create({ 'name': 'box250', 'category_id': self.env.ref('uom.product_uom_categ_unit').id, 'ratio': 250.0, 'uom_type': 'bigger', 'rounding': 1.0, }) test_bom = self.env['mrp.bom'].create({ 'product_tmpl_id': self.product_7_template.id, 'product_uom_id': self.uom_unit.id, 'product_qty': 250.0, 'type': 'normal', 'bom_line_ids': [ (0, 0, {'product_id': self.product_2.id, 'product_qty': 1.0, 'product_uom_id': self.box250.id}), ] }) self.env['stock.quant'].create({ 'location_id':self.env.ref('stock.stock_location_stock').id, 'product_id': self.product_2.id, 'inventory_quantity': 500 }).action_apply_inventory() mo_form = Form(self.env['mrp.production']) mo_form.bom_id = test_bom mo = mo_form.save() mo.action_confirm() self.assertEqual(mo.move_raw_ids.product_uom_qty, 1) self.assertEqual(mo.move_raw_ids.move_line_ids.quantity, mo.move_raw_ids.product_uom_qty) self.assertEqual(mo.move_raw_ids.availability, 250) update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': mo.id, 'product_qty': 300, }) update_quantity_wizard.change_prod_qty() self.assertEqual(mo.move_raw_ids.product_uom_qty, 2) self.assertEqual(mo.move_raw_ids.move_line_ids.quantity, mo.move_raw_ids.product_uom_qty) self.assertEqual(mo.move_raw_ids.availability, 0) def test_update_qty_to_consume_of_component(self): """ The UoM of the finished product has a rounding precision equal to 1.0 and the UoM of the component has a decimal one. When the producing qty is set, an onchange autocomplete the consumed quantity of the component. Then, when updating the 'to consume' quantity of the components, their consumed quantity is updated again. The test ensures that this update respects the rounding precisions """ self.uom_dozen.rounding = 1 self.bom_4.product_uom_id = self.uom_dozen mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_4 mo = mo_form.save() mo.action_confirm() mo.action_toggle_is_locked() with Form(mo) as mo_form: mo_form.qty_producing = 1 with mo_form.move_raw_ids.edit(0) as raw: raw.product_uom_qty = 1.25 self.assertEqual(mo.move_raw_ids.quantity, 1.25) def test_clear_finished_move(self): """ Test that the finished moves created by the compute are correctly erased after changing the finished product""" mo_form = Form(self.env['mrp.production']) mo_form.product_id = self.product_1 mo = mo_form.save() self.assertEqual(len(mo.move_finished_ids), 1) mo.product_id = self.product_2 self.assertEqual(len(mo.move_finished_ids), 1) self.assertFalse(self.env['stock.move'].search([ ('product_id', '=', self.product_1.id), ('state', '=', 'draft'), ])) def test_compute_picking_type_id(self): """ Test that the operation type set on the bom is set in the manufacturing order when selecting the BoM""" self.env.user.groups_id += self.env.ref("stock.group_adv_location") picking_type = self.env['stock.picking.type'].create({ 'name': 'new_picking_type', 'code': 'internal', 'sequence_code': 'NPT', 'default_location_src_id': self.env.ref('stock.stock_location_stock').id, 'default_location_dest_id': self.stock_location_components.id, 'warehouse_id': self.warehouse_1.id, }) self.bom_1.picking_type_id = picking_type mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_1 mo = mo_form.save() self.assertEqual(mo.picking_type_id.id, picking_type.id) # MO_2 self.assertFalse(self.bom_2.picking_type_id) mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_2 mo_2 = mo_form.save() picking_type_company = self.env['stock.picking.type'].search_read([ ('code', '=', 'mrp_operation'), ('warehouse_id.company_id', 'in', mo_2.company_id.ids), ], ['company_id'], load=False, limit=1) self.assertEqual(mo_2.picking_type_id.id, picking_type_company[0]['id']) def test_onchange_bom_ids_and_picking_type(self): warehouse01 = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1) warehouse02, warehouse03 = self.env['stock.warehouse'].create([ {'name': 'Second Warehouse', 'code': 'WH02'}, {'name': 'Third Warehouse', 'code': 'WH03'}, ]) finished_product = self.env['product.product'].create({'name': 'finished product'}) bom_wh01, bom_wh02 = self.env['mrp.bom'].create([{ 'product_id': finished_product.id, 'product_tmpl_id': finished_product.product_tmpl_id.id, 'product_uom_id': self.uom_unit.id, 'product_qty': 1.0, 'bom_line_ids': [(0, 0, {'product_id': self.product.id, 'product_qty': 1})], 'picking_type_id': wh.manu_type_id.id, 'sequence': wh.id, } for wh in [warehouse01, warehouse02]]) # Prioritize BoM of WH02 bom_wh01.sequence = bom_wh02.sequence + 1 mo_form = Form(self.env['mrp.production']) mo_form.product_id = finished_product self.assertEqual(mo_form.bom_id, bom_wh02, 'Should select the first BoM in the list, whatever the picking type is') self.assertEqual(mo_form.picking_type_id, warehouse02.manu_type_id) mo_form.bom_id = bom_wh01 self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should be adapted because of the found BoM') mo_form.bom_id = bom_wh02 self.assertEqual(mo_form.picking_type_id, warehouse02.manu_type_id, 'Should be adapted because of the found BoM') mo_form.picking_type_id = warehouse01.manu_type_id self.assertEqual(mo_form.bom_id, bom_wh02, 'Should not change') self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should not change') mo_form.picking_type_id = warehouse03.manu_type_id mo_form.bom_id = bom_wh01 self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should be adapted because of the found BoM ' '(the selected picking type should be ignored)') mo_form = Form(self.env['mrp.production'].with_context(default_picking_type_id=warehouse03.manu_type_id.id)) mo_form.product_id = finished_product self.assertFalse(mo_form.bom_id, 'Should not find any BoM, because of the defined picking type') self.assertEqual(mo_form.picking_type_id, warehouse03.manu_type_id) mo_form = Form(self.env['mrp.production'].with_context(default_picking_type_id=warehouse01.manu_type_id.id)) mo_form.product_id = finished_product self.assertEqual(mo_form.bom_id, bom_wh01, 'Should select the BoM that matches the default picking type') self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should be the default one') mo_form.bom_id = bom_wh02 self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should not change, because of default value') mo_form.picking_type_id = warehouse02.manu_type_id self.assertEqual(mo_form.bom_id, bom_wh02, 'Should not change') self.assertEqual(mo_form.picking_type_id, warehouse02.manu_type_id, 'Should not change') mo_form.picking_type_id = warehouse02.manu_type_id mo_form.bom_id = bom_wh02 self.assertEqual(mo_form.picking_type_id, warehouse01.manu_type_id, 'Should be adapted because of the default value') def test_workcenter_specific_capacities(self): """ Test that the duraction expected is correctly computed when specific capacities are defined on the workcenter. """ # Required for `workorder_ids` to be visible in the view self.env.user.groups_id += self.env.ref('mrp.group_mrp_routings') self.workcenter_2.update({ 'time_start': 10, 'time_stop': 20, }) self.env['mrp.workcenter.capacity'].create({ 'product_id': self.product_4.id, 'workcenter_id': self.workcenter_2.id, 'time_start': 5, 'time_stop': 10, }) production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_5 production = production_form.save() with Form(production) as mo_form: with mo_form.workorder_ids.new() as wo: wo.name = "OP1" wo.workcenter_id = self.workcenter_2 # Since no duration was given, only duration from the workcenter setup/cleanup time should be added in : 10 + 20 = 30 self.assertEqual(production.workorder_ids[0].duration_expected, 30.0, "Workcenter setup time (10) + workcenter cleanup time (20)") # Change the product so that it uses a specific capacity of that workcenter with Form(production) as mo_form: mo_form.product_id = self.product_4 with mo_form.workorder_ids.new() as wo: wo.name = "OP1" wo.workcenter_id = self.workcenter_2 # Only duration from the workcenter specific capacity setup/cleanup times since there is one defined for this product. self.assertEqual(production.workorder_ids[0].duration_expected, 15.0, "Capacity setup time (5) + capacity cleanup time (10)") def test_unlink_workorder_with_consumed_operations(self): self.bom_3.bom_line_ids[0].operation_id = self.bom_3.operation_ids[0].id self.bom_3.bom_line_ids[1].operation_id = self.bom_3.operation_ids[1].id mo_form = Form(self.env['mrp.production']) mo_form.bom_id = self.bom_3 mo = mo_form.save() mo.workorder_ids[1].unlink() mo.action_confirm() self.assertEqual(mo.state, 'confirmed') self.assertEqual(len(mo.workorder_ids), 2) def test_consumption_action_set_qty_and_validate(self): """ Check `To Consume` and `Consumed` qty are correctly updated to match the consumption warning values under 5 use cases: scenario 1: - bom is changed after MO is created => action_set_qty = match BoM scenario 2 (combined 3 use cases since they shouldn't affect each other): - a component move is deleted before MO is confirmed => action_set_qty = add missing BoM component - a component's UoM is changed after MO is created => action_set_qty = match BoM qty, but leave UoM unchanged (i.e. correctly convert) - add a new component not part of the BOM scenario 3: - a component has 2 moves in a MO => action_set_qty = set the 1st move to the correct qty, set 2nd move to 0 (i.e. no way to know how to distribute quantity across these moves since warning aggregates qty by product) """ mo, bom, p_final, p1, p2 = self.generate_mo(consumption='warning', qty_final=10, qty_base_1=12, qty_base_2=20) #### scenario 1 - change BoM after MO created #### mo_form = Form(mo) mo_form.qty_producing = 4 mo = mo_form.save() # mo.move_raw_ids[0] = p2 => 20 qty_base, mo.move_raw_ids[1] = p1 => 12 qty_base self.assertEqual(mo.move_raw_ids[0].product_uom_qty, 200, "current MO To Consume qty should match expected qty to produce") self.assertEqual(mo.move_raw_ids[0].quantity, 80, "current MO Consumed qty should match expected qty to produce") self.assertEqual(mo.move_raw_ids[1].product_uom_qty, 120, "current MO To Consume qty should match expected qty produced") self.assertEqual(mo.move_raw_ids[1].quantity, 48, "current MO Consumed qty should match expected qty produced") # bom changes won't auto-update MO, it will only show diff in consumption warning bom.bom_line_ids[0].product_qty = 10 self.assertEqual(mo.move_raw_ids[0].product_uom_qty, 200) self.assertEqual(mo.move_raw_ids[0].quantity, 80) action = mo.button_mark_done() warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])) consumption = warning.save() self.assertEqual(consumption.mrp_consumption_warning_line_ids.product_consumed_qty_uom, 80, "qty consumed incorrectly passed to wizard") self.assertEqual(consumption.mrp_consumption_warning_line_ids.product_expected_qty_uom, 40, "expected qty should match current BoM qty for qty being produced") action = consumption.action_set_qty() backorder = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder.save().action_backorder() self.assertEqual(mo.move_raw_ids[0].product_uom_qty, 80, "current bom expected qty should remain unchanged") self.assertEqual(mo.move_raw_ids[0].quantity, 40, "current bom expected qty was not applied as qty to be done") self.assertEqual(mo.move_raw_ids[1].product_uom_qty, 48, "line without consumption issue was incorrectly changed") self.assertEqual(mo.move_raw_ids[1].quantity, 48, "line without consumption issue was incorrectly changed") self.assertEqual(mo.state, 'done') # double check that backorder qtys are also correct mo_backorder = mo.procurement_group_id.mrp_production_ids[-1] self.assertEqual(mo_backorder.move_raw_ids[0].product_uom_qty, 120, "backorder values are based on original MO, not current bom") self.assertEqual(mo_backorder.move_raw_ids[1].product_uom_qty, 72, "backorder values incorrectly calculated") #### scenario 2 - that removing a line in the MO + changing the uom of a line + adding a component not on the BOM #### mo2_form = Form(self.env['mrp.production']) mo2_form.product_id = p_final mo2_form.bom_id = bom mo2_form.product_qty = 5.0 with mo2_form.move_raw_ids.new() as move: move.product_id = self.product_1 move.product_uom_qty = 50 mo2 = mo2_form.save() for move in mo2.move_raw_ids: if move.product_id == p2: move.unlink() elif move.product_id == p1: # p1 = qty_base_1 = 12 => now 12 dozens instead of units move.product_uom = self.env.ref('uom.product_uom_dozen') mo2.action_confirm() mo2_form = Form(mo2) mo2_form.qty_producing = 4 mo2 = mo2_form.save() self.assertEqual(len(mo2.move_raw_ids), 2, "current MO should still have 1 component from its BoM deleted + 1 additional component") self.assertEqual(mo2.move_raw_ids[0].product_uom_qty, 60, "current MO To Consume qty should match manually set expected qty produced") self.assertEqual(mo2.move_raw_ids[0].quantity, 48, "current MO Consumed qty should match expected qty to produce based on manually set value") self.assertEqual(mo2.move_raw_ids[1].product_uom_qty, 50, "current MO To Consume qty should match manually set expected qty produced") self.assertEqual(mo2.move_raw_ids[1].quantity, 40, "current MO Consumed qty should match expected qty to produce based on manually set value") action = mo2.button_mark_done() warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])) consumption = warning.save() self.assertEqual(len(consumption.mrp_consumption_warning_line_ids), 3, "deleted move should also show as an consumption line diff from BoM") # mrp_consumption_warning_line_ids[2] = p1 => 12 unit qty_base # mrp_consumption_warning_line_ids[1] = p2 => 10 unit qty_base # mrp_consumption_warning_line_ids[0] = self.product_1 => 10 unit qty_base self.assertEqual(consumption.mrp_consumption_warning_line_ids[0].product_consumed_qty_uom, 40, "additional component was not correctly passed to wizard") self.assertEqual(consumption.mrp_consumption_warning_line_ids[0].product_expected_qty_uom, 0, "additional component should have no expected qty") self.assertEqual(consumption.mrp_consumption_warning_line_ids[1].product_consumed_qty_uom, 0, "missing line was not correctly passed to wizard") self.assertEqual(consumption.mrp_consumption_warning_line_ids[1].product_expected_qty_uom, 40, "expected qty should match current BoM qty for qty being produced") self.assertEqual(consumption.mrp_consumption_warning_line_ids[2].product_consumed_qty_uom, 576, "qty consumed was not correctly converted to product's uom before passing to wizard") self.assertEqual(consumption.mrp_consumption_warning_line_ids[2].product_expected_qty_uom, 48, "expected qty should match current BoM qty for qty being produced") action = consumption.action_set_qty() backorder2 = Form(self.env['mrp.production.backorder'].with_context(**action['context'])) backorder2.save().action_backorder() # expect 3 moves: 1 for the originally missing product p2 with qty demand/done = 40 # 1 for the overused product p1 one with qty demand/done = 48/12 = 4 dozens # 1 for the additional product self.product_1 with demand/done = 40/0 self.assertEqual(len(mo2.move_raw_ids), 3, "missing line was not correctly added") for move in mo2.move_raw_ids: if move.product_id == p2: self.assertEqual(move.product_uom_qty, 40, "missing line values were not correctly added") self.assertEqual(move.quantity, 40, "missing line values were not correctly added") elif move.product_id == p1: self.assertEqual(move.product_uom_qty, 48, "expected qty should be unchanged") self.assertEqual(move.quantity, 4, "expected qty was not applied as qty to be done (UoM was possibly not correctly converted)") else: self.assertEqual(move.product_uom_qty, 40, "additional component's demand should have carried over") self.assertEqual(move.quantity, 0, "additional component should have nothing reserved") self.assertEqual(mo2.state, 'done') # double check that backorder qtys are also correct mo2_backorder = mo2.procurement_group_id.mrp_production_ids[-1] self.assertEqual(len(mo2_backorder.move_raw_ids), 2, "missing line should NOT have been added in but additional line should") self.assertEqual(mo2_backorder.move_raw_ids.product_id.ids, [p1.id, self.product_1.id]) self.assertEqual(mo2_backorder.move_raw_ids[0].product_uom_qty, 12, "backorder values are based on original MO, not current bom") #### scenario 3 - repeated comp move #### # bom.bom_line_ids[0]/product_id = p2 bom.bom_line_ids[0].unlink() mo3 = self.env['mrp.production'].create({ 'product_id': p_final.id, 'bom_id': bom.id, 'product_qty': 1, 'product_uom_id': p_final.uom_id.id, }) mo3_form = Form(mo3) with mo3_form.move_raw_ids.new() as line: line.product_id = p1 line.product_uom_qty = 5 mo3 = mo3_form.save() mo3.action_confirm() self.assertEqual(len(mo3.move_raw_ids), 2, "there should be 2 comp lines") self.assertEqual(len(mo3.move_raw_ids.product_id), 1, "comp lines should have same product") mo3_form = Form(mo3) mo3_form.qty_producing = 1 mo3 = mo3_form.save() self.assertEqual(mo3.move_raw_ids[0].product_uom_qty, 12, "BoM created comp move does not match expected To Consume qty") self.assertEqual(mo3.move_raw_ids[0].quantity, 12, "BoM created comp move does not match expected Consumed qty") self.assertEqual(mo3.move_raw_ids[1].product_uom_qty, 5, "Manually added comp move does not match original To Consume qty") self.assertEqual(mo3.move_raw_ids[1].quantity, 5, "Manually added comp move was not Consumed") action = mo3.button_mark_done() warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])) consumption = warning.save() self.assertEqual(len(consumption.mrp_consumption_warning_line_ids), 1, "warning lines should be grouped by product") self.assertEqual(consumption.mrp_consumption_warning_line_ids[0].product_expected_qty_uom, 12, "BoM expected qty not correctly passed to wizard") self.assertEqual(consumption.mrp_consumption_warning_line_ids[0].product_consumed_qty_uom, 17, "total Consumed qty not correctly passed to wizard") action = consumption.action_set_qty() self.assertEqual(mo3.move_raw_ids[0].product_uom_qty, 12, "BoM created comp move does not match expected To Consume qty") self.assertEqual(mo3.move_raw_ids[0].quantity, 12, "BoM created comp move does not match expected Consumed qty") self.assertEqual(mo3.move_raw_ids[1].product_uom_qty, 5, "Manually added comp move To Consume qty should be unchanged") self.assertEqual(mo3.move_raw_ids[1].quantity, 0, "Extra line Consumed qty not correctly zero-ed") self.assertEqual(mo3.state, 'done') def test_exceeded_consumed_qty_and_duplicated_lines(self): """ Two components C01, C02. C01 has the MTO route. MO with 1 x C01, 1 x C02, 1 x C02. Process the MO and set a high consumed qty for C01. Ensure that the MO can still be processed and that the consumed quantities are correct. """ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1) mto_route = warehouse.mto_pull_id.route_id manufacture_route = warehouse.manufacture_pull_id.route_id mto_route.active = True product01, product02, product03 = self.env['product.product'].create([{ 'name': 'Product %s' % (i + 1), 'type': 'product', } for i in range(3)]) product02.route_ids = [(6, 0, (mto_route | manufacture_route).ids)] mo_form = Form(self.env['mrp.production']) mo_form.product_id = product01 mo_form.product_qty = 1 for component in (product02, product03, product03): with mo_form.move_raw_ids.new() as line: line.product_id = component line.product_uom_qty = 1 mo = mo_form.save() mo.action_confirm() mo_form = Form(mo) mo_form.qty_producing = 1.0 mo = mo_form.save() mo.move_raw_ids[0].move_line_ids.quantity = 1.5 mo.button_mark_done() self.assertEqual(mo.state, 'done') p02_raws = mo.move_raw_ids.filtered(lambda m: m.product_id == product02) p03_raws = mo.move_raw_ids.filtered(lambda m: m.product_id == product03) self.assertEqual(sum(p02_raws.mapped('quantity')), 1.5) self.assertEqual(sum(p03_raws.mapped('quantity')), 2) def test_validation_mo_with_tracked_component(self): """ check that the verification of SN for tracked component is ignored when the quantity to consume is 0. """ self.product_2.tracking = 'serial' bom = self.env["mrp.bom"].create({ 'product_tmpl_id': self.product_4.product_tmpl_id.id, 'product_qty': 1.0, 'bom_line_ids': [(0, 0, { 'product_id': self.product_2.id, 'product_qty': 1.0, }), (0, 0, { 'product_id': self.product_3.id, 'product_qty': 1.0, })] }) # create the MO and confirm it mo = self.env['mrp.production'].create({ 'product_id': self.product_4.id, 'bom_id': bom.id, 'product_qty': 1.0, }) mo.action_confirm() self.assertEqual(mo.state, 'confirmed') # set the qty to consume of the tracked product to 0 mo.move_raw_ids[0].product_uom_qty = 0 mo.move_raw_ids[0].quantity = 0 # Set MO Done and create backorder action = mo.button_mark_done() consumption_warning = Form(self.env['mrp.consumption.warning'].with_context(**action['context'])).save() self.assertEqual(len(consumption_warning.mrp_consumption_warning_line_ids), 1) self.assertEqual(consumption_warning.mrp_consumption_warning_line_ids[0].product_consumed_qty_uom, 0) self.assertEqual(consumption_warning.mrp_consumption_warning_line_ids[0].product_expected_qty_uom, 1) # Force the warning consumption_warning.action_confirm() self.assertEqual(mo.state, 'done') def test_cancel_return(self): """ check that the return picking is not created on done state transfer when reducing MO quantity. """ self.stock_location = self.env.ref('stock.stock_location_stock') warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1) warehouse.manufacture_steps = 'pbm' mo, _bom, _p_final, p1, p2 = self.generate_mo(qty_final=5.0, qty_base_1=1.0, qty_base_2=1.0) mo.action_confirm() self.env['stock.quant']._update_available_quantity(p1, self.stock_location, 5.0) self.env['stock.quant']._update_available_quantity(p2, self.stock_location, 5.0) mo.picking_ids.move_ids[0].quantity = 5.0 mo.picking_ids.move_ids[1].quantity = 5.0 mo.picking_ids.button_validate() update_quantity_wizard = self.env['change.production.qty'].create({ 'mo_id': mo.id, 'product_qty': 4.0, }) update_quantity_wizard.change_prod_qty() new_picking = mo.picking_ids self.assertEqual(len(new_picking), 1, "Return picking should not be created in done Transfer") def test_manufacture_lead_days(self): """Test the lead days computation for manufacturing route. """ rule = self.env['stock.rule'].search([('action', '=', 'manufacture')], limit=1) self.env.company.manufacturing_lead = 1 self.bom_1.days_to_prepare_mo = 2 self.bom_1.produce_delay = 3 delays, _ = rule._get_lead_days(self.bom_1.product_id, bom=self.bom_1) self.assertEqual(delays['total_delay'], self.env.company.manufacturing_lead + self.bom_1.days_to_prepare_mo + self.bom_1.produce_delay) # switch to the 3 steps, only pre-production rules delays will be taken into account warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1) warehouse.manufacture_steps = 'pbm_sam' warehouse.pbm_route_id.rule_ids.delay = 100 delays, _ = rule._get_lead_days(self.bom_1.product_id, bom=self.bom_1) self.assertEqual(delays['total_delay'], self.env.company.manufacturing_lead + self.bom_1.days_to_prepare_mo + self.bom_1.produce_delay + 100 * 2) def test_use_kit_as_component_in_production_without_bom(self): """ Test that a MO is not cancelled when a kit is added in a MO without a BoM. """ finished, component, kit = self.env['product.product'].create([{ 'name': 'Product %s' % (i + 1), 'type': 'product', } for i in range(3)]) self.env['mrp.bom'].create({ 'product_id': kit.id, 'product_tmpl_id': kit.product_tmpl_id.id, 'type': 'phantom', 'bom_line_ids': [(0, 0, { 'product_id': component.id, 'product_qty': 1, })], }) mo_form = Form(self.env['mrp.production']) mo_form.product_id = finished mo_form.product_qty = 1 with mo_form.move_raw_ids.new() as line: line.product_id = kit mo = mo_form.save() mo.action_confirm() self.assertEqual(mo.state, 'confirmed') self.assertEqual(mo.move_raw_ids.product_id, component) def test_product_variants_in_mo(self): """ Test that the moves are corrltly removed when the poduct variant is changed """ # Add another attribute line to test efficiency the function bom_line check size_attribute_line = self.env['product.template.attribute.line'].create([{ 'product_tmpl_id': self.product_7_template.id, 'attribute_id': self.size_attribute.id, 'value_ids': [(6, 0, self.size_attribute.value_ids.ids)] }]) c1, c2, c3 = self.env['product.product'].create([{ 'name': i, 'type': 'product', } for i in range(3)]) 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', 'bom_line_ids': [ Command.create({ 'product_id': c1.id, 'product_qty': 1, 'bom_product_template_attribute_value_ids': [(4, self.product_7_attr1_v2.id)]}), # Blue color Command.create({ 'product_id': c2.id, 'product_qty': 1, 'bom_product_template_attribute_value_ids': [ (4, self.product_7_attr1_v1.id), # Red color (4, size_attribute_line.product_template_value_ids[2].id) # size L ]}), Command.create({ 'product_id': c3.id, 'product_qty': 1, 'bom_product_template_attribute_value_ids': [(4, self.product_7_attr1_v1.id)]}), # Red color ] }) mo_form = Form(self.env['mrp.production']) # select a product with a blue and s size attribute mo_form.product_id = self.product_7_template.product_variant_ids[1] mo_form.product_qty = 1 mo = mo_form.save() self.assertEqual(mo.move_raw_ids.product_id, c1) # select a product with a red and L attribute (the compoent C1 should be removed and C2, C3 added) mo_form.product_id = self.product_7_template.product_variant_ids[6] mo = mo_form.save() self.assertEqual(mo.move_raw_ids.product_id, (c2 | c3)) # select the product with red and s attribute (C2 and C3 should be removed and C1 added) mo_form.product_id = self.product_7_template.product_variant_ids[0] mo = mo_form.save() self.assertEqual(mo.move_raw_ids.product_id, c3) def test_mo_duration_expected(self): """ Test to verify that the 'duration_expected' on a work order in a manufacturing order correctly remains as manually set after completion. This test involves creating a product with a Bill of Materials (BOM) and an operation with an initial expected duration. A manufacturing order is then created for this product, the expected duration of the work order is manually changed, and the order is completed. The test checks that the expected duration remains as manually set and does not revert to the original value. """ production_form = Form(self.env['mrp.production']) production_form.product_id = self.product_5 production_form.bom_id = self.bom_2 production_form.product_qty = 1.0 production = production_form.save() production.action_confirm() init_duration_expected = production.workorder_ids.duration_expected production.workorder_ids.duration_expected = init_duration_expected + 5 production_form = Form(production) production_form.qty_producing = 1.0 production = production_form.save() production.button_mark_done() self.assertEqual(production.workorder_ids.duration_expected, init_duration_expected + 5) def test_multi_edit_start_date_wo(self): """ Test setting the start date for multiple workorders, checking if the finish date will be set too. As if the finish date is not set the planned workorder will not be shown in planning gantt view """ mo = self.env['mrp.production'].create({ 'product_id': self.product.id, 'product_uom_id': self.bom_1.product_uom_id.id, }) wos = self.env['mrp.workorder'].create([ { 'name': 'Test order', 'workcenter_id': self.workcenter_1.id, 'product_uom_id': self.bom_1.product_uom_id.id, 'production_id': mo.id, 'duration_expected': 1.0 }, { 'name': 'Test order2', 'workcenter_id': self.workcenter_2.id, 'product_uom_id': self.bom_1.product_uom_id.id, 'production_id': mo.id, 'duration_expected': 2.0 } ]) dt = datetime(2024, 1, 17, 11) wos.date_start = dt self.assertEqual(wos[0].date_start, dt) self.assertEqual(wos[1].date_start, dt) self.assertEqual(wos[0].date_finished, dt + timedelta(hours=1, minutes=1)) self.assertEqual(wos[1].date_finished, dt + timedelta(hours=1, minutes=2))