diff options
| author | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
|---|---|---|
| committer | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
| commit | 3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch) | |
| tree | a44932296ef4a9b71d5f010906253d8c53727726 /addons/stock_picking_batch/tests/test_batch_picking.py | |
| parent | 0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff) | |
initial commit 2
Diffstat (limited to 'addons/stock_picking_batch/tests/test_batch_picking.py')
| -rw-r--r-- | addons/stock_picking_batch/tests/test_batch_picking.py | 367 |
1 files changed, 367 insertions, 0 deletions
diff --git a/addons/stock_picking_batch/tests/test_batch_picking.py b/addons/stock_picking_batch/tests/test_batch_picking.py new file mode 100644 index 00000000..ecf50379 --- /dev/null +++ b/addons/stock_picking_batch/tests/test_batch_picking.py @@ -0,0 +1,367 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +from datetime import datetime, timedelta + +from odoo.tests import Form +from odoo.tests.common import TransactionCase + + +class TestBatchPicking(TransactionCase): + + def setUp(self): + """ Create a picking batch with two pickings from stock to customer """ + super(TestBatchPicking, self).setUp() + self.stock_location = self.env.ref('stock.stock_location_stock') + self.customer_location = self.env.ref('stock.stock_location_customers') + self.picking_type_out = self.env['ir.model.data'].xmlid_to_res_id('stock.picking_type_out') + self.productA = self.env['product.product'].create({ + 'name': 'Product A', + 'type': 'product', + 'categ_id': self.env.ref('product.product_category_all').id, + }) + self.productB = self.env['product.product'].create({ + 'name': 'Product B', + 'type': 'product', + 'categ_id': self.env.ref('product.product_category_all').id, + }) + + self.picking_client_1 = self.env['stock.picking'].create({ + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + 'picking_type_id': self.picking_type_out, + 'company_id': self.env.company.id, + }) + + self.env['stock.move'].create({ + 'name': self.productA.name, + 'product_id': self.productA.id, + 'product_uom_qty': 10, + 'product_uom': self.productA.uom_id.id, + 'picking_id': self.picking_client_1.id, + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + }) + + self.picking_client_2 = self.env['stock.picking'].create({ + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + 'picking_type_id': self.picking_type_out, + 'company_id': self.env.company.id, + }) + + self.env['stock.move'].create({ + 'name': self.productB.name, + 'product_id': self.productB.id, + 'product_uom_qty': 10, + 'product_uom': self.productA.uom_id.id, + 'picking_id': self.picking_client_2.id, + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + }) + + self.picking_client_3 = self.env['stock.picking'].create({ + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + 'picking_type_id': self.picking_type_out, + 'company_id': self.env.company.id, + }) + + self.env['stock.move'].create({ + 'name': self.productB.name, + 'product_id': self.productB.id, + 'product_uom_qty': 10, + 'product_uom': self.productA.uom_id.id, + 'picking_id': self.picking_client_3.id, + 'location_id': self.stock_location.id, + 'location_dest_id': self.customer_location.id, + }) + + self.batch = self.env['stock.picking.batch'].create({ + 'name': 'Batch 1', + 'company_id': self.env.company.id, + 'picking_ids': [(4, self.picking_client_1.id), (4, self.picking_client_2.id)] + }) + + def test_batch_scheduled_date(self): + """ Test to make sure the correct scheduled date is set for both a batch and its pickings. + Setting a batch's scheduled date manually has different behavior from when it is automatically + set/updated via compute. + """ + + now = datetime.now().replace(microsecond=0) + self.batch.scheduled_date = now + + # TODO: this test cannot currently handle the onchange scheduled_date logic because of test form + # view not handling the M2M widget assigned to picking_ids (O2M). Hopefully if this changes then + # commented parts of this test can be used later. + + + # manually set batch scheduled date => picking's scheduled dates auto update to match (onchange logic test) + # with Form(self.batch) as batch_form: + # batch_form.scheduled_date = now - timedelta(days=1) + # batch_form.save() + # self.assertEqual(self.batch.scheduled_date, self.picking_client_1.scheduled_date) + # self.assertEqual(self.batch.scheduled_date, self.picking_client_2.scheduled_date) + + picking1_scheduled_date = now - timedelta(days=2) + picking2_scheduled_date = now - timedelta(days=3) + picking3_scheduled_date = now - timedelta(days=4) + + # manually update picking scheduled dates => batch's scheduled date auto update to match lowest value + self.picking_client_1.scheduled_date = picking1_scheduled_date + self.picking_client_2.scheduled_date = picking2_scheduled_date + self.assertEqual(self.batch.scheduled_date, self.picking_client_2.scheduled_date) + # but individual pickings keep original scheduled dates + self.assertEqual(self.picking_client_1.scheduled_date, picking1_scheduled_date) + self.assertEqual(self.picking_client_2.scheduled_date, picking2_scheduled_date) + + # add a new picking with an earlier scheduled date => batch's scheduled date should auto-update + self.picking_client_3.scheduled_date = picking3_scheduled_date + self.batch.write({'picking_ids': [(4, self.picking_client_3.id)]}) + self.assertEqual(self.batch.scheduled_date, self.picking_client_3.scheduled_date) + + # remove that picking and batch scheduled date should auto-update to next min date + self.batch.write({'picking_ids': [(3, self.picking_client_3.id)]}) + self.assertEqual(self.batch.scheduled_date, self.picking_client_2.scheduled_date) + + # directly add new picking with an earlier scheduled date => batch's scheduled date auto updates to match, + # but existing pickings do not (onchange logic test) + # with Form(self.batch) as batch_form: + # batch_form.picking_ids.add(self.picking_client_3) + # batch_form.save() + # # individual pickings keep original scheduled dates + self.assertEqual(self.picking_client_1.scheduled_date, picking1_scheduled_date) + self.assertEqual(self.picking_client_2.scheduled_date, picking2_scheduled_date) + # self.assertEqual(self.batch.scheduled_date, self.picking_client_3.scheduled_date) + # self.batch.write({'picking_ids': [(3, self.picking_client_3.id)]}) + + + # remove all pickings and batch scheduled date should default to none + self.batch.write({'picking_ids': [(3, self.picking_client_1.id)]}) + self.batch.write({'picking_ids': [(3, self.picking_client_2.id)]}) + self.assertEqual(self.batch.scheduled_date, False) + + def test_simple_batch_with_manual_qty_done(self): + """ Test a simple batch picking with all quantity for picking available. + The user set all the quantity_done on picking manually and no wizard are used. + """ + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 10.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + self.picking_client_1.move_lines.quantity_done = 10 + self.picking_client_2.move_lines.quantity_done = 10 + self.batch.action_done() + + self.assertEqual(self.picking_client_1.state, 'done', 'Picking 1 should be done') + self.assertEqual(self.picking_client_2.state, 'done', 'Picking 2 should be done') + + quant_A = self.env['stock.quant']._gather(self.productA, self.stock_location) + quant_B = self.env['stock.quant']._gather(self.productB, self.stock_location) + + # ensure that quantity for picking has been moved + self.assertFalse(sum(quant_A.mapped('quantity'))) + self.assertFalse(sum(quant_B.mapped('quantity'))) + + def test_simple_batch_with_wizard(self): + """ Test a simple batch picking with all quantity for picking available. + The user use the wizard in order to complete automatically the quantity_done to + the initial demand (or reserved quantity in this test). + """ + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 10.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + # There should be a wizard asking to process picking without quantity done + immediate_transfer_wizard_dict = self.batch.action_done() + self.assertTrue(immediate_transfer_wizard_dict) + immediate_transfer_wizard = Form(self.env[(immediate_transfer_wizard_dict.get('res_model'))].with_context(immediate_transfer_wizard_dict['context'])).save() + self.assertEqual(len(immediate_transfer_wizard.pick_ids), 2) + immediate_transfer_wizard.process() + + self.assertEqual(self.picking_client_1.state, 'done', 'Picking 1 should be done') + self.assertEqual(self.picking_client_2.state, 'done', 'Picking 2 should be done') + + quant_A = self.env['stock.quant']._gather(self.productA, self.stock_location) + quant_B = self.env['stock.quant']._gather(self.productB, self.stock_location) + + # ensure that quantity for picking has been moved + self.assertFalse(sum(quant_A.mapped('quantity'))) + self.assertFalse(sum(quant_B.mapped('quantity'))) + + def test_batch_with_backorder_wizard(self): + """ Test a simple batch picking with only one quantity fully available. + The user will set by himself the quantity reserved for each picking and + run the picking batch. There should be a wizard asking for a backorder. + """ + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 5.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + self.picking_client_1.move_lines.quantity_done = 5 + self.picking_client_2.move_lines.quantity_done = 10 + + # There should be a wizard asking to process picking without quantity done + back_order_wizard_dict = self.batch.action_done() + self.assertTrue(back_order_wizard_dict) + back_order_wizard = Form(self.env[(back_order_wizard_dict.get('res_model'))].with_context(back_order_wizard_dict['context'])).save() + self.assertEqual(len(back_order_wizard.pick_ids), 1) + back_order_wizard.process() + + self.assertEqual(self.picking_client_2.state, 'done', 'Picking 2 should be done') + self.assertEqual(self.picking_client_1.state, 'done', 'Picking 1 should be done') + self.assertEqual(self.picking_client_1.move_lines.product_uom_qty, 5, 'initial demand should be 5 after picking split') + self.assertTrue(self.env['stock.picking'].search([('backorder_id', '=', self.picking_client_1.id)]), 'no back order created') + + quant_A = self.env['stock.quant']._gather(self.productA, self.stock_location) + quant_B = self.env['stock.quant']._gather(self.productB, self.stock_location) + + # ensure that quantity for picking has been moved + self.assertFalse(sum(quant_A.mapped('quantity'))) + self.assertFalse(sum(quant_B.mapped('quantity'))) + + def test_batch_with_immediate_transfer_and_backorder_wizard(self): + """ Test a simple batch picking with only one product fully available. + Everything should be automatically. First one backorder in order to set quantity_done + to reserved quantity. After a second wizard asking for a backorder for the quantity that + has not been fully transfered. + """ + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 5.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + # There should be a wizard asking to process picking without quantity done + immediate_transfer_wizard_dict = self.batch.action_done() + self.assertTrue(immediate_transfer_wizard_dict) + immediate_transfer_wizard = Form(self.env[(immediate_transfer_wizard_dict.get('res_model'))].with_context(immediate_transfer_wizard_dict['context'])).save() + self.assertEqual(len(immediate_transfer_wizard.pick_ids), 2) + back_order_wizard_dict = immediate_transfer_wizard.process() + self.assertTrue(back_order_wizard_dict) + back_order_wizard = Form(self.env[(back_order_wizard_dict.get('res_model'))].with_context(back_order_wizard_dict['context'])).save() + self.assertEqual(len(back_order_wizard.pick_ids), 1) + back_order_wizard.process() + + self.assertEqual(self.picking_client_1.state, 'done', 'Picking 1 should be done') + self.assertEqual(self.picking_client_1.move_lines.product_uom_qty, 5, 'initial demand should be 5 after picking split') + self.assertTrue(self.env['stock.picking'].search([('backorder_id', '=', self.picking_client_1.id)]), 'no back order created') + + quant_A = self.env['stock.quant']._gather(self.productA, self.stock_location) + quant_B = self.env['stock.quant']._gather(self.productB, self.stock_location) + + # ensure that quantity for picking has been moved + self.assertFalse(sum(quant_A.mapped('quantity'))) + self.assertFalse(sum(quant_B.mapped('quantity'))) + + def test_batch_with_immediate_transfer_and_backorder_wizard_with_manual_operations(self): + """ Test a simple batch picking with only one quantity fully available. + The user set the quantity done only for the partially available picking. + The test should run the immediate transfer for the first picking and then + the backorder wizard for the second picking. + """ + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 5.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + self.picking_client_1.move_lines.quantity_done = 5 + # There should be a wizard asking to process picking without quantity done + immediate_transfer_wizard_dict = self.batch.action_done() + self.assertTrue(immediate_transfer_wizard_dict) + immediate_transfer_wizard = Form(self.env[(immediate_transfer_wizard_dict.get('res_model'))].with_context(immediate_transfer_wizard_dict['context'])).save() + self.assertEqual(len(immediate_transfer_wizard.pick_ids), 1) + back_order_wizard_dict = immediate_transfer_wizard.process() + self.assertTrue(back_order_wizard_dict) + back_order_wizard = Form(self.env[(back_order_wizard_dict.get('res_model'))].with_context(back_order_wizard_dict['context'])).save() + self.assertEqual(len(back_order_wizard.pick_ids), 1) + back_order_wizard.process() + + self.assertEqual(self.picking_client_1.state, 'done', 'Picking 1 should be done') + self.assertEqual(self.picking_client_1.move_lines.product_uom_qty, 5, 'initial demand should be 5 after picking split') + self.assertTrue(self.env['stock.picking'].search([('backorder_id', '=', self.picking_client_1.id)]), 'no back order created') + + quant_A = self.env['stock.quant']._gather(self.productA, self.stock_location) + quant_B = self.env['stock.quant']._gather(self.productB, self.stock_location) + + # ensure that quantity for picking has been moved + self.assertFalse(sum(quant_A.mapped('quantity'))) + self.assertFalse(sum(quant_B.mapped('quantity'))) + + def test_put_in_pack(self): + self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 10.0) + self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 10.0) + + # Confirm batch, pickings should not be automatically assigned. + self.batch.action_confirm() + self.assertEqual(self.picking_client_1.state, 'confirmed', 'Picking 1 should be confirmed') + self.assertEqual(self.picking_client_2.state, 'confirmed', 'Picking 2 should be confirmed') + # Ask to assign, so pickings should be assigned now. + self.batch.action_assign() + self.assertEqual(self.picking_client_1.state, 'assigned', 'Picking 1 should be ready') + self.assertEqual(self.picking_client_2.state, 'assigned', 'Picking 2 should be ready') + + # only do part of pickings + assign different destinations + try to pack (should get wizard to correct destination) + self.batch.move_line_ids.qty_done = 5 + self.batch.move_line_ids[0].location_dest_id = self.stock_location.id + wizard_values = self.batch.action_put_in_pack() + wizard = self.env[(wizard_values.get('res_model'))].browse(wizard_values.get('res_id')) + wizard.location_dest_id = self.customer_location.id + package = wizard.action_done() + + # a new package is made and done quantities should be in same package + self.assertTrue(package) + done_qty_move_lines = self.batch.move_line_ids.filtered(lambda ml: ml.qty_done == 5) + self.assertEqual(done_qty_move_lines[0].result_package_id.id, package.id) + self.assertEqual(done_qty_move_lines[1].result_package_id.id, package.id) + + # not done quantities should be split into separate lines + self.assertEqual(len(self.batch.move_line_ids), 4) + + # confirm w/ backorder + back_order_wizard_dict = self.batch.action_done() + self.assertTrue(back_order_wizard_dict) + back_order_wizard = Form(self.env[(back_order_wizard_dict.get('res_model'))].with_context(back_order_wizard_dict['context'])).save() + self.assertEqual(len(back_order_wizard.pick_ids), 2) + back_order_wizard.process() + + # final package location should be correctly set based on wizard + self.assertEqual(package.location_id.id, self.customer_location.id) |
