# 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)