2018-01-16 11:28:15 +05:30
|
|
|
# -*- coding: utf-8 -*-
|
2018-01-16 02:34:37 -08:00
|
|
|
# Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details.
|
|
|
|
from flectra.exceptions import ValidationError
|
|
|
|
from flectra.tests.common import TransactionCase
|
2018-01-16 11:28:15 +05:30
|
|
|
|
|
|
|
|
|
|
|
class TestInventory(TransactionCase):
|
|
|
|
def setUp(self):
|
|
|
|
super(TestInventory, self).setUp()
|
|
|
|
self.stock_location = self.env.ref('stock.stock_location_stock')
|
|
|
|
self.pack_location = self.env.ref('stock.location_pack_zone')
|
|
|
|
self.pack_location.active = True
|
|
|
|
self.customer_location = self.env.ref('stock.stock_location_customers')
|
|
|
|
self.uom_unit = self.env.ref('product.product_uom_unit')
|
|
|
|
self.product1 = self.env['product.product'].create({
|
|
|
|
'name': 'Product A',
|
|
|
|
'type': 'product',
|
|
|
|
'categ_id': self.env.ref('product.product_category_all').id,
|
|
|
|
})
|
|
|
|
self.product2 = self.env['product.product'].create({
|
|
|
|
'name': 'Product A',
|
|
|
|
'type': 'product',
|
|
|
|
'tracking': 'serial',
|
|
|
|
'categ_id': self.env.ref('product.product_category_all').id,
|
|
|
|
})
|
|
|
|
|
|
|
|
def test_inventory_1(self):
|
|
|
|
""" Check that making an inventory adjustment to remove all products from stock is working
|
|
|
|
as expected.
|
|
|
|
"""
|
|
|
|
# make some stock
|
|
|
|
self.env['stock.quant']._update_available_quantity(self.product1, self.stock_location, 100)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product1, self.stock_location)), 1.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.stock_location), 100.0)
|
|
|
|
|
|
|
|
# remove them with an inventory adjustment
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 100)
|
|
|
|
inventory.line_ids.product_qty = 0 # Put the quantity back to 0
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
# check
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.stock_location), 0.0)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product1, self.stock_location)), 0.0)
|
|
|
|
|
|
|
|
def test_inventory_2(self):
|
|
|
|
""" Check that adding a tracked product through an inventory adjustment work as expected.
|
|
|
|
"""
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
'exhausted': True, # should be set by an onchange
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 0)
|
|
|
|
|
|
|
|
lot1 = self.env['stock.production.lot'].create({
|
|
|
|
'name': 'sn2',
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
})
|
|
|
|
|
|
|
|
inventory.line_ids.prod_lot_id = lot1
|
|
|
|
inventory.line_ids.product_qty = 1
|
|
|
|
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
# check
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product2, self.stock_location, lot_id=lot1), 1.0)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product2, self.stock_location, lot_id=lot1)), 1.0)
|
|
|
|
self.assertEqual(lot1.product_qty, 1.0)
|
|
|
|
|
|
|
|
def test_inventory_3(self):
|
|
|
|
""" Check that it's not posisble to have multiple products with a serial number through an
|
|
|
|
inventory adjustment
|
|
|
|
"""
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
'exhausted': True, # should be set by an onchange
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 0)
|
|
|
|
|
|
|
|
lot1 = self.env['stock.production.lot'].create({
|
|
|
|
'name': 'sn2',
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
})
|
|
|
|
|
|
|
|
inventory.line_ids.prod_lot_id = lot1
|
|
|
|
inventory.line_ids.product_qty = 2
|
|
|
|
|
|
|
|
with self.assertRaises(ValidationError):
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
def test_inventory_4(self):
|
|
|
|
""" Check that even if a product is tracked by serial number, it's possible to add
|
|
|
|
untracked one in an inventory adjustment.
|
|
|
|
"""
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
'exhausted': True, # should be set by an onchange
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 0)
|
|
|
|
|
|
|
|
lot1 = self.env['stock.production.lot'].create({
|
|
|
|
'name': 'sn2',
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
})
|
|
|
|
|
|
|
|
inventory.line_ids.prod_lot_id = lot1
|
|
|
|
inventory.line_ids.product_qty = 1
|
|
|
|
|
|
|
|
self.env['stock.inventory.line'].create({
|
|
|
|
'inventory_id': inventory.id,
|
|
|
|
'product_id': self.product2.id,
|
|
|
|
'product_uom_id': self.uom_unit.id,
|
|
|
|
'product_qty': 10,
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
})
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
# check
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product2, self.stock_location, lot_id=lot1, strict=True), 1.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product2, self.stock_location, strict=True), 10.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product2, self.stock_location), 11.0)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product2, self.stock_location, lot_id=lot1, strict=True)), 1.0)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product2, self.stock_location, strict=True)), 1.0)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product2, self.stock_location)), 2.0)
|
|
|
|
|
|
|
|
def test_inventory_5(self):
|
|
|
|
""" Check that assigning an owner does work.
|
|
|
|
"""
|
|
|
|
owner1 = self.env['res.partner'].create({'name': 'test_inventory_5'})
|
|
|
|
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
'exhausted': True,
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 0)
|
|
|
|
inventory.line_ids.partner_id = owner1
|
|
|
|
inventory.line_ids.product_qty = 5
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
quant = self.env['stock.quant']._gather(self.product1, self.stock_location)
|
|
|
|
self.assertEqual(len(quant), 1)
|
|
|
|
self.assertEqual(quant.quantity, 5)
|
|
|
|
self.assertEqual(quant.owner_id.id, owner1.id)
|
|
|
|
|
|
|
|
def test_inventory_6(self):
|
|
|
|
""" Test that for chained moves, making an inventory adjustment to reduce a quantity that
|
|
|
|
has been reserved correctly free the reservation. After that, add products in stock and check
|
|
|
|
that they're used if the user encodes more than what's available through the chain
|
|
|
|
"""
|
|
|
|
# add 10 products in stock
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'add 10 products 1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
'exhausted': True, # should be set by an onchange
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
inventory.line_ids.product_qty = 10
|
|
|
|
inventory.action_done()
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.stock_location), 10.0)
|
|
|
|
|
|
|
|
# Make a chain of two moves, validate the first and check that 10 products are reserved
|
|
|
|
# in the second one.
|
|
|
|
move_stock_pack = self.env['stock.move'].create({
|
|
|
|
'name': 'test_link_2_1',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'location_dest_id': self.pack_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
'product_uom': self.uom_unit.id,
|
|
|
|
'product_uom_qty': 10.0,
|
|
|
|
})
|
|
|
|
move_pack_cust = self.env['stock.move'].create({
|
|
|
|
'name': 'test_link_2_2',
|
|
|
|
'location_id': self.pack_location.id,
|
|
|
|
'location_dest_id': self.customer_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
'product_uom': self.uom_unit.id,
|
|
|
|
'product_uom_qty': 10.0,
|
|
|
|
})
|
|
|
|
move_stock_pack.write({'move_dest_ids': [(4, move_pack_cust.id, 0)]})
|
|
|
|
move_pack_cust.write({'move_orig_ids': [(4, move_stock_pack.id, 0)]})
|
|
|
|
(move_stock_pack + move_pack_cust)._action_confirm()
|
|
|
|
move_stock_pack._action_assign()
|
|
|
|
self.assertEqual(move_stock_pack.state, 'assigned')
|
|
|
|
move_stock_pack.move_line_ids.qty_done = 10
|
|
|
|
move_stock_pack._action_done()
|
|
|
|
self.assertEqual(move_stock_pack.state, 'done')
|
|
|
|
self.assertEqual(move_pack_cust.state, 'assigned')
|
|
|
|
self.assertEqual(self.env['stock.quant']._gather(self.product1, self.pack_location).quantity, 10.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.pack_location), 0.0)
|
|
|
|
|
|
|
|
# Make and inventory adjustment and remove two products from the pack location. This should
|
|
|
|
# free the reservation of the second move.
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove 2 products 1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.pack_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
inventory.line_ids.product_qty = 8
|
|
|
|
inventory.action_done()
|
|
|
|
self.assertEqual(self.env['stock.quant']._gather(self.product1, self.pack_location).quantity, 8.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.pack_location), 0)
|
|
|
|
self.assertEqual(move_pack_cust.state, 'partially_available')
|
|
|
|
self.assertEqual(move_pack_cust.reserved_availability, 8)
|
|
|
|
|
|
|
|
# If the user tries to assign again, only 8 products are available and thus the reservation
|
|
|
|
# state should not change.
|
|
|
|
move_pack_cust._action_assign()
|
|
|
|
self.assertEqual(move_pack_cust.state, 'partially_available')
|
|
|
|
self.assertEqual(move_pack_cust.reserved_availability, 8)
|
|
|
|
|
|
|
|
# Make a new inventory adjustment and bring two now products.
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'remove 2 products 1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.pack_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
inventory.line_ids.product_qty = 10
|
|
|
|
inventory.action_done()
|
|
|
|
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.pack_location), 2)
|
|
|
|
|
|
|
|
# Nothing should have changed for our pack move
|
|
|
|
self.assertEqual(move_pack_cust.state, 'partially_available')
|
|
|
|
self.assertEqual(move_pack_cust.reserved_availability, 8)
|
|
|
|
|
|
|
|
# Running _action_assign will now find the new available quantity. Indeed, as the products
|
|
|
|
# are not discernabl (not lot/pack/owner), even if the new available quantity is not directly
|
|
|
|
# brought by the chain, the system fill take them into account.
|
|
|
|
move_pack_cust._action_assign()
|
|
|
|
self.assertEqual(move_pack_cust.state, 'assigned')
|
|
|
|
|
|
|
|
# move all the things
|
|
|
|
move_pack_cust.move_line_ids.qty_done = 10
|
|
|
|
move_stock_pack._action_done()
|
|
|
|
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.pack_location), 0)
|
|
|
|
|
|
|
|
def test_inventory_7(self):
|
|
|
|
""" Check that duplicated quants create a single inventory line.
|
|
|
|
"""
|
|
|
|
owner1 = self.env['res.partner'].create({'name': 'test_inventory_7'})
|
|
|
|
vals = {
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
'product_uom_id': self.uom_unit.id,
|
|
|
|
'owner_id': owner1.id,
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'quantity': 1,
|
|
|
|
'reserved_quantity': 0,
|
|
|
|
}
|
|
|
|
self.env['stock.quant'].create(vals)
|
|
|
|
self.env['stock.quant'].create(vals)
|
|
|
|
self.assertEqual(len(self.env['stock.quant']._gather(self.product1, self.stock_location)), 2.0)
|
|
|
|
self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product1, self.stock_location), 2.0)
|
|
|
|
|
|
|
|
inventory = self.env['stock.inventory'].create({
|
|
|
|
'name': 'product1',
|
|
|
|
'filter': 'product',
|
|
|
|
'location_id': self.stock_location.id,
|
|
|
|
'product_id': self.product1.id,
|
|
|
|
})
|
|
|
|
inventory.action_start()
|
|
|
|
self.assertEqual(len(inventory.line_ids), 1)
|
|
|
|
self.assertEqual(inventory.line_ids.theoretical_qty, 2)
|
|
|
|
|