Files
2025-12-26 13:11:43 +00:00

184 lines
6.8 KiB
Python
Executable File

# 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 trytond.model import Check, fields
from trytond.pool import Pool, PoolMeta
from trytond.transaction import Transaction
import logging
logger = logging.getLogger(__name__)
def _get_field(type_):
if type_.startswith('in_'):
return 'in_anglo_saxon_quantity'
else:
return 'out_anglo_saxon_quantity'
class Move(metaclass=PoolMeta):
__name__ = 'stock.move'
in_anglo_saxon_quantity = fields.Float('Input Anglo-Saxon Quantity',
required=True)
out_anglo_saxon_quantity = fields.Float('Output Anglo-Saxon Quantity',
required=True)
@classmethod
def __setup__(cls):
super(Move, cls).__setup__()
cls._allow_modify_closed_period.update(['in_anglo_saxon_quantity',
'out_anglo_saxon_quantity'])
t = cls.__table__()
cls._sql_constraints += [
('check_in_anglo_saxon_quantity',
Check(t, t.quantity >= t.in_anglo_saxon_quantity),
'account_stock_anglo_saxon.msg_move_quantity_greater'),
('check_out_anglo_saxon_quantity',
Check(t, t.quantity >= t.out_anglo_saxon_quantity),
'account_stock_anglo_saxon.msg_move_quantity_greater'),
]
@staticmethod
def default_in_anglo_saxon_quantity():
return 0.0
@staticmethod
def default_out_anglo_saxon_quantity():
return 0.0
def _get_account_stock_move_lines(self, type_,fee=None):
pool = Pool()
Uom = pool.get('product.uom')
AccountMoveLine = pool.get('account.move.line')
Currency = pool.get('currency.currency')
lines = super(Move, self)._get_account_stock_move_lines(type_,fee)
logger.info("FROM_MOVE_GET:%s",lines)
cost_price_method = self.product.get_multivalue(
'cost_price_method', **self._cost_price_pattern)
# if type_.endswith('supplier') and cost_price_method == 'fixed':
# cost_price = Uom.compute_price(
# self.product.default_uom, self.cost_price, self.unit)
# with Transaction().set_context(date=self.effective_date):
# unit_price = Currency.compute(self.currency, self.unit_price,
# self.company.currency, round=False)
# amount = self.company.currency.round(
# Decimal(str(self.quantity)) * (unit_price - cost_price))
# if self.company.currency.is_zero(amount):
# return lines
# account = self.product.account_stock_in_used
# for move_line in lines:
# if move_line.account == account:
# break
# else:
# return lines
# if type_.startswith('in_'):
# move_line.credit += amount
# debit = amount
# credit = Decimal(0)
# else:
# move_line.debit += amount
# debit = Decimal(0)
# credit = amount
# if amount < Decimal(0):
# debit, credit = -credit, -debit
# move_line = AccountMoveLine(
# debit=debit,
# credit=credit,
# account=self.product.account_expense_used,
# )
# lines.append(move_line)
return lines
@classmethod
def _get_anglo_saxon_move(cls, moves, quantity, type_):
'''
Generator of (move, qty, cost_price) where move is the move to be
consumed, qty is the quantity (in the product default uom) to be
consumed on this move and cost_price is in the company currency.
'''
pool = Pool()
Uom = pool.get('product.uom')
Currency = pool.get('currency.currency')
as_qty_field = _get_field(type_)
consumed_qty = 0.0
for move in moves:
qty = Uom.compute_qty(
move.unit,
move.quantity - getattr(move, as_qty_field),
move.product.default_uom, round=False)
if qty <= 0.0:
continue
if qty > quantity - consumed_qty:
qty = quantity - consumed_qty
if consumed_qty >= quantity:
break
# if type_.endswith('supplier'):
# with Transaction().set_context(date=move.effective_date):
# unit_price = Currency.compute(move.currency,
# move.unit_price, move.company.currency, round=False)
# cost_price = Uom.compute_price(
# move.unit, unit_price, move.product.default_uom)
# else:
# cost_price = move.cost_price
cost_price = move.unit_price
yield (move, qty, cost_price)
consumed_qty += qty
@classmethod
def update_anglo_saxon_quantity_product_cost(cls, product, moves,
quantity, unit, type_):
'''
Return the cost for quantity based on lines.
Update anglo_saxon_quantity on the concerned moves.
'''
pool = Pool()
Uom = pool.get('product.uom')
assert all(m.product == product for m in moves), 'wrong product'
assert type_.startswith('in_') or type_.startswith('out_'), \
'wrong type'
total_qty = Uom.compute_qty(
unit, quantity, product.default_uom, round=False)
as_qty_field = _get_field(type_)
cost = Decimal(0)
consumed_qty = 0.0
for move, move_qty, move_cost_price in cls._get_anglo_saxon_move(
moves, total_qty, type_):
consumed_qty += move_qty
logger.info("ANGLO_UPDATE:%s",move_cost_price)
cost += move_cost_price * Decimal(str(move_qty))
move_qty = Uom.compute_qty(
product.default_uom, move_qty, move.unit, round=False)
# Avoid float rounding issue but allow only rounding precision lost
new_qty = (getattr(move, as_qty_field) or 0.0) + move_qty
assert move.unit.round(new_qty) <= move.quantity
new_qty = min(new_qty, move.quantity)
cls.write([move], {
as_qty_field: new_qty,
})
if consumed_qty < total_qty:
qty = total_qty - consumed_qty
consumed_qty += qty
cost += product.cost_price * Decimal(str(qty))
return cost
@classmethod
def copy(cls, moves, default=None):
if default is None:
default = {}
else:
default = default.copy()
for prefix in ('in_', 'out_'):
default.setdefault(prefix + 'anglo_saxon_quantity',
getattr(cls, 'default_%sanglo_saxon_quantity' % prefix)())
return super(Move, cls).copy(moves, default=default)