summaryrefslogtreecommitdiff
path: root/addons/stock/tests/test_move2.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/tests/test_move2.py
parent0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff)
initial commit 2
Diffstat (limited to 'addons/stock/tests/test_move2.py')
-rw-r--r--addons/stock/tests/test_move2.py3146
1 files changed, 3146 insertions, 0 deletions
diff --git a/addons/stock/tests/test_move2.py b/addons/stock/tests/test_move2.py
new file mode 100644
index 00000000..32c8f094
--- /dev/null
+++ b/addons/stock/tests/test_move2.py
@@ -0,0 +1,3146 @@
+# -*- coding: utf-8 -*-
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+
+from datetime import timedelta
+
+from odoo.addons.stock.tests.common import TestStockCommon
+from odoo.exceptions import UserError
+
+from odoo.tests import Form
+from odoo.tools import float_is_zero, float_compare
+
+from odoo.tests.common import Form
+
+class TestPickShip(TestStockCommon):
+ def create_pick_ship(self):
+ picking_client = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ dest = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_client.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'state': 'waiting',
+ 'procure_method': 'make_to_order',
+ })
+
+ picking_pick = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_pick.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'move_dest_ids': [(4, dest.id)],
+ 'state': 'confirmed',
+ })
+ return picking_pick, picking_client
+
+ def create_pick_pack_ship(self):
+ picking_ship = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ ship = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_ship.id,
+ 'location_id': self.output_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ picking_pack = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ pack = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_pack.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.output_location,
+ 'move_dest_ids': [(4, ship.id)],
+ })
+
+ picking_pick = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_pick.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'move_dest_ids': [(4, pack.id)],
+ 'state': 'confirmed',
+ })
+ return picking_pick, picking_pack, picking_ship
+
+ def test_mto_moves(self):
+ """
+ 10 in stock, do pick->ship and check ship is assigned when pick is done, then backorder of ship
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+
+ self.assertEqual(picking_client.state, 'assigned', 'The state of the client should be assigned')
+
+ # Now partially transfer the ship
+ picking_client.move_lines[0].move_line_ids[0].qty_done = 5
+ picking_client._action_done() # no new in order to create backorder
+
+ backorder = self.env['stock.picking'].search([('backorder_id', '=', picking_client.id)])
+ self.assertEqual(backorder.state, 'waiting', 'Backorder should be waiting for reservation')
+
+ def test_mto_moves_transfer(self):
+ """
+ 10 in stock, 5 in pack. Make sure it does not assign the 5 pieces in pack
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 5.0)
+
+ self.assertEqual(len(self.env['stock.quant']._gather(self.productA, stock_location)), 1.0)
+ self.assertEqual(len(self.env['stock.quant']._gather(self.productA, pack_location)), 1.0)
+
+ (picking_pick + picking_client).action_assign()
+
+ move_pick = picking_pick.move_lines
+ move_cust = picking_client.move_lines
+ self.assertEqual(move_pick.state, 'assigned')
+ self.assertEqual(picking_pick.state, 'assigned')
+ self.assertEqual(move_cust.state, 'waiting')
+ self.assertEqual(picking_client.state, 'waiting', 'The picking should not assign what it does not have')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, stock_location), 0.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 5.0)
+
+ move_pick.move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+
+ self.assertEqual(move_pick.state, 'done')
+ self.assertEqual(picking_pick.state, 'done')
+ self.assertEqual(move_cust.state, 'assigned')
+ self.assertEqual(picking_client.state, 'assigned', 'The picking should not assign what it does not have')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, stock_location), 0.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 5.0)
+ self.assertEqual(sum(self.env['stock.quant']._gather(self.productA, stock_location).mapped('quantity')), 0.0)
+ self.assertEqual(len(self.env['stock.quant']._gather(self.productA, pack_location)), 1.0)
+
+ def test_mto_moves_return(self):
+ picking_pick, picking_client = self.create_pick_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+ self.assertEqual(picking_pick.state, 'done')
+ self.assertEqual(picking_client.state, 'assigned')
+
+ # return a part of what we've done
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_pick.ids, active_id=picking_pick.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 2.0 # Return only 2
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_pick = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+ return_pick.move_lines[0].move_line_ids[0].qty_done = 2.0
+ return_pick._action_done()
+ # the client picking should not be assigned anymore, as we returned partially what we took
+ self.assertEqual(picking_client.state, 'confirmed')
+
+ def test_mto_moves_extra_qty(self):
+ """ Ensure that a move in MTO will support an extra quantity. The extra
+ move should be created in MTS and should not be merged in the initial
+ move if it's in MTO. It should also avoid to trigger the rules.
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.productA.write({'route_ids': [(4, self.env.ref('stock.route_warehouse0_mto').id)]})
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 15.0
+ picking_pick._action_done()
+ self.assertEqual(picking_pick.state, 'done')
+ self.assertEqual(picking_client.state, 'assigned')
+
+ picking_client.move_lines[0].move_line_ids[0].qty_done = 15.0
+ picking_client.move_lines._action_done()
+ self.assertEqual(len(picking_client.move_lines), 2)
+ move_lines = picking_client.move_lines.sorted()
+ self.assertEqual(move_lines.mapped('procure_method'), ['make_to_order', 'make_to_stock'])
+ self.assertEqual(move_lines.mapped('product_uom_qty'), [10.0, 5.0])
+
+ def test_mto_moves_return_extra(self):
+ picking_pick, picking_client = self.create_pick_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+ self.assertEqual(picking_pick.state, 'done')
+ self.assertEqual(picking_client.state, 'assigned')
+
+ # return more than we've done
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_pick.ids, active_id=picking_pick.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 12.0 # Return 2 extra
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_pick = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ # Verify the extra move has been merged with the original move
+ self.assertAlmostEqual(return_pick.move_lines.product_uom_qty, 12.0)
+ self.assertAlmostEqual(return_pick.move_lines.quantity_done, 0.0)
+ self.assertAlmostEqual(return_pick.move_lines.reserved_availability, 10.0)
+
+ def test_mto_resupply_cancel_ship(self):
+ """ This test simulates a pick pack ship with a resupply route
+ set. Pick and pack are validated, ship is cancelled. This test
+ ensure that new picking are not created from the cancelled
+ ship after the scheduler task. The supply route is only set in
+ order to make the scheduler run without mistakes (no next
+ activity).
+ """
+ picking_pick, picking_pack, picking_ship = self.create_pick_pack_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ warehouse_1 = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ warehouse_1.write({'delivery_steps': 'pick_pack_ship'})
+ warehouse_2 = self.env['stock.warehouse'].create({
+ 'name': 'Small Warehouse',
+ 'code': 'SWH'
+ })
+ warehouse_1.write({
+ 'resupply_wh_ids': [(6, 0, [warehouse_2.id])]
+ })
+ resupply_route = self.env['stock.location.route'].search([('supplier_wh_id', '=', warehouse_2.id), ('supplied_wh_id', '=', warehouse_1.id)])
+ self.assertTrue(resupply_route)
+ self.productA.write({'route_ids': [(4, resupply_route.id), (4, self.env.ref('stock.route_warehouse0_mto').id)]})
+
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+
+ picking_pack.action_assign()
+ picking_pack.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pack._action_done()
+
+ picking_ship.action_cancel()
+ picking_ship.move_lines.write({'procure_method': 'make_to_order'})
+
+ self.env['procurement.group'].run_scheduler()
+ next_activity = self.env['mail.activity'].search([('res_model', '=', 'product.template'), ('res_id', '=', self.productA.product_tmpl_id.id)])
+ self.assertEqual(picking_ship.state, 'cancel')
+ self.assertFalse(next_activity, 'If a next activity has been created if means that scheduler failed\
+ and the end of this test do not have sense.')
+ self.assertEqual(len(picking_ship.move_lines.mapped('move_orig_ids')), 0,
+ 'Scheduler should not create picking pack and pick since ship has been manually cancelled.')
+
+ def test_no_backorder_1(self):
+ """ Check the behavior of doing less than asked in the picking pick and chosing not to
+ create a backorder. In this behavior, the second picking should obviously only be able to
+ reserve what was brought, but its initial demand should stay the same and the system will
+ ask the user will have to consider again if he wants to create a backorder or not.
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 5.0
+
+ # create a backorder
+ picking_pick._action_done()
+ picking_pick_backorder = self.env['stock.picking'].search([('backorder_id', '=', picking_pick.id)])
+ self.assertEqual(picking_pick_backorder.state, 'confirmed')
+ self.assertEqual(picking_pick_backorder.move_lines.product_qty, 5.0)
+
+ self.assertEqual(picking_client.state, 'assigned')
+
+ # cancel the backorder
+ picking_pick_backorder.action_cancel()
+ self.assertEqual(picking_client.state, 'assigned')
+
+ def test_edit_done_chained_move(self):
+ """ Let’s say two moves are chained: the first is done and the second is assigned.
+ Editing the move line of the first move should impact the reservation of the second one.
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+
+ self.assertEqual(picking_pick.state, 'done', 'The state of the pick should be done')
+ self.assertEqual(picking_client.state, 'assigned', 'The state of the client should be assigned')
+ self.assertEqual(picking_pick.move_lines.quantity_done, 10.0, 'Wrong quantity_done for pick move')
+ self.assertEqual(picking_client.move_lines.product_qty, 10.0, 'Wrong initial demand for client move')
+ self.assertEqual(picking_client.move_lines.reserved_availability, 10.0, 'Wrong quantity already reserved for client move')
+
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 5.0
+ self.assertEqual(picking_pick.state, 'done', 'The state of the pick should be done')
+ self.assertEqual(picking_client.state, 'assigned', 'The state of the client should be partially available')
+ self.assertEqual(picking_pick.move_lines.quantity_done, 5.0, 'Wrong quantity_done for pick move')
+ self.assertEqual(picking_client.move_lines.product_qty, 10.0, 'Wrong initial demand for client move')
+ self.assertEqual(picking_client.move_lines.reserved_availability, 5.0, 'Wrong quantity already reserved for client move')
+
+ # Check if run action_assign does not crash
+ picking_client.action_assign()
+
+ def test_edit_done_chained_move_with_lot(self):
+ """ Let’s say two moves are chained: the first is done and the second is assigned.
+ Editing the lot on the move line of the first move should impact the reservation of the second one.
+ """
+ self.productA.tracking = 'lot'
+ lot1 = self.env['stock.production.lot'].create({
+ 'name': 'lot1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ lot2 = self.env['stock.production.lot'].create({
+ 'name': 'lot2',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ picking_pick, picking_client = self.create_pick_ship()
+ location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].write({
+ 'qty_done': 10.0,
+ 'lot_id': lot1.id,
+ })
+ picking_pick._action_done()
+
+ self.assertEqual(picking_pick.state, 'done', 'The state of the pick should be done')
+ self.assertEqual(picking_client.state, 'assigned', 'The state of the client should be assigned')
+ self.assertEqual(picking_pick.move_lines.quantity_done, 10.0, 'Wrong quantity_done for pick move')
+ self.assertEqual(picking_client.move_lines.product_qty, 10.0, 'Wrong initial demand for client move')
+ self.assertEqual(picking_client.move_lines.move_line_ids.lot_id, lot1, 'Wrong lot for client move line')
+ self.assertEqual(picking_client.move_lines.reserved_availability, 10.0, 'Wrong quantity already reserved for client move')
+
+ picking_pick.move_lines[0].move_line_ids[0].lot_id = lot2.id
+ self.assertEqual(picking_pick.state, 'done', 'The state of the pick should be done')
+ self.assertEqual(picking_client.state, 'assigned', 'The state of the client should be partially available')
+ self.assertEqual(picking_pick.move_lines.quantity_done, 10.0, 'Wrong quantity_done for pick move')
+ self.assertEqual(picking_client.move_lines.product_qty, 10.0, 'Wrong initial demand for client move')
+ self.assertEqual(picking_client.move_lines.move_line_ids.lot_id, lot2, 'Wrong lot for client move line')
+ self.assertEqual(picking_client.move_lines.reserved_availability, 10.0, 'Wrong quantity already reserved for client move')
+
+ # Check if run action_assign does not crash
+ picking_client.action_assign()
+
+ def test_chained_move_with_uom(self):
+ """ Create pick ship with a different uom than the once used for quant.
+ Check that reserved quantity and flow work correctly.
+ """
+ picking_client = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ dest = self.MoveObj.create({
+ 'name': self.gB.name,
+ 'product_id': self.gB.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.uom_kg.id,
+ 'picking_id': picking_client.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'state': 'waiting',
+ })
+
+ picking_pick = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ self.MoveObj.create({
+ 'name': self.gB.name,
+ 'product_id': self.gB.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.uom_kg.id,
+ 'picking_id': picking_pick.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'move_dest_ids': [(4, dest.id)],
+ 'state': 'confirmed',
+ })
+ location = self.env['stock.location'].browse(self.stock_location)
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.gB, location, 10000.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.gB, pack_location), 0.0)
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 5.0
+ picking_pick._action_done()
+
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.gB, location), 5000.0)
+ self.assertEqual(self.env['stock.quant']._gather(self.gB, pack_location).quantity, 5000.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.gB, pack_location), 0.0)
+ self.assertEqual(picking_client.state, 'assigned')
+ self.assertEqual(picking_client.move_lines.reserved_availability, 5.0)
+
+ def test_pick_ship_return(self):
+ """ Create pick and ship. Bring it ot the customer and then return
+ it to stock. This test check the state and the quantity after each move in
+ order to ensure that it is correct.
+ """
+ picking_pick, picking_ship = self.create_pick_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ customer_location = self.env['stock.location'].browse(self.customer_location)
+ self.productA.tracking = 'lot'
+ lot = self.env['stock.production.lot'].create({
+ 'product_id': self.productA.id,
+ 'name': '123456789',
+ 'company_id': self.env.company.id,
+ })
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0, lot_id=lot)
+
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+ self.assertEqual(picking_pick.state, 'done')
+ self.assertEqual(picking_ship.state, 'assigned')
+
+ picking_ship.action_assign()
+ picking_ship.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_ship._action_done()
+
+ customer_quantity = self.env['stock.quant']._get_available_quantity(self.productA, customer_location, lot_id=lot)
+ self.assertEqual(customer_quantity, 10, 'It should be one product in customer')
+
+ """ First we create the return picking for pick pinking.
+ Since we do not have created the return between customer and
+ output. This return should not be available and should only have
+ picking pick as origin move.
+ """
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_pick.ids, active_id=picking_pick.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 10.0
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_pick_picking = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ self.assertEqual(return_pick_picking.state, 'waiting')
+
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_ship.ids, active_id=picking_ship.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 10.0
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_ship_picking = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ self.assertEqual(return_ship_picking.state, 'assigned', 'Return ship picking should automatically be assigned')
+ """ We created the return for ship picking. The origin/destination
+ link between return moves should have been created during return creation.
+ """
+ self.assertTrue(return_ship_picking.move_lines in return_pick_picking.move_lines.mapped('move_orig_ids'),
+ 'The pick return picking\'s moves should have the ship return picking\'s moves as origin')
+
+ self.assertTrue(return_pick_picking.move_lines in return_ship_picking.move_lines.mapped('move_dest_ids'),
+ 'The ship return picking\'s moves should have the pick return picking\'s moves as destination')
+
+ return_ship_picking.move_lines[0].move_line_ids[0].write({
+ 'qty_done': 10.0,
+ 'lot_id': lot.id,
+ })
+ return_ship_picking._action_done()
+ self.assertEqual(return_ship_picking.state, 'done')
+ self.assertEqual(return_pick_picking.state, 'assigned')
+
+ customer_quantity = self.env['stock.quant']._get_available_quantity(self.productA, customer_location, lot_id=lot)
+ self.assertEqual(customer_quantity, 0, 'It should be one product in customer')
+
+ pack_quantity = self.env['stock.quant']._get_available_quantity(self.productA, pack_location, lot_id=lot)
+ self.assertEqual(pack_quantity, 0, 'It should be one product in pack location but is reserved')
+
+ # Should use previous move lot.
+ return_pick_picking.move_lines[0].move_line_ids[0].qty_done = 10.0
+ return_pick_picking._action_done()
+ self.assertEqual(return_pick_picking.state, 'done')
+
+ stock_quantity = self.env['stock.quant']._get_available_quantity(self.productA, stock_location, lot_id=lot)
+ self.assertEqual(stock_quantity, 10, 'The product is not back in stock')
+
+ def test_pick_pack_ship_return(self):
+ """ This test do a pick pack ship delivery to customer and then
+ return it to stock. Once everything is done, this test will check
+ if all the link orgini/destination between moves are correct.
+ """
+ picking_pick, picking_pack, picking_ship = self.create_pick_pack_ship()
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.productA.tracking = 'serial'
+ lot = self.env['stock.production.lot'].create({
+ 'product_id': self.productA.id,
+ 'name': '123456789',
+ 'company_id': self.env.company.id,
+ })
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 1.0, lot_id=lot)
+
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 1.0
+ picking_pick._action_done()
+
+ picking_pack.action_assign()
+ picking_pack.move_lines[0].move_line_ids[0].qty_done = 1.0
+ picking_pack._action_done()
+
+ picking_ship.action_assign()
+ picking_ship.move_lines[0].move_line_ids[0].qty_done = 1.0
+ picking_ship._action_done()
+
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_ship.ids, active_id=picking_ship.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 1.0
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_ship_picking = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ return_ship_picking.move_lines[0].move_line_ids[0].write({
+ 'qty_done': 1.0,
+ 'lot_id': lot.id,
+ })
+ return_ship_picking._action_done()
+
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_pack.ids, active_id=picking_pack.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 1.0
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_pack_picking = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ return_pack_picking.move_lines[0].move_line_ids[0].qty_done = 1.0
+ return_pack_picking._action_done()
+
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_pick.ids, active_id=picking_pick.ids[0],
+ active_model='stock.picking'))
+ stock_return_picking = stock_return_picking_form.save()
+ stock_return_picking.product_return_moves.quantity = 1.0
+ stock_return_picking_action = stock_return_picking.create_returns()
+ return_pick_picking = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
+
+ return_pick_picking.move_lines[0].move_line_ids[0].qty_done = 1.0
+ return_pick_picking._action_done()
+
+ # Now that everything is returned we will check if the return moves are correctly linked between them.
+ # +--------------------------------------------------------------------------------------------------------+
+ # | -- picking_pick(1) --> -- picking_pack(2) --> -- picking_ship(3) -->
+ # | Stock Pack Output Customer
+ # | <--- return pick(6) -- <--- return pack(5) -- <--- return ship(4) --
+ # +--------------------------------------------------------------------------------------------------------+
+ # Recaps of final link (MO = move_orig_ids, MD = move_dest_ids)
+ # picking_pick(1) : MO = (), MD = (2,6)
+ # picking_pack(2) : MO = (1), MD = (3,5)
+ # picking ship(3) : MO = (2), MD = (4)
+ # return ship(4) : MO = (3), MD = (5)
+ # return pack(5) : MO = (2, 4), MD = (6)
+ # return pick(6) : MO = (1, 5), MD = ()
+
+ self.assertEqual(len(picking_pick.move_lines.move_orig_ids), 0, 'Picking pick should not have origin moves')
+ self.assertEqual(set(picking_pick.move_lines.move_dest_ids.ids), set((picking_pack.move_lines | return_pick_picking.move_lines).ids))
+
+ self.assertEqual(set(picking_pack.move_lines.move_orig_ids.ids), set(picking_pick.move_lines.ids))
+ self.assertEqual(set(picking_pack.move_lines.move_dest_ids.ids), set((picking_ship.move_lines | return_pack_picking.move_lines).ids))
+
+ self.assertEqual(set(picking_ship.move_lines.move_orig_ids.ids), set(picking_pack.move_lines.ids))
+ self.assertEqual(set(picking_ship.move_lines.move_dest_ids.ids), set(return_ship_picking.move_lines.ids))
+
+ self.assertEqual(set(return_ship_picking.move_lines.move_orig_ids.ids), set(picking_ship.move_lines.ids))
+ self.assertEqual(set(return_ship_picking.move_lines.move_dest_ids.ids), set(return_pack_picking.move_lines.ids))
+
+ self.assertEqual(set(return_pack_picking.move_lines.move_orig_ids.ids), set((picking_pack.move_lines | return_ship_picking.move_lines).ids))
+ self.assertEqual(set(return_pack_picking.move_lines.move_dest_ids.ids), set(return_pick_picking.move_lines.ids))
+
+ self.assertEqual(set(return_pick_picking.move_lines.move_orig_ids.ids), set((picking_pick.move_lines | return_pack_picking.move_lines).ids))
+ self.assertEqual(len(return_pick_picking.move_lines.move_dest_ids), 0)
+
+ def test_merge_move_mto_mts(self):
+ """ Create 2 moves of the same product in the same picking with
+ one in 'MTO' and the other one in 'MTS'. The moves shouldn't be merged
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 3,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_client.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'origin': 'MPS',
+ 'procure_method': 'make_to_stock',
+ })
+ picking_client.action_confirm()
+ self.assertEqual(len(picking_client.move_lines), 2, 'Moves should not be merged')
+
+ def test_mto_cancel_move_line(self):
+ """ Create a pick ship situation. Then process the pick picking
+ with a backorder. Then try to unlink the move line created on
+ the ship and check if the picking and move state are updated.
+ Then validate the backorder and unlink the ship move lines in
+ order to check again if the picking and state are updated.
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.move_lines.quantity_done = 5.0
+ backorder_wizard_values = picking_pick.button_validate()
+ backorder_wizard = self.env[(backorder_wizard_values.get('res_model'))].browse(backorder_wizard_values.get('res_id')).with_context(backorder_wizard_values['context'])
+ backorder_wizard.process()
+
+ self.assertTrue(picking_client.move_line_ids, 'A move line should be created.')
+ self.assertEqual(picking_client.move_line_ids.product_uom_qty, 5, 'The move line should have 5 unit reserved.')
+
+ # Directly delete the move lines on the picking. (Use show detail operation on picking type)
+ # Should do the same behavior than unreserve
+ picking_client.move_line_ids.unlink()
+
+ self.assertEqual(picking_client.move_lines.state, 'waiting', 'The move state should be waiting since nothing is reserved and another origin move still in progess.')
+ self.assertEqual(picking_client.state, 'waiting', 'The picking state should not be ready anymore.')
+
+ picking_client.action_assign()
+
+ back_order = self.env['stock.picking'].search([('backorder_id', '=', picking_pick.id)])
+ back_order.move_lines.quantity_done = 5
+ back_order.button_validate()
+
+ self.assertEqual(picking_client.move_lines.reserved_availability, 10, 'The total quantity should be reserved since everything is available.')
+ picking_client.move_line_ids.unlink()
+
+ self.assertEqual(picking_client.move_lines.state, 'confirmed', 'The move should be confirmed since all the origin moves are processed.')
+ self.assertEqual(picking_client.state, 'confirmed', 'The picking should be confirmed since all the moves are confirmed.')
+
+ def test_unreserve(self):
+ picking_pick, picking_client = self.create_pick_ship()
+
+ self.assertEqual(picking_pick.state, 'confirmed')
+ picking_pick.do_unreserve()
+ self.assertEqual(picking_pick.state, 'confirmed')
+ location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, location, 10.0)
+ picking_pick.action_assign()
+ self.assertEqual(picking_pick.state, 'assigned')
+ picking_pick.do_unreserve()
+ self.assertEqual(picking_pick.state, 'confirmed')
+
+ self.assertEqual(picking_client.state, 'waiting')
+ picking_client.do_unreserve()
+ self.assertEqual(picking_client.state, 'waiting')
+
+ def test_return_location(self):
+ """ In a pick ship scenario, send two items to the customer, then return one in the ship
+ location and one in a return location that is located in another warehouse.
+ """
+ pick_location = self.env['stock.location'].browse(self.stock_location)
+ pick_location.return_location = True
+
+ return_warehouse = self.env['stock.warehouse'].create({'name': 'return warehouse', 'code': 'rw'})
+ return_location = self.env['stock.location'].create({
+ 'name': 'return internal',
+ 'usage': 'internal',
+ 'location_id': return_warehouse.view_location_id.id
+ })
+
+ self.env['stock.quant']._update_available_quantity(self.productA, pick_location, 10.0)
+ picking_pick, picking_client = self.create_pick_ship()
+
+ # send the items to the customer
+ picking_pick.action_assign()
+ picking_pick.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick._action_done()
+ picking_client.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_client._action_done()
+
+ # return half in the pick location
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_client.ids, active_id=picking_client.ids[0],
+ active_model='stock.picking'))
+ return1 = stock_return_picking_form.save()
+ return1.product_return_moves.quantity = 5.0
+ return1.location_id = pick_location.id
+ return_to_pick_picking_action = return1.create_returns()
+
+ return_to_pick_picking = self.env['stock.picking'].browse(return_to_pick_picking_action['res_id'])
+ return_to_pick_picking.move_lines[0].move_line_ids[0].qty_done = 5.0
+ return_to_pick_picking._action_done()
+
+ # return the remainig products in the return warehouse
+ stock_return_picking_form = Form(self.env['stock.return.picking']
+ .with_context(active_ids=picking_client.ids, active_id=picking_client.ids[0],
+ active_model='stock.picking'))
+ return2 = stock_return_picking_form.save()
+ return2.product_return_moves.quantity = 5.0
+ return2.location_id = return_location.id
+ return_to_return_picking_action = return2.create_returns()
+
+ return_to_return_picking = self.env['stock.picking'].browse(return_to_return_picking_action['res_id'])
+ return_to_return_picking.move_lines[0].move_line_ids[0].qty_done = 5.0
+ return_to_return_picking._action_done()
+
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pick_location), 5.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, return_location), 5.0)
+ self.assertEqual(len(self.env['stock.quant'].search([('product_id', '=', self.productA.id), ('quantity', '!=', 0)])), 2)
+
+
+class TestSinglePicking(TestStockCommon):
+ def test_backorder_1(self):
+ """ Check the good behavior of creating a backorder for an available stock move.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # make some stock
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 2)
+
+ # assign
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ # valid with backorder creation
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 1
+ delivery_order._action_done()
+ self.assertNotEqual(delivery_order.date_done, False)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 1.0)
+
+ backorder = self.env['stock.picking'].search([('backorder_id', '=', delivery_order.id)])
+ self.assertEqual(backorder.state, 'confirmed')
+ backorder.action_assign()
+ self.assertEqual(backorder.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ def test_backorder_2(self):
+ """ Check the good behavior of creating a backorder for a partially available stock move.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # make some stock
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 1)
+
+ # assign to partially available
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ # valid with backorder creation
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 1
+ delivery_order._action_done()
+ self.assertNotEqual(delivery_order.date_done, False)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ backorder = self.env['stock.picking'].search([('backorder_id', '=', delivery_order.id)])
+ self.assertEqual(backorder.state, 'confirmed')
+
+ def test_backorder_3(self):
+ """ Check the good behavior of creating a backorder for an available move on a picking with
+ two available moves.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # make some stock
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 2)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 2)
+
+ # assign to partially available
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 2
+ delivery_order._action_done()
+
+ backorder = self.env['stock.picking'].search([('backorder_id', '=', delivery_order.id)])
+ self.assertEqual(backorder.state, 'confirmed')
+
+ def test_backorder_4(self):
+ """ Check the good behavior if no backorder are created
+ for a picking with a missing product.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # Update available quantities for each products
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 2)
+ self.env['stock.quant']._update_available_quantity(self.productB, pack_location, 2)
+
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+
+ # Process only one product without creating a backorder
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 2
+ res_dict = delivery_order.button_validate()
+ backorder_wizard = Form(self.env['stock.backorder.confirmation'].with_context(res_dict['context'])).save()
+ backorder_wizard.process_cancel_backorder()
+
+ # No backorder should be created and the move corresponding to the missing product should be cancelled
+ backorder = self.env['stock.picking'].search([('backorder_id', '=', delivery_order.id)])
+ self.assertFalse(backorder)
+ self.assertEqual(delivery_order.state, 'done')
+ self.assertEqual(delivery_order.move_lines[1].state, 'cancel')
+
+ def test_extra_move_1(self):
+ """ Check the good behavior of creating an extra move in a delivery order. This usecase
+ simulates the delivery of 2 item while the initial stock move had to move 1 and there's
+ only 1 in stock.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # make some stock
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 1)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 1.0)
+
+ # assign to available
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ # valid with backorder creation
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 2
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+ delivery_order._action_done()
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location, allow_negative=True), -1.0)
+
+ self.assertEqual(move1.product_qty, 2.0)
+ self.assertEqual(move1.quantity_done, 2.0)
+ self.assertEqual(move1.reserved_availability, 0.0)
+ self.assertEqual(move1.move_line_ids.product_qty, 0.0) # change reservation to 0 for done move
+ self.assertEqual(sum(move1.move_line_ids.mapped('qty_done')), 2.0)
+ self.assertEqual(move1.state, 'done')
+
+ def test_extra_move_2(self):
+ """ Check the good behavior of creating an extra move in a delivery order. This usecase
+ simulates the delivery of 3 item while the initial stock move had to move 1 and there's
+ only 1 in stock.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ # make some stock
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, pack_location, 1)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 1.0)
+
+ # assign to available
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+
+ # valid with backorder creation
+ delivery_order.move_lines[0].move_line_ids[0].qty_done = 3
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+ delivery_order._action_done()
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 0.0)
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location, allow_negative=True), -2.0)
+
+ self.assertEqual(move1.product_qty, 3.0)
+ self.assertEqual(move1.quantity_done, 3.0)
+ self.assertEqual(move1.reserved_availability, 0.0)
+ self.assertEqual(move1.move_line_ids.product_qty, 0.0) # change reservation to 0 for done move
+ self.assertEqual(sum(move1.move_line_ids.mapped('qty_done')), 3.0)
+ self.assertEqual(move1.state, 'done')
+
+ def test_extra_move_3(self):
+ """ Check the good behavior of creating an extra move in a receipt. This usecase simulates
+ the receipt of 2 item while the initial stock move had to move 1.
+ """
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+
+ # assign to available
+ receipt.action_confirm()
+ receipt.action_assign()
+ self.assertEqual(receipt.state, 'assigned')
+
+ # valid with backorder creation
+ receipt.move_lines[0].move_line_ids[0].qty_done = 2
+ receipt._action_done()
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, stock_location), 2.0)
+
+ self.assertEqual(move1.product_qty, 2.0)
+ self.assertEqual(move1.quantity_done, 2.0)
+ self.assertEqual(move1.reserved_availability, 0.0)
+ self.assertEqual(move1.move_line_ids.product_qty, 0.0) # change reservation to 0 for done move
+ self.assertEqual(sum(move1.move_line_ids.mapped('qty_done')), 2.0)
+ self.assertEqual(move1.state, 'done')
+
+ def test_extra_move_4(self):
+ """ Create a picking with similar moves (created after
+ confirmation). Action done should propagate all the extra
+ quantity and only merge extra moves in their original moves.
+ """
+ delivery = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 5,
+ 'quantity_done': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 5)
+ delivery.action_confirm()
+ delivery.action_assign()
+
+ delivery.write({
+ 'move_lines': [(0, 0, {
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 0,
+ 'quantity_done': 10,
+ 'state': 'assigned',
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })]
+ })
+ delivery._action_done()
+ self.assertEqual(len(delivery.move_lines), 2, 'Move should not be merged together')
+ for move in delivery.move_lines:
+ self.assertEqual(move.quantity_done, move.product_uom_qty, 'Initial demand should be equals to quantity done')
+
+ def test_extra_move_5(self):
+ """ Create a picking a move that is problematic with
+ rounding (5.95 - 5.5 = 0.4500000000000002). Ensure that
+ initial demand is corrct afer action_done and backoder
+ are not created.
+ """
+ delivery = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ product = self.kgB
+ self.MoveObj.create({
+ 'name': product.name,
+ 'product_id': product.id,
+ 'product_uom_qty': 5.5,
+ 'quantity_done': 5.95,
+ 'product_uom': product.uom_id.id,
+ 'picking_id': delivery.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(product, stock_location, 5.5)
+ delivery.action_confirm()
+ delivery.action_assign()
+ delivery._action_done()
+ self.assertEqual(delivery.move_lines.product_uom_qty, 5.95, 'Move initial demand should be 5.95')
+
+ back_order = self.env['stock.picking'].search([('backorder_id', '=', delivery.id)])
+ self.assertFalse(back_order, 'There should be no back order')
+
+ def test_recheck_availability_1(self):
+ """ Check the good behavior of check availability. I create a DO for 2 unit with
+ only one in stock. After the first check availability, I should have 1 reserved
+ product with one move line. After adding a second unit in stock and recheck availability.
+ The DO should have 2 reserved unit, be in available state and have only one move line.
+ """
+ self.env['stock.quant']._update_available_quantity(self.productA, self.env['stock.location'].browse(self.stock_location), 1.0)
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ # Check State
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'partially_available')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 1.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.product_qty, 1)
+
+ inventory = self.env['stock.inventory'].create({
+ 'name': 'remove product1',
+ 'location_ids': [(4, self.stock_location)],
+ 'product_ids': [(4, self.productA.id)],
+ })
+ inventory.action_start()
+ inventory.line_ids.product_qty = 2
+ inventory.action_validate()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'assigned')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 2.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.product_qty, 2)
+
+ def test_recheck_availability_2(self):
+ """ Same check than test_recheck_availability_1 but with lot this time.
+ If the new product has the same lot that already reserved one, the move lines
+ reserved quantity should be updated.
+ Otherwise a new move lines with the new lot should be added.
+ """
+ self.productA.tracking = 'lot'
+ lot1 = self.env['stock.production.lot'].create({
+ 'name': 'lot1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 1.0, lot_id=lot1)
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ # Check State
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'partially_available')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 1.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.product_qty, 1)
+
+ inventory = self.env['stock.inventory'].create({
+ 'name': 'remove product1',
+ 'location_ids': [(4, self.stock_location)],
+ 'product_ids': [(4, self.productA.id)],
+ })
+ inventory.action_start()
+ inventory.line_ids.prod_lot_id = lot1
+ inventory.line_ids.product_qty = 2
+ inventory.action_validate()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'assigned')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 2.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.lot_id.id, lot1.id)
+ self.assertEqual(move1.move_line_ids.product_qty, 2)
+
+ def test_recheck_availability_3(self):
+ """ Same check than test_recheck_availability_2 but with different lots.
+ """
+ self.productA.tracking = 'lot'
+ lot1 = self.env['stock.production.lot'].create({
+ 'name': 'lot1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ lot2 = self.env['stock.production.lot'].create({
+ 'name': 'lot2',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 1.0, lot_id=lot1)
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ # Check State
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'partially_available')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 1.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.product_qty, 1)
+
+ inventory = self.env['stock.inventory'].create({
+ 'name': 'remove product1',
+ 'location_ids': [(4, self.stock_location)],
+ 'product_ids': [(4, self.productA.id)],
+ })
+ inventory.action_start()
+ self.env['stock.inventory.line'].create({
+ 'inventory_id': inventory.id,
+ 'location_id': inventory.location_ids[0].id,
+ 'prod_lot_id': lot2.id,
+ 'product_id': self.productA.id,
+ 'product_qty': 1,
+ })
+ inventory.action_validate()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'assigned')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 2.0)
+ self.assertEqual(len(move1.move_line_ids), 2)
+ move_lines = move1.move_line_ids.sorted()
+ self.assertEqual(move_lines[0].lot_id.id, lot1.id)
+ self.assertEqual(move_lines[1].lot_id.id, lot2.id)
+
+ def test_recheck_availability_4(self):
+ """ Same check than test_recheck_availability_2 but with serial number this time.
+ Serial number reservation should always create a new move line.
+ """
+ self.productA.tracking = 'serial'
+ serial1 = self.env['stock.production.lot'].create({
+ 'name': 'serial1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ serial2 = self.env['stock.production.lot'].create({
+ 'name': 'serial2',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 1.0, lot_id=serial1)
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ # Check State
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'partially_available')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 1.0)
+ self.assertEqual(len(move1.move_line_ids), 1)
+ self.assertEqual(move1.move_line_ids.product_qty, 1)
+
+ inventory = self.env['stock.inventory'].create({
+ 'name': 'remove product1',
+ 'location_ids': [(4, self.stock_location)],
+ 'product_ids': [(4, self.productA.id)],
+ })
+ inventory.action_start()
+ self.env['stock.inventory.line'].create({
+ 'inventory_id': inventory.id,
+ 'location_id': inventory.location_ids[0].id,
+ 'prod_lot_id': serial2.id,
+ 'product_id': self.productA.id,
+ 'product_qty': 1,
+ })
+ inventory.action_validate()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(move1.state, 'assigned')
+
+ # Check reserved quantity
+ self.assertEqual(move1.reserved_availability, 2.0)
+ self.assertEqual(len(move1.move_line_ids), 2)
+ move_lines = move1.move_line_ids.sorted()
+ self.assertEqual(move_lines[0].lot_id.id, serial1.id)
+ self.assertEqual(move_lines[1].lot_id.id, serial2.id)
+
+ def test_use_create_lot_use_existing_lot_1(self):
+ """ Check the behavior of a picking when `use_create_lot` and `use_existing_lot` are
+ set to False and there's a move for a tracked product.
+ """
+ self.env['stock.picking.type']\
+ .browse(self.picking_type_out)\
+ .write({
+ 'use_create_lots': False,
+ 'use_existing_lots': False,
+ })
+ self.productA.tracking = 'lot'
+
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'picking_type_id': self.picking_type_out,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ delivery_order.action_confirm()
+ delivery_order.move_lines.quantity_done = 2
+ # do not set a lot_id or lot_name, it should work
+ delivery_order._action_done()
+
+ def test_use_create_lot_use_existing_lot_2(self):
+ """ Check the behavior of a picking when `use_create_lot` and `use_existing_lot` are
+ set to True and there's a move for a tracked product.
+ """
+ self.env['stock.picking.type']\
+ .browse(self.picking_type_out)\
+ .write({
+ 'use_create_lots': True,
+ 'use_existing_lots': True,
+ })
+ self.productA.tracking = 'lot'
+
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'picking_type_id': self.picking_type_out,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ delivery_order.action_confirm()
+ delivery_order.move_lines.quantity_done = 2
+ move_line = delivery_order.move_lines.move_line_ids
+
+ # not lot_name set, should raise
+ with self.assertRaises(UserError):
+ delivery_order._action_done()
+
+ # enter a new lot name, should work
+ move_line.lot_name = 'newlot'
+ delivery_order._action_done()
+
+ def test_use_create_lot_use_existing_lot_3(self):
+ """ Check the behavior of a picking when `use_create_lot` is set to True and
+ `use_existing_lot` is set to False and there's a move for a tracked product.
+ """
+ self.env['stock.picking.type']\
+ .browse(self.picking_type_out)\
+ .write({
+ 'use_create_lots': True,
+ 'use_existing_lots': False,
+ })
+ self.productA.tracking = 'lot'
+
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'picking_type_id': self.picking_type_out,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ delivery_order.action_confirm()
+ delivery_order.move_lines.quantity_done = 2
+ move_line = delivery_order.move_lines.move_line_ids
+
+ # not lot_name set, should raise
+ with self.assertRaises(UserError):
+ delivery_order._action_done()
+
+ # enter a new lot name, should work
+ move_line.lot_name = 'newlot'
+ delivery_order._action_done()
+
+ def test_use_create_lot_use_existing_lot_4(self):
+ """ Check the behavior of a picking when `use_create_lot` is set to False and
+ `use_existing_lot` is set to True and there's a move for a tracked product.
+ """
+ self.env['stock.picking.type']\
+ .browse(self.picking_type_out)\
+ .write({
+ 'use_create_lots': False,
+ 'use_existing_lots': True,
+ })
+ self.productA.tracking = 'lot'
+
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 2,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'picking_type_id': self.picking_type_out,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })
+
+ delivery_order.action_confirm()
+ delivery_order.move_lines.quantity_done = 2
+ move_line = delivery_order.move_lines.move_line_ids
+
+ # not lot_name set, should raise
+ with self.assertRaises(UserError):
+ delivery_order._action_done()
+
+ # creating a lot from the view should raise
+ with self.assertRaises(UserError):
+ self.env['stock.production.lot']\
+ .with_context(active_picking_id=delivery_order.id)\
+ .create({
+ 'name': 'lot1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+
+ # enter an existing lot_id, should work
+ lot1 = self.env['stock.production.lot'].create({
+ 'name': 'lot1',
+ 'product_id': self.productA.id,
+ 'company_id': self.env.company.id,
+ })
+ move_line.lot_id = lot1
+ delivery_order._action_done()
+
+ def test_merge_moves_1(self):
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 3,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productB.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ receipt.action_confirm()
+ self.assertEqual(len(receipt.move_lines), 2, 'Moves were not merged')
+ self.assertEqual(receipt.move_lines.filtered(lambda m: m.product_id == self.productA).product_uom_qty, 9, 'Merged quantity is not correct')
+ self.assertEqual(receipt.move_lines.filtered(lambda m: m.product_id == self.productB).product_uom_qty, 5, 'Merge should not impact product B reserved quantity')
+
+ def test_merge_moves_2(self):
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 3,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'origin': 'MPS'
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'origin': 'PO0001'
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 3,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'origin': 'MPS'
+ })
+ receipt.action_confirm()
+ self.assertEqual(len(receipt.move_lines), 1, 'Moves were not merged')
+ self.assertEqual(receipt.move_lines.origin.count('MPS'), 1, 'Origin not merged together or duplicated')
+ self.assertEqual(receipt.move_lines.origin.count('PO0001'), 1, 'Origin not merged together or duplicated')
+
+ def test_merge_moves_3(self):
+ """ Create 2 moves without initial_demand and already a
+ quantity done. Check that we still have only 2 moves after
+ validation.
+ """
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ move_1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'origin': 'MPS'
+ })
+ move_2 = self.MoveObj.create({
+ 'name': self.productB.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'origin': 'PO0001'
+ })
+ move_1.quantity_done = 5
+ move_2.quantity_done = 5
+ receipt.button_validate()
+ self.assertEqual(len(receipt.move_lines), 2, 'Moves were not merged')
+
+ def test_merge_chained_moves(self):
+ """ Imagine multiple step delivery. Two different receipt picking for the same product should only generate
+ 1 picking from input to QC and another from QC to stock. The link at the end should follow this scheme.
+ Move receipt 1 \
+ Move Input-> QC - Move QC -> Stock
+ Move receipt 2 /
+ """
+ warehouse = self.env['stock.warehouse'].create({
+ 'name': 'TEST WAREHOUSE',
+ 'code': 'TEST1',
+ 'reception_steps': 'three_steps',
+ })
+ receipt1 = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': warehouse.wh_input_stock_loc_id.id,
+ 'picking_type_id': warehouse.in_type_id.id,
+ })
+ move_receipt_1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 5,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt1.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': warehouse.wh_input_stock_loc_id.id,
+ })
+ receipt2 = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': warehouse.wh_input_stock_loc_id.id,
+ 'picking_type_id': warehouse.in_type_id.id,
+ })
+ move_receipt_2 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 3,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt2.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': warehouse.wh_input_stock_loc_id.id,
+ })
+ receipt1.action_confirm()
+ receipt2.action_confirm()
+
+ # Check following move has been created and grouped in one picking.
+ self.assertTrue(move_receipt_1.move_dest_ids, 'No move created from push rules')
+ self.assertTrue(move_receipt_2.move_dest_ids, 'No move created from push rules')
+ self.assertEqual(move_receipt_1.move_dest_ids.picking_id, move_receipt_2.move_dest_ids.picking_id, 'Destination moves should be in the same picking')
+
+ # Check link for input move are correct.
+ input_move = move_receipt_2.move_dest_ids
+ self.assertEqual(len(input_move.move_dest_ids), 1)
+ self.assertEqual(set(input_move.move_orig_ids.ids), set((move_receipt_2 | move_receipt_1).ids),
+ 'Move from input to QC should be merged and have the two receipt moves as origin.')
+ self.assertEqual(move_receipt_1.move_dest_ids, input_move)
+ self.assertEqual(move_receipt_2.move_dest_ids, input_move)
+
+ # Check link for quality check move are also correct.
+ qc_move = input_move.move_dest_ids
+ self.assertEqual(len(qc_move), 1)
+ self.assertTrue(qc_move.move_orig_ids == input_move, 'Move between QC and stock should only have the input move as origin')
+
+ def test_empty_moves_validation_1(self):
+ """ Use button validate on a picking that contains only moves
+ without initial demand and without quantity done should be
+ impossible and raise a usererror.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ self.MoveObj.create({
+ 'name': self.productB.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ with self.assertRaises(UserError):
+ delivery_order.button_validate()
+
+ def test_empty_moves_validation_2(self):
+ """ Use button validate on a picking that contains only moves
+ without initial demand but at least one with a quantity done
+ should process the move with quantity done and cancel the
+ other.
+ """
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ move_a = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ move_b = self.MoveObj.create({
+ 'name': self.productB.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 0,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ move_a.quantity_done = 1
+ delivery_order.button_validate()
+
+ self.assertEqual(move_a.state, 'done')
+ self.assertEqual(move_b.state, 'cancel')
+ back_order = self.env['stock.picking'].search([('backorder_id', '=', delivery_order.id)])
+ self.assertFalse(back_order, 'There should be no back order')
+
+ def test_unlink_move_1(self):
+ picking = Form(self.env['stock.picking'])
+ ptout = self.env['stock.picking.type'].browse(self.picking_type_out)
+ picking.picking_type_id = ptout
+ with picking.move_ids_without_package.new() as move:
+ move.product_id = self.productA
+ move.product_uom_qty = 10
+ picking = picking.save()
+ self.assertEqual(picking.immediate_transfer, False)
+ self.assertEqual(picking.state, 'draft')
+
+ picking = Form(picking)
+ picking.move_ids_without_package.remove(0)
+ picking = picking.save()
+ self.assertEqual(len(picking.move_ids_without_package), 0)
+
+ def test_additional_move_1(self):
+ """ On a planned trasfer, add a stock move when the picking is already ready. Check that
+ the check availability button appears and work.
+ """
+ # Make some stock for productA and productB.
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ move_1 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ move_2 = self.MoveObj.create({
+ 'name': self.productB.name,
+ 'product_id': self.productB.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productB.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ receipt.action_confirm()
+ move_1.quantity_done = 10
+ move_2.quantity_done = 10
+ receipt.button_validate()
+ self.assertEqual(self.productA.qty_available, 10)
+ self.assertEqual(self.productB.qty_available, 10)
+
+ # Create a delivery for 1 productA, reserve, check the picking is ready
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ 'move_type': 'one',
+ })
+ move_3 = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': delivery_order.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ })
+ delivery_order.action_confirm()
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+
+ # Add a unit of productB, the check_availability button should appear.
+ delivery_order = Form(delivery_order)
+ with delivery_order.move_ids_without_package.new() as move:
+ move.product_id = self.productB
+ move.product_uom_qty = 10
+ delivery_order = delivery_order.save()
+
+ # The autocoform ran, the picking shoud be confirmed and reservable.
+ self.assertEqual(delivery_order.state, 'confirmed')
+ self.assertEqual(delivery_order.show_mark_as_todo, False)
+ self.assertEqual(delivery_order.show_check_availability, True)
+
+ delivery_order.action_assign()
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(delivery_order.show_check_availability, False)
+ self.assertEqual(delivery_order.show_mark_as_todo, False)
+
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.assertEqual(self.env['stock.quant']._gather(self.productA, stock_location).reserved_quantity, 10.0)
+ self.assertEqual(self.env['stock.quant']._gather(self.productB, stock_location).reserved_quantity, 10.0)
+
+ def test_additional_move_2(self):
+ """ On an immediate trasfer, add a stock move when the picking is already ready. Check that
+ the check availability button doest not appear.
+ """
+ # Create a delivery for 1 productA, check the picking is ready
+ delivery_order = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ 'immediate_transfer': True,
+ 'move_ids_without_package': [(0, 0, {
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom': self.productA.uom_id.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'quantity_done': 5,
+ })],
+ })
+ self.assertEqual(delivery_order.state, 'assigned')
+
+ # Add a unit of productB, the check_availability button should not appear.
+ delivery_order = Form(delivery_order)
+ with delivery_order.move_ids_without_package.new() as move:
+ move.product_id = self.productB
+ delivery_order = delivery_order.save()
+
+ self.assertEqual(delivery_order.state, 'assigned')
+ self.assertEqual(delivery_order.show_check_availability, False)
+ self.assertEqual(delivery_order.show_mark_as_todo, False)
+
+ def test_owner_1(self):
+ """Make a receipt, set an owner and validate"""
+ owner1 = self.env['res.partner'].create({'name': 'owner'})
+ receipt = self.env['stock.picking'].create({
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ move1 = self.env['stock.move'].create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 1,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': receipt.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ })
+ receipt.action_confirm()
+ receipt = Form(receipt)
+ receipt.owner_id = owner1
+ receipt = receipt.save()
+ wiz = receipt.button_validate()
+ wiz = Form(self.env['stock.immediate.transfer'].with_context(wiz['context'])).save()
+ wiz.process()
+
+ supplier_location = self.env['stock.location'].browse(self.supplier_location)
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ supplier_quant = self.env['stock.quant']._gather(self.productA, supplier_location)
+ stock_quant = self.env['stock.quant']._gather(self.productA, stock_location)
+
+ self.assertEqual(supplier_quant.owner_id, owner1)
+ self.assertEqual(supplier_quant.quantity, -1)
+ self.assertEqual(stock_quant.owner_id, owner1)
+ self.assertEqual(stock_quant.quantity, 1)
+
+ def test_putaway_for_picking_sml(self):
+ """ Checks picking's move lines will take in account the putaway rules
+ to define the `location_dest_id`.
+ """
+ partner = self.env['res.partner'].create({'name': 'Partner'})
+ supplier_location = self.env['stock.location'].browse(self.supplier_location)
+ stock_location = self.env['stock.location'].create({
+ 'name': 'test-stock',
+ 'usage': 'internal',
+ })
+ shelf_location = self.env['stock.location'].create({
+ 'name': 'shelf1',
+ 'usage': 'internal',
+ 'location_id': stock_location.id,
+ })
+
+ # We need to activate multi-locations to use putaway rules.
+ grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
+ self.env.user.write({'groups_id': [(4, grp_multi_loc.id)]})
+ putaway_product = self.env['stock.putaway.rule'].create({
+ 'product_id': self.productA.id,
+ 'location_in_id': stock_location.id,
+ 'location_out_id': shelf_location.id,
+ })
+ # Changes config of receipt type to allow to edit move lines directly.
+ picking_type = self.env['stock.picking.type'].browse(self.picking_type_in)
+ picking_type.show_operations = True
+
+ receipt_form = Form(self.env['stock.picking'].with_context(
+ force_detailed_view=True
+ ), view='stock.view_picking_form')
+ receipt_form.partner_id = partner
+ receipt_form.picking_type_id = picking_type
+ receipt_form.location_id = supplier_location
+ receipt_form.location_dest_id = stock_location
+ receipt = receipt_form.save()
+ with receipt_form.move_line_nosuggest_ids.new() as move_line:
+ move_line.product_id = self.productA
+
+ receipt = receipt_form.save()
+ # Checks receipt has still its destination location and checks its move
+ # line took the one from the putaway rule.
+ self.assertEqual(receipt.location_dest_id.id, stock_location.id)
+ self.assertEqual(receipt.move_line_ids.location_dest_id.id, shelf_location.id)
+
+ def test_cancel_plan_transfer(self):
+ """ Test canceling plan transfer """
+ # Create picking with stock move.
+ picking = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ 'move_lines': [(0, 0, {
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })]
+ })
+ # Confirm the outgoing picking, state should be changed.
+ picking.action_confirm()
+ self.assertEqual(picking.state, 'confirmed', "Picking should be in a confirmed state.")
+
+ # Picking in a confirmed state and try to cancel it.
+ picking.action_cancel()
+ self.assertEqual(picking.state, 'cancel', "Picking should be in a cancel state.")
+
+ def test_immediate_transfer(self):
+ """ Test picking should be in ready state if immediate transfer and SML is created via view +
+ Test picking cancelation with immediate transfer and done quantity"""
+ # create picking with stock move line
+ picking = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ 'immediate_transfer': True,
+ 'move_line_ids': [(0, 0, {
+ 'product_id': self.productA.id,
+ 'qty_done': 10,
+ 'product_uom_id': self.productA.uom_id.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ })]
+ })
+
+ self.assertEqual(picking.state, 'assigned', "Picking should not be in a draft state.")
+ self.assertEqual(len(picking.move_lines), 1, "Picking should have stock move.")
+ picking.action_cancel()
+ self.assertEqual(picking.move_lines.state, 'cancel', "Stock move should be in a cancel state.")
+ self.assertEqual(picking.state, 'cancel', "Picking should be in a cancel state.")
+
+
+class TestStockUOM(TestStockCommon):
+ def setUp(self):
+ super(TestStockUOM, self).setUp()
+ dp = self.env.ref('product.decimal_product_uom')
+ dp.digits = 7
+
+ def test_pickings_transfer_with_different_uom_and_back_orders(self):
+ """ Picking transfer with diffrent unit of meassure. """
+ # weight category
+ categ_test = self.env['uom.category'].create({'name': 'Bigger than tons'})
+
+ T_LBS = self.env['uom.uom'].create({
+ 'name': 'T-LBS',
+ 'category_id': categ_test.id,
+ 'uom_type': 'reference',
+ 'rounding': 0.01
+ })
+ T_GT = self.env['uom.uom'].create({
+ 'name': 'T-GT',
+ 'category_id': categ_test.id,
+ 'uom_type': 'bigger',
+ 'rounding': 0.0000001,
+ 'factor_inv': 2240.00,
+ })
+ T_TEST = self.env['product.product'].create({
+ 'name': 'T_TEST',
+ 'type': 'product',
+ 'uom_id': T_LBS.id,
+ 'uom_po_id': T_LBS.id,
+ 'tracking': 'lot',
+ })
+
+ picking_in = self.env['stock.picking'].create({
+ 'picking_type_id': self.picking_type_in,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location
+ })
+ move = self.env['stock.move'].create({
+ 'name': 'First move with 60 GT',
+ 'product_id': T_TEST.id,
+ 'product_uom_qty': 60,
+ 'product_uom': T_GT.id,
+ 'picking_id': picking_in.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location
+ })
+ picking_in.action_confirm()
+
+ self.assertEqual(move.product_uom_qty, 60.00, 'Wrong T_GT quantity')
+ self.assertEqual(move.product_qty, 134400.00, 'Wrong T_LBS quantity')
+
+ lot = self.env['stock.production.lot'].create({'name': 'Lot TEST', 'product_id': T_TEST.id, 'company_id': self.env.company.id, })
+ self.env['stock.move.line'].create({
+ 'move_id': move.id,
+ 'product_id': T_TEST.id,
+ 'product_uom_id': T_LBS.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.stock_location,
+ 'qty_done': 42760.00,
+ 'lot_id': lot.id,
+ })
+
+ picking_in._action_done()
+ back_order_in = self.env['stock.picking'].search([('backorder_id', '=', picking_in.id)])
+
+ self.assertEqual(len(back_order_in), 1.00, 'There should be one back order created')
+ self.assertEqual(back_order_in.move_lines.product_qty, 91640.00, 'There should be one back order created')
+
+ def test_move_product_with_different_uom(self):
+ """ Product defined in g with 0.01 rounding
+ Decimal Accuracy (DA) 3 digits.
+ Quantity on hand: 149.88g
+ Picking of 1kg
+ kg has 0.0001 rounding
+ Due to conversions, we may end up reserving 150g
+ (more than the quantity in stock), we check that
+ we reserve less than the quantity in stock
+ """
+ precision = self.env.ref('product.decimal_product_uom')
+ precision.digits = 3
+ precision_digits = precision.digits
+
+ self.uom_kg.rounding = 0.0001
+ self.uom_gm.rounding = 0.01
+
+ product_G = self.env['product.product'].create({
+ 'name': 'Product G',
+ 'type': 'product',
+ 'categ_id': self.env.ref('product.product_category_all').id,
+ 'uom_id': self.uom_gm.id,
+ 'uom_po_id': self.uom_gm.id,
+ })
+
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+ self.env['stock.quant']._update_available_quantity(product_G, stock_location, 149.88)
+ self.assertEqual(len(product_G.stock_quant_ids), 1, 'One quant should exist for the product.')
+ quant = product_G.stock_quant_ids
+
+ # transfer 1kg of product_G
+ picking = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.env.ref('stock.picking_type_out').id,
+ })
+
+ move = self.env['stock.move'].create({
+ 'name': 'test_reserve_product_G',
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_id': picking.id,
+ 'product_id': product_G.id,
+ 'product_uom': self.uom_kg.id,
+ 'product_uom_qty': 1,
+ })
+
+ self.assertEqual(move.product_uom.id, self.uom_kg.id)
+ self.assertEqual(move.product_uom_qty, 1.0)
+
+ picking.action_confirm()
+ picking.action_assign()
+
+ self.assertEqual(product_G.uom_id.rounding, 0.01)
+ self.assertEqual(move.product_uom.rounding, 0.0001)
+
+ self.assertEqual(len(picking.move_line_ids), 1, 'One move line should exist for the picking.')
+ move_line = picking.move_line_ids
+ # check that we do not reserve more (in the same UOM) than the quantity in stock
+ self.assertEqual(float_compare(move_line.product_qty, quant.quantity, precision_digits=precision_digits), -1, "We do not reserve more (in the same UOM) than the quantity in stock")
+ # check that we reserve the same quantity in the ml and the quant
+ self.assertTrue(float_is_zero(move_line.product_qty - quant.reserved_quantity, precision_digits=precision_digits))
+
+ def test_update_product_move_line_with_different_uom(self):
+ """ Check that when the move line and corresponding
+ product have different UOM with possibly conflicting
+ precisions, we do not reserve more than the quantity
+ in stock. Similar initial configuration as
+ test_move_product_with_different_uom.
+ """
+ precision = self.env.ref('product.decimal_product_uom')
+ precision.digits = 3
+ precision_digits = precision.digits
+
+ self.uom_kg.rounding = 0.0001
+ self.uom_gm.rounding = 0.01
+
+ product_LtDA = self.env['product.product'].create({
+ 'name': 'Product Less than DA',
+ 'type': 'product',
+ 'categ_id': self.env.ref('product.product_category_all').id,
+ 'uom_id': self.uom_gm.id,
+ 'uom_po_id': self.uom_gm.id,
+ })
+
+ product_GtDA = self.env['product.product'].create({
+ 'name': 'Product Greater than DA',
+ 'type': 'product',
+ 'categ_id': self.env.ref('product.product_category_all').id,
+ 'uom_id': self.uom_gm.id,
+ 'uom_po_id': self.uom_gm.id,
+ })
+
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+
+ # quantity in hand converted to kg is not more precise than the DA
+ self.env['stock.quant']._update_available_quantity(product_LtDA, stock_location, 149)
+ # quantity in hand converted to kg is more precise than the DA
+ self.env['stock.quant']._update_available_quantity(product_GtDA, stock_location, 149.88)
+
+ self.assertEqual(len(product_LtDA.stock_quant_ids), 1, 'One quant should exist for the product.')
+ self.assertEqual(len(product_GtDA.stock_quant_ids), 1, 'One quant should exist for the product.')
+ quant_LtDA = product_LtDA.stock_quant_ids
+ quant_GtDA = product_GtDA.stock_quant_ids
+
+ # create 2 moves of 1kg
+ move_LtDA = self.env['stock.move'].create({
+ 'name': 'test_reserve_product_LtDA',
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'product_id': product_LtDA.id,
+ 'product_uom': self.uom_kg.id,
+ 'product_uom_qty': 1,
+ })
+
+ move_GtDA = self.env['stock.move'].create({
+ 'name': 'test_reserve_product_GtDA',
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'product_id': product_GtDA.id,
+ 'product_uom': self.uom_kg.id,
+ 'product_uom_qty': 1,
+ })
+
+ self.assertEqual(move_LtDA.state, 'draft')
+ self.assertEqual(move_GtDA.state, 'draft')
+ move_LtDA._action_confirm()
+ move_GtDA._action_confirm()
+ self.assertEqual(move_LtDA.state, 'confirmed')
+ self.assertEqual(move_GtDA.state, 'confirmed')
+ # check availability, less than initial demand
+ move_LtDA._action_assign()
+ move_GtDA._action_assign()
+ self.assertEqual(move_LtDA.state, 'partially_available')
+ self.assertEqual(move_GtDA.state, 'partially_available')
+ # the initial demand is 1kg
+ self.assertEqual(move_LtDA.product_uom.id, self.uom_kg.id)
+ self.assertEqual(move_GtDA.product_uom.id, self.uom_kg.id)
+ self.assertEqual(move_LtDA.product_uom_qty, 1.0)
+ self.assertEqual(move_GtDA.product_uom_qty, 1.0)
+ # one move line is created
+ self.assertEqual(len(move_LtDA.move_line_ids), 1)
+ self.assertEqual(len(move_GtDA.move_line_ids), 1)
+
+ # increase quantity by 0.14988 kg (more precise than DA)
+ self.env['stock.quant']._update_available_quantity(product_LtDA, stock_location, 149.88)
+ self.env['stock.quant']._update_available_quantity(product_GtDA, stock_location, 149.88)
+
+ # _update_reserved_quantity is called on a move only in _action_assign
+ move_LtDA._action_assign()
+ move_GtDA._action_assign()
+
+ # as the move line for LtDA and its corresponding quant can be
+ # in different UOMs, a new move line can be created
+ # from _update_reserved_quantity
+ move_lines_LtDA = self.env["stock.move.line"].search([
+ ('product_id', '=', quant_LtDA.product_id.id),
+ ('location_id', '=', quant_LtDA.location_id.id),
+ ('lot_id', '=', quant_LtDA.lot_id.id),
+ ('package_id', '=', quant_LtDA.package_id.id),
+ ('owner_id', '=', quant_LtDA.owner_id.id),
+ ('product_qty', '!=', 0)
+ ])
+ reserved_on_move_lines_LtDA = sum(move_lines_LtDA.mapped('product_qty'))
+
+ move_lines_GtDA = self.env["stock.move.line"].search([
+ ('product_id', '=', quant_GtDA.product_id.id),
+ ('location_id', '=', quant_GtDA.location_id.id),
+ ('lot_id', '=', quant_GtDA.lot_id.id),
+ ('package_id', '=', quant_GtDA.package_id.id),
+ ('owner_id', '=', quant_GtDA.owner_id.id),
+ ('product_qty', '!=', 0)
+ ])
+ reserved_on_move_lines_GtDA = sum(move_lines_GtDA.mapped('product_qty'))
+
+ # check that we do not reserve more (in the same UOM) than the quantity in stock
+ self.assertEqual(float_compare(reserved_on_move_lines_LtDA, quant_LtDA.quantity, precision_digits=precision_digits), -1, "We do not reserve more (in the same UOM) than the quantity in stock")
+ self.assertEqual(float_compare(reserved_on_move_lines_GtDA, quant_GtDA.quantity, precision_digits=precision_digits), -1, "We do not reserve more (in the same UOM) than the quantity in stock")
+
+ # check that we reserve the same quantity in the ml and the quant
+ self.assertTrue(float_is_zero(reserved_on_move_lines_LtDA - quant_LtDA.reserved_quantity, precision_digits=precision_digits))
+ self.assertTrue(float_is_zero(reserved_on_move_lines_GtDA - quant_GtDA.reserved_quantity, precision_digits=precision_digits))
+
+
+class TestRoutes(TestStockCommon):
+ def setUp(self):
+ super(TestRoutes, self).setUp()
+ self.product1 = self.env['product.product'].create({
+ 'name': 'product a',
+ 'type': 'product',
+ 'categ_id': self.env.ref('product.product_category_all').id,
+ })
+ self.uom_unit = self.env.ref('uom.product_uom_unit')
+ self.partner = self.env['res.partner'].create({'name': 'Partner'})
+
+ def _enable_pick_ship(self):
+ self.wh = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+
+ # create and get back the pick ship route
+ self.wh.write({'delivery_steps': 'pick_ship'})
+ self.pick_ship_route = self.wh.route_ids.filtered(lambda r: '(pick + ship)' in r.name)
+
+ def test_pick_ship_1(self):
+ """ Enable the pick ship route, force a procurement group on the
+ pick. When a second move is added, make sure the `partner_id` and
+ `origin` fields are erased.
+ """
+ self._enable_pick_ship()
+
+ # create a procurement group and set in on the pick stock rule
+ procurement_group0 = self.env['procurement.group'].create({})
+ pick_rule = self.pick_ship_route.rule_ids.filtered(lambda rule: 'Stock → Output' in rule.name)
+ push_rule = self.pick_ship_route.rule_ids - pick_rule
+ pick_rule.write({
+ 'group_propagation_option': 'fixed',
+ 'group_id': procurement_group0.id,
+ })
+
+ ship_location = pick_rule.location_id
+ customer_location = push_rule.location_id
+ partners = self.env['res.partner'].search([], limit=2)
+ partner0 = partners[0]
+ partner1 = partners[1]
+ procurement_group1 = self.env['procurement.group'].create({'partner_id': partner0.id})
+ procurement_group2 = self.env['procurement.group'].create({'partner_id': partner1.id})
+
+ move1 = self.env['stock.move'].create({
+ 'name': 'first out move',
+ 'procure_method': 'make_to_order',
+ 'location_id': ship_location.id,
+ 'location_dest_id': customer_location.id,
+ 'product_id': self.product1.id,
+ 'product_uom': self.uom_unit.id,
+ 'product_uom_qty': 1.0,
+ 'warehouse_id': self.wh.id,
+ 'group_id': procurement_group1.id,
+ 'origin': 'origin1',
+ })
+
+ move2 = self.env['stock.move'].create({
+ 'name': 'second out move',
+ 'procure_method': 'make_to_order',
+ 'location_id': ship_location.id,
+ 'location_dest_id': customer_location.id,
+ 'product_id': self.product1.id,
+ 'product_uom': self.uom_unit.id,
+ 'product_uom_qty': 1.0,
+ 'warehouse_id': self.wh.id,
+ 'group_id': procurement_group2.id,
+ 'origin': 'origin2',
+ })
+
+ # first out move, the "pick" picking should have a partner and an origin
+ move1._action_confirm()
+ picking_pick = move1.move_orig_ids.picking_id
+ self.assertEqual(picking_pick.partner_id.id, procurement_group1.partner_id.id)
+ self.assertEqual(picking_pick.origin, move1.group_id.name)
+
+ # second out move, the "pick" picking should have lost its partner and origin
+ move2._action_confirm()
+ self.assertEqual(picking_pick.partner_id.id, False)
+ self.assertEqual(picking_pick.origin, False)
+
+ def test_replenish_pick_ship_1(self):
+ """ Creates 2 warehouses and make a replenish using one warehouse
+ to ressuply the other one, Then check if the quantity and the product are matching
+ """
+ self.product_uom_qty = 42
+
+ warehouse_1 = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ warehouse_2 = self.env['stock.warehouse'].create({
+ 'name': 'Small Warehouse',
+ 'code': 'SWH'
+ })
+ warehouse_1.write({
+ 'resupply_wh_ids': [(6, 0, [warehouse_2.id])]
+ })
+ resupply_route = self.env['stock.location.route'].search([('supplier_wh_id', '=', warehouse_2.id), ('supplied_wh_id', '=', warehouse_1.id)])
+ self.assertTrue(resupply_route, "Ressuply route not found")
+ self.product1.write({'route_ids': [(4, resupply_route.id), (4, self.env.ref('stock.route_warehouse0_mto').id)]})
+ self.wh = warehouse_1
+
+ replenish_wizard = self.env['product.replenish'].create({
+ 'product_id': self.product1.id,
+ 'product_tmpl_id': self.product1.product_tmpl_id.id,
+ 'product_uom_id': self.uom_unit.id,
+ 'quantity': self.product_uom_qty,
+ 'warehouse_id': self.wh.id,
+ })
+
+ replenish_wizard.launch_replenishment()
+ last_picking_id = self.env['stock.picking'].search([('origin', '=', 'Manual Replenishment')])[-1]
+ self.assertTrue(last_picking_id, 'Picking not found')
+ move_line = last_picking_id.move_lines.search([('product_id','=', self.product1.id)])
+ self.assertTrue(move_line,'The product is not in the picking')
+ self.assertEqual(move_line[0].product_uom_qty, self.product_uom_qty, 'Quantities does not match')
+ self.assertEqual(move_line[1].product_uom_qty, self.product_uom_qty, 'Quantities does not match')
+
+ def test_push_rule_on_move_1(self):
+ """ Create a route with a push rule, force it on a move, check that it is applied.
+ """
+ self._enable_pick_ship()
+ stock_location = self.env.ref('stock.stock_location_stock')
+
+ push_location = self.env['stock.location'].create({
+ 'location_id': stock_location.location_id.id,
+ 'name': 'push location',
+ })
+
+ # TODO: maybe add a new type on the "applicable on" fields?
+ route = self.env['stock.location.route'].create({
+ 'name': 'new route',
+ 'rule_ids': [(0, False, {
+ 'name': 'create a move to push location',
+ 'location_src_id': stock_location.id,
+ 'location_id': push_location.id,
+ 'company_id': self.env.company.id,
+ 'action': 'push',
+ 'auto': 'manual',
+ 'picking_type_id': self.env.ref('stock.picking_type_in').id,
+ })],
+ })
+ move1 = self.env['stock.move'].create({
+ 'name': 'move with a route',
+ 'location_id': stock_location.id,
+ 'location_dest_id': stock_location.id,
+ 'product_id': self.product1.id,
+ 'product_uom': self.uom_unit.id,
+ 'product_uom_qty': 1.0,
+ 'route_ids': [(4, route.id)]
+ })
+ move1._action_confirm()
+
+ pushed_move = move1.move_dest_ids
+ self.assertEqual(pushed_move.location_dest_id.id, push_location.id)
+
+ def test_location_dest_update(self):
+ """ Check the location dest of a stock move changed by a push rule
+ with auto field set to transparent is done correctly. The stock_move
+ is create with the move line directly to pass into action_confirm() via
+ action_done(). """
+ self.wh = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ new_loc = self.env['stock.location'].create({
+ 'name': 'New_location',
+ 'usage': 'internal',
+ 'location_id': self.env.ref('stock.stock_location_locations').id,
+ })
+ 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': new_loc.id,
+ 'warehouse_id': self.wh.id,
+ })
+ route = self.env['stock.location.route'].create({
+ 'name': 'new route',
+ 'rule_ids': [(0, False, {
+ 'name': 'create a move to push location',
+ 'location_src_id': self.env.ref('stock.stock_location_stock').id,
+ 'location_id': new_loc.id,
+ 'company_id': self.env.company.id,
+ 'action': 'push',
+ 'auto': 'transparent',
+ 'picking_type_id': picking_type.id,
+ })],
+ })
+ product = self.env['product.product'].create({
+ 'name': 'new_product',
+ 'type': 'product',
+ 'route_ids': [(4, route.id)]
+ })
+ move1 = self.env['stock.move'].create({
+ 'name': 'move with a route',
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.env.ref('stock.stock_location_stock').id,
+ 'product_id': product.id,
+ 'product_uom_qty': 1.0,
+ 'product_uom': self.uom_unit.id,
+ 'move_line_ids': [(0, 0, {
+ 'product_id': product.id,
+ 'product_uom_id': self.uom_unit.id,
+ 'location_id': self.supplier_location,
+ 'location_dest_id': self.env.ref('stock.stock_location_stock').id,
+ 'qty_done': 1.00,
+ })],
+ })
+ move1._action_done()
+ self.assertEqual(move1.location_dest_id, new_loc)
+ positive_quant = product.stock_quant_ids.filtered(lambda q: q.quantity > 0)
+ self.assertEqual(positive_quant.location_id, new_loc)
+
+ def test_mtso_mto(self):
+ """ Run a procurement for 5 products when there are only 4 in stock then
+ check that MTO is applied on the moves when the rule is set to 'mts_else_mto'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ warehouse.delivery_steps = 'pick_pack_ship'
+ partner_demo_customer = self.partner
+ final_location = partner_demo_customer.property_stock_customer
+ product_a = self.env['product.product'].create({
+ 'name': 'ProductA',
+ 'type': 'product',
+ })
+
+ self.env['stock.quant']._update_available_quantity(product_a, warehouse.wh_output_stock_loc_id, 4.0)
+
+ # We set quantities in the stock location to avoid warnings
+ # triggered by '_onchange_product_id_check_availability'
+ self.env['stock.quant']._update_available_quantity(product_a, warehouse.lot_stock_id, 4.0)
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ values = {'warehouse_id': warehouse}
+ rule = self.env['procurement.group']._get_rule(product_a, final_location, values)
+ rule.procure_method = 'mts_else_mto'
+
+ pg = self.env['procurement.group'].create({'name': 'Test-pg-mtso-mto'})
+
+ self.env['procurement.group'].run([
+ pg.Procurement(
+ product_a,
+ 5.0,
+ product_a.uom_id,
+ final_location,
+ 'test_mtso_mto',
+ 'test_mtso_mto',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg
+ }
+ )
+ ])
+
+ qty_available = self.env['stock.quant']._get_available_quantity(product_a, warehouse.wh_output_stock_loc_id)
+
+ # 3 pickings should be created.
+ picking_ids = self.env['stock.picking'].search([('group_id', '=', pg.id)])
+ self.assertEqual(len(picking_ids), 3)
+ for picking in picking_ids:
+ # Only the picking from Stock to Pack should be MTS
+ if picking.location_id == warehouse.lot_stock_id:
+ self.assertEqual(picking.move_lines.procure_method, 'make_to_stock')
+ else:
+ self.assertEqual(picking.move_lines.procure_method, 'make_to_order')
+
+ self.assertEqual(len(picking.move_lines), 1)
+ self.assertEqual(picking.move_lines.product_uom_qty, 5, 'The quantity of the move should be the same as on the SO')
+ self.assertEqual(qty_available, 4, 'The 4 products should still be available')
+
+ def test_mtso_mts(self):
+ """ Run a procurement for 4 products when there are 4 in stock then
+ check that MTS is applied on the moves when the rule is set to 'mts_else_mto'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ warehouse.delivery_steps = 'pick_pack_ship'
+ partner_demo_customer = self.partner
+ final_location = partner_demo_customer.property_stock_customer
+ product_a = self.env['product.product'].create({
+ 'name': 'ProductA',
+ 'type': 'product',
+ })
+
+ self.env['stock.quant']._update_available_quantity(product_a, warehouse.wh_output_stock_loc_id, 4.0)
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ values = {'warehouse_id': warehouse}
+ rule = self.env['procurement.group']._get_rule(product_a, final_location, values)
+ rule.procure_method = 'mts_else_mto'
+
+ pg = self.env['procurement.group'].create({'name': 'Test-pg-mtso-mts'})
+
+ self.env['procurement.group'].run([
+ pg.Procurement(
+ product_a,
+ 4.0,
+ product_a.uom_id,
+ final_location,
+ 'test_mtso_mts',
+ 'test_mtso_mts',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg
+ }
+ )
+ ])
+
+ # A picking should be created with its move having MTS as procure method.
+ picking_ids = self.env['stock.picking'].search([('group_id', '=', pg.id)])
+ self.assertEqual(len(picking_ids), 1)
+ picking = picking_ids
+ self.assertEqual(picking.move_lines.procure_method, 'make_to_stock')
+ self.assertEqual(len(picking.move_lines), 1)
+ self.assertEqual(picking.move_lines.product_uom_qty, 4)
+
+ def test_mtso_multi_pg(self):
+ """ Run 3 procurements for 2 products at the same times when there are 4 in stock then
+ check that MTS is applied on the moves when the rule is set to 'mts_else_mto'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ warehouse.delivery_steps = 'pick_pack_ship'
+ partner_demo_customer = self.partner
+ final_location = partner_demo_customer.property_stock_customer
+ product_a = self.env['product.product'].create({
+ 'name': 'ProductA',
+ 'type': 'product',
+ })
+
+ self.env['stock.quant']._update_available_quantity(product_a, warehouse.wh_output_stock_loc_id, 4.0)
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ values = {'warehouse_id': warehouse}
+ rule = self.env['procurement.group']._get_rule(product_a, final_location, values)
+ rule.procure_method = 'mts_else_mto'
+
+ pg1 = self.env['procurement.group'].create({'name': 'Test-pg-mtso-mts-1'})
+ pg2 = self.env['procurement.group'].create({'name': 'Test-pg-mtso-mts-2'})
+ pg3 = self.env['procurement.group'].create({'name': 'Test-pg-mtso-mts-3'})
+
+ self.env['procurement.group'].run([
+ pg1.Procurement(
+ product_a,
+ 2.0,
+ product_a.uom_id,
+ final_location,
+ 'test_mtso_mts_1',
+ 'test_mtso_mts_1',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg1
+ }
+ ),
+ pg2.Procurement(
+ product_a,
+ 2.0,
+ product_a.uom_id,
+ final_location,
+ 'test_mtso_mts_2',
+ 'test_mtso_mts_2',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg2
+ }
+ ),
+ pg3.Procurement(
+ product_a,
+ 2.0,
+ product_a.uom_id,
+ final_location,
+ 'test_mtso_mts_3',
+ 'test_mtso_mts_3',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg3
+ }
+ )
+ ])
+
+ pickings_pg1 = self.env['stock.picking'].search([('group_id', '=', pg1.id)])
+ pickings_pg2 = self.env['stock.picking'].search([('group_id', '=', pg2.id)])
+ pickings_pg3 = self.env['stock.picking'].search([('group_id', '=', pg3.id)])
+
+ # The 2 first procurements should have create only 1 picking since enough quantities
+ # are left in the delivery location
+ self.assertEqual(len(pickings_pg1), 1)
+ self.assertEqual(len(pickings_pg2), 1)
+ self.assertEqual(pickings_pg1.move_lines.procure_method, 'make_to_stock')
+ self.assertEqual(pickings_pg2.move_lines.procure_method, 'make_to_stock')
+
+ # The last one should have 3 pickings as there's nothing left in the delivery location
+ self.assertEqual(len(pickings_pg3), 3)
+ for picking in pickings_pg3:
+ # Only the picking from Stock to Pack should be MTS
+ if picking.location_id == warehouse.lot_stock_id:
+ self.assertEqual(picking.move_lines.procure_method, 'make_to_stock')
+ else:
+ self.assertEqual(picking.move_lines.procure_method, 'make_to_order')
+
+ # All the moves should be should have the same quantity as it is on each procurements
+ self.assertEqual(len(picking.move_lines), 1)
+ self.assertEqual(picking.move_lines.product_uom_qty, 2)
+
+ def test_mtso_mto_adjust_01(self):
+ """ Run '_adjust_procure_method' for products A & B:
+ - Product A has 5.0 available
+ - Product B has 3.0 available
+ Stock moves (SM) are created for 4.0 units
+ After '_adjust_procure_method':
+ - SM for A is 'make_to_stock'
+ - SM for B is 'make_to_order'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ final_location = self.partner.property_stock_customer
+ product_A = self.env['product.product'].create({
+ 'name': 'Product A',
+ 'type': 'product',
+ })
+ product_B = self.env['product.product'].create({
+ 'name': 'Product B',
+ 'type': 'product',
+ })
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ rule = self.env['procurement.group']._get_rule(product_A, final_location, {'warehouse_id': warehouse})
+ rule.procure_method = 'mts_else_mto'
+
+ self.env['stock.quant']._update_available_quantity(product_A, warehouse.lot_stock_id, 5.0)
+ self.env['stock.quant']._update_available_quantity(product_B, warehouse.lot_stock_id, 3.0)
+
+ move_tmpl = {
+ 'name': 'Product',
+ 'product_uom': self.uom_unit.id,
+ 'product_uom_qty': 4.0,
+ 'location_id': warehouse.lot_stock_id.id,
+ 'location_dest_id': self.partner.property_stock_customer.id,
+ 'warehouse_id': warehouse.id,
+ }
+ move_A_vals = dict(move_tmpl)
+ move_A_vals.update({
+ 'product_id': product_A.id,
+ })
+ move_A = self.env['stock.move'].create(move_A_vals)
+ move_B_vals = dict(move_tmpl)
+ move_B_vals.update({
+ 'product_id': product_B.id,
+ })
+ move_B = self.env['stock.move'].create(move_B_vals)
+ moves = move_A + move_B
+
+ self.assertEqual(move_A.procure_method, 'make_to_stock', 'Move A should be "make_to_stock"')
+ self.assertEqual(move_B.procure_method, 'make_to_stock', 'Move A should be "make_to_order"')
+ moves._adjust_procure_method()
+ self.assertEqual(move_A.procure_method, 'make_to_stock', 'Move A should be "make_to_stock"')
+ self.assertEqual(move_B.procure_method, 'make_to_order', 'Move A should be "make_to_order"')
+
+ def test_mtso_mto_adjust_02(self):
+ """ Run '_adjust_procure_method' for products A & B:
+ - Product A has 5.0 available
+ - Product B has 3.0 available
+ Stock moves (SM) are created for 2.0 + 2.0 units
+ After '_adjust_procure_method':
+ - SM for A is 'make_to_stock'
+ - SM for B is 'make_to_stock' and 'make_to_order'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ final_location = self.partner.property_stock_customer
+ product_A = self.env['product.product'].create({
+ 'name': 'Product A',
+ 'type': 'product',
+ })
+ product_B = self.env['product.product'].create({
+ 'name': 'Product B',
+ 'type': 'product',
+ })
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ rule = self.env['procurement.group']._get_rule(product_A, final_location, {'warehouse_id': warehouse})
+ rule.procure_method = 'mts_else_mto'
+
+ self.env['stock.quant']._update_available_quantity(product_A, warehouse.lot_stock_id, 5.0)
+ self.env['stock.quant']._update_available_quantity(product_B, warehouse.lot_stock_id, 3.0)
+
+ move_tmpl = {
+ 'name': 'Product',
+ 'product_uom': self.uom_unit.id,
+ 'product_uom_qty': 2.0,
+ 'location_id': warehouse.lot_stock_id.id,
+ 'location_dest_id': self.partner.property_stock_customer.id,
+ 'warehouse_id': warehouse.id,
+ }
+ move_A1_vals = dict(move_tmpl)
+ move_A1_vals.update({
+ 'product_id': product_A.id,
+ })
+ move_A1 = self.env['stock.move'].create(move_A1_vals)
+ move_A2_vals = dict(move_tmpl)
+ move_A2_vals.update({
+ 'product_id': product_A.id,
+ })
+ move_A2 = self.env['stock.move'].create(move_A2_vals)
+ move_B1_vals = dict(move_tmpl)
+ move_B1_vals.update({
+ 'product_id': product_B.id,
+ })
+ move_B1 = self.env['stock.move'].create(move_B1_vals)
+ move_B2_vals = dict(move_tmpl)
+ move_B2_vals.update({
+ 'product_id': product_B.id,
+ })
+ move_B2 = self.env['stock.move'].create(move_B2_vals)
+ moves = move_A1 + move_A2 + move_B1 + move_B2
+
+ self.assertEqual(move_A1.procure_method, 'make_to_stock', 'Move A1 should be "make_to_stock"')
+ self.assertEqual(move_A2.procure_method, 'make_to_stock', 'Move A2 should be "make_to_stock"')
+ self.assertEqual(move_B1.procure_method, 'make_to_stock', 'Move B1 should be "make_to_stock"')
+ self.assertEqual(move_B2.procure_method, 'make_to_stock', 'Move B2 should be "make_to_stock"')
+ moves._adjust_procure_method()
+ self.assertEqual(move_A1.procure_method, 'make_to_stock', 'Move A1 should be "make_to_stock"')
+ self.assertEqual(move_A2.procure_method, 'make_to_stock', 'Move A2 should be "make_to_stock"')
+ self.assertEqual(move_B1.procure_method, 'make_to_stock', 'Move B1 should be "make_to_stock"')
+ self.assertEqual(move_B2.procure_method, 'make_to_order', 'Move B2 should be "make_to_order"')
+
+ def test_mtso_mto_adjust_03(self):
+ """ Run '_adjust_procure_method' for products A with 4.0 available
+ 2 Stock moves (SM) are created:
+ - SM1 for 5.0 Units
+ - SM2 for 3.0 Units
+ SM1 is confirmed, so 'virtual_available' is -1.0.
+ SM1 should become 'make_to_order'
+ SM2 should remain 'make_to_stock'
+ """
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.user.id)], limit=1)
+ final_location = self.partner.property_stock_customer
+ product_A = self.env['product.product'].create({
+ 'name': 'Product A',
+ 'type': 'product',
+ })
+
+ # We alter one rule and we set it to 'mts_else_mto'
+ rule = self.env['procurement.group']._get_rule(product_A, final_location, {'warehouse_id': warehouse})
+ rule.procure_method = 'mts_else_mto'
+
+ self.env['stock.quant']._update_available_quantity(product_A, warehouse.lot_stock_id, 4.0)
+
+ move_tmpl = {
+ 'name': 'Product',
+ 'product_id': product_A.id,
+ 'product_uom': self.uom_unit.id,
+ 'location_id': warehouse.lot_stock_id.id,
+ 'location_dest_id': self.partner.property_stock_customer.id,
+ 'warehouse_id': warehouse.id,
+ }
+ move_A1_vals = dict(move_tmpl)
+ move_A1_vals.update({
+ 'product_uom_qty': 5.0,
+ })
+ move_A1 = self.env['stock.move'].create(move_A1_vals)
+ move_A2_vals = dict(move_tmpl)
+ move_A2_vals.update({
+ 'product_uom_qty': 3.0,
+ })
+ move_A2 = self.env['stock.move'].create(move_A2_vals)
+ moves = move_A1 + move_A2
+
+ self.assertEqual(move_A1.procure_method, 'make_to_stock', 'Move A1 should be "make_to_stock"')
+ self.assertEqual(move_A2.procure_method, 'make_to_stock', 'Move A2 should be "make_to_stock"')
+ move_A1._action_confirm()
+ moves._adjust_procure_method()
+ self.assertEqual(move_A1.procure_method, 'make_to_order', 'Move A should be "make_to_stock"')
+ self.assertEqual(move_A2.procure_method, 'make_to_stock', 'Move A should be "make_to_order"')
+
+ def test_delay_alert_3(self):
+ warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1)
+ warehouse.delivery_steps = 'pick_pack_ship'
+ partner_demo_customer = self.partner
+ final_location = partner_demo_customer.property_stock_customer
+ product_a = self.env['product.product'].create({
+ 'name': 'ProductA',
+ 'type': 'product',
+ })
+ pg = self.env['procurement.group'].create({'name': 'Test-delay_alert_3'})
+ self.env['procurement.group'].run([
+ pg.Procurement(
+ product_a,
+ 4.0,
+ product_a.uom_id,
+ final_location,
+ 'delay',
+ 'delay',
+ warehouse.company_id,
+ {
+ 'warehouse_id': warehouse,
+ 'group_id': pg
+ }
+ ),
+ ])
+ ship, pack, pick = self.env['stock.move'].search([('product_id', '=', product_a.id)])
+
+ # by default they all the same `date`
+ self.assertEqual(set((ship + pack + pick).mapped('date')), {pick.date})
+
+ # pick - pack - ship
+ ship.date += timedelta(days=2)
+ pack.date += timedelta(days=1)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertFalse(pack.delay_alert_date)
+ self.assertFalse(ship.delay_alert_date)
+
+ # move the pack after the ship
+ # pick - ship - pack
+ pack.date += timedelta(days=2)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertFalse(pack.delay_alert_date)
+ self.assertTrue(ship.delay_alert_date)
+ self.assertAlmostEqual(ship.delay_alert_date, pack.date)
+
+ # restore the pack before the ship
+ # pick - pack - ship
+ pack.date -= timedelta(days=2)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertFalse(pack.delay_alert_date)
+ self.assertFalse(ship.delay_alert_date)
+
+ # move the pick after the pack
+ # pack - ship - pick
+ pick.date += timedelta(days=3)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertTrue(pack.delay_alert_date)
+ self.assertFalse(ship.delay_alert_date)
+ self.assertAlmostEqual(pack.delay_alert_date, pick.date)
+
+ # move the ship before the pack
+ # ship - pack - pick
+ ship.date -= timedelta(days=2)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertTrue(pack.delay_alert_date)
+ self.assertTrue(ship.delay_alert_date)
+ self.assertAlmostEqual(pack.delay_alert_date, pick.date)
+ self.assertAlmostEqual(ship.delay_alert_date, pack.date)
+
+ # move the pack at the end
+ # ship - pick - pack
+ pack.date = pick.date + timedelta(days=2)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertFalse(pack.delay_alert_date)
+ self.assertTrue(ship.delay_alert_date)
+ self.assertAlmostEqual(ship.delay_alert_date, pack.date)
+
+ # fix the ship
+ ship.date = pack.date + timedelta(days=2)
+ self.assertFalse(pick.delay_alert_date)
+ self.assertFalse(pack.delay_alert_date)
+ self.assertFalse(ship.delay_alert_date)
+
+
+class TestAutoAssign(TestStockCommon):
+ def create_pick_ship(self):
+ picking_client = self.env['stock.picking'].create({
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ dest = self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_client.id,
+ 'location_id': self.pack_location,
+ 'location_dest_id': self.customer_location,
+ 'state': 'waiting',
+ 'procure_method': 'make_to_order',
+ })
+
+ picking_pick = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_pick.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'move_dest_ids': [(4, dest.id)],
+ 'state': 'confirmed',
+ })
+ return picking_pick, picking_client
+
+ def test_auto_assign_0(self):
+ """Create a outgoing MTS move without enough products in stock, then
+ validate a incoming move to check if the outgoing move is automatically
+ assigned.
+ """
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+
+ # create customer picking and move
+ customer_picking = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ customer_move = self.env['stock.move'].create({
+ 'name': 'customer move',
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.customer_location,
+ 'product_id': self.productA.id,
+ 'product_uom': self.productA.uom_id.id,
+ 'product_uom_qty': 10.0,
+ 'picking_id': customer_picking.id,
+ 'picking_type_id': self.picking_type_out,
+ })
+ customer_picking.action_confirm()
+ customer_picking.action_assign()
+ self.assertEqual(customer_move.state, 'confirmed')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, stock_location), 0)
+
+ # create supplier picking and move
+ supplier_picking = self.env['stock.picking'].create({
+ 'location_id': self.customer_location,
+ 'location_dest_id': self.stock_location,
+ 'picking_type_id': self.picking_type_in,
+ })
+ supplier_move = self.env['stock.move'].create({
+ 'name': 'test_transit_1',
+ 'location_id': self.customer_location,
+ 'location_dest_id': self.stock_location,
+ 'product_id': self.productA.id,
+ 'product_uom': self.productA.uom_id.id,
+ 'product_uom_qty': 10.0,
+ 'picking_id': supplier_picking.id,
+ })
+ customer_picking.action_confirm()
+ customer_picking.action_assign()
+ supplier_move.quantity_done = 10
+ supplier_picking._action_done()
+
+ # customer move should be automatically assigned and no more available product in stock
+ self.assertEqual(customer_move.state, 'assigned')
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, stock_location), 0)
+
+ def test_auto_assign_1(self):
+ """Create a outgoing MTO move without enough products, then validate a
+ move to make it available to check if the outgoing move is not
+ automatically assigned.
+ """
+ picking_pick, picking_client = self.create_pick_ship()
+ pack_location = self.env['stock.location'].browse(self.pack_location)
+ stock_location = self.env['stock.location'].browse(self.stock_location)
+
+ # make some stock
+ self.env['stock.quant']._update_available_quantity(self.productA, stock_location, 10.0)
+
+ # create another move to make product available in pack_location
+ picking_pick_2 = self.env['stock.picking'].create({
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'picking_type_id': self.picking_type_out,
+ })
+ self.MoveObj.create({
+ 'name': self.productA.name,
+ 'product_id': self.productA.id,
+ 'product_uom_qty': 10,
+ 'product_uom': self.productA.uom_id.id,
+ 'picking_id': picking_pick_2.id,
+ 'location_id': self.stock_location,
+ 'location_dest_id': self.pack_location,
+ 'state': 'confirmed',
+ })
+ picking_pick_2.action_assign()
+ picking_pick_2.move_lines[0].move_line_ids[0].qty_done = 10.0
+ picking_pick_2._action_done()
+
+ self.assertEqual(picking_client.state, 'waiting', "MTO moves can't be automatically assigned.")
+ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.productA, pack_location), 10.0)
+
+ def test_serial_lot_ids(self):
+ self.stock_location = self.env.ref('stock.stock_location_stock')
+ self.customer_location = self.env.ref('stock.stock_location_customers')
+ self.supplier_location = self.env.ref('stock.stock_location_suppliers')
+ self.uom_unit = self.env.ref('uom.product_uom_unit')
+ self.product_serial = self.env['product.product'].create({
+ 'name': 'PSerial',
+ 'type': 'product',
+ 'tracking': 'serial',
+ 'categ_id': self.env.ref('product.product_category_all').id,
+ })
+
+ move = self.env['stock.move'].create({
+ 'name': 'TestReceive',
+ 'location_id': self.supplier_location.id,
+ 'location_dest_id': self.stock_location.id,
+ 'product_id': self.product_serial.id,
+ 'product_uom': self.uom_unit.id,
+ 'picking_type_id': self.env.ref('stock.picking_type_in').id,
+ })
+ self.assertEqual(move.state, 'draft')
+ lot1 = self.env['stock.production.lot'].create({
+ 'name': 'serial1',
+ 'product_id': self.product_serial.id,
+ 'company_id': self.env.company.id,
+ })
+ lot2 = self.env['stock.production.lot'].create({
+ 'name': 'serial2',
+ 'product_id': self.product_serial.id,
+ 'company_id': self.env.company.id,
+ })
+ lot3 = self.env['stock.production.lot'].create({
+ 'name': 'serial3',
+ 'product_id': self.product_serial.id,
+ 'company_id': self.env.company.id,
+ })
+ move.lot_ids = [(4, lot1.id)]
+ move.lot_ids = [(4, lot2.id)]
+ move.lot_ids = [(4, lot3.id)]
+ self.assertEqual(move.quantity_done, 3.0)
+ move.lot_ids = [(3, lot2.id)]
+ self.assertEqual(move.quantity_done, 2.0)
+
+ self.uom_dozen = self.env.ref('uom.product_uom_dozen')
+ move = self.env['stock.move'].create({
+ 'name': 'TestReceiveDozen',
+ 'location_id': self.supplier_location.id,
+ 'location_dest_id': self.stock_location.id,
+ 'product_id': self.product_serial.id,
+ 'product_uom': self.uom_dozen.id,
+ 'picking_type_id': self.env.ref('stock.picking_type_in').id,
+ })
+ move.lot_ids = [(4, lot1.id)]
+ move.lot_ids = [(4, lot2.id)]
+ move.lot_ids = [(4, lot3.id)]
+ self.assertEqual(move.quantity_done, 3.0/12.0)
+
+ def test_update_description(self):
+ """ Create an empty picking. Adds a move on product1, select the picking type, add
+ again a move on product1. Confirm the picking. The two stock moves should be merged. """
+ product1 = self.env['product.product'].create({
+ 'name': 'product',
+ 'type':'product',
+ })
+ picking_form = Form(self.env['stock.picking'])
+ with picking_form.move_ids_without_package.new() as move:
+ move.product_id = product1
+ move.product_uom_qty = 10
+ move.location_id = self.env.ref('stock.stock_location_suppliers')
+ move.location_dest_id = self.env.ref('stock.stock_location_stock')
+ picking_form.picking_type_id = self.env.ref('stock.picking_type_in')
+ with picking_form.move_ids_without_package.new() as move:
+ move.product_id = product1
+ move.product_uom_qty = 15
+
+ picking = picking_form.save()
+ picking.action_confirm()
+
+ self.assertEqual(len(picking.move_lines), 1)
+ self.assertEqual(picking.move_lines.product_uom_qty, 25)