odoo/addons/stock/models/procurement.py

398 lines
21 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from collections import defaultdict
from datetime import datetime
from dateutil.relativedelta import relativedelta
from psycopg2 import OperationalError
from odoo import api, fields, models, registry, _
from odoo.osv import expression
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_compare, float_round
import logging
_logger = logging.getLogger(__name__)
class ProcurementGroup(models.Model):
_inherit = 'procurement.group'
partner_id = fields.Many2one('res.partner', 'Partner')
class ProcurementRule(models.Model):
""" Pull rules """
_inherit = 'procurement.rule'
location_id = fields.Many2one('stock.location', 'Procurement Location')
location_src_id = fields.Many2one('stock.location', 'Source Location', help="Source location is action=move")
route_id = fields.Many2one('stock.location.route', 'Route', help="If route_id is False, the rule is global")
procure_method = fields.Selection([
('make_to_stock', 'Take From Stock'),
('make_to_order', 'Create Procurement')], string='Move Supply Method',
default='make_to_stock', required=True,
help="""Determines the procurement method of the stock move that will be generated: whether it will need to 'take from the available stock' in its source location or needs to ignore its stock and create a procurement over there.""")
route_sequence = fields.Integer('Route Sequence', related='route_id.sequence', store=True)
picking_type_id = fields.Many2one(
'stock.picking.type', 'Picking Type',
required=True,
help="Picking Type determines the way the picking should be shown in the view, reports, ...")
delay = fields.Integer('Number of Days', default=0)
partner_address_id = fields.Many2one('res.partner', 'Partner Address')
propagate = fields.Boolean(
'Propagate cancel and split', default=True,
help='If checked, when the previous move of the move (which was generated by a next procurement) is cancelled or split, the move generated by this move will too')
warehouse_id = fields.Many2one('stock.warehouse', 'Served Warehouse', help='The warehouse this rule is for')
propagate_warehouse_id = fields.Many2one(
'stock.warehouse', 'Warehouse to Propagate',
help="The warehouse to propagate on the created move/procurement, which can be different of the warehouse this rule is for (e.g for resupplying rules from another warehouse)")
@api.model
def _get_action(self):
result = super(ProcurementRule, self)._get_action()
return result + [('move', _('Move From Another Location'))]
class ProcurementOrder(models.Model):
_inherit = "procurement.order"
location_id = fields.Many2one('stock.location', 'Procurement Location') # not required because task may create procurements that aren't linked to a location with sale_service
partner_dest_id = fields.Many2one('res.partner', 'Customer Address', help="In case of dropshipping, we need to know the destination address more precisely")
move_ids = fields.One2many('stock.move', 'procurement_id', 'Moves', help="Moves created by the procurement")
move_dest_id = fields.Many2one('stock.move', 'Destination Move', help="Move which caused (created) the procurement")
route_ids = fields.Many2many(
'stock.location.route', 'stock_location_route_procurement', 'procurement_id', 'route_id', 'Preferred Routes',
help="Preferred route to be followed by the procurement order. Usually copied from the generating document (SO) but could be set up manually.")
warehouse_id = fields.Many2one('stock.warehouse', 'Warehouse', help="Warehouse to consider for the route selection")
orderpoint_id = fields.Many2one('stock.warehouse.orderpoint', 'Minimum Stock Rule')
@api.onchange('warehouse_id')
def onchange_warehouse_id(self):
if self.warehouse_id:
self.location_id = self.warehouse_id.lot_stock_id.id
@api.multi
def propagate_cancels(self):
# set the context for the propagation of the procurement cancellation
# TDE FIXME: was in cancel, moved here for consistency
cancel_moves = self.with_context(cancel_procurement=True).filtered(lambda order: order.rule_id.action == 'move').mapped('move_ids')
if cancel_moves:
cancel_moves.action_cancel()
return self.search([('move_dest_id', 'in', cancel_moves.filtered(lambda move: move.propagate).ids)])
@api.multi
def cancel(self):
propagated_procurements = self.filtered(lambda order: order.state != 'done').propagate_cancels()
if propagated_procurements:
propagated_procurements.cancel()
return super(ProcurementOrder, self).cancel()
@api.multi
def do_view_pickings(self):
""" Return an action to display the pickings belonging to the same
procurement group of given ids. """
action = self.env.ref('stock.do_view_pickings').read()[0]
action['domain'] = [('group_id', 'in', self.mapped('group_id').ids)]
return action
@api.multi
@api.returns('procurement.rule', lambda value: value.id if value else False)
def _find_suitable_rule(self):
rule = super(ProcurementOrder, self)._find_suitable_rule()
if not rule:
# a rule defined on 'Stock' is suitable for a procurement in 'Stock\Bin A'
all_parent_location_ids = self._find_parent_locations()
rule = self._search_suitable_rule([('location_id', 'in', all_parent_location_ids.ids)])
return rule
def _find_parent_locations(self):
parent_locations = self.env['stock.location']
location = self.location_id
while location:
parent_locations |= location
location = location.location_id
return parent_locations
def _search_suitable_rule(self, domain):
""" First find a rule among the ones defined on the procurement order
group; then try on the routes defined for the product; finally fallback
on the default behavior """
if self.warehouse_id:
domain = expression.AND([['|', ('warehouse_id', '=', self.warehouse_id.id), ('warehouse_id', '=', False)], domain])
Pull = self.env['procurement.rule']
res = self.env['procurement.rule']
if self.route_ids:
res = Pull.search(expression.AND([[('route_id', 'in', self.route_ids.ids)], domain]), order='route_sequence, sequence', limit=1)
if not res:
product_routes = self.product_id.route_ids | self.product_id.categ_id.total_route_ids
if product_routes:
res = Pull.search(expression.AND([[('route_id', 'in', product_routes.ids)], domain]), order='route_sequence, sequence', limit=1)
if not res:
warehouse_routes = self.warehouse_id.route_ids
if warehouse_routes:
res = Pull.search(expression.AND([[('route_id', 'in', warehouse_routes.ids)], domain]), order='route_sequence, sequence', limit=1)
if not res:
res = Pull.search(expression.AND([[('route_id', '=', False)], domain]), order='sequence', limit=1)
return res
def _get_stock_move_values(self):
''' Returns a dictionary of values that will be used to create a stock move from a procurement.
This function assumes that the given procurement has a rule (action == 'move') set on it.
:param procurement: browse record
:rtype: dictionary
'''
group_id = False
if self.rule_id.group_propagation_option == 'propagate':
group_id = self.group_id.id
elif self.rule_id.group_propagation_option == 'fixed':
group_id = self.rule_id.group_id.id
date_expected = (datetime.strptime(self.date_planned, DEFAULT_SERVER_DATETIME_FORMAT) - relativedelta(days=self.rule_id.delay or 0)).strftime(DEFAULT_SERVER_DATETIME_FORMAT)
# it is possible that we've already got some move done, so check for the done qty and create
# a new move with the correct qty
qty_done = sum(self.move_ids.filtered(lambda move: move.state == 'done').mapped('product_uom_qty'))
qty_left = max(self.product_qty - qty_done, 0)
return {
'name': self.name[:2000],
'company_id': self.rule_id.company_id.id or self.rule_id.location_src_id.company_id.id or self.rule_id.location_id.company_id.id or self.company_id.id,
'product_id': self.product_id.id,
'product_uom': self.product_uom.id,
'product_uom_qty': qty_left,
'partner_id': self.rule_id.partner_address_id.id or (self.group_id and self.group_id.partner_id.id) or False,
'location_id': self.rule_id.location_src_id.id,
'location_dest_id': self.location_id.id,
'move_dest_id': self.move_dest_id and self.move_dest_id.id or False,
'procurement_id': self.id,
'rule_id': self.rule_id.id,
'procure_method': self.rule_id.procure_method,
'origin': self.origin,
'picking_type_id': self.rule_id.picking_type_id.id,
'group_id': group_id,
'route_ids': [(4, route.id) for route in self.route_ids],
'warehouse_id': self.rule_id.propagate_warehouse_id.id or self.rule_id.warehouse_id.id,
'date': date_expected,
'date_expected': date_expected,
'propagate': self.rule_id.propagate,
'priority': self.priority,
}
def _run_move_create(self):
# FIXME - remove me in master/saas-14
_logger.warning("'_run_move_create' has been renamed into '_get_stock_move_values'... Overrides are ignored")
return self._get_stock_move_values()
@api.multi
def _run(self):
if self.rule_id.action == 'move':
if not self.rule_id.location_src_id:
self.message_post(body=_('No source location defined!'))
return False
# create the move as SUPERUSER because the current user may not have the rights to do it (mto product launched by a sale for example)
self.env['stock.move'].sudo().create(self._get_stock_move_values())
return True
return super(ProcurementOrder, self)._run()
@api.multi
def run(self, autocommit=False):
# TDE CLEANME: unused context key procurement_auto_defer remove
new_self = self.filtered(lambda order: order.state not in ['running', 'done', 'cancel'])
res = True
if new_self:
res = super(ProcurementOrder, new_self).run(autocommit=autocommit)
# after all the procurements are run, check if some created a draft stock move that needs to be confirmed
# (we do that in batch because it fasts the picking assignation and the picking state computation)
move_ids = new_self.filtered(lambda order: order.state == 'running' and order.rule_id.action == 'move').mapped('move_ids').filtered(lambda move: move.state == 'draft')
if move_ids:
move_ids.action_confirm()
# TDE FIXME: action_confirm in stock_move already call run() ... necessary ??
# If procurements created other procurements, run the created in batch
new_procurements = self.search([('move_dest_id.procurement_id', 'in', new_self.ids)], order='id')
if new_procurements:
res = new_procurements.run(autocommit=autocommit)
return res
@api.multi
def _check(self):
""" Checking rules of type 'move': satisfied only if all related moves
are done/cancel and if the requested quantity is moved. """
if self.rule_id.action == 'move':
# In case Phantom BoM splits only into procurements
if not self.move_ids:
return True
move_all_done_or_cancel = all(move.state in ['done', 'cancel'] for move in self.move_ids)
move_all_cancel = all(move.state == 'cancel' for move in self.move_ids)
if not move_all_done_or_cancel:
return False
elif move_all_done_or_cancel and not move_all_cancel:
return True
else:
self.message_post(body=_('All stock moves have been cancelled for this procurement.'))
# TDE FIXME: strange that a check method actually modified the procurement...
self.write({'state': 'cancel'})
return False
return super(ProcurementOrder, self)._check()
@api.model
def run_scheduler(self, use_new_cursor=False, company_id=False):
''' Call the scheduler in order to check the running procurements (super method), to check the minimum stock rules
and the availability of moves. This function is intended to be run for all the companies at the same time, so
we run functions as SUPERUSER to avoid intercompanies and access rights issues. '''
super(ProcurementOrder, self).run_scheduler(use_new_cursor=use_new_cursor, company_id=company_id)
try:
if use_new_cursor:
cr = registry(self._cr.dbname).cursor()
self = self.with_env(self.env(cr=cr)) # TDE FIXME
# Minimum stock rules
self.sudo()._procure_orderpoint_confirm(use_new_cursor=use_new_cursor, company_id=company_id)
# Search all confirmed stock_moves and try to assign them
confirmed_moves = self.env['stock.move'].search([('state', '=', 'confirmed'), ('product_uom_qty', '!=', 0.0)], limit=None, order='priority desc, date_expected asc')
for x in xrange(0, len(confirmed_moves.ids), 100):
# TDE CLEANME: muf muf
self.env['stock.move'].browse(confirmed_moves.ids[x:x + 100]).action_assign()
if use_new_cursor:
self._cr.commit()
if use_new_cursor:
self._cr.commit()
finally:
if use_new_cursor:
try:
self._cr.close()
except Exception:
pass
return {}
@api.model
def _procurement_from_orderpoint_get_order(self):
return 'location_id'
@api.model
def _procurement_from_orderpoint_get_grouping_key(self, orderpoint_ids):
orderpoints = self.env['stock.warehouse.orderpoint'].browse(orderpoint_ids)
return orderpoints.location_id.id
@api.model
def _procurement_from_orderpoint_get_groups(self, orderpoint_ids):
""" Make groups for a given orderpoint; by default schedule all operations in one without date """
return [{'to_date': False, 'procurement_values': dict()}]
@api.model
def _procurement_from_orderpoint_post_process(self, orderpoint_ids):
return True
def _get_orderpoint_domain(self, company_id=False):
domain = [('company_id', '=', company_id)] if company_id else []
domain += [('product_id.active', '=', True)]
return domain
@api.model
def _procure_orderpoint_confirm(self, use_new_cursor=False, company_id=False):
""" Create procurements based on orderpoints.
:param bool use_new_cursor: if set, use a dedicated cursor and auto-commit after processing
1000 orderpoints.
This is appropriate for batch jobs only.
"""
if company_id and self.env.user.company_id.id != company_id:
# To ensure that the company_id is taken into account for
# all the processes triggered by this method
# i.e. If a PO is generated by the run of the procurements the
# sequence to use is the one for the specified company not the
# one of the user's company
self = self.with_context(company_id=company_id, force_company=company_id)
OrderPoint = self.env['stock.warehouse.orderpoint']
domain = self._get_orderpoint_domain(company_id=company_id)
orderpoints_noprefetch = OrderPoint.with_context(prefetch_fields=False).search(domain,
order=self._procurement_from_orderpoint_get_order()).ids
while orderpoints_noprefetch:
if use_new_cursor:
cr = registry(self._cr.dbname).cursor()
self = self.with_env(self.env(cr=cr))
OrderPoint = self.env['stock.warehouse.orderpoint']
Procurement = self.env['procurement.order']
ProcurementAutorundefer = Procurement.with_context(procurement_autorun_defer=True)
procurement_list = []
orderpoints = OrderPoint.browse(orderpoints_noprefetch[:1000])
orderpoints_noprefetch = orderpoints_noprefetch[1000:]
# Calculate groups that can be executed together
location_data = defaultdict(lambda: dict(products=self.env['product.product'], orderpoints=self.env['stock.warehouse.orderpoint'], groups=list()))
for orderpoint in orderpoints:
key = self._procurement_from_orderpoint_get_grouping_key([orderpoint.id])
location_data[key]['products'] += orderpoint.product_id
location_data[key]['orderpoints'] += orderpoint
location_data[key]['groups'] = self._procurement_from_orderpoint_get_groups([orderpoint.id])
for location_id, location_data in location_data.iteritems():
location_orderpoints = location_data['orderpoints']
product_context = dict(self._context, location=location_orderpoints[0].location_id.id)
substract_quantity = location_orderpoints.subtract_procurements_from_orderpoints()
for group in location_data['groups']:
if group.get('from_date'):
product_context['from_date'] = group['from_date'].strftime(DEFAULT_SERVER_DATETIME_FORMAT)
if group['to_date']:
product_context['to_date'] = group['to_date'].strftime(DEFAULT_SERVER_DATETIME_FORMAT)
product_quantity = location_data['products'].with_context(product_context)._product_available()
for orderpoint in location_orderpoints:
try:
op_product_virtual = product_quantity[orderpoint.product_id.id]['virtual_available']
if op_product_virtual is None:
continue
if float_compare(op_product_virtual, orderpoint.product_min_qty, precision_rounding=orderpoint.product_uom.rounding) <= 0:
qty = max(orderpoint.product_min_qty, orderpoint.product_max_qty) - op_product_virtual
remainder = orderpoint.qty_multiple > 0 and qty % orderpoint.qty_multiple or 0.0
if float_compare(remainder, 0.0, precision_rounding=orderpoint.product_uom.rounding) > 0:
qty += orderpoint.qty_multiple - remainder
if float_compare(qty, 0.0, precision_rounding=orderpoint.product_uom.rounding) < 0:
continue
qty -= substract_quantity[orderpoint.id]
qty_rounded = float_round(qty, precision_rounding=orderpoint.product_uom.rounding)
if qty_rounded > 0:
new_procurement = ProcurementAutorundefer.create(
orderpoint._prepare_procurement_values(qty_rounded, **group['procurement_values']))
procurement_list.append(new_procurement)
new_procurement.message_post_with_view('mail.message_origin_link',
values={'self': new_procurement, 'origin': orderpoint},
subtype_id=self.env.ref('mail.mt_note').id)
self._procurement_from_orderpoint_post_process([orderpoint.id])
if use_new_cursor:
cr.commit()
except OperationalError:
if use_new_cursor:
orderpoints_noprefetch += [orderpoint.id]
cr.rollback()
continue
else:
raise
try:
# TDE CLEANME: use record set ?
procurement_list.reverse()
procurements = self.env['procurement.order']
for p in procurement_list:
procurements += p
procurements.run()
if use_new_cursor:
cr.commit()
except OperationalError:
if use_new_cursor:
cr.rollback()
continue
else:
raise
if use_new_cursor:
cr.commit()
cr.close()
return {}