mirror of
https://github.com/OCA/manufacture.git
synced 2025-01-28 16:37:15 +02:00
[11.0][REW/IMP] multi_level_mrp: major overhault
This commit is contained in:
committed by
Jordi Ballester Alomar
parent
523ca538fe
commit
a3a6992ba7
@@ -2,7 +2,7 @@
|
||||
# (http://www.eficent.com)
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from datetime import datetime, timedelta
|
||||
from datetime import date, datetime, timedelta
|
||||
|
||||
from odoo.tests.common import SavepointCase
|
||||
from odoo import fields
|
||||
@@ -13,27 +13,36 @@ class TestMultiLevelMRP(SavepointCase):
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super(TestMultiLevelMRP, cls).setUpClass()
|
||||
cls.wiz_multi_level_mrp_model = cls.env['multi.level.mrp']
|
||||
cls.stock_picking_model = cls.env['stock.picking']
|
||||
cls.mrp_inventory_model = cls.env['mrp.inventory']
|
||||
cls.mo_obj = cls.env['mrp.production']
|
||||
cls.po_obj = cls.env['purchase.order']
|
||||
cls.stock_picking_obj = cls.env['stock.picking']
|
||||
cls.multi_level_mrp_wiz = cls.env['multi.level.mrp']
|
||||
cls.mrp_inventory_procure_wiz = cls.env['mrp.inventory.procure']
|
||||
cls.mrp_inventory_obj = cls.env['mrp.inventory']
|
||||
cls.mrp_product_obj = cls.env['mrp.product']
|
||||
cls.mrp_move_obj = cls.env['mrp.move']
|
||||
|
||||
cls.fp_1 = cls.env.ref('multi_level_mrp.product_product_fp_1')
|
||||
cls.fp_2 = cls.env.ref('multi_level_mrp.product_product_fp_2')
|
||||
cls.sf_1 = cls.env.ref('multi_level_mrp.product_product_sf_1')
|
||||
cls.sf_2 = cls.env.ref('multi_level_mrp.product_product_sf_2')
|
||||
cls.pp_1 = cls.env.ref('multi_level_mrp.product_product_pp_1')
|
||||
cls.pp_2 = cls.env.ref('multi_level_mrp.product_product_pp_2')
|
||||
cls.vendor = cls.env.ref('multi_level_mrp.res_partner_lazer_tech')
|
||||
cls.wh = cls.env.ref('stock.warehouse0')
|
||||
cls.stock_location = cls.wh.lot_stock_id
|
||||
cls.customer_location = cls.env.ref(
|
||||
'stock.stock_location_customers')
|
||||
|
||||
# Create test picking:
|
||||
date_move = datetime.today() + timedelta(days=7)
|
||||
cls.picking_1 = cls.stock_picking_model.create({
|
||||
cls.picking_1 = cls.stock_picking_obj.create({
|
||||
'picking_type_id': cls.env.ref('stock.picking_type_out').id,
|
||||
'location_id': cls.stock_location.id,
|
||||
'location_dest_id': cls.customer_location.id,
|
||||
'move_lines': [
|
||||
(0, 0, {
|
||||
'name': 'Test move pf-1',
|
||||
'name': 'Test move fp-1',
|
||||
'product_id': cls.fp_1.id,
|
||||
'date_expected': date_move,
|
||||
'date': date_move,
|
||||
@@ -54,7 +63,44 @@ class TestMultiLevelMRP(SavepointCase):
|
||||
})]
|
||||
})
|
||||
cls.picking_1.action_confirm()
|
||||
cls.wiz_multi_level_mrp_model.create({}).run_multi_level_mrp()
|
||||
|
||||
# Create Test PO:
|
||||
date_po = datetime.today() + timedelta(days=1)
|
||||
cls.po = cls.po_obj.create({
|
||||
'name': 'Test PO-001',
|
||||
'partner_id': cls.vendor.id,
|
||||
'order_line': [
|
||||
(0, 0, {
|
||||
'name': 'Test PP-2 line',
|
||||
'product_id': cls.pp_2.id,
|
||||
'date_planned': date_po,
|
||||
'product_qty': 5.0,
|
||||
'product_uom': cls.pp_2.uom_id.id,
|
||||
'price_unit': 25.0,
|
||||
})],
|
||||
})
|
||||
|
||||
# Create test MO:
|
||||
date_mo = datetime.today() + timedelta(days=9)
|
||||
bom_fp_2 = cls.env.ref('multi_level_mrp.mrp_bom_fp_2')
|
||||
cls.mo = cls.mo_obj.create({
|
||||
'product_id': cls.fp_2.id,
|
||||
'bom_id': bom_fp_2.id,
|
||||
'product_qty': 12.0,
|
||||
'product_uom_id': cls.fp_2.uom_id.id,
|
||||
'date_planned_start': date_mo,
|
||||
})
|
||||
cls.multi_level_mrp_wiz.create({}).run_multi_level_mrp()
|
||||
|
||||
# Dates (Strings):
|
||||
today = datetime.today()
|
||||
cls.date_3 = fields.Date.to_string(today + timedelta(days=3))
|
||||
cls.date_5 = fields.Date.to_string(today + timedelta(days=5))
|
||||
cls.date_6 = fields.Date.to_string(today + timedelta(days=6))
|
||||
cls.date_7 = fields.Date.to_string(today + timedelta(days=7))
|
||||
cls.date_8 = fields.Date.to_string(today + timedelta(days=8))
|
||||
cls.date_9 = fields.Date.to_string(today + timedelta(days=9))
|
||||
cls.date_10 = fields.Date.to_string(today + timedelta(days=10))
|
||||
|
||||
def test_01_mrp_levels(self):
|
||||
"""Tests computation of MRP levels."""
|
||||
@@ -65,61 +111,172 @@ class TestMultiLevelMRP(SavepointCase):
|
||||
self.assertEqual(self.pp_1.llc, 2)
|
||||
self.assertEqual(self.pp_2.llc, 2)
|
||||
|
||||
def test_02_multi_level_mrp(self):
|
||||
def test_02_mrp_product(self):
|
||||
"""Tests that mrp products are generated correctly."""
|
||||
mrp_product = self.mrp_product_obj.search([
|
||||
('product_id', '=', self.pp_1.id)])
|
||||
self.assertEqual(mrp_product.supply_method, 'buy')
|
||||
self.assertEqual(mrp_product.main_supplier_id, self.vendor)
|
||||
self.assertEqual(mrp_product.mrp_qty_available, 10.0)
|
||||
mrp_product = self.mrp_product_obj.search([
|
||||
('product_id', '=', self.sf_1.id)])
|
||||
self.assertEqual(mrp_product.supply_method, 'manufacture')
|
||||
|
||||
def test_03_mrp_moves(self):
|
||||
"""Tests for mrp moves generated."""
|
||||
moves = self.mrp_move_obj.search([
|
||||
('product_id', '=', self.pp_1.id),
|
||||
('mrp_action', '=', 'none'),
|
||||
])
|
||||
self.assertEqual(len(moves), 3)
|
||||
self.assertNotIn('s', moves.mapped('mrp_type'))
|
||||
for move in moves:
|
||||
self.assertTrue(move.mrp_move_up_ids)
|
||||
if move.mrp_move_up_ids.mrp_product_id.product_id == self.fp_1:
|
||||
# Demand coming from FP-1
|
||||
self.assertEqual(move.mrp_move_up_ids.mrp_action, 'mo')
|
||||
self.assertEqual(move.mrp_qty, -200.0)
|
||||
elif move.mrp_move_up_ids.mrp_product_id.product_id == self.sf_1:
|
||||
# Demand coming from FP-2 -> SF-1
|
||||
self.assertEqual(move.mrp_move_up_ids.mrp_action, 'mo')
|
||||
if move.mrp_date == self.date_5:
|
||||
self.assertEqual(move.mrp_qty, -90.0)
|
||||
elif move.mrp_date == self.date_8:
|
||||
self.assertEqual(move.mrp_qty, -72.0)
|
||||
# Check actions:
|
||||
moves = self.mrp_move_obj.search([
|
||||
('product_id', '=', self.pp_1.id),
|
||||
('mrp_action', '!=', 'none'),
|
||||
])
|
||||
self.assertEqual(len(moves), 3)
|
||||
for move in moves:
|
||||
self.assertEqual(move.mrp_action, 'po')
|
||||
self.assertEqual(move.mrp_type, 's')
|
||||
# Check PP-2 PO being accounted:
|
||||
po_move = self.mrp_move_obj.search([
|
||||
('product_id', '=', self.pp_2.id),
|
||||
('mrp_action', '=', 'none'),
|
||||
('mrp_type', '=', 's'),
|
||||
])
|
||||
self.assertEqual(len(po_move), 1)
|
||||
self.assertEqual(po_move.purchase_order_id, self.po)
|
||||
self.assertEqual(po_move.purchase_line_id, self.po.order_line)
|
||||
|
||||
def test_04_multi_level_mrp(self):
|
||||
"""Tests MRP inventories created."""
|
||||
# FP-1
|
||||
fp_1_inventory_lines = self.mrp_inventory_model.search(
|
||||
fp_1_inventory_lines = self.mrp_inventory_obj.search(
|
||||
[('mrp_product_id.product_id', '=', self.fp_1.id)])
|
||||
self.assertEqual(len(fp_1_inventory_lines), 1)
|
||||
date_7 = fields.Date.to_string(datetime.today() + timedelta(days=7))
|
||||
self.assertEqual(fp_1_inventory_lines.date, date_7)
|
||||
self.assertEqual(fp_1_inventory_lines.date, self.date_7)
|
||||
self.assertEqual(fp_1_inventory_lines.demand_qty, 100.0)
|
||||
self.assertEqual(fp_1_inventory_lines.to_procure, 100.0)
|
||||
# FP-2
|
||||
fp_2_inventory_lines = self.mrp_inventory_model.search(
|
||||
[('mrp_product_id.product_id', '=', self.fp_2.id)])
|
||||
self.assertEqual(len(fp_2_inventory_lines), 1)
|
||||
self.assertEqual(fp_2_inventory_lines.date, date_7)
|
||||
self.assertEqual(fp_2_inventory_lines.demand_qty, 15.0)
|
||||
self.assertEqual(fp_2_inventory_lines.to_procure, 15.0)
|
||||
fp_2_line_1 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.fp_2.id),
|
||||
('date', '=', self.date_7)])
|
||||
self.assertEqual(len(fp_2_line_1), 1)
|
||||
self.assertEqual(fp_2_line_1.demand_qty, 15.0)
|
||||
self.assertEqual(fp_2_line_1.to_procure, 15.0)
|
||||
# TODO: ask odoo to fix it... should be date10
|
||||
fp_2_line_2 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.fp_2.id),
|
||||
('date', '=', self.date_9)])
|
||||
self.assertEqual(len(fp_2_line_2), 1)
|
||||
self.assertEqual(fp_2_line_2.demand_qty, 0.0)
|
||||
self.assertEqual(fp_2_line_2.to_procure, 0.0)
|
||||
self.assertEqual(fp_2_line_2.supply_qty, 12.0)
|
||||
|
||||
# SF-1
|
||||
sf_1_inventory_lines = self.mrp_inventory_model.search(
|
||||
[('mrp_product_id.product_id', '=', self.sf_1.id)])
|
||||
self.assertEqual(len(sf_1_inventory_lines), 1)
|
||||
date_6 = fields.Date.to_string(datetime.today() + timedelta(days=6))
|
||||
self.assertEqual(sf_1_inventory_lines.date, date_6)
|
||||
self.assertEqual(sf_1_inventory_lines.demand_qty, 30.0)
|
||||
self.assertEqual(sf_1_inventory_lines.to_procure, 30.0)
|
||||
sf_1_line_1 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.sf_1.id),
|
||||
('date', '=', self.date_6)])
|
||||
self.assertEqual(len(sf_1_line_1), 1)
|
||||
self.assertEqual(sf_1_line_1.demand_qty, 30.0)
|
||||
self.assertEqual(sf_1_line_1.to_procure, 30.0)
|
||||
sf_1_line_2 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.sf_1.id),
|
||||
('date', '=', self.date_9)])
|
||||
self.assertEqual(len(sf_1_line_2), 1)
|
||||
self.assertEqual(sf_1_line_2.demand_qty, 24.0)
|
||||
self.assertEqual(sf_1_line_2.to_procure, 24.0)
|
||||
# SF-2
|
||||
sf_2_inventory_lines = self.mrp_inventory_model.search(
|
||||
[('mrp_product_id.product_id', '=', self.sf_2.id)])
|
||||
self.assertEqual(len(sf_2_inventory_lines), 1)
|
||||
self.assertEqual(sf_2_inventory_lines.date, date_6)
|
||||
self.assertEqual(sf_2_inventory_lines.demand_qty, 45.0)
|
||||
self.assertEqual(sf_2_inventory_lines.to_procure, 30.0)
|
||||
sf_2_line_1 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.sf_2.id),
|
||||
('date', '=', self.date_6)])
|
||||
self.assertEqual(len(sf_2_line_1), 1)
|
||||
self.assertEqual(sf_2_line_1.demand_qty, 45.0)
|
||||
self.assertEqual(sf_2_line_1.to_procure, 30.0)
|
||||
sf_2_line_2 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.sf_2.id),
|
||||
('date', '=', self.date_9)])
|
||||
self.assertEqual(len(sf_2_line_2), 1)
|
||||
self.assertEqual(sf_2_line_2.demand_qty, 36.0)
|
||||
self.assertEqual(sf_2_line_2.to_procure, 36.0)
|
||||
|
||||
# PP-1
|
||||
pp_1_inventory_lines = self.mrp_inventory_model.search(
|
||||
[('mrp_product_id.product_id', '=', self.pp_1.id)])
|
||||
self.assertEqual(len(pp_1_inventory_lines), 1)
|
||||
date_5 = fields.Date.to_string(datetime.today() + timedelta(days=5))
|
||||
self.assertEqual(pp_1_inventory_lines.date, date_5)
|
||||
self.assertEqual(pp_1_inventory_lines.demand_qty, 290.0)
|
||||
self.assertEqual(pp_1_inventory_lines.to_procure, 280.0)
|
||||
pp_1_line_1 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_1.id),
|
||||
('date', '=', self.date_5)])
|
||||
self.assertEqual(len(pp_1_line_1), 1)
|
||||
self.assertEqual(pp_1_line_1.demand_qty, 290.0)
|
||||
self.assertEqual(pp_1_line_1.to_procure, 280.0)
|
||||
pp_1_line_2 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_1.id),
|
||||
('date', '=', self.date_8)])
|
||||
self.assertEqual(len(pp_1_line_2), 1)
|
||||
self.assertEqual(pp_1_line_2.demand_qty, 72.0)
|
||||
self.assertEqual(pp_1_line_2.to_procure, 72.0)
|
||||
# PP-2
|
||||
pp_2_line_1 = self.mrp_inventory_model.search([
|
||||
pp_2_line_1 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_2.id),
|
||||
('date', '=', date_5)])
|
||||
('date', '=', self.date_3)])
|
||||
self.assertEqual(len(pp_2_line_1), 1)
|
||||
self.assertEqual(pp_2_line_1.demand_qty, 360.0)
|
||||
self.assertEqual(pp_2_line_1.to_procure, 360.0)
|
||||
date_3 = fields.Date.to_string(datetime.today() + timedelta(days=3))
|
||||
pp_2_line_2 = self.mrp_inventory_model.search([
|
||||
self.assertEqual(pp_2_line_1.demand_qty, 90.0)
|
||||
# 90.0 demand - 20.0 on hand - 5.0 on PO = 65.0
|
||||
self.assertEqual(pp_2_line_1.to_procure, 65.0)
|
||||
pp_2_line_2 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_2.id),
|
||||
('date', '=', date_3)])
|
||||
('date', '=', self.date_5)])
|
||||
self.assertEqual(len(pp_2_line_2), 1)
|
||||
self.assertEqual(pp_2_line_2.demand_qty, 90.0)
|
||||
self.assertEqual(pp_2_line_2.to_procure, 70.0)
|
||||
self.assertEqual(pp_2_line_2.demand_qty, 360.0)
|
||||
self.assertEqual(pp_2_line_2.to_procure, 360.0)
|
||||
pp_2_line_3 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_2.id),
|
||||
('date', '=', self.date_6)])
|
||||
self.assertEqual(len(pp_2_line_3), 1)
|
||||
self.assertEqual(pp_2_line_3.demand_qty, 108.0)
|
||||
self.assertEqual(pp_2_line_3.to_procure, 108.0)
|
||||
pp_2_line_4 = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.pp_2.id),
|
||||
('date', '=', self.date_8)])
|
||||
self.assertEqual(len(pp_2_line_4), 1)
|
||||
self.assertEqual(pp_2_line_4.demand_qty, 48.0)
|
||||
self.assertEqual(pp_2_line_4.to_procure, 48.0)
|
||||
|
||||
# TODO: test procure wizard: dates...
|
||||
def test_05_actions_count(self):
|
||||
"""Test actions counters."""
|
||||
pass
|
||||
|
||||
def test_06_procure_mo(self):
|
||||
"""Test procurement wizard with MOs."""
|
||||
mos = self.mo_obj.search([
|
||||
('product_id', '=', self.fp_1.id)])
|
||||
self.assertFalse(mos)
|
||||
mrp_inv = self.mrp_inventory_obj.search([
|
||||
('mrp_product_id.product_id', '=', self.fp_1.id)])
|
||||
self.mrp_inventory_procure_wiz.with_context({
|
||||
'active_model': 'mrp.inventory',
|
||||
'active_ids': mrp_inv.ids,
|
||||
'active_id': mrp_inv.id,
|
||||
}).create({}).make_procurement()
|
||||
mos = self.mo_obj.search([
|
||||
('product_id', '=', self.fp_1.id)])
|
||||
self.assertTrue(mos)
|
||||
self.assertEqual(mos.product_qty, 100.0)
|
||||
datetime_5 = fields.Datetime.to_string(
|
||||
date.today() + timedelta(days=5))
|
||||
self.assertEqual(mos.date_planned_start, datetime_5)
|
||||
|
||||
# TODO: test procure wizard: pos, multiple...
|
||||
# TODO: test multiple destination IDS:...
|
||||
Reference in New Issue
Block a user