Files
2026-03-14 09:42:12 +00:00

163 lines
5.9 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 For, Literal, Null
from sql.aggregate import Sum
from sql.conditionals import Case, Coalesce
from trytond import backend
from trytond.i18n import gettext
from trytond.model import fields
from trytond.modules.currency.fields import Monetary
from trytond.modules.party.exceptions import EraseError
from trytond.pool import Pool, PoolMeta
from trytond.tools import grouped_slice, reduce_ids, sqlite_apply_types
from trytond.transaction import Transaction
class Party(metaclass=PoolMeta):
__name__ = 'party.party'
deposit = fields.Function(Monetary(
"Deposit", currency='currency', digits='currency'),
'get_deposit', searcher='search_deposit')
@classmethod
def get_deposit(cls, parties, name):
pool = Pool()
MoveLine = pool.get('account.move.line')
Account = pool.get('account.account')
AccountType = pool.get('account.account.type')
User = pool.get('res.user')
cursor = Transaction().connection.cursor()
line = MoveLine.__table__()
account = Account.__table__()
account_type = AccountType.__table__()
values = {p.id: Decimal(0) for p in parties}
user = User(Transaction().user)
if not user.company:
return values
currency = user.company.currency
line_clause, _ = MoveLine.query_get(line)
for sub_parties in grouped_slice(parties):
party_clause = reduce_ids(line.party, [p.id for p in sub_parties])
query = (line
.join(account, condition=account.id == line.account)
.join(account_type, condition=account.type == account_type.id)
.select(line.party,
# Use credit - debit to positive deposit amount
Sum(Coalesce(line.credit, 0) - Coalesce(line.debit, 0)
).as_('debit'),
where=account_type.deposit
& party_clause
& (line.reconciliation == Null)
& (account.company == user.company.id)
& line_clause,
group_by=line.party))
if backend.name == 'sqlite':
sqlite_apply_types(query, [None, 'NUMERIC'])
cursor.execute(*query)
values.update((p, currency.round(a)) for p, a in cursor)
return values
@classmethod
def search_deposit(cls, name, clause):
pool = Pool()
MoveLine = pool.get('account.move.line')
Account = pool.get('account.account')
AccountType = pool.get('account.account.type')
User = pool.get('res.user')
line = MoveLine.__table__()
account = Account.__table__()
account_type = AccountType.__table__()
user = User(Transaction().user)
if not user.company:
return []
line_clause, _ = MoveLine.query_get(line)
Operator = fields.SQL_OPERATORS[clause[1]]
query = (line
.join(account, condition=account.id == line.account)
.join(account_type, condition=account.type == account_type.id)
.select(line.party,
where=account.active
& account_type.deposit
& (line.party != Null)
& (line.reconciliation == Null)
& (account.company == user.company.id)
& line_clause,
group_by=line.party,
having=Operator(
# Use credit - debit to positive deposit amount
Sum(Coalesce(line.credit, 0) - Coalesce(line.debit, 0)),
Decimal(clause[2] or 0))))
return [('id', 'in', query)]
def get_deposit_balance(self, deposit_account, currency=None):
'Return the deposit account balance (debit - credit) for the party'
pool = Pool()
MoveLine = pool.get('account.move.line')
transaction = Transaction()
cursor = transaction.connection.cursor()
line = MoveLine.__table__()
if currency is None:
currency = deposit_account.currency
assert deposit_account.type.deposit
where = ((line.account == deposit_account.id)
& (line.party == self.id)
& (line.reconciliation == Null))
if transaction.database.has_select_for():
cursor.execute(*line.select(
Literal(1),
where=where,
for_=For('UPDATE', nowait=True)))
else:
MoveLine.lock()
if currency == deposit_account.currency:
amount = Sum(Coalesce(line.debit, 0) - Coalesce(line.credit, 0))
else:
amount = Sum(Case(
(line.second_currency == currency.id,
line.amount_second_currency),
else_=0))
query = line.select(amount.as_('deposit_balance'), where=where)
if backend.name == 'sqlite':
sqlite_apply_types(query, ['NUMERIC'])
cursor.execute(*query)
amount, = cursor.fetchone()
if amount is None:
amount = Decimal(0)
return currency.round(amount)
def check_deposit(self, deposit_account, sign=1):
'''Check if the deposit account balance (debit - credit) has the same
sign for the party'''
assert sign in (1, -1)
amount = self.get_deposit_balance(
deposit_account, currency=deposit_account.second_currency)
return not amount or ((amount < 0) == (sign < 0))
class Erase(metaclass=PoolMeta):
__name__ = 'party.erase'
def check_erase_company(self, party, company):
if party.deposit:
raise EraseError(
gettext('account_deposit.msg_erase_party_deposit',
party=party.rec_name,
company=company.rec_name))