stock/tests/test_packing.py

1642 lines
77 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo.tests import Form
from odoo.tests.common import TransactionCase
from odoo.tools import float_round
from odoo.exceptions import UserError
class TestPackingCommon(TransactionCase):
@classmethod
def setUpClass(cls):
super(TestPackingCommon, cls).setUpClass()
cls.stock_location = cls.env.ref('stock.stock_location_stock')
cls.warehouse = cls.env['stock.warehouse'].search([('lot_stock_id', '=', cls.stock_location.id)], limit=1)
cls.warehouse.write({'delivery_steps': 'pick_pack_ship'})
cls.warehouse.int_type_id.reservation_method = 'manual'
cls.pack_location = cls.warehouse.wh_pack_stock_loc_id
cls.ship_location = cls.warehouse.wh_output_stock_loc_id
cls.customer_location = cls.env.ref('stock.stock_location_customers')
cls.picking_type_in = cls.env.ref('stock.picking_type_in')
cls.productA = cls.env['product.product'].create({'name': 'Product A', 'type': 'product'})
cls.productB = cls.env['product.product'].create({'name': 'Product B', 'type': 'product'})
class TestPacking(TestPackingCommon):
def test_put_in_pack(self):
""" In a pick pack ship scenario, create two packs in pick and check that
they are correctly recognised and handled by the pack and ship picking.
Along this test, we'll use action_toggle_processed to process a pack
from the entire_package_ids one2many and we'll directly fill the move
lines, the latter is the behavior when the user did not enable the display
of entire packs on the picking type.
"""
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20.0)
self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 20.0)
ship_move_a = self.env['stock.move'].create({
'name': 'The ship move',
'product_id': self.productA.id,
'product_uom_qty': 5.0,
'product_uom': self.productA.uom_id.id,
'location_id': self.ship_location.id,
'location_dest_id': self.customer_location.id,
'warehouse_id': self.warehouse.id,
'picking_type_id': self.warehouse.out_type_id.id,
'procure_method': 'make_to_order',
'state': 'draft',
})
ship_move_b = self.env['stock.move'].create({
'name': 'The ship move',
'product_id': self.productB.id,
'product_uom_qty': 5.0,
'product_uom': self.productB.uom_id.id,
'location_id': self.ship_location.id,
'location_dest_id': self.customer_location.id,
'warehouse_id': self.warehouse.id,
'picking_type_id': self.warehouse.out_type_id.id,
'procure_method': 'make_to_order',
'state': 'draft',
})
ship_move_a._assign_picking()
ship_move_b._assign_picking()
ship_move_a._action_confirm()
ship_move_b._action_confirm()
pack_move_a = ship_move_a.move_orig_ids[0]
pick_move_a = pack_move_a.move_orig_ids[0]
pick_picking = pick_move_a.picking_id
packing_picking = pack_move_a.picking_id
shipping_picking = ship_move_a.picking_id
pick_picking.picking_type_id.show_entire_packs = True
packing_picking.picking_type_id.show_entire_packs = True
shipping_picking.picking_type_id.show_entire_packs = True
pick_picking.action_assign()
self.assertEqual(len(pick_picking.move_ids_without_package), 2)
pick_picking.move_line_ids.filtered(lambda ml: ml.product_id == self.productA).quantity = 1.0
pick_picking.move_line_ids.filtered(lambda ml: ml.product_id == self.productB).quantity = 2.0
pick_picking.move_ids_without_package.picked = True
first_pack = pick_picking.action_put_in_pack()
self.assertEqual(len(pick_picking.package_level_ids), 1, 'Put some products in pack should create a package_level')
self.assertEqual(pick_picking.package_level_ids[0].state, 'new', 'A new pack should be in state "new"')
ml = pick_picking.move_line_ids[0].copy()
ml.write({
'quantity': 4.0,
'result_package_id': False,
})
ml = pick_picking.move_line_ids[1].copy()
ml.write({
'quantity': 3.0,
'result_package_id': False,
})
second_pack = pick_picking.action_put_in_pack()
self.assertEqual(len(pick_picking.move_ids_without_package), 2)
self.assertEqual(len(packing_picking.move_ids_without_package), 2)
pick_picking.move_ids.picked = True
pick_picking.button_validate()
self.assertEqual(len(packing_picking.move_ids_without_package), 2)
self.assertEqual(len(first_pack.quant_ids), 2)
self.assertEqual(len(second_pack.quant_ids), 2)
packing_picking.action_assign()
self.assertEqual(len(packing_picking.package_level_ids), 2, 'Two package levels must be created after assigning picking')
packing_picking.package_level_ids.write({'is_done': True})
packing_picking._action_done()
def test_pick_a_pack_confirm(self):
pack = self.env['stock.quant.package'].create({'name': 'The pack to pick'})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20.0, package_id=pack)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.int_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
picking.picking_type_id.show_entire_packs = True
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'company_id': picking.company_id.id,
})
self.assertEqual(package_level.state, 'draft',
'The package_level should be in draft as it has no moves, move lines and is not confirmed')
picking.action_confirm()
self.assertEqual(len(picking.move_ids_without_package), 0)
self.assertEqual(len(picking.move_ids), 1,
'One move should be created when the package_level has been confirmed')
self.assertEqual(len(package_level.move_ids), 1,
'The move should be in the package level')
self.assertEqual(package_level.state, 'confirmed',
'The package level must be state confirmed when picking is confirmed')
picking.action_assign()
self.assertEqual(len(picking.move_ids), 1,
'You still have only one move when the picking is assigned')
self.assertEqual(len(picking.move_ids.move_line_ids), 1,
'The move should have one move line which is the reservation')
self.assertEqual(picking.move_line_ids.package_level_id.id, package_level.id,
'The move line created should be linked to the package level')
self.assertEqual(picking.move_line_ids.package_id.id, pack.id,
'The move line must have been reserved on the package of the package_level')
self.assertEqual(picking.move_line_ids.result_package_id.id, pack.id,
'The move line must have the same package as result package')
self.assertEqual(package_level.state, 'assigned', 'The package level must be in state assigned')
package_level.write({'is_done': True})
self.assertEqual(len(package_level.move_line_ids), 1,
'The package level should still keep one move line after have been set to "done"')
self.assertEqual(package_level.move_line_ids[0].quantity, 20.0,
'All quantity in package must be procesed in move line')
picking.button_validate()
self.assertEqual(len(picking.move_ids), 1,
'You still have only one move when the picking is assigned')
self.assertEqual(len(picking.move_ids.move_line_ids), 1,
'The move should have one move line which is the reservation')
self.assertEqual(package_level.state, 'done', 'The package level must be in state done')
self.assertEqual(pack.location_id.id, picking.location_dest_id.id,
'The quant package must be in the destination location')
self.assertEqual(pack.quant_ids[0].location_id.id, picking.location_dest_id.id,
'The quant must be in the destination location')
def test_pick_a_pack_cancel(self):
"""Cancel a reserved operation with a not-done package level (is_done=False)."""
pack = self.env['stock.quant.package'].create({'name': 'The pack to pick'})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20.0, package_id=pack)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.int_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
picking.picking_type_id.show_entire_packs = True
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'location_dest_id': self.stock_location.id,
'company_id': picking.company_id.id,
})
picking.action_confirm()
picking.action_assign()
self.assertEqual(package_level.state, 'assigned')
self.assertTrue(package_level.move_line_ids)
picking.action_cancel()
self.assertEqual(package_level.state, 'cancel')
self.assertFalse(package_level.move_line_ids)
def test_pick_a_pack_cancel_is_done(self):
"""Cancel a reserved operation with a package level that is done (is_done=True)."""
pack = self.env['stock.quant.package'].create({'name': 'The pack to pick'})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20.0, package_id=pack)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.int_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
picking.picking_type_id.show_entire_packs = True
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'location_dest_id': self.stock_location.id,
'company_id': picking.company_id.id,
})
picking.action_confirm()
picking.action_assign()
self.assertEqual(package_level.state, 'assigned')
self.assertTrue(package_level.move_line_ids)
# By setting the package_level as 'done', all related lines will be kept
# when cancelling the transfer
package_level.is_done = True
picking.action_cancel()
self.assertEqual(picking.state, 'cancel')
self.assertEqual(package_level.state, 'cancel')
def test_multi_pack_reservation(self):
""" When we move entire packages, it is possible to have a multiple times
the same package in package level list, we make sure that only one is reserved,
and that the location_id of the package is the one where the package is once it
is reserved.
"""
pack = self.env['stock.quant.package'].create({'name': 'The pack to pick'})
shelf1_location = self.env['stock.location'].create({
'name': 'shelf1',
'usage': 'internal',
'location_id': self.stock_location.id,
})
self.env['stock.quant']._update_available_quantity(self.productA, shelf1_location, 20.0, package_id=pack)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.int_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'company_id': picking.company_id.id,
})
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'company_id': picking.company_id.id,
})
picking.action_confirm()
self.assertEqual(picking.package_level_ids.mapped('location_id.id'), [shelf1_location.id],
'The package levels should still in the same location after confirmation.')
picking.action_assign()
package_level_reserved = picking.package_level_ids.filtered(lambda pl: pl.state == 'assigned')
self.assertEqual(package_level_reserved.location_id.id, shelf1_location.id, 'The reserved package level must be reserved in shelf1')
picking.do_unreserve()
self.assertEqual(picking.package_level_ids.mapped('location_id.id'), [shelf1_location.id],
'The package levels should have back the original location.')
picking.package_level_ids.write({'is_done': True})
picking.action_assign()
package_level_reserved = picking.package_level_ids.filtered(lambda pl: pl.state == 'assigned')
self.assertEqual(package_level_reserved.location_id.id, shelf1_location.id, 'The reserved package level must be reserved in shelf1')
self.assertEqual(picking.package_level_ids.mapped('is_done'), [True, True], 'Both package should still done')
def test_put_in_pack_to_different_location(self):
""" Hitting 'Put in pack' button while some move lines go to different
location should trigger a wizard. This wizard applies the same destination
location to all the move lines
"""
self.warehouse.in_type_id.show_reserved = True
shelf1_location = self.env['stock.location'].create({
'name': 'shelf1',
'usage': 'internal',
'location_id': self.stock_location.id,
})
shelf2_location = self.env['stock.location'].create({
'name': 'shelf2',
'usage': 'internal',
'location_id': self.stock_location.id,
})
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': self.customer_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
ship_move_a = self.env['stock.move'].create({
'name': 'move 1',
'product_id': self.productA.id,
'product_uom_qty': 5.0,
'product_uom': self.productA.uom_id.id,
'location_id': self.customer_location.id,
'location_dest_id': shelf1_location.id,
'picking_id': picking.id,
'state': 'draft',
})
picking.action_confirm()
picking.action_assign()
picking.move_ids.filtered(lambda ml: ml.product_id == self.productA).picked = True
picking.action_put_in_pack()
pack1 = self.env['stock.quant.package'].search([], order='id')[-1]
picking.write({
'move_line_ids': [(0, 0, {
'product_id': self.productB.id,
'quantity': 7.0,
'product_uom_id': self.productB.uom_id.id,
'location_id': self.customer_location.id,
'location_dest_id': shelf2_location.id,
'picking_id': picking.id,
'state': 'confirmed',
})]
})
picking.write({
'move_line_ids': [(0, 0, {
'product_id': self.productA.id,
'quantity': 5.0,
'product_uom_id': self.productA.uom_id.id,
'location_id': self.customer_location.id,
'location_dest_id': shelf1_location.id,
'picking_id': picking.id,
'state': 'confirmed',
})]
})
picking.move_ids.picked = True
wizard_values = picking.action_put_in_pack()
wizard = self.env[(wizard_values.get('res_model'))].browse(wizard_values.get('res_id'))
wizard.location_dest_id = shelf2_location.id
wizard.action_done()
picking._action_done()
pack2 = self.env['stock.quant.package'].search([], order='id')[-1]
self.assertEqual(pack2.location_id.id, shelf2_location.id, 'The package must be stored in shelf2')
self.assertEqual(pack1.location_id.id, shelf1_location.id, 'The package must be stored in shelf1')
qp1 = pack2.quant_ids[0]
qp2 = pack2.quant_ids[1]
self.assertEqual(qp1.quantity + qp2.quantity, 12, 'The quant has not the good quantity')
def test_move_picking_with_package(self):
"""
355.4 rounded with 0.01 precision is 355.40000000000003.
check that nonetheless, moving a picking is accepted
"""
self.assertEqual(self.productA.uom_id.rounding, 0.01)
self.assertEqual(
float_round(355.4, precision_rounding=self.productA.uom_id.rounding),
355.40000000000003,
)
location_dict = {
'location_id': self.stock_location.id,
}
quant = self.env['stock.quant'].create({
**location_dict,
**{'product_id': self.productA.id, 'quantity': 355.4}, # important number
})
package = self.env['stock.quant.package'].create({
**location_dict, **{'quant_ids': [(6, 0, [quant.id])]},
})
location_dict.update({
'state': 'draft',
'location_dest_id': self.ship_location.id,
})
move = self.env['stock.move'].create({
**location_dict,
**{
'name': "XXX",
'product_id': self.productA.id,
'product_uom': self.productA.uom_id.id,
'product_uom_qty': 355.40000000000003, # other number
}})
picking = self.env['stock.picking'].create({
**location_dict,
**{
'picking_type_id': self.warehouse.in_type_id.id,
'move_ids': [(6, 0, [move.id])],
}})
picking.action_confirm()
picking.action_assign()
move.picked = True
picking._action_done()
# if we managed to get there, there was not any exception
# complaining that 355.4 is not 355.40000000000003. Good job!
def test_move_picking_with_package_2(self):
""" Generate two move lines going to different location in the same
package.
"""
shelf1 = self.env['stock.location'].create({
'location_id': self.stock_location.id,
'name': 'Shelf 1',
})
shelf2 = self.env['stock.location'].create({
'location_id': self.stock_location.id,
'name': 'Shelf 2',
})
package = self.env['stock.quant.package'].create({})
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.stock_location.id,
'state': 'draft',
})
self.env['stock.move.line'].create({
'location_id': self.stock_location.id,
'location_dest_id': shelf1.id,
'product_id': self.productA.id,
'product_uom_id': self.productA.uom_id.id,
'quantity': 5.0,
'picking_id': picking.id,
'result_package_id': package.id,
})
self.env['stock.move.line'].create({
'location_id': self.stock_location.id,
'location_dest_id': shelf2.id,
'product_id': self.productA.id,
'product_uom_id': self.productA.uom_id.id,
'quantity': 5.0,
'picking_id': picking.id,
'result_package_id': package.id,
})
picking.action_confirm()
picking.move_ids.picked = True
with self.assertRaises(UserError):
picking._action_done()
def test_pack_delivery_three_step_propagate_package_consumable(self):
""" Checks all works right in the following case:
* For a three-step delivery
* Put products in a package then validate the receipt.
* The automatically generated internal transfer should have package set by default.
"""
prod = self.env['product.product'].create({'name': 'bad dragon', 'type': 'consu'})
ship_move = self.env['stock.move'].create({
'name': 'The ship move',
'product_id': prod.id,
'product_uom_qty': 5.0,
'product_uom': prod.uom_id.id,
'location_id': self.warehouse.wh_output_stock_loc_id.id,
'location_dest_id': self.env.ref('stock.stock_location_customers').id,
'warehouse_id': self.warehouse.id,
'picking_type_id': self.warehouse.out_type_id.id,
'procure_method': 'make_to_order',
'state': 'draft',
})
# create chained pick/pack moves to test with
ship_move._assign_picking()
ship_move._action_confirm()
pack_move = ship_move.move_orig_ids[0]
pick_move = pack_move.move_orig_ids[0]
picking = pick_move.picking_id
picking.action_confirm()
picking.action_put_in_pack()
self.assertTrue(picking.move_line_ids.result_package_id)
picking.button_validate()
self.assertEqual(pack_move.move_line_ids.result_package_id, picking.move_line_ids.result_package_id)
def test_pack_in_receipt_two_step_single_putway(self):
""" Checks all works right in the following specific corner case:
* For a two-step receipt, receives two products using the same putaway
* Puts these products in a package then valid the receipt.
* Cancels the automatically generated internal transfer then create a new one.
* In this internal transfer, adds the package then valid it.
"""
grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
grp_multi_step_rule = self.env.ref('stock.group_adv_location')
grp_pack = self.env.ref('stock.group_tracking_lot')
self.env.user.write({'groups_id': [(3, grp_multi_loc.id)]})
self.env.user.write({'groups_id': [(3, grp_multi_step_rule.id)]})
self.env.user.write({'groups_id': [(3, grp_pack.id)]})
self.warehouse.reception_steps = 'two_steps'
# Settings of receipt.
self.warehouse.in_type_id.show_entire_packs = True
self.warehouse.in_type_id.show_reserved = True
# Settings of internal transfer.
self.warehouse.int_type_id.show_entire_packs = True
self.warehouse.int_type_id.show_reserved = True
# Creates two new locations for putaway.
location_form = Form(self.env['stock.location'])
location_form.name = 'Shelf A'
location_form.location_id = self.stock_location
loc_shelf_A = location_form.save()
# Creates a new putaway rule for productA and productB.
putaway_A = self.env['stock.putaway.rule'].create({
'product_id': self.productA.id,
'location_in_id': self.stock_location.id,
'location_out_id': loc_shelf_A.id,
})
putaway_B = self.env['stock.putaway.rule'].create({
'product_id': self.productB.id,
'location_in_id': self.stock_location.id,
'location_out_id': loc_shelf_A.id,
})
self.stock_location.putaway_rule_ids = [(4, putaway_A.id, 0), (4, putaway_B.id, 0)]
# Create a new receipt with the two products.
receipt_form = Form(self.env['stock.picking'])
receipt_form.picking_type_id = self.warehouse.in_type_id
# Add 2 lines
with receipt_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productA
move_line.product_uom_qty = 1
with receipt_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productB
move_line.product_uom_qty = 1
receipt = receipt_form.save()
receipt.action_confirm()
# Adds quantities then packs them and valids the receipt.
move_form = Form(receipt.move_ids_without_package[0], view='stock.view_stock_move_operations')
with move_form.move_line_ids.edit(0) as move_line:
move_line.quantity = 1
move_form = Form(receipt.move_ids_without_package[1], view='stock.view_stock_move_operations')
with move_form.move_line_ids.edit(0) as move_line:
move_line.quantity = 1
move_form.save()
receipt = receipt_form.save()
receipt.move_ids.picked = True
receipt.action_put_in_pack()
receipt.button_validate()
receipt_package = receipt.package_level_ids[0]
self.assertEqual(receipt_package.location_dest_id.id, receipt.location_dest_id.id)
self.assertEqual(
receipt_package.move_line_ids[0].location_dest_id.id,
receipt.location_dest_id.id)
self.assertEqual(
receipt_package.move_line_ids[1].location_dest_id.id,
receipt.location_dest_id.id)
# Checks an internal transfer was created following the validation of the receipt.
internal_transfer = self.env['stock.picking'].search([
('picking_type_id', '=', self.warehouse.int_type_id.id)
], order='id desc', limit=1)
self.assertEqual(internal_transfer.origin, receipt.name)
self.assertEqual(
len(internal_transfer.package_level_ids), 1)
internal_package = internal_transfer.package_level_ids[0]
self.assertNotEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
self.assertEqual(
internal_package.location_dest_id.id,
putaway_A.location_out_id.id,
"The package destination location must be the one from the putaway.")
self.assertEqual(
internal_package.move_line_ids[0].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the putaway.")
self.assertEqual(
internal_package.move_line_ids[1].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the putaway.")
# Cancels the internal transfer and creates a new one.
internal_transfer.action_cancel()
picking = self.env['stock.picking'].create({
'state': 'draft',
'picking_type_id': self.warehouse.int_type_id.id,
})
internal_form = Form(picking)
# The test specifically removes the ability to see the location fields
# grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
# self.env.user.write({'groups_id': [(3, grp_multi_loc.id)]})
# Hence, `internal_form.location_id` shouldn't be changed
with internal_form.package_level_ids.new() as pack_line:
pack_line.package_id = receipt_package.package_id
internal_transfer = internal_form.save()
# Checks the package fields have been correctly set.
internal_package = internal_transfer.package_level_ids[0]
self.assertEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
internal_transfer.action_assign()
self.assertNotEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
self.assertEqual(
internal_package.location_dest_id.id,
putaway_A.location_out_id.id,
"The package destination location must be the one from the putaway.")
self.assertEqual(
internal_package.move_line_ids[0].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the putaway.")
self.assertEqual(
internal_package.move_line_ids[1].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the putaway.")
internal_transfer.button_validate()
def test_pack_in_receipt_two_step_multi_putaway(self):
""" Checks all works right in the following specific corner case:
* For a two-step receipt, receives two products using two putaways
targeting different locations.
* Puts these products in a package then valid the receipt.
* Cancels the automatically generated internal transfer then create a new one.
* In this internal transfer, adds the package then valid it.
"""
grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
grp_multi_step_rule = self.env.ref('stock.group_adv_location')
grp_pack = self.env.ref('stock.group_tracking_lot')
self.env.user.write({'groups_id': [(3, grp_multi_loc.id)]})
self.env.user.write({'groups_id': [(3, grp_multi_step_rule.id)]})
self.env.user.write({'groups_id': [(3, grp_pack.id)]})
self.warehouse.reception_steps = 'two_steps'
# Settings of receipt.
self.warehouse.in_type_id.show_entire_packs = True
self.warehouse.in_type_id.show_reserved = True
# Settings of internal transfer.
self.warehouse.int_type_id.show_entire_packs = True
self.warehouse.int_type_id.show_reserved = True
# Creates two new locations for putaway.
location_form = Form(self.env['stock.location'])
location_form.name = 'Shelf A'
location_form.location_id = self.stock_location
loc_shelf_A = location_form.save()
location_form = Form(self.env['stock.location'])
location_form.name = 'Shelf B'
location_form.location_id = self.stock_location
loc_shelf_B = location_form.save()
# Creates a new putaway rule for productA and productB.
putaway_A = self.env['stock.putaway.rule'].create({
'product_id': self.productA.id,
'location_in_id': self.stock_location.id,
'location_out_id': loc_shelf_A.id,
})
putaway_B = self.env['stock.putaway.rule'].create({
'product_id': self.productB.id,
'location_in_id': self.stock_location.id,
'location_out_id': loc_shelf_B.id,
})
self.stock_location.putaway_rule_ids = [(4, putaway_A.id, 0), (4, putaway_B.id, 0)]
# location_form = Form(self.stock_location)
# location_form.putaway_rule_ids = [(4, putaway_A.id, 0), (4, putaway_B.id, 0), ],
# self.stock_location = location_form.save()
# Create a new receipt with the two products.
receipt_form = Form(self.env['stock.picking'])
receipt_form.picking_type_id = self.warehouse.in_type_id
# Add 2 lines
with receipt_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productA
move_line.product_uom_qty = 1
with receipt_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productB
move_line.product_uom_qty = 1
receipt = receipt_form.save()
receipt.action_confirm()
# Adds quantities then packs them and valids the receipt.
move_form = Form(receipt.move_ids_without_package[0], view='stock.view_stock_move_operations')
with move_form.move_line_ids.edit(0) as move_line:
move_line.quantity = 1
move_form = Form(receipt.move_ids_without_package[1], view='stock.view_stock_move_operations')
with move_form.move_line_ids.edit(0) as move_line:
move_line.quantity = 1
move_form.save()
receipt = receipt_form.save()
receipt.move_ids.picked = True
receipt.action_put_in_pack()
receipt.button_validate()
receipt_package = receipt.package_level_ids[0]
self.assertEqual(receipt_package.location_dest_id.id, receipt.location_dest_id.id)
self.assertEqual(
receipt_package.move_line_ids[0].location_dest_id.id,
receipt.location_dest_id.id)
self.assertEqual(
receipt_package.move_line_ids[1].location_dest_id.id,
receipt.location_dest_id.id)
# Checks an internal transfer was created following the validation of the receipt.
internal_transfer = self.env['stock.picking'].search([
('picking_type_id', '=', self.warehouse.int_type_id.id)
], order='id desc', limit=1)
self.assertEqual(internal_transfer.origin, receipt.name)
self.assertEqual(
len(internal_transfer.package_level_ids), 1)
internal_package = internal_transfer.package_level_ids[0]
self.assertEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
self.assertNotEqual(
internal_package.location_dest_id.id,
putaway_A.location_out_id.id,
"The package destination location must be the one from the picking.")
self.assertNotEqual(
internal_package.move_line_ids[0].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the picking.")
self.assertNotEqual(
internal_package.move_line_ids[1].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the picking.")
# Cancels the internal transfer and creates a new one.
internal_transfer.action_cancel()
picking = self.env['stock.picking'].create({
'state': 'draft',
'picking_type_id': self.warehouse.int_type_id.id,
})
internal_form = Form(picking)
# The test specifically removes the ability to see the location fields
# grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
# self.env.user.write({'groups_id': [(3, grp_multi_loc.id)]})
# Hence, `internal_form.location_id` shouldn't be changed
with internal_form.package_level_ids.new() as pack_line:
pack_line.package_id = receipt_package.package_id
internal_transfer = internal_form.save()
# Checks the package fields have been correctly set.
internal_package = internal_transfer.package_level_ids[0]
self.assertEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
internal_transfer.action_assign()
self.assertEqual(
internal_package.location_dest_id.id,
internal_transfer.location_dest_id.id)
self.assertNotEqual(
internal_package.location_dest_id.id,
putaway_A.location_out_id.id,
"The package destination location must be the one from the picking.")
self.assertNotEqual(
internal_package.move_line_ids[0].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the picking.")
self.assertNotEqual(
internal_package.move_line_ids[1].location_dest_id.id,
putaway_A.location_out_id.id,
"The move line destination location must be the one from the picking.")
internal_transfer.button_validate()
def test_partial_put_in_pack(self):
""" Create a simple move in a delivery. Reserve the quantity but set as quantity done only a part.
Call Put In Pack button. """
self.productA.tracking = 'lot'
lot1 = self.env['stock.lot'].create({
'product_id': self.productA.id,
'name': '00001',
'company_id': self.warehouse.company_id.id
})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20.0, lot_id=lot1)
ship_move_a = self.env['stock.move'].create({
'name': 'The ship move',
'product_id': self.productA.id,
'product_uom_qty': 5.0,
'product_uom': self.productA.uom_id.id,
'location_id': self.ship_location.id,
'location_dest_id': self.customer_location.id,
'warehouse_id': self.warehouse.id,
'picking_type_id': self.warehouse.out_type_id.id,
'procure_method': 'make_to_order',
'state': 'draft',
})
ship_move_a._assign_picking()
ship_move_a._action_confirm()
pack_move_a = ship_move_a.move_orig_ids[0]
pick_move_a = pack_move_a.move_orig_ids[0]
pick_picking = pick_move_a.picking_id
pick_picking.picking_type_id.show_entire_packs = True
pick_picking.action_assign()
pick_picking.move_line_ids.quantity = 3
first_pack = pick_picking.action_put_in_pack()
def test_action_assign_package_level(self):
"""calling _action_assign on move does not erase lines' "result_package_id"
At the end of the method ``StockMove._action_assign()``, the method
``StockPicking._check_entire_pack()`` is called. This method compares
the move lines with the quants of their source package, and if the entire
package is moved at once in the same transfer, a ``stock.package_level`` is
created. On creation of a ``stock.package_level``, the result package of
the move lines is directly updated with the entire package.
This is good on the first assign of the move, but when we call assign for
the second time on a move, for instance because it was made partially available
and we want to assign the remaining, it can override the result package we
selected before.
An override of ``StockPicking._check_move_lines_map_quant_package()`` ensures
that we ignore:
* picked lines (quantity > 0)
* lines with a different result package already
"""
package = self.env["stock.quant.package"].create({"name": "Src Pack"})
dest_package1 = self.env["stock.quant.package"].create({"name": "Dest Pack1"})
# Create new picking: 120 productA
picking = self.env['stock.picking'].create({
'state': 'draft',
'picking_type_id': self.warehouse.pick_type_id.id,
})
picking_form = Form(picking)
with picking_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productA
move_line.product_uom_qty = 120
picking = picking_form.save()
# mark as TO-DO
picking.action_confirm()
# Update quantity on hand: 100 units in package
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 100, package_id=package)
# Check Availability
picking.action_assign()
self.assertEqual(picking.state, "assigned")
self.assertEqual(picking.package_level_ids.package_id, package)
move = picking.move_ids
line = move.move_line_ids
# change the result package and set a quantity
line.quantity = 100
line.result_package_id = dest_package1
# Update quantity on hand: 20 units in new_package
new_package = self.env["stock.quant.package"].create({"name": "New Pack"})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 20, package_id=new_package)
# Check Availability
picking.action_assign()
# Check that result package is not changed on first line
new_line = move.move_line_ids - line
self.assertRecordValues(
line + new_line,
[
{"quantity": 100, "result_package_id": dest_package1.id},
{"quantity": 20, "result_package_id": new_package.id},
],
)
def test_entire_pack_overship(self):
"""
Test the scenario of overshipping: we send the customer an entire package, even though it might be more than
what they initially ordered, and update the quantity on the sales order to reflect what was actually sent.
"""
self.warehouse.delivery_steps = 'ship_only'
package = self.env["stock.quant.package"].create({"name": "Src Pack"})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 100, package_id=package)
self.warehouse.out_type_id.show_entire_packs = True
picking = self.env['stock.picking'].create({
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'picking_type_id': self.warehouse.out_type_id.id,
'state': 'draft',
})
with Form(picking) as picking_form:
with picking_form.move_ids_without_package.new() as move:
move.product_id = self.productA
move.product_uom_qty = 75
picking.action_confirm()
picking.action_assign()
with Form(picking) as picking_form:
with picking_form.package_level_ids.new() as package_level:
package_level.package_id = package
self.assertEqual(len(picking.move_ids), 1, 'Should have only 1 stock move')
self.assertEqual(len(picking.move_ids), 1, 'Should have only 1 stock move')
with Form(picking) as picking_form:
with picking_form.package_level_ids.edit(0) as package_level:
package_level.is_done = True
action = picking.button_validate()
self.assertEqual(action, True, 'Should not open wizard')
for ml in picking.move_line_ids:
self.assertEqual(ml.package_id, package, 'move_line.package')
self.assertEqual(ml.result_package_id, package, 'move_line.result_package')
self.assertEqual(ml.state, 'done', 'move_line.state')
quant = package.quant_ids.filtered(lambda q: q.location_id == self.customer_location)
self.assertEqual(len(quant), 1, 'Should have quant at customer location')
self.assertEqual(quant.reserved_quantity, 0, 'quant.reserved_quantity should = 0')
self.assertEqual(quant.quantity, 100.0, 'quant.quantity should = 100')
self.assertEqual(sum(ml.quantity for ml in picking.move_line_ids), 100.0, 'total move_line.quantity should = 100')
backorders = self.env['stock.picking'].search([('backorder_id', '=', picking.id)])
self.assertEqual(len(backorders), 0, 'Should not create a backorder')
def test_remove_package(self):
"""
In the overshipping scenario, if I remove the package after adding it, we should not remove the associated
stock move.
"""
self.warehouse.delivery_steps = 'ship_only'
package = self.env["stock.quant.package"].create({"name": "Src Pack"})
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 100, package_id=package)
self.warehouse.out_type_id.show_entire_packs = True
picking = self.env['stock.picking'].create({
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'picking_type_id': self.warehouse.out_type_id.id,
'state': 'draft',
})
with Form(picking) as picking_form:
with picking_form.move_ids_without_package.new() as move:
move.product_id = self.productA
move.product_uom_qty = 75
picking.action_assign()
with Form(picking) as picking_form:
with picking_form.package_level_ids.new() as package_level:
package_level.package_id = package
with Form(picking) as picking_form:
picking_form.package_level_ids.remove(0)
self.assertEqual(len(picking.move_ids), 1, 'Should have only 1 stock move')
def test_picking_state_with_null_qty(self):
""" Exclude empty stock move of the picking state computation """
delivery_form = Form(self.env['stock.picking'])
picking_type_id = self.warehouse.out_type_id
delivery_form.picking_type_id = picking_type_id
with delivery_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productA
move_line.product_uom_qty = 10
with delivery_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productB
move_line.product_uom_qty = 10
delivery = delivery_form.save()
delivery.action_confirm()
self.assertEqual(delivery.state, 'confirmed')
delivery.move_ids_without_package[1].product_uom_qty = 0
self.assertEqual(delivery.state, 'confirmed')
delivery_form = Form(self.env['stock.picking'])
picking_type_id = self.warehouse.out_type_id
delivery_form.picking_type_id = picking_type_id
with delivery_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productA
move_line.quantity = 10
with delivery_form.move_ids_without_package.new() as move_line:
move_line.product_id = self.productB
move_line.quantity = 10
delivery = delivery_form.save()
self.assertEqual(delivery.state, 'assigned')
delivery.move_ids_without_package[1].quantity = 0
self.assertEqual(delivery.state, 'assigned')
def test_2_steps_and_backorder(self):
""" When creating a backorder with a package, the latter should be reserved in the new picking. Moreover,
the initial picking shouldn't have any line about this package """
def create_picking(pick_type, from_loc, to_loc):
picking = self.env['stock.picking'].create({
'picking_type_id': pick_type.id,
'location_id': from_loc.id,
'location_dest_id': to_loc.id,
'state': 'draft',
})
move_A, move_B = 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': picking.id,
'location_id': from_loc.id,
'location_dest_id': to_loc.id,
}, {
'name': self.productB.name,
'product_id': self.productB.id,
'product_uom_qty': 1,
'product_uom': self.productB.uom_id.id,
'picking_id': picking.id,
'location_id': from_loc.id,
'location_dest_id': to_loc.id,
}])
picking.action_confirm()
picking.action_assign()
return picking, move_A, move_B
self.warehouse.delivery_steps = 'pick_ship'
pick_type = self.warehouse.pick_type_id
delivery_type = self.warehouse.out_type_id
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 1)
self.env['stock.quant']._update_available_quantity(self.productB, self.stock_location, 1)
picking, moveA, moveB = create_picking(pick_type, pick_type.default_location_src_id, pick_type.default_location_dest_id)
moveA.picked = True
picking.action_put_in_pack()
moveB.picked = True
picking.action_put_in_pack()
picking.button_validate()
delivery_type.show_entire_packs = True
picking, _, _ = create_picking(delivery_type, delivery_type.default_location_src_id, self.customer_location)
packB = picking.package_level_ids[1]
picking.package_level_ids_details[0].is_done = True
action_data = picking.button_validate()
backorder_wizard = Form(self.env['stock.backorder.confirmation'].with_context(action_data['context'])).save()
backorder_wizard.process()
bo = self.env['stock.picking'].search([('backorder_id', '=', picking.id)])
self.assertNotIn(packB, picking.package_level_ids)
self.assertEqual(packB, bo.package_level_ids)
self.assertEqual(bo.package_level_ids.state, 'assigned')
def test_package_and_sub_location(self):
"""
Suppose there are some products P available in shelf1, a child location of the pack location.
When moving these P to another child location of pack location, the source location of the
related package level should be shelf1
"""
shelf1_location = self.env['stock.location'].create({
'name': 'shelf1',
'usage': 'internal',
'location_id': self.pack_location.id,
})
shelf2_location = self.env['stock.location'].create({
'name': 'shelf2',
'usage': 'internal',
'location_id': self.pack_location.id,
})
pack = self.env['stock.quant.package'].create({'name': 'Super Package'})
self.env['stock.quant']._update_available_quantity(self.productA, shelf1_location, 20.0, package_id=pack)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': self.pack_location.id,
'location_dest_id': shelf2_location.id,
'state': 'draft',
})
package_level = self.env['stock.package_level'].create({
'package_id': pack.id,
'picking_id': picking.id,
'company_id': picking.company_id.id,
})
self.assertEqual(package_level.location_id, shelf1_location)
picking.action_confirm()
package_level.is_done = True
picking.button_validate()
self.assertEqual(package_level.location_id, shelf1_location)
def test_pack_in_receipt_two_step_multi_putaway_02(self):
"""
Suppose a product P, its weight is equal to 1kg
We have 100 x P on two pallets.
Receipt in two steps + Sub locations in WH/Stock + Storage Category
The Storage Category adds some constraints on weight/pallets capacity
"""
warehouse = self.stock_location.warehouse_id
warehouse.reception_steps = "two_steps"
self.picking_type_in.show_reserved = True
self.productA.weight = 1.0
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_stock_storage_categories').id)]})
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_stock_multi_locations').id)]})
# Required for `result_package_id` to be visible in the view
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_tracking_lot').id)]})
package_type = self.env['stock.package.type'].create({
'name': "Super Pallet",
})
package_01, package_02 = self.env['stock.quant.package'].create([{
'name': 'Pallet %s' % i,
'package_type_id': package_type.id,
} for i in [1, 2]])
# max 100kg (so 100 x P) and max 1 pallet -> we will work with pallets,
# so the pallet capacity constraint should be the effective one
stor_category = self.env['stock.storage.category'].create({
'name': 'Super Storage Category',
'max_weight': 100,
'package_capacity_ids': [(0, 0, {
'package_type_id': package_type.id,
'quantity': 1,
})]
})
# 3 sub locations with the storage category
# (the third location should never be used)
sub_loc_01, sub_loc_02, dummy = self.env['stock.location'].create([{
'name': 'Sub Location %s' % i,
'usage': 'internal',
'location_id': self.stock_location.id,
'storage_category_id': stor_category.id,
} for i in [1, 2, 3]])
self.env['stock.putaway.rule'].create({
'location_in_id': self.stock_location.id,
'location_out_id': self.stock_location.id,
'package_type_ids': [(4, package_type.id)],
'storage_category_id': stor_category.id,
})
# Receive 100 x P
receipt_picking = self.env['stock.picking'].create({
'picking_type_id': warehouse.in_type_id.id,
'location_id': self.env.ref('stock.stock_location_suppliers').id,
'location_dest_id': warehouse.wh_input_stock_loc_id.id,
'state': 'draft',
})
self.env['stock.move'].create({
'name': self.productA.name,
'product_id': self.productA.id,
'product_uom': self.productA.uom_id.id,
'product_uom_qty': 100.0,
'picking_id': receipt_picking.id,
'location_id': receipt_picking.location_id.id,
'location_dest_id': receipt_picking.location_dest_id.id,
})
receipt_picking.action_confirm()
# Distribute the products on two pallets, one with 49 x P and a second
# one with 51 x P (to easy the debugging in case of trouble)
move_form = Form(receipt_picking.move_ids, view="stock.view_stock_move_operations")
with move_form.move_line_ids.edit(0) as line:
line.quantity = 49
line.result_package_id = package_01
with move_form.move_line_ids.new() as line:
line.quantity = 51
line.result_package_id = package_02
move_form.save()
receipt_picking.move_ids.picked = True
receipt_picking.button_validate()
# We are in two-steps receipt -> check the internal picking
internal_picking = self.env['stock.picking'].search([], order='id desc', limit=1)
self.assertRecordValues(internal_picking.move_line_ids, [
{'quantity': 51, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_01.id},
{'quantity': 49, 'result_package_id': package_01.id, 'location_dest_id': sub_loc_02.id},
])
# Change the constraints of the storage category:
# max 75kg (so 75 x P) and max 2 pallet -> this time, the weight
# constraint should be the effective one
stor_category.max_weight = 75
stor_category.package_capacity_ids.quantity = 2
internal_picking.do_unreserve()
internal_picking.action_assign()
self.assertRecordValues(internal_picking.move_line_ids, [
{'quantity': 51, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_01.id},
{'quantity': 49, 'result_package_id': package_01.id, 'location_dest_id': sub_loc_02.id},
])
def test_pack_in_receipt_two_step_multi_putaway_03(self):
"""
Two sublocations (max 100kg, max 2 pallet)
Two products P1, P2, weight = 1kg
There are 10 x P1 on a pallet in the first sub location
Receive a pallet of 50 x P1 + 50 x P2 => because of weight constraint, should be redirected to the
second sub location
Then, same with max 200kg max 1 pallet => same result, this time because of pallet count constraint
"""
warehouse = self.stock_location.warehouse_id
warehouse.reception_steps = "two_steps"
self.picking_type_in.show_reserved = True
self.productA.weight = 1.0
self.productB.weight = 1.0
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_stock_storage_categories').id)]})
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_stock_multi_locations').id)]})
# Required for `result_package_id` to be visible in the view
self.env.user.write({'groups_id': [(4, self.env.ref('stock.group_tracking_lot').id)]})
package_type = self.env['stock.package.type'].create({
'name': "Super Pallet",
})
package_01, package_02 = self.env['stock.quant.package'].create([{
'name': 'Pallet %s' % i,
'package_type_id': package_type.id,
} for i in [1, 2]])
# max 100kg and max 2 pallets
stor_category = self.env['stock.storage.category'].create({
'name': 'Super Storage Category',
'max_weight': 100,
'package_capacity_ids': [(0, 0, {
'package_type_id': package_type.id,
'quantity': 2,
})]
})
# 3 sub locations with the storage category
# (the third location should never be used)
sub_loc_01, sub_loc_02, dummy = self.env['stock.location'].create([{
'name': 'Sub Location %s' % i,
'usage': 'internal',
'location_id': self.stock_location.id,
'storage_category_id': stor_category.id,
} for i in [1, 2, 3]])
self.env['stock.quant']._update_available_quantity(self.productA, sub_loc_01, 10, package_id=package_01)
self.env['stock.putaway.rule'].create({
'location_in_id': self.stock_location.id,
'location_out_id': self.stock_location.id,
'package_type_ids': [(4, package_type.id)],
'storage_category_id': stor_category.id,
})
# Receive 50 x P_A and 50 x P_B
receipt_picking = self.env['stock.picking'].create({
'picking_type_id': warehouse.in_type_id.id,
'location_id': self.env.ref('stock.stock_location_suppliers').id,
'location_dest_id': warehouse.wh_input_stock_loc_id.id,
'state': 'draft',
})
self.env['stock.move'].create([{
'name': p.name,
'product_id': p.id,
'product_uom': p.uom_id.id,
'product_uom_qty': 50,
'picking_id': receipt_picking.id,
'location_id': receipt_picking.location_id.id,
'location_dest_id': receipt_picking.location_dest_id.id,
} for p in [self.productA, self.productB]])
receipt_picking.action_confirm()
move_form = Form(receipt_picking.move_ids[0], view="stock.view_stock_move_operations")
with move_form.move_line_ids.edit(0) as line:
line.quantity = 50
line.result_package_id = package_02
move_form.save()
move_form = Form(receipt_picking.move_ids[1], view="stock.view_stock_move_operations")
with move_form.move_line_ids.edit(0) as line:
line.quantity = 50
line.result_package_id = package_02
move_form.save()
receipt_picking.move_ids.picked = True
receipt_picking.button_validate()
# We are in two-steps receipt -> check the internal picking
internal_picking = self.env['stock.picking'].search([], order='id desc', limit=1)
self.assertRecordValues(internal_picking.move_line_ids, [
{'product_id': self.productA.id, 'quantity': 50, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_02.id},
{'product_id': self.productB.id, 'quantity': 50, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_02.id},
])
# Change the constraints of the storage category:
# max 200kg and max 1 pallet
stor_category.max_weight = 200
stor_category.package_capacity_ids.quantity = 1
internal_picking.do_unreserve()
internal_picking.action_assign()
self.assertRecordValues(internal_picking.move_line_ids, [
{'product_id': self.productA.id, 'quantity': 50, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_02.id},
{'product_id': self.productB.id, 'quantity': 50, 'result_package_id': package_02.id, 'location_dest_id': sub_loc_02.id},
])
def test_pack_in_receipt_two_step_multi_putaway_04(self):
"""
Create a putaway rules for package type T and storage category SC. SC
only allows same products and has a maximum of 2 x T. Four SC locations
L1, L2, L3 and L4.
First, move a package that contains two different products: should not
redirect to L1/L2 because of the "same products" contraint.
Then, add one T-package (with product P01) at L1 and move 2 T-packages
(both with product P01): one should be redirected to L1 and the second
one to L2
Finally, move 3 T-packages (two with 1xP01, one with 1xP02): one P01
should be redirected to L2 and the second one to L3 (because of capacity
constraint), then P02 should be redirected to L4 (because of "same
product" policy)
"""
self.warehouse.reception_steps = "two_steps"
supplier_location = self.env.ref('stock.stock_location_suppliers')
input_location = self.warehouse.wh_input_stock_loc_id
package_type = self.env['stock.package.type'].create({
'name': "package type",
})
storage_category = self.env['stock.storage.category'].create({
'name': "storage category",
'allow_new_product': "same",
'max_weight': 1000,
'package_capacity_ids': [(0, 0, {
'package_type_id': package_type.id,
'quantity': 2,
})],
})
loc01, loc02, loc03, loc04 = self.env['stock.location'].create([{
'name': 'loc 0%d' % i,
'usage': 'internal',
'location_id': self.stock_location.id,
'storage_category_id': storage_category.id,
} for i in range(1, 5)])
self.env['stock.putaway.rule'].create({
'location_in_id': self.stock_location.id,
'location_out_id': self.stock_location.id,
'storage_category_id': storage_category.id,
'package_type_ids': [(4, package_type.id, 0)],
})
receipt = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'state': 'draft',
'move_ids': [(0, 0, {
'name': p.name,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'product_id': p.id,
'product_uom': p.uom_id.id,
'product_uom_qty': 1.0,
}) for p in (self.productA, self.productB)],
})
receipt.action_confirm()
moves = receipt.move_ids
moves.move_line_ids.quantity = 1
moves.move_line_ids.result_package_id = self.env['stock.quant.package'].create({'package_type_id': package_type.id})
moves.picked = True
receipt.button_validate()
internal_picking = moves.move_dest_ids.picking_id
self.assertEqual(internal_picking.move_line_ids.location_dest_id, self.stock_location,
'Storage location only accepts one same product. Here the package contains two different '
'products so it should not be redirected.')
internal_picking.action_cancel()
# Second test part
package = self.env['stock.quant.package'].create({'package_type_id': package_type.id})
self.env['stock.quant']._update_available_quantity(self.productA, loc01, 1.0, package_id=package)
receipt = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'move_ids': [(0, 0, {
'name': self.productA.name,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'product_id': self.productA.id,
'product_uom': self.productA.uom_id.id,
'product_uom_qty': 2.0,
})],
})
receipt.action_confirm()
receipt.do_unreserve()
self.env['stock.move.line'].create([{
'move_id': receipt.move_ids.id,
'quantity': 1,
'product_id': self.productA.id,
'product_uom_id': self.productA.uom_id.id,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'result_package_id': self.env['stock.quant.package'].create({'package_type_id': package_type.id}).id,
'picking_id': receipt.id,
} for _ in range(2)])
receipt.move_ids.picked = True
receipt.button_validate()
internal_transfer = receipt.move_ids.move_dest_ids.picking_id
self.assertEqual(internal_transfer.move_line_ids.location_dest_id, loc01 | loc02,
'There is already one package at L1, so the first SML should be redirected to L1 '
'and the second one to L2')
internal_transfer.move_line_ids.quantity = 1
internal_transfer.move_ids.picked = True
internal_transfer.button_validate()
# Third part (move 3 packages, 2 x P01 and 1 x P02)
receipt = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.in_type_id.id,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'move_ids': [(0, 0, {
'name': product.name,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'product_id': product.id,
'product_uom': product.uom_id.id,
'product_uom_qty': qty,
}) for qty, product in [(2.0, self.productA), (1.0, self.productB)]],
})
receipt.action_confirm()
receipt.do_unreserve()
moves = receipt.move_ids
self.env['stock.move.line'].create([{
'move_id': move.id,
'quantity': 1,
'product_id': product.id,
'product_uom_id': product.uom_id.id,
'location_id': supplier_location.id,
'location_dest_id': input_location.id,
'result_package_id': self.env['stock.quant.package'].create({'package_type_id': package_type.id}).id,
'picking_id': receipt.id,
} for product, move in [
(self.productA, moves[0]),
(self.productA, moves[0]),
(self.productB, moves[1]),
]])
receipt.move_ids.picked = True
receipt.button_validate()
internal_transfer = receipt.move_ids.move_dest_ids.picking_id
self.assertRecordValues(internal_transfer.move_line_ids, [
{'product_id': self.productA.id, 'quantity': 1.0, 'location_dest_id': loc02.id},
{'product_id': self.productA.id, 'quantity': 1.0, 'location_dest_id': loc03.id},
{'product_id': self.productB.id, 'quantity': 1.0, 'location_dest_id': loc04.id},
])
def test_rounding_and_reserved_qty(self):
"""
Basic use case: deliver a storable product put in two packages. This
test actually ensures that the process 'put in pack' handles some
possible issues with the floating point representation
"""
self.env['stock.quant']._update_available_quantity(self.productA, self.stock_location, 0.4)
picking = self.env['stock.picking'].create({
'picking_type_id': self.warehouse.out_type_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'move_ids': [(0, 0, {
'name': self.productA.name,
'product_id': self.productA.id,
'product_uom_qty': 0.4,
'product_uom': self.productA.uom_id.id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'picking_type_id': self.warehouse.out_type_id.id,
})],
'state': 'draft',
})
picking.action_confirm()
picking.move_line_ids.quantity = 0.3
picking.move_ids.picked = True
picking.action_put_in_pack()
ml = picking.move_line_ids.copy()
ml.write({
'quantity': 0.1,
'result_package_id': False,
})
picking.action_put_in_pack()
quant = self.env['stock.quant'].search([('product_id', '=', self.productA.id), ('location_id', '=', self.stock_location.id)])
self.assertEqual(quant.available_quantity, 0)
picking.button_validate()
self.assertEqual(picking.state, 'done')
self.assertEqual(picking.move_ids.quantity, 0.4)
self.assertEqual(len(picking.move_line_ids.result_package_id), 2)
def test_put_out_of_pack_transfer(self):
""" When a transfer has multiple products all in the same package, removing a product from the destination package
(i.e. removing it from the package but still putting it in the same location) shouldn't remove it for other products. """
loc_1 = self.env['stock.location'].create({
'name': 'Location A',
'location_id': self.stock_location.id,
})
loc_2 = self.env['stock.location'].create({
'name': 'Location B',
'location_id': self.stock_location.id,
})
pack = self.env['stock.quant.package'].create({'name': 'New Package'})
self.env['stock.quant']._update_available_quantity(self.productA, loc_1, 5, package_id=pack)
self.env['stock.quant']._update_available_quantity(self.productB, loc_1, 4, package_id=pack)
picking = self.env['stock.picking'].create({
'location_id': loc_1.id,
'location_dest_id': loc_2.id,
'picking_type_id': self.warehouse.int_type_id.id,
'state': 'draft',
})
moveA = self.env['stock.move'].create({
'name': self.productA.name,
'product_id': self.productA.id,
'product_uom_qty': 5,
'product_uom': self.productA.uom_id.id,
'picking_id': picking.id,
'location_id': loc_1.id,
'location_dest_id': loc_2.id,
})
moveB = self.env['stock.move'].create({
'name': self.productB.name,
'product_id': self.productB.id,
'product_uom_qty': 4,
'product_uom': self.productB.uom_id.id,
'picking_id': picking.id,
'location_id': loc_1.id,
'location_dest_id': loc_2.id,
})
# Check availabilities
picking.action_assign()
self.assertEqual(len(moveA.move_line_ids), 1, "A move line should have been created for the reservation of the package.")
self.assertEqual(moveA.move_line_ids.package_id.id, pack.id, "The package should have been reserved for both products.")
self.assertEqual(moveB.move_line_ids.package_id.id, pack.id, "The package should have been reserved for both products.")
pack_level = moveA.move_line_ids.package_level_id
# Remove the product A from the package in the destination.
moveA.move_line_ids.result_package_id = False
self.assertEqual(moveA.move_line_ids.result_package_id.id, False, "No package should be linked in the destination.")
self.assertEqual(moveA.move_line_ids.package_level_id.id, False, "Package level should have been unlinked from this move line.")
self.assertEqual(moveB.move_line_ids.result_package_id.id, pack.id, "Package should have stayed the same.")
self.assertEqual(moveB.move_line_ids.package_level_id.id, pack_level.id, "Package level should have stayed the same.")
# Validate the picking
picking.move_ids.picked = True
picking.button_validate()
# Check that the quants have their expected location/package/quantities
quantA = self.env['stock.quant'].search([('product_id', '=', self.productA.id), ('location_id', '=', loc_2.id)])
quantB = self.env['stock.quant'].search([('product_id', '=', self.productB.id), ('location_id', '=', loc_2.id)])
self.assertEqual(pack.location_id.id, loc_2.id, "Package should have been moved to Location B.")
self.assertEqual(quantA.quantity, 5, "All 5 units of product A should be in location B")
self.assertEqual(quantA.package_id.id, False, "There should be no package for product A as it was removed in the move.")
self.assertEqual(quantB.quantity, 4, "All 4 units of product B should be in location B")
self.assertEqual(quantB.package_id.id, pack.id, "Product B should still be in the initial package.")
def test_package_selection(self):
"""
Test that the package selection is correct when using the least_package_strategy:
- Pack 1 -> 10 unit, PAck 2 -> 10 unit, Pack 3 -> 20 unit
- SO 1 -> 20 unit, SO 2 -> 10 unit, SO 3 -> 10 unit
SO 1 should be in Pack 3, SO 2 in Pack 1 and SO 3 in Pack 2
"""
product = self.env['product.product'].create({
'name': 'Product',
'type': 'product',
})
# Set the removal strategy to 'least_packages'
least_package_strategy = self.env['product.removal'].search(
[('method', '=', 'least_packages')])
product.categ_id.removal_strategy_id = least_package_strategy.id
# Create three packages with different quantities: 10, 10 and 20
pack_1 = self.env['stock.quant.package'].create({
'name': 'Pack 1',
'quant_ids': [(0, 0, {
'product_id': product.id,
'quantity': 10,
'location_id': self.stock_location.id,
})],
})
pack_2 = self.env['stock.quant.package'].create({
'name': 'Pack 2',
'quant_ids': [(0, 0, {
'product_id': product.id,
'quantity': 10,
'location_id': self.stock_location.id,
})],
})
pack_3 = self.env['stock.quant.package'].create({
'name': 'Pack 3',
'quant_ids': [(0, 0, {
'product_id': product.id,
'quantity': 20,
'location_id': self.stock_location.id,
})],
})
# Create a quant without package to include none element in the selection of the package
self.env['stock.quant'].create({
'product_id': product.id,
'quantity': 5,
'location_id': self.stock_location.id,
})
# Check that the total quantity of the product is 40
self.assertEqual(product.qty_available, 45)
picking = self.env['stock.picking'].create({
'picking_type_id': self.env.ref('stock.picking_type_out').id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
move = self.env['stock.move'].create({
'name': product.name,
'product_id': product.id,
'product_uom': product.uom_id.id,
'picking_id': picking.id,
'product_uom_qty': 20,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
picking.action_confirm()
self.assertEqual(move.move_line_ids.result_package_id, pack_3)
picking_02 = self.env['stock.picking'].create({
'picking_type_id': self.env.ref('stock.picking_type_out').id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
move_02 = self.env['stock.move'].create({
'name': product.name,
'product_id': product.id,
'product_uom': product.uom_id.id,
'picking_id': picking_02.id,
'product_uom_qty': 10,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
picking_02.action_confirm()
self.assertEqual(move_02.move_line_ids.result_package_id, pack_1)
picking_03 = self.env['stock.picking'].create({
'picking_type_id': self.env.ref('stock.picking_type_out').id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
move_03 = self.env['stock.move'].create({
'name': product.name,
'product_id': product.id,
'product_uom': product.uom_id.id,
'picking_id': picking_03.id,
'product_uom_qty': 10,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
picking_03.action_confirm()
self.assertEqual(move_03.move_line_ids.result_package_id, pack_2)
def test_compute_hide_picking_type_multiple_records(self):
"""
Create two pickings and compute their respective hide picking types together.
"""
picking1 = self.env['stock.picking'].create({
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'picking_type_id': self.env.ref('stock.picking_type_out').id,
})
self.env['stock.move'].create({
'name': self.productA.name,
'product_id': self.productA.id,
'product_uom_qty': 10,
'product_uom': self.productA.uom_id.id,
'picking_id': picking1.id,
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
})
picking1.action_confirm()
picking2 = self.env['stock.picking'].create({
'location_id': self.stock_location.id,
'location_dest_id': self.customer_location.id,
'picking_type_id': self.env.ref('stock.picking_type_out').id,
})
(picking1 | picking2).with_context(default_picking_type_id=self.ref('stock.picking_type_out'))._compute_hide_picking_type()
self.assertTrue(picking1.hide_picking_type)
self.assertFalse(picking2.hide_picking_type)