Files
tradon/modules/account_fr/account.py
2025-12-26 13:11:43 +00:00

338 lines
12 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.
import csv
from io import BytesIO, TextIOWrapper
from sql import Table
from sql.aggregate import Sum
from sql.conditionals import Coalesce
from trytond.config import config
from trytond.model import ModelStorage, ModelView, fields
from trytond.pool import Pool, PoolMeta
from trytond.pyson import Eval
from trytond.transaction import Transaction
from trytond.wizard import Button, StateTransition, StateView, Wizard
OPENING_CODE = config.get('account_fr', 'fec_opening_code', default="OUV")
OPENING_NAME = config.get(
'account_fr', 'fec_opening_name', default="Balance Initiale")
OPENING_NUMBER = config.get('account_fr', 'fec_opening_number', default="0")
class AccountTemplate(metaclass=PoolMeta):
__name__ = 'account.account.template'
@classmethod
def __register__(cls, module_name):
cursor = Transaction().connection.cursor()
model_data = Table('ir_model_data')
# Migration from 6.0: rename ids
if module_name == 'account_fr':
for old_id, new_id in (
('fr_pcg_pay', '4011'),
('fr_pcg_recv', '4111'),
('fr_pcg_cash', '512'),
('fr_pcg_expense', '607'),
):
cursor.execute(*model_data.select(model_data.id,
where=(model_data.fs_id == new_id)
& (model_data.module == module_name)))
if cursor.fetchone():
continue
cursor.execute(*model_data.update(
columns=[model_data.fs_id],
values=[new_id],
where=(model_data.fs_id == old_id)
& (model_data.module == module_name)))
super().__register__(module_name)
class CreateChart(metaclass=PoolMeta):
__name__ = 'account.create_chart'
def default_properties(self, fields):
pool = Pool()
ModelData = pool.get('ir.model.data')
defaults = super().default_properties(fields)
template_id = ModelData.get_id('account_fr.root')
if self.account.account_template.id == template_id:
defaults['account_receivable'] = self.get_account(
'account_fr.4111')
defaults['account_payable'] = self.get_account(
'account_fr.4011')
return defaults
class FrFEC(Wizard):
'Generate FEC'
__name__ = 'account.fr.fec'
start = StateView('account.fr.fec.start',
'account_fr.fec_start_view_form', [
Button('Cancel', 'end', 'tryton-cancel'),
Button('Generate', 'generate', 'tryton-ok', default=True),
])
generate = StateTransition()
result = StateView('account.fr.fec.result',
'account_fr.fec_result_view_form', [
Button('Close', 'end', 'tryton-close'),
])
def transition_generate(self):
fec = BytesIO()
writer = self.get_writer(
TextIOWrapper(fec, encoding='utf-8', write_through=True))
writer.writerow(self.get_header())
format_date = self.get_format_date()
format_number = self.get_format_number()
def convert(c):
delimiter = writer.dialect.delimiter
return (c or '').replace(delimiter, ' ').replace('\n', ' ')
for row in self.get_start_balance():
writer.writerow(map(convert, row))
for line in self.get_lines():
row = self.get_row(line, format_date, format_number)
writer.writerow(map(convert, row))
self.result.file = self.result.__class__.file.cast(fec.getvalue())
return 'result'
def default_result(self, fields):
file_ = self.result.file
self.result.file = None # No need to store it in session
format_date = self.get_format_date()
if self.start.fiscalyear.state == 'locked':
filename = '%sFEC%s.csv' % (
self.start.fiscalyear.company.party.siren or '',
format_date(self.start.fiscalyear.end_date),
)
else:
filename = None
return {
'file': file_,
'filename': filename,
}
def get_writer(self, fd):
return csv.writer(
fd, delimiter='\t', doublequote=False, escapechar='\\',
quoting=csv.QUOTE_NONE)
def get_format_date(self):
pool = Pool()
Lang = pool.get('ir.lang')
fr = Lang(code='fr_FR', date='%d.%m.%Y')
return lambda value: fr.strftime(value, '%Y%m%d')
def get_format_number(self):
pool = Pool()
Lang = pool.get('ir.lang')
fr = Lang(
decimal_point=',',
thousands_sep='',
grouping='[]',
)
return lambda value: fr.format('%.2f', value)
def get_header(self):
return [
'JournalCode',
'JournalLib',
'EcritureNum',
'EcritureDate',
'CompteNum',
'CompteLib',
'CompAuxNum',
'CompAuxLib',
'PieceRef',
'PieceDate',
'EcritureLib',
'Debit',
'Credit',
'EcritureLet',
'DateLet',
'ValidDate',
'Montantdevise',
'Idevise',
]
def get_start_balance(self):
pool = Pool()
Account = pool.get('account.account')
Move = pool.get('account.move')
Line = pool.get('account.move.line')
FiscalYear = pool.get('account.fiscalyear')
Period = pool.get('account.period')
Party = pool.get('party.party')
account = Account.__table__()
move = Move.__table__()
line = Line.__table__()
period = Period.__table__()
party = Party.__table__()
cursor = Transaction().connection.cursor()
format_date = self.get_format_date()
format_number = self.get_format_number()
company = self.start.fiscalyear.company
fiscalyears = FiscalYear.search([
('end_date', '<', self.start.fiscalyear.start_date),
('company', '=', company.id),
])
fiscalyear_ids = list(map(int, fiscalyears))
if not fiscalyear_ids:
return
query = (account
.join(line, condition=line.account == account.id)
.join(move, condition=line.move == move.id)
.join(period, condition=move.period == period.id)
.join(party, 'LEFT', condition=line.party == party.id)
.select(
account.id,
account.code,
account.name,
party.code,
party.name,
Sum(Coalesce(line.debit, 0) - Coalesce(line.credit, 0)),
where=(account.company == company.id)
& (move.state == 'posted')
& (line.state != 'draft')
& period.fiscalyear.in_(fiscalyear_ids),
group_by=[
account.id, account.code, account.name,
party.code, party.name],
order_by=account.code))
cursor.execute(*query)
for row in cursor:
_, code, name, party_code, party_name, balance = row
if not balance:
continue
if balance > 0:
debit, credit = balance, 0
else:
debit, credit = 0, -balance
yield [
OPENING_CODE,
OPENING_NAME,
OPENING_NUMBER,
format_date(self.start.fiscalyear.start_date),
code,
name,
party_code or '',
party_name or '',
'-',
format_date(self.start.fiscalyear.start_date),
'-',
format_number(debit),
format_number(credit),
'',
'',
format_date(self.start.fiscalyear.start_date),
'',
'',
]
def get_lines(self):
pool = Pool()
Line = pool.get('account.move.line')
domain = [
('move.period.fiscalyear', '=', self.start.fiscalyear.id),
('move.state', '=', 'posted'),
['OR', ('debit', '!=', 0), ('credit', '!=', 0)],
]
if self.start.deferral_period:
domain.append(('move.period', '!=', self.start.deferral_period.id))
return Line.search(
domain,
order=[
('move.post_date', 'ASC'),
('move.post_number', 'ASC'),
('id', 'ASC'),
])
def get_row(self, line, format_date, format_number):
def description():
value = line.move.description or ''
if line.description:
if value:
value += ' - '
value += line.description
return value
end_date = self.start.fiscalyear.end_date
reconciliation = None
if line.reconciliation and line.reconciliation.date <= end_date:
reconciliation = line.reconciliation
return [
line.move.journal.code or line.move.journal.name,
line.move.journal.name,
line.move.post_number,
format_date(line.move.date),
line.account.code,
line.account.name,
line.party.code if line.party else '',
line.party.name if line.party else '',
self.get_reference(line) or '-',
format_date(self.get_reference_date(line)),
description() or '-',
format_number(line.debit or 0),
format_number(line.credit or 0),
reconciliation.rec_name if reconciliation else '',
format_date(reconciliation.create_date.date())
if reconciliation else '',
format_date(line.move.post_date),
format_number(line.amount_second_currency)
if line.amount_second_currency else '',
line.second_currency.code if line.amount_second_currency else '',
]
def get_reference(self, line):
if isinstance(line.move.origin, ModelStorage):
return line.move.origin.rec_name
return line.move.origin
def get_reference_date(self, line):
pool = Pool()
try:
Invoice = pool.get('account.invoice')
except KeyError:
Invoice = None
if Invoice and isinstance(line.move.origin, Invoice):
return line.move.origin.invoice_date
return line.move.date
class FrFECStart(ModelView):
'Generate FEC'
__name__ = 'account.fr.fec.start'
fiscalyear = fields.Many2One(
'account.fiscalyear', 'Fiscal Year', required=True)
type = fields.Selection([
('is-bic', 'IS-BIC'),
], 'Type', required=True)
deferral_period = fields.Many2One(
'account.period', "Deferral Period", required=True,
domain=[
('fiscalyear', '=', Eval('fiscalyear')),
('type', '=', 'adjustment'),
],
help="The period to exclude which contains "
"the moves to balance non-deferral account")
@classmethod
def default_type(cls):
return 'is-bic'
class FrFECResult(ModelView):
'Generate FEC'
__name__ = 'account.fr.fec.result'
file = fields.Binary('File', readonly=True, filename='filename')
filename = fields.Char('File Name', readonly=True)