summaryrefslogtreecommitdiff
path: root/addons/stock_picking_batch/tests/test_batch_picking.py
diff options
context:
space:
mode:
authorstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
committerstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
commit3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch)
treea44932296ef4a9b71d5f010906253d8c53727726 /addons/stock_picking_batch/tests/test_batch_picking.py
parent0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (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.py367
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)