# -*- coding: utf-8 -*- # Part of Odoo. See LICENSE file for full copyright and licensing details. from contextlib import closing from datetime import datetime, timedelta from unittest.mock import patch from odoo import fields from odoo.addons.mail.tests.common import mail_new_test_user from odoo.exceptions import ValidationError from odoo.tests.common import Form, TransactionCase from odoo.exceptions import AccessError, RedirectWarning, UserError class StockQuant(TransactionCase): @classmethod def setUpClass(cls): super(StockQuant, cls).setUpClass() cls.demo_user = mail_new_test_user( cls.env, name='Pauline Poivraisselle', login='pauline', email='p.p@example.com', notification_type='inbox', groups='base.group_user', ) cls.stock_user = mail_new_test_user( cls.env, name='Pauline Poivraisselle', login='pauline2', email='p.p@example.com', notification_type='inbox', groups='stock.group_stock_user', ) cls.product = cls.env['product.product'].create({ 'name': 'Product A', 'type': 'product', }) cls.product_lot = cls.env['product.product'].create({ 'name': 'Product A', 'type': 'product', 'tracking': 'lot', }) cls.product_consu = cls.env['product.product'].create({ 'name': 'Product A', 'type': 'consu', }) cls.product_serial = cls.env['product.product'].create({ 'name': 'Product A', 'type': 'product', 'tracking': 'serial', }) cls.stock_location = cls.env['stock.location'].create({ 'name': 'stock_location', 'usage': 'internal', }) cls.stock_subloc3 = cls.env['stock.location'].create({ 'name': 'subloc3', 'usage': 'internal', 'location_id': cls.stock_location.id }) cls.stock_subloc2 = cls.env['stock.location'].create({ 'name': 'subloc2', 'usage': 'internal', 'location_id': cls.stock_location.id, }) def gather_relevant(self, product_id, location_id, lot_id=None, package_id=None, owner_id=None, strict=False): quants = self.env['stock.quant']._gather(product_id, location_id, lot_id=lot_id, package_id=package_id, owner_id=owner_id, strict=strict) return quants.filtered(lambda q: not (q.quantity == 0 and q.reserved_quantity == 0)) def test_get_available_quantity_1(self): """ Quantity availability with only one quant in a location. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 1.0) def test_get_available_quantity_2(self): """ Quantity availability with multiple quants in a location. """ for i in range(3): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 3.0) def test_get_available_quantity_3(self): """ Quantity availability with multiple quants (including negatives ones) in a location. """ for i in range(3): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': -3.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) def test_get_available_quantity_4(self): """ Quantity availability with no quants in a location. """ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) def test_get_available_quantity_5(self): """ Quantity availability with multiple partially reserved quants in a location. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 10.0, 'reserved_quantity': 9.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, 'reserved_quantity': 1.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 1.0) def test_get_available_quantity_6(self): """ Quantity availability with multiple partially reserved quants in a location. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 10.0, 'reserved_quantity': 20.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 5.0, 'reserved_quantity': 0.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location, allow_negative=True), -5.0) def test_get_available_quantity_7(self): """ Quantity availability with only one tracked quant in a location. """ lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_lot.id, 'company_id': self.env.company.id, }) self.env['stock.quant'].create({ 'product_id': self.product_lot.id, 'location_id': self.stock_location.id, 'quantity': 10.0, 'reserved_quantity': 20.0, 'lot_id': lot1.id, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_lot, self.stock_location, lot_id=lot1), 0.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_lot, self.stock_location, lot_id=lot1, allow_negative=True), -10.0) def test_get_available_quantity_8(self): """ Quantity availability with a consumable product. """ self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_consu, self.stock_location), 0.0) self.assertEqual(len(self.gather_relevant(self.product_consu, self.stock_location)), 0) with self.assertRaises(ValidationError): self.env['stock.quant']._update_available_quantity(self.product_consu, self.stock_location, 1.0) def test_get_available_quantity_9(self): """ Quantity availability by a demo user with access rights/rules. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.env = self.env(user=self.demo_user) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 1.0) def test_increase_available_quantity_1(self): """ Increase the available quantity when no quants are already in a location. """ self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 1.0) def test_increase_available_quantity_2(self): """ Increase the available quantity when multiple quants are already in a location. """ for i in range(2): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 2.0) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 3.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 2) def test_increase_available_quantity_3(self): """ Increase the available quantity when a concurrent transaction is already increasing the reserved quanntity for the same product. """ quant = self.env['stock.quant'].search([('location_id', '=', self.stock_location.id)], limit=1) if not quant: self.skipTest('Cannot test concurrent transactions without demo data.') product = quant.product_id available_quantity = self.env['stock.quant']._get_available_quantity(product, self.stock_location, allow_negative=True) # opens a new cursor and SELECT FOR UPDATE the quant, to simulate another concurrent reserved # quantity increase with closing(self.registry.cursor()) as cr: cr.execute("SELECT id FROM stock_quant WHERE product_id=%s AND location_id=%s", (product.id, self.stock_location.id)) quant_id = cr.fetchone() cr.execute("SELECT 1 FROM stock_quant WHERE id=%s FOR UPDATE", quant_id) self.env['stock.quant']._update_available_quantity(product, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(product, self.stock_location, allow_negative=True), available_quantity + 1) self.assertEqual(len(self.gather_relevant(product, self.stock_location, strict=True)), 2) def test_increase_available_quantity_4(self): """ Increase the available quantity when no quants are already in a location with a user without access right. """ self.env = self.env(user=self.demo_user) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) def test_increase_available_quantity_5(self): """ Increase the available quantity when no quants are already in stock. Increase a subLocation and check that quants are in this location. Also test inverse. """ stock_sub_location = self.stock_location.child_ids[0] product2 = self.env['product.product'].create({ 'name': 'Product B', 'type': 'product', }) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) self.env['stock.quant']._update_available_quantity(self.product, stock_sub_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, stock_sub_location), 1.0) self.env['stock.quant']._update_available_quantity(product2, stock_sub_location, 1.0) self.env['stock.quant']._update_available_quantity(product2, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(product2, self.stock_location), 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(product2, stock_sub_location), 1.0) def test_increase_available_quantity_6(self): """ Increasing the available quantity in a view location should be forbidden. """ location1 = self.env['stock.location'].create({ 'name': 'viewloc1', 'usage': 'view', 'location_id': self.stock_location.id, }) with self.assertRaises(ValidationError): self.env['stock.quant']._update_available_quantity(self.product, location1, 1.0) def test_increase_available_quantity_7(self): """ Setting a location's usage as "view" should be forbidden if it already contains quant. """ self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) self.assertTrue(len(self.stock_location.quant_ids.ids) > 0) with self.assertRaises(UserError): self.stock_location.usage = 'view' def test_decrease_available_quantity_1(self): """ Decrease the available quantity when no quants are already in a location. """ self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, -1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location, allow_negative=True), -1.0) def test_decrease_available_quantity_2(self): """ Decrease the available quantity when multiple quants are already in a location. """ for i in range(2): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 2.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 2) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, -1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 1.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 1) def test_decrease_available_quantity_3(self): """ Decrease the available quantity when a concurrent transaction is already increasing the reserved quanntity for the same product. """ quant = self.env['stock.quant'].search([('location_id', '=', self.stock_location.id)], limit=1) if not quant: self.skipTest('Cannot test concurrent transactions without demo data.') product = quant.product_id available_quantity = self.env['stock.quant']._get_available_quantity(product, self.stock_location, allow_negative=True) # opens a new cursor and SELECT FOR UPDATE the quant, to simulate another concurrent reserved # quantity increase with closing(self.registry.cursor()) as cr: cr.execute("SELECT 1 FROM stock_quant WHERE id = %s FOR UPDATE", quant.ids) self.env['stock.quant']._update_available_quantity(product, self.stock_location, -1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(product, self.stock_location, allow_negative=True), available_quantity - 1) self.assertEqual(len(self.gather_relevant(product, self.stock_location, strict=True)), 2) def test_decrease_available_quantity_4(self): """ Decrease the available quantity that delete the quant. The active user should have read,write and unlink rights """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.env = self.env(user=self.demo_user) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, -1.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 0) def test_increase_reserved_quantity_1(self): """ Increase the reserved quantity of quantity x when there's a single quant in a given location which has an available quantity of x. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 10.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 10.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 1) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 10.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 1) def test_increase_reserved_quantity_2(self): """ Increase the reserved quantity of quantity x when there's two quants in a given location which have an available quantity of x together. """ for i in range(2): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 5.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 10.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 2) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 10.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 2) def test_increase_reserved_quantity_3(self): """ Increase the reserved quantity of quantity x when there's multiple quants in a given location which have an available quantity of x together. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 5.0, 'reserved_quantity': 2.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 10.0, 'reserved_quantity': 12.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 8.0, 'reserved_quantity': 3.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 35.0, 'reserved_quantity': 12.0, }) # total quantity: 58 # total reserved quantity: 29 self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 29.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 4) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 10.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 19.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 4) def test_increase_reserved_quantity_4(self): """ Increase the reserved quantity of quantity x when there's multiple quants in a given location which have an available quantity of x together. """ self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 5.0, 'reserved_quantity': 7.0, }) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 12.0, 'reserved_quantity': 10.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 2) with self.assertRaises(UserError): self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 10.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) def test_increase_reserved_quantity_5(self): """ Decrease the available quantity when no quant are in a location. """ with self.assertRaises(UserError): self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) def test_decrease_reserved_quantity_1(self): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 10.0, 'reserved_quantity': 10.0, }) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 1) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, -10.0, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 10.0) self.assertEqual(len(self.gather_relevant(self.product, self.stock_location)), 1) def test_increase_decrease_reserved_quantity_1(self): """ Decrease then increase reserved quantity when no quant are in a location. """ with self.assertRaises(UserError): self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) with self.assertRaises(RedirectWarning): self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, -1.0, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) def test_action_done_1(self): pack_location = self.env.ref('stock.location_pack_zone') pack_location.active = True self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 2.0) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, -2.0, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 2.0) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, -2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, self.stock_location), 0.0) self.env['stock.quant']._update_available_quantity(self.product, pack_location, 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product, pack_location), 2.0) def test_mix_tracked_untracked_1(self): lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) # add one tracked, one untracked self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0) self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot1) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location), 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, strict=True), 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, lot_id=lot1), 2.0) self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot1, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location), 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, strict=True), 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, lot_id=lot1), 1.0) self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, -1.0, lot_id=lot1, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location), 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, strict=True), 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, lot_id=lot1), 2.0) with self.assertRaises(RedirectWarning): self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, -1.0, strict=True) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location), 2.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, strict=True), 1.0) self.assertEqual(self.env['stock.quant']._get_available_quantity(self.product_serial, self.stock_location, lot_id=lot1), 2.0) def test_access_rights_1(self): """ Directly update the quant with a user with or without stock access rights should not raise an AccessError only deletion will. """ quant = self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) self.env = self.env(user=self.demo_user) with self.assertRaises(AccessError): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) with self.assertRaises(AccessError): quant.with_user(self.demo_user).write({'quantity': 2.0}) with self.assertRaises(UserError): quant.with_user(self.demo_user).unlink() self.env = self.env(user=self.stock_user) self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) quant.with_user(self.stock_user).with_context(inventory_mode=True).write({'inventory_quantity': 3.0}) with self.assertRaises(AccessError): quant.with_user(self.stock_user).unlink() def test_in_date_1(self): """ Check that no incoming date is set when updating the quantity of an untracked quant. """ quantity, in_date = self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) self.assertEqual(quantity, 1) self.assertNotEqual(in_date, None) def test_in_date_1b(self): self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_location.id, 'quantity': 1.0, }) quantity, in_date = self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 2.0) self.assertEqual(quantity, 3) self.assertNotEqual(in_date, None) def test_in_date_2(self): """ Check that an incoming date is correctly set when updating the quantity of a tracked quant. """ lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) quantity, in_date = self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot1) self.assertEqual(quantity, 1) self.assertNotEqual(in_date, None) def test_in_date_3(self): """ Check that the FIFO strategies correctly applies when you have multiple lot received at different times for a tracked product. """ lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) lot2 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) in_date_lot1 = datetime.now() in_date_lot2 = datetime.now() - timedelta(days=5) self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot1, in_date=in_date_lot1) self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot2, in_date=in_date_lot2) quants = self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, 1) # Default removal strategy is FIFO, so lot2 should be received as it was received earlier. self.assertEqual(quants[0][0].lot_id.id, lot2.id) def test_in_date_4(self): """ Check that the LIFO strategies correctly applies when you have multiple lot received at different times for a tracked product. """ lifo_strategy = self.env['product.removal'].search([('method', '=', 'lifo')]) self.stock_location.removal_strategy_id = lifo_strategy lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) lot2 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) in_date_lot1 = datetime.now() in_date_lot2 = datetime.now() - timedelta(days=5) self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot1, in_date=in_date_lot1) self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_location, 1.0, lot_id=lot2, in_date=in_date_lot2) quants = self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, 1) # Removal strategy is LIFO, so lot1 should be received as it was received later. self.assertEqual(quants[0][0].lot_id.id, lot1.id) def test_in_date_5(self): """ Receive the same lot at different times, once they're in the same location, the quants are merged and only the earliest incoming date is kept. """ lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_lot.id, 'company_id': self.env.company.id, }) from odoo.fields import Datetime in_date1 = Datetime.now() self.env['stock.quant']._update_available_quantity(self.product_lot, self.stock_location, 1.0, lot_id=lot1, in_date=in_date1) quant = self.env['stock.quant'].search([ ('product_id', '=', self.product_lot.id), ('location_id', '=', self.stock_location.id), ]) self.assertEqual(len(quant), 1) self.assertEqual(quant.quantity, 1) self.assertEqual(quant.lot_id.id, lot1.id) self.assertEqual(quant.in_date, in_date1) in_date2 = Datetime.now() - timedelta(days=5) self.env['stock.quant']._update_available_quantity(self.product_lot, self.stock_location, 1.0, lot_id=lot1, in_date=in_date2) quant = self.env['stock.quant'].search([ ('product_id', '=', self.product_lot.id), ('location_id', '=', self.stock_location.id), ]) self.assertEqual(len(quant), 1) self.assertEqual(quant.quantity, 2) self.assertEqual(quant.lot_id.id, lot1.id) self.assertEqual(quant.in_date, in_date2) def test_closest_removal_strategy_tracked(self): """ Check that the Closest location strategy correctly applies when you have multiple lot received at different locations for a tracked product. """ closest_strategy = self.env['product.removal'].search([('method', '=', 'closest')]) self.stock_location.removal_strategy_id = closest_strategy lot1 = self.env['stock.lot'].create({ 'name': 'lot1', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) lot2 = self.env['stock.lot'].create({ 'name': 'lot2', 'product_id': self.product_serial.id, 'company_id': self.env.company.id, }) in_date = datetime.now() # Add a product from lot1 in stock_location/subloc2 self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_subloc2, 1.0, lot_id=lot1, in_date=in_date) # Add a product from lot2 in stock_location/subloc3 self.env['stock.quant']._update_available_quantity(self.product_serial, self.stock_subloc3, 1.0, lot_id=lot2, in_date=in_date) # Require one unit of the product quants = self.env['stock.quant']._update_reserved_quantity(self.product_serial, self.stock_location, 1) # Default removal strategy is 'Closest location', so lot1 should be received as it was put in a closer location. (stock_location/subloc2 < stock_location/subloc3) self.assertEqual(quants[0][0].lot_id.id, lot1.id) def test_closest_removal_strategy_untracked(self): """ Check that the Closest location strategy correctly applies when you have multiple products received at different locations for untracked products.""" closest_strategy = self.env['product.removal'].search([('method', '=', 'closest')]) self.stock_location.removal_strategy_id = closest_strategy # Add 2 units of product into stock_location/subloc2 self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_subloc2.id, 'quantity': 2.0, }) # Add 3 units of product into stock_location/subloc3 self.env['stock.quant'].create({ 'product_id': self.product.id, 'location_id': self.stock_subloc3.id, 'quantity': 3.0 }) # Request 3 units of product, with 'Closest location' as removal strategy quants = self.env['stock.quant']._update_reserved_quantity(self.product, self.stock_location, 3) # The 2 in stock_location/subloc2 should be taken first, as the location name is smaller alphabetically self.assertEqual(quants[0][0].reserved_quantity, 2) # The last one should then be taken in stock_location/subloc3 since the first location doesn't have enough products self.assertEqual(quants[1][0].reserved_quantity, 1) def test_in_date_6(self): """ One P in stock, P is delivered. Later on, a stock adjustement adds one P. This test checks the date value of the related quant """ self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0) move = self.env['stock.move'].create({ 'name': 'OUT 1 product', 'product_id': self.product.id, 'product_uom_qty': 1, 'product_uom': self.product.uom_id.id, 'location_id': self.stock_location.id, 'location_dest_id': self.ref('stock.stock_location_customers'), }) move._action_confirm() move._action_assign() move.quantity_done = 1 move._action_done() tomorrow = fields.Datetime.now() + timedelta(days=1) with patch.object(fields.Datetime, 'now', lambda: tomorrow): move = self.env['stock.move'].create({ 'name': 'IN 1 product', 'product_id': self.product.id, 'product_uom_qty': 1, 'product_uom': self.product.uom_id.id, 'location_id': self.ref('stock.stock_location_suppliers'), 'location_dest_id': self.stock_location.id, }) move._action_confirm() move._action_assign() move.quantity_done = 1 move._action_done() quant = self.env['stock.quant'].search([('product_id', '=', self.product.id), ('location_id', '=', self.stock_location.id), ('quantity', '>', 0)]) self.assertEqual(quant.in_date, tomorrow) def test_quant_creation(self): """ This test ensures that, after an internal transfer, the values of the created quand are correct """ self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 10.0) move = self.env['stock.move'].create({ 'name': 'Move 1 product', 'product_id': self.product.id, 'product_uom_qty': 1, 'product_uom': self.product.uom_id.id, 'location_id': self.stock_location.id, 'location_dest_id': self.stock_subloc2.id, }) move._action_confirm() move._action_assign() move.quantity_done = 1 move._action_done() quant = self.gather_relevant(self.product, self.stock_subloc2) self.assertFalse(quant.inventory_quantity_set) def test_unpack_and_quants_merging(self): """ When unpacking a package, if there are already some quantities of the packed product in the stock, the quant of the on hand quantity and the one of the package should be merged """ stock_location = self.env['stock.warehouse'].search([], limit=1).lot_stock_id supplier_location = self.env.ref('stock.stock_location_suppliers') picking_type_in = self.env.ref('stock.picking_type_in') self.env['stock.quant']._update_available_quantity(self.product, stock_location, 1.0) picking = self.env['stock.picking'].create({ 'picking_type_id': picking_type_in.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'move_ids': [(0, 0, { 'name': 'In 10 x %s' % self.product.name, 'product_id': self.product.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'product_uom_qty': 10, 'product_uom': self.product.uom_id.id, })], }) picking.action_confirm() package = self.env['stock.quant.package'].create({ 'name': 'Super Package', }) picking.move_ids.move_line_ids.write({ 'qty_done': 10, 'result_package_id': package.id, }) picking.button_validate() package.unpack() quant = self.env['stock.quant'].search([('product_id', '=', self.product.id), ('on_hand', '=', True)]) self.assertEqual(len(quant), 1) # The quants merging is processed thanks to a SQL query (see StockQuant._merge_quants). # At that point, the ORM is not aware of the new value. So we need to invalidate the # cache to ensure that the value will be the newest quant.invalidate_recordset(['quantity']) self.assertEqual(quant.quantity, 11) def test_clean_quant_after_package_move(self): """ A product is at WH/Stock in a package PK. We deliver PK. The user should not find any quant at WH/Stock with PK anymore. """ package = self.env['stock.quant.package'].create({}) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0, package_id=package) move = self.env['stock.move'].create({ 'name': 'OUT 1 product', 'product_id': self.product.id, 'product_uom_qty': 1, 'product_uom': self.product.uom_id.id, 'location_id': self.stock_location.id, 'location_dest_id': self.ref('stock.stock_location_customers'), }) move._action_confirm() move._action_assign() move.move_line_ids.write({ 'result_package_id': package.id, 'qty_done': 1, }) move._action_done() self.assertFalse(self.env['stock.quant'].search_count([ ('product_id', '=', self.product.id), ('package_id', '=', package.id), ('location_id', '=', self.stock_location.id), ])) def test_serial_constraint_with_package_and_return(self): """ Receive product with serial S Return it in a package Confirm a new receipt with S """ stock_location = self.env['stock.warehouse'].search([], limit=1).lot_stock_id supplier_location = self.env.ref('stock.stock_location_suppliers') picking_type_in = self.env.ref('stock.picking_type_in') receipt01 = self.env['stock.picking'].create({ 'picking_type_id': picking_type_in.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'move_ids': [(0, 0, { 'name': self.product_serial.name, 'product_id': self.product_serial.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'product_uom_qty': 1, 'product_uom': self.product_serial.uom_id.id, })], }) receipt01.action_confirm() receipt01.move_line_ids.write({ 'lot_name': 'Michel', 'qty_done': 1.0 }) receipt01.button_validate() quant = self.env['stock.quant'].search([('product_id', '=', self.product_serial.id), ('location_id', '=', stock_location.id)]) wizard_form = Form(self.env['stock.return.picking'].with_context(active_ids=receipt01.ids, active_id=receipt01.ids[0], active_model='stock.picking')) wizard = wizard_form.save() wizard.product_return_moves.quantity = 1.0 stock_return_picking_action = wizard.create_returns() return_pick = self.env['stock.picking'].browse(stock_return_picking_action['res_id']) return_pick.move_ids.move_line_ids.qty_done = 1.0 return_pick.action_put_in_pack() return_pick._action_done() self.assertEqual(return_pick.move_line_ids.lot_id, quant.lot_id) self.assertTrue(return_pick.move_line_ids.result_package_id, quant.lot_id) receipt02 = self.env['stock.picking'].create({ 'picking_type_id': picking_type_in.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'move_ids': [(0, 0, { 'name': self.product_serial.name, 'product_id': self.product_serial.id, 'location_id': supplier_location.id, 'location_dest_id': stock_location.id, 'product_uom_qty': 1, 'product_uom': self.product_serial.uom_id.id, })], }) receipt02.action_confirm() receipt02.move_line_ids.write({ 'lot_name': 'Michel', 'qty_done': 1.0 }) receipt02.button_validate() quant = self.env['stock.quant'].search([('product_id', '=', self.product_serial.id), ('location_id', '=', stock_location.id)]) self.assertEqual(len(quant), 1) self.assertEqual(quant.lot_id.name, 'Michel') def test_update_quant_with_forbidden_field(self): """ Test that updating a quant with a forbidden field raise an error. """ product = self.env['product.product'].create({ 'name': 'Product', 'type': 'product', 'tracking': 'serial', }) sn1 = self.env['stock.lot'].create({ 'name': 'SN1', 'product_id': product.id, }) self.env['stock.quant']._update_available_quantity(product, self.stock_subloc2, 1.0, lot_id=sn1) self.assertEqual(len(product.stock_quant_ids), 1) self.env['stock.quant']._update_available_quantity(product, self.stock_subloc3, 1.0, lot_id=sn1) self.assertEqual(len(product.stock_quant_ids), 2) quant_2 = product.stock_quant_ids[1] self.assertEqual(quant_2.with_context(inventory_mode=True).sn_duplicated, True) with self.assertRaises(UserError): quant_2.with_context(inventory_mode=True).write({'location_id': self.stock_subloc2}) def test_update_quant_with_forbidden_field_02(self): """ Test that updating the package from the quant raise an error but if the package is unpacked, the quant can be updated. """ package = self.env['stock.quant.package'].create({ 'name': 'Package', }) self.env['stock.quant']._update_available_quantity(self.product, self.stock_location, 1.0, package_id=package) quant = self.product.stock_quant_ids self.assertEqual(len(self.product.stock_quant_ids), 1) with self.assertRaises(UserError): quant.with_context(inventory_mode=True).write({'package_id': False}) package.with_context(inventory_mode=True).unpack() self.assertFalse(quant.package_id) self.assertTrue(True) def test_unpack_and_quants_history(self): """ Test that after unpacking the quant history is preserved """ product = self.env['product.product'].create({ 'name': 'Product', 'type': 'product', 'tracking': 'lot', }) lot_a = self.env['stock.lot'].create({ 'name': 'A', 'product_id': product.id, 'product_qty': 5, }) package = self.env['stock.quant.package'].create({ 'name': 'Super Package', }) stock_location = self.stock_location dst_location = self.stock_subloc2 picking_type = self.env.ref('stock.picking_type_internal') self.env['stock.quant']._update_available_quantity(product, stock_location, 5.0, lot_id=lot_a, package_id=package) picking = self.env['stock.picking'].create({ 'picking_type_id': picking_type.id, 'location_id': dst_location.id, 'location_dest_id': stock_location.id, 'move_ids': [(0, 0, { 'name': 'In 5 x %s' % product.name, 'product_id': product.id, 'location_id': stock_location.id, 'location_dest_id': dst_location.id, 'product_uom_qty': 5, 'product_uom': product.uom_id.id, })], }) picking.action_confirm() picking.move_ids.move_line_ids.write({ 'qty_done': 5, 'lot_id': lot_a.id, 'package_id': package.id, 'result_package_id': package.id, }) picking.button_validate() package.unpack() quant = self.env['stock.quant'].search([ ('product_id', '=', product.id), ('location_id', '=', dst_location.id), ]) action = quant.action_view_stock_moves() history = self.env['stock.move.line'].search(action['domain']) self.assertTrue(history)