stock/tests/test_product.py

344 lines
15 KiB
Python

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
# Author: Leonardo Pistone
# Copyright 2015 Camptocamp SA
from odoo.addons.stock.tests.common2 import TestStockCommon
from odoo.exceptions import UserError
from odoo.tests.common import Form
class TestVirtualAvailable(TestStockCommon):
@classmethod
def setUpClass(cls):
super().setUpClass()
# Make `product3` a storable product for this test. Indeed, creating quants
# and playing with owners is not possible for consumables.
cls.product_3.type = 'product'
cls.env['stock.picking.type'].browse(cls.env.ref('stock.picking_type_out').id).reservation_method = 'manual'
cls.env['stock.quant'].create({
'product_id': cls.product_3.id,
'location_id': cls.env.ref('stock.stock_location_stock').id,
'quantity': 30.0})
cls.env['stock.quant'].create({
'product_id': cls.product_3.id,
'location_id': cls.env.ref('stock.stock_location_stock').id,
'quantity': 10.0,
'owner_id': cls.user_stock_user.partner_id.id})
cls.picking_out = cls.env['stock.picking'].create({
'state': 'draft',
'picking_type_id': cls.env.ref('stock.picking_type_out').id
})
cls.env['stock.move'].create({
'name': 'a move',
'product_id': cls.product_3.id,
'product_uom_qty': 3.0,
'product_uom': cls.product_3.uom_id.id,
'picking_id': cls.picking_out.id,
'location_id': cls.env.ref('stock.stock_location_stock').id,
'location_dest_id': cls.env.ref('stock.stock_location_customers').id})
cls.picking_out_2 = cls.env['stock.picking'].create({
'state': 'draft',
'picking_type_id': cls.env.ref('stock.picking_type_out').id})
cls.env['stock.move'].create({
'restrict_partner_id': cls.user_stock_user.partner_id.id,
'name': 'another move',
'product_id': cls.product_3.id,
'product_uom_qty': 5.0,
'product_uom': cls.product_3.uom_id.id,
'picking_id': cls.picking_out_2.id,
'location_id': cls.env.ref('stock.stock_location_stock').id,
'location_dest_id': cls.env.ref('stock.stock_location_customers').id})
def test_without_owner(self):
self.assertAlmostEqual(40.0, self.product_3.virtual_available)
self.picking_out.action_assign()
self.picking_out_2.action_assign()
self.assertAlmostEqual(32.0, self.product_3.virtual_available)
def test_with_owner(self):
prod_context = self.product_3.with_context(owner_id=self.user_stock_user.partner_id.id)
self.assertAlmostEqual(10.0, prod_context.virtual_available)
self.picking_out.action_assign()
self.picking_out_2.action_assign()
self.assertAlmostEqual(5.0, prod_context.virtual_available)
def test_free_quantity(self):
""" Test the value of product.free_qty. Free_qty = qty_on_hand - qty_reserved"""
self.assertAlmostEqual(40.0, self.product_3.free_qty)
self.picking_out.action_confirm()
self.picking_out_2.action_confirm()
# No reservation so free_qty is unchanged
self.assertAlmostEqual(40.0, self.product_3.free_qty)
self.picking_out.action_assign()
self.picking_out_2.action_assign()
# 8 units are now reserved
self.assertAlmostEqual(32.0, self.product_3.free_qty)
self.picking_out.do_unreserve()
self.picking_out_2.do_unreserve()
# 8 units are available again
self.assertAlmostEqual(40.0, self.product_3.free_qty)
def test_archive_product_1(self):
"""`qty_available` and `virtual_available` are computed on archived products"""
self.assertTrue(self.product_3.active)
self.assertAlmostEqual(40.0, self.product_3.qty_available)
self.assertAlmostEqual(40.0, self.product_3.virtual_available)
self.product_3.active = False
self.assertAlmostEqual(40.0, self.product_3.qty_available)
self.assertAlmostEqual(40.0, self.product_3.virtual_available)
def test_archive_product_2(self):
"""Archiving a product should archive its reordering rules"""
self.assertTrue(self.product_3.active)
orderpoint_form = Form(self.env['stock.warehouse.orderpoint'])
orderpoint_form.product_id = self.product_3
orderpoint_form.location_id = self.env.ref('stock.stock_location_stock')
orderpoint_form.product_min_qty = 0.0
orderpoint_form.product_max_qty = 5.0
orderpoint = orderpoint_form.save()
self.assertTrue(orderpoint.active)
self.product_3.active = False
self.assertFalse(orderpoint.active)
def test_change_product_company(self):
""" Checks we can't change the product's company if this product has
quant in another company. """
company1 = self.env.ref('base.main_company')
company2 = self.env['res.company'].create({'name': 'Second Company'})
product = self.env['product.product'].create({
'name': 'Product [TEST - Change Company]',
'type': 'product',
})
# Creates a quant for productA in the first company.
self.env['stock.quant'].create({
'product_id': product.id,
'product_uom_id': self.uom_unit.id,
'location_id': self.location_1.id,
'quantity': 7,
'reserved_quantity': 0,
})
# Assigns a company: should be OK for company1 but should raise an error for company2.
product.company_id = company1.id
with self.assertRaises(UserError):
product.company_id = company2.id
# Checks we can assing company2 for the product once there is no more quant for it.
quant = self.env['stock.quant'].search([('product_id', '=', product.id)])
quant.quantity = 0
self.env['stock.quant']._unlink_zero_quants()
product.company_id = company2.id # Should work this time.
def test_change_product_company_02(self):
""" Checks we can't change the product's company if this product has
stock move line in another company. """
company1 = self.env.ref('base.main_company')
company2 = self.env['res.company'].create({'name': 'Second Company'})
product = self.env['product.product'].create({
'name': 'Product [TEST - Change Company]',
'type': 'consu',
})
picking = self.env['stock.picking'].create({
'location_id': self.env.ref('stock.stock_location_customers').id,
'location_dest_id': self.env.ref('stock.stock_location_stock').id,
'picking_type_id': self.ref('stock.picking_type_in'),
'state': 'draft',
})
self.env['stock.move'].create({
'name': 'test',
'location_id': self.env.ref('stock.stock_location_customers').id,
'location_dest_id': self.env.ref('stock.stock_location_stock').id,
'product_id': product.id,
'product_uom': product.uom_id.id,
'product_uom_qty': 1,
'picking_id': picking.id,
})
picking.action_confirm()
picking.button_validate()
product.company_id = company1.id
with self.assertRaises(UserError):
product.company_id = company2.id
def test_change_product_company_exclude_vendor_and_customer_location(self):
""" Checks we can change product company where only exist single company
and exist quant in vendor/customer location"""
company1 = self.env.ref('base.main_company')
customer_location = self.env.ref('stock.stock_location_customers')
supplier_location = self.env.ref('stock.stock_location_suppliers')
product = self.env['product.product'].create({
'name': 'Product Single Company',
'type': 'product',
})
# Creates a quant for company 1.
self.env['stock.quant'].create({
'product_id': product.id,
'product_uom_id': self.uom_unit.id,
'location_id': self.location_1.id,
'quantity': 5,
})
# Creates a quant for vendor location.
self.env['stock.quant'].create({
'product_id': product.id,
'product_uom_id': self.uom_unit.id,
'location_id': supplier_location.id,
'quantity': -15,
})
# Creates a quant for customer location.
self.env['stock.quant'].create({
'product_id': product.id,
'product_uom_id': self.uom_unit.id,
'location_id': customer_location.id,
'quantity': 10,
})
# Assigns a company: should be ok because only exist one company (exclude vendor and customer location)
product.company_id = company1.id
# Reset product company to empty
product.company_id = False
company2 = self.env['res.company'].create({'name': 'Second Company'})
# Assigns to another company: should be not okay because exist quants in defferent company (exclude vendor and customer location)
with self.assertRaises(UserError):
product.company_id = company2.id
def test_search_qty_available(self):
product = self.env['product.product'].create({
'name': 'Brand new product',
'type': 'product',
})
result = self.env['product.product'].search([
('qty_available', '=', 0),
('id', 'in', product.ids),
])
self.assertEqual(product, result)
def test_search_product_template(self):
"""
Suppose a variant V01 that can not be deleted because it is used by a
lot [1]. Then, the variant's template T is changed: we add a dynamic
attribute. Because of [1], V01 is archived. This test ensures that
`name_search` still finds T.
Then, we create a new variant V02 of T. This test also ensures that
calling `name_search` with a negative operator will exclude T from the
result.
"""
self.env.ref('base.group_user').write({'implied_ids': [(4, self.env.ref('product.group_product_variant').id)]})
template = self.env['product.template'].create({
'name': 'Super Product',
})
product01 = template.product_variant_id
self.env['stock.lot'].create({
'name': 'lot1',
'product_id': product01.id,
'company_id': self.env.company.id,
})
product_attribute = self.env['product.attribute'].create({
'name': 'PA',
'create_variant': 'dynamic'
})
self.env['product.attribute.value'].create([{
'name': 'PAV' + str(i),
'attribute_id': product_attribute.id
} for i in range(2)])
tmpl_attr_lines = self.env['product.template.attribute.line'].create({
'attribute_id': product_attribute.id,
'product_tmpl_id': product01.product_tmpl_id.id,
'value_ids': [(6, 0, product_attribute.value_ids.ids)],
})
self.assertFalse(product01.active)
self.assertTrue(template.active)
self.assertFalse(template.product_variant_ids)
res = self.env['product.template'].name_search(name='super', operator='ilike')
res_ids = [r[0] for r in res]
self.assertIn(template.id, res_ids)
product02 = self.env['product.product'].create({
'default_code': '123',
'product_tmpl_id': template.id,
'product_template_attribute_value_ids': [(6, 0, tmpl_attr_lines.product_template_value_ids[0].ids)]
})
self.assertFalse(product01.active)
self.assertTrue(product02.active)
self.assertTrue(template)
self.assertEqual(template.product_variant_ids, product02)
res = self.env['product.template'].name_search(name='123', operator='not ilike')
res_ids = [r[0] for r in res]
self.assertNotIn(template.id, res_ids)
def test_product_qty_field_and_context(self):
main_warehouse = self.warehouse_1
other_warehouse = self.env['stock.warehouse'].search([('id', '!=', main_warehouse.id)], limit=1)
warehouses = main_warehouse | other_warehouse
main_loc = main_warehouse.lot_stock_id
other_loc = other_warehouse.lot_stock_id
self.assertTrue(other_warehouse, 'The test needs another warehouse')
(main_loc | other_loc).name = 'Stock'
sub_loc01, sub_loc02, sub_loc03 = self.env['stock.location'].create([{
'name': 'Sub0%s' % (i + 1),
'location_id': main_loc.id,
} for i in range(3)])
self.env['stock.quant'].search([('product_id', '=', self.product_3.id)]).unlink()
self.env['stock.quant']._update_available_quantity(self.product_3, other_loc, 1000)
self.env['stock.quant']._update_available_quantity(self.product_3, main_loc, 100)
self.env['stock.quant']._update_available_quantity(self.product_3, sub_loc01, 10)
self.env['stock.quant']._update_available_quantity(self.product_3, sub_loc02, 1)
for wh, loc, expected in [
(False, False, 1111.0),
(False, other_loc.id, 1000.0),
(False, main_loc.id, 111.0),
(False, sub_loc01.id, 10.0),
(False, sub_loc01.name, 10.0),
(False, 'sub', 11.0),
(False, main_loc.name, 1111.0),
(False, (sub_loc01 | sub_loc02 | sub_loc03).ids, 11.0),
(main_warehouse.id, main_loc.name, 111.0),
(main_warehouse.id, main_loc.id, 111.0),
(main_warehouse.id, (main_loc | other_loc).ids, 111.0),
(main_warehouse.id, sub_loc01.id, 10.0),
(main_warehouse.id, (sub_loc01 | sub_loc02).ids, 11.0),
(other_warehouse.id, main_loc.name, 1000.0),
(other_warehouse.id, main_loc.id, 0.0),
(main_warehouse.name, False, 111.0),
(main_warehouse.id, False, 111.0),
(warehouses.ids, False, 1111.0),
(warehouses.ids, (other_loc | sub_loc02).ids, 1001),
]:
product_qty = self.product_3.with_context(warehouse=wh, location=loc).qty_available
self.assertEqual(product_qty, expected)
def test_change_type_tracked_product(self):
product = self.env['product.template'].create({
'name': 'Brand new product',
'type': 'product',
'tracking': 'serial',
})
product_form = Form(product)
product_form.detailed_type = 'service'
product = product_form.save()
self.assertEqual(product.tracking, 'none')
product.detailed_type = 'product'
product.tracking = 'serial'
self.assertEqual(product.tracking, 'serial')
# change the type from "product.product" form
product_form = Form(product.product_variant_id)
product_form.detailed_type = 'service'
product = product_form.save()
self.assertEqual(product.tracking, 'none')