Files
tradon/modules/product_cost_fifo/move.py
2026-03-14 09:42:12 +00:00

176 lines
6.6 KiB
Python

# This file is part of Tryton. The COPYRIGHT file at the top level of
# this repository contains the full copyright notices and license terms.
from decimal import Decimal
from sql import Literal, operators
from trytond.i18n import gettext
from trytond.model import Check, ModelView, Workflow, fields
from trytond.model.exceptions import AccessError
from trytond.modules.product import round_price
from trytond.pool import Pool, PoolMeta
from trytond.pyson import Eval
from trytond.transaction import Transaction
class Move(metaclass=PoolMeta):
__name__ = 'stock.move'
fifo_quantity = fields.Float(
"FIFO Quantity", required=True,
domain=[
('fifo_quantity', '<=', Eval('quantity', 0)),
],
help="Quantity used by FIFO.")
fifo_quantity_available = fields.Function(fields.Float(
"FIFO Quantity Available",
help="Quantity available for FIFO"),
'get_fifo_quantity_available')
@classmethod
def __setup__(cls):
super().__setup__()
cls._allow_modify_closed_period.add('fifo_quantity')
t = cls.__table__()
cls._sql_constraints += [
('check_fifo_quantity',
Check(t, t.quantity >= t.fifo_quantity),
'product_cost_fifo.msg_move_fifo_quantity_greater'),
]
@classmethod
def __register__(cls, module):
table_h = cls.__table_handler__(module)
super().__register__(module)
# Migration from 6.6: rename check_fifo_quantity_out to
# check_fifo_quantity
table_h.drop_constraint('check_fifo_quantity_out')
@staticmethod
def default_fifo_quantity():
return 0.0
def get_fifo_quantity_available(self, name):
return self.quantity - (self.fifo_quantity or 0)
@classmethod
def domain_fifo_quantity_available(cls, domain, tables):
table, _ = tables[None]
name, operator, value = domain
field = cls.fifo_quantity_available._field
Operator = fields.SQL_OPERATORS[operator]
column = (
cls.quantity.sql_column(table)
- cls.fifo_quantity.sql_column(table))
expression = Operator(column, field._domain_value(operator, value))
if isinstance(expression, operators.In) and not expression.right:
expression = Literal(False)
elif isinstance(expression, operators.NotIn) and not expression.right:
expression = Literal(True)
expression = field._domain_add_null(
column, operator, value, expression)
return expression
def _update_fifo_out_product_cost_price(self):
'''
Update the product cost price of the given product on the move. Update
fifo_quantity on the concerned incoming moves. Return the
cost price for outputing the given product and quantity.
'''
pool = Pool()
Uom = pool.get('product.uom')
total_qty = Uom.compute_qty(
self.unit, self.quantity, self.product.default_uom, round=False)
with Transaction().set_context(company=self.company.id):
fifo_moves = self.product.get_fifo_move(total_qty)
cost_price = Decimal(0)
consumed_qty = 0.0
to_save = []
for move, move_qty in fifo_moves:
consumed_qty += move_qty
cost_price += move.get_cost_price() * Decimal(str(move_qty))
move_qty = Uom.compute_qty(
self.product.default_uom, move_qty, move.unit, round=False)
move.fifo_quantity = (move.fifo_quantity or 0.0) + move_qty
# Due to float, the fifo quantity result can exceed the quantity.
assert move.quantity >= move.fifo_quantity - move.unit.rounding
move.fifo_quantity = min(move.fifo_quantity, move.quantity)
to_save.append(move)
if consumed_qty:
cost_price = cost_price / Decimal(str(consumed_qty))
else:
cost_price = self.product.get_multivalue(
'cost_price', **self._cost_price_pattern)
# Compute average cost price
average_cost_price = self._compute_product_cost_price(
'out', product_cost_price=cost_price)
if cost_price:
cost_price = round_price(cost_price)
else:
cost_price = average_cost_price
return cost_price, average_cost_price, to_save
def _do(self):
cost_price, to_save = super()._do()
cost_price_method = self.product.get_multivalue(
'cost_price_method', **self._cost_price_pattern)
if (self.from_location.type != 'storage'
and self.to_location.type == 'storage'
and cost_price_method == 'fifo'):
cost_price = self._compute_product_cost_price('in')
elif (self.to_location.type == 'supplier'
and self.from_location.type == 'storage'
and cost_price_method == 'fifo'):
cost_price = self._compute_product_cost_price('out')
elif (self.from_location.type == 'storage'
and self.to_location.type != 'storage'
and cost_price_method == 'fifo'):
fifo_cost_price, cost_price, moves = (
self._update_fifo_out_product_cost_price())
if self.cost_price_required:
if self.cost_price is None:
self.cost_price = fifo_cost_price
if self.product_cost_price is None:
self.product_cost_price = cost_price
to_save.extend(moves)
return cost_price, to_save
@classmethod
@ModelView.button
@Workflow.transition('cancelled')
def cancel(cls, moves):
for move in moves:
if move.fifo_quantity:
raise AccessError(
gettext('product_cost_fifo.msg_move_cancel_fifo',
move=move.rec_name))
super().cancel(moves)
@classmethod
def check_modification(cls, mode, moves, values=None, external=False):
super().check_modification(
mode, moves, values=values, external=external)
if mode == 'delete':
for move in moves:
if move.fifo_quantity:
raise AccessError(gettext(
'product_cost_fifo.msg_move_delete_fifo',
move=move.rec_name))
@classmethod
def copy(cls, moves, default=None):
if default is None:
default = {}
else:
default = default.copy()
default.setdefault('fifo_quantity', cls.default_fifo_quantity())
return super().copy(moves, default=default)