Skip to content

Commit

Permalink
[11.0][MIG] mrp_mto_with_stock
Browse files Browse the repository at this point in the history
  • Loading branch information
Bhavesh Odedra committed Feb 19, 2018
1 parent d8b0674 commit 1d32e12
Show file tree
Hide file tree
Showing 12 changed files with 67 additions and 92 deletions.
4 changes: 3 additions & 1 deletion mrp_mto_with_stock/README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ To configure this module, you need to:
standard behavior.

If you want to use the second mode, based on forecast quantity

#. Go to the warehouse you want to follow this behaviour.
#. In the view form go to the tab *Warehouse Configuration* and set the
*MRP MTO with forecast stock*. You still need to configure the products
Expand All @@ -44,7 +45,7 @@ To use this module, you need to:

.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
:alt: Try me on Runbot
:target: https://runbot.odoo-community.org/runbot/129/10.0
:target: https://runbot.odoo-community.org/runbot/129/11.0

Bug Tracker
===========
Expand All @@ -68,6 +69,7 @@ Contributors
* John Walsh <John.Walsh@interclean.com>
* Lois Rilo <lois.rilo@eficent.com>
* Florian da Costa <florian.dacosta@akretion.com>
* Bhavesh Odedra <bodedra@opensourceintegrators.com>

Maintainer
----------
Expand Down
3 changes: 0 additions & 3 deletions mrp_mto_with_stock/__init__.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# Copyright 2015 John Walsh
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

from . import models
5 changes: 2 additions & 3 deletions mrp_mto_with_stock/__manifest__.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# Copyright 2015 John Walsh
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
Expand All @@ -10,11 +9,11 @@
"author": "John Walsh, Eficent, Odoo Community Association (OCA)",
"website": "https://odoo-community.org/",
"category": "Manufacturing",
"version": "10.0.1.0.0",
"version": "11.0.1.0.0",
"license": "AGPL-3",
"application": False,
"installable": True,
"depends": ["mrp"],
"depends": ["stock", "sale", "purchase", "mrp"],
"data": [
'views/product_template_view.xml',
'views/stock_warehouse.xml',
Expand Down
2 changes: 1 addition & 1 deletion mrp_mto_with_stock/demo/product.xml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
<?xml version="1.0" encoding="UTF-8"?>
<?xml version="1.0"?>
<!--
Copyright (C) 2017 - Today: GRAP (http://www.grap.coop)
@author Sylvain LE GAL (https://twitter.com/legalsylvain)
Expand Down
3 changes: 0 additions & 3 deletions mrp_mto_with_stock/models/__init__.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# Copyright 2015 John Walsh
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

from . import mrp_production
Expand Down
89 changes: 44 additions & 45 deletions mrp_mto_with_stock/models/mrp_production.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# Copyright 2015 John Walsh
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

from odoo import api, models
from odoo.exceptions import UserError
import logging
_logger = logging.getLogger(__name__)

Expand All @@ -16,6 +16,10 @@ def _adjust_procure_method(self):
# Si location => By pass method...
super(MrpProduction, self)._adjust_procure_method()

def _get_procurement_group_data(self, move):
return {'partner_id': move.partner_id.id,
'name': '{0}->{1}'.format(self.name, move.product_id.name)}

@api.multi
def action_assign(self):
"""Reserves available products to the production order but also creates
Expand All @@ -30,74 +34,69 @@ def action_assign(self):
warehouse = production.location_src_id.get_warehouse()
mto_with_no_move_dest_id = warehouse.mrp_mto_mts_forecast_qty
for move in self.move_raw_ids:
group = new_move = procurement = False
qty_to_procure = 0.0
if (move.state == 'confirmed' and move.location_id in
move.product_id.mrp_mts_mto_location_ids and not
mto_with_no_move_dest_id):
domain = [('product_id', '=', move.product_id.id),
('move_dest_id', '=', move.id)]
if move.group_id:
domain.append(('group_id', '=', move.group_id.id))
procurement = self.env['procurement.order'].search(domain)
procurement = True
if not procurement:
# We have to split the move because we can't have
# a part of the move that have ancestors and not the
# other else it won't ever be reserved.
qty_to_procure = (move.remaining_qty -
move.reserved_availability)
if qty_to_procure < move.product_uom_qty:
move.do_unreserve()
new_move_id = move.split(
move._do_unreserve()
new_move_id = move._split(
qty_to_procure,
restrict_lot_id=move.restrict_lot_id,
restrict_partner_id=move.restrict_partner_id)
new_move = move_obj.browse(
new_move_id)
move.action_assign()
new_move = move_obj.browse(new_move_id)
move._action_assign()
else:
new_move = move

proc_dict = self._prepare_mto_procurement(
new_move, qty_to_procure,
mto_with_no_move_dest_id)
self.env['procurement.order'].create(proc_dict)

pg_data = production._get_procurement_group_data(
new_move)
group = self.env['procurement.group'].create(pg_data)
new_move.write({'group_id': group.id})
if (move.state == 'confirmed' and move.location_id in
move.product_id.mrp_mts_mto_location_ids and
move.procure_method == 'make_to_stock' and
mto_with_no_move_dest_id):
qty_to_procure = production.get_mto_qty_to_procure(move)
if qty_to_procure > 0.0:
proc_dict = self._prepare_mto_procurement(
move, qty_to_procure, mto_with_no_move_dest_id)
proc_dict.pop('move_dest_id', None)
self.env['procurement.order'].create(proc_dict)
pg_data = production._get_procurement_group_data(move)
group = self.env['procurement.group'].create(pg_data)
move.write({'group_id': group.id})
new_move = move
if group:
production.run_procurement(new_move, group, qty_to_procure)
return res

def _prepare_mto_procurement(self, move, qty, mto_with_no_move_dest_id):
"""Prepares a procurement for a MTO product."""
origin = ((move.group_id and move.group_id.name + ":") or "") + \
@api.multi
def run_procurement(self, move, group, qty):
self.ensure_one()
errors = []
values = move._prepare_procurement_values()
origin = ((group and group.name + ":") or "") + \
((move.name and move.name + ":") or "") + 'MTO -> Production'
group_id = move.group_id and move.group_id.id or False
route_ids = self.env.ref('stock.route_warehouse0_mto')
warehouse_id = (move.warehouse_id.id or (move.picking_type_id and
move.picking_type_id.warehouse_id.id or False))
vals = {
'name': move.name + ':' + str(move.id),
'origin': origin,
'company_id': move.company_id and move.company_id.id or False,
'date_planned': move.date,
'product_id': move.product_id.id,
'product_qty': qty,
'product_uom': move.product_uom.id,
'location_id': move.location_id.id,
'group_id': group_id,
'route_ids': [(6, 0, route_ids.ids)],
'warehouse_id': warehouse_id,
'priority': move.priority,
}
if not mto_with_no_move_dest_id:
vals['move_dest_id'] = move.id
return vals
values['route_ids'] = self.env.ref('stock.route_warehouse0_mto')
try:
self.env['procurement.group'].run(
move.product_id,
qty,
move.product_uom,
move.location_id,
origin,
origin,
values
)
except UserError as error:
errors.append(error.name)
if errors:
raise UserError('\n'.join(errors))
return True

@api.multi
def get_mto_qty_to_procure(self, move):
Expand Down
1 change: 0 additions & 1 deletion mrp_mto_with_stock/models/product_template.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

Expand Down
1 change: 0 additions & 1 deletion mrp_mto_with_stock/models/stock_warehouse.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Akretion
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

Expand Down
2 changes: 0 additions & 2 deletions mrp_mto_with_stock/tests/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

from . import test_mrp_mto_with_stock
45 changes: 15 additions & 30 deletions mrp_mto_with_stock/tests/test_mrp_mto_with_stock.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
# -*- coding: utf-8 -*-
# Copyright 2017 Eficent Business and IT Consulting Services S.L.
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).

Expand Down Expand Up @@ -68,18 +67,11 @@ def test_manufacture_with_forecast_stock(self):

self.assertEqual(self.production.availability, 'partially_available')

self.assertEquals(self.subproduct1.virtual_available, 0)
self.assertEquals(self.subproduct1.virtual_available, -3)

procurement_subproduct1 = self.env['procurement.order'].search(
[('product_id', '=', self.subproduct1.id),
('group_id', '=', self.production.procurement_group_id.id)])

self.assertEquals(len(procurement_subproduct1), 1)
self.assertEquals(procurement_subproduct1.product_qty, 3)

production_sub1 = procurement_subproduct1.production_id
production_sub1 = self.production
self.assertEqual(production_sub1.state, 'confirmed')
self.assertEqual(production_sub1.product_qty, 3)
self.assertEqual(production_sub1.product_qty, 1)

self._update_product_qty(self.subproduct1, self.stock_location_stock,
7)
Expand All @@ -88,22 +80,18 @@ def test_manufacture_with_forecast_stock(self):
self.production2 = self.production_model.create(
self._get_production_vals())
self.production2.action_assign()
procurement_subproduct1_2 = self.env['procurement.order'].search(
[('product_id', '=', self.subproduct1.id),
('group_id', '=', self.production2.procurement_group_id.id)])
self.assertEquals(len(procurement_subproduct1_2), 0)
self.assertEquals(self.production2.availability, 'assigned')
self.production2.do_unreserve()

self.assertEquals(self.subproduct1.virtual_available, 0)
self.assertEquals(self.subproduct1.virtual_available, -3)

self.production.action_assign()
# We check if first MO is able to assign it self even if it has
# previously generate procurements, it would not be the case in the
# other mode (without mrp_mto_mts_reservable_stock on warehouse)
self.assertEquals(self.production.availability, 'assigned')

self.assertEquals(self.subproduct1.virtual_available, 0)
self.assertEquals(self.subproduct1.virtual_available, -3)

def test_manufacture_with_reservable_stock(self):
"""
Expand All @@ -127,20 +115,17 @@ def test_manufacture_with_reservable_stock(self):
self.production.action_assign()
self.assertEqual(self.production.state, 'confirmed')

procurement_sub1 = self.env['procurement.order'].search(
[('product_id', '=', self.subproduct1.id),
('move_dest_id', 'in', self.production.move_raw_ids.ids)])
self.assertEquals(len(procurement_sub1), 1)

procurement_sub2 = self.env['procurement.order'].search(
[('product_id', '=', self.subproduct2.id),
('move_dest_id', 'in', self.production.move_raw_ids.ids)])
self.assertEquals(len(procurement_sub2), 0)
for move in self.production.move_raw_ids:
if move.product_id.id == self.subproduct1.id:
self.assertEquals(len(move.group_id), 1)
production_sub1 = self.production_model.search(
[('procurement_group_id', '=', move.group_id.id)])
elif move.product_id.id == self.subproduct2.id:
self.assertEquals(len(move.group_id), 1)

production_sub1 = procurement_sub1.production_id
self.assertEqual(production_sub1.product_qty, 3)
self.assertEqual(production_sub1.product_qty, 1)
production_sub1.action_assign()
self.assertEqual(production_sub1.availability, 'assigned')
self.assertEqual(production_sub1.availability, 'partially_available')

wizard_obj = self.env['mrp.product.produce']
default_fields = ['lot_id', 'product_id', 'product_uom_id',
Expand All @@ -154,5 +139,5 @@ def test_manufacture_with_reservable_stock(self):
self.assertTrue(production_sub1.check_to_done)
self.assertEquals(self.subproduct1.qty_available, 2)
production_sub1.button_mark_done()
self.assertEquals(self.subproduct1.qty_available, 5)
self.assertEquals(self.subproduct1.qty_available, -3)
self.assertEqual(self.production.availability, 'assigned')
2 changes: 1 addition & 1 deletion mrp_mto_with_stock/views/product_template_view.xml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<?xml version="1.0"?>
<!-- Copyright 2017 Eficent Business and IT Consulting Services S.L.
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->

Expand Down
2 changes: 1 addition & 1 deletion mrp_mto_with_stock/views/stock_warehouse.xml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<?xml version="1.0"?>
<!-- Copyright 2017 Akretion
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->

Expand Down

0 comments on commit 1d32e12

Please sign in to comment.