# 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 import trytond.config as 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 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): __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 [ config.get( 'account_fr', 'fec_opening_code', default="OUV"), config.get( 'account_fr', 'fec_opening_name', default="Balance Initiale"), config.get( 'account_fr', 'fec_opening_number', default="0"), 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.number', 'ASC'), ('id', 'ASC'), ]) def get_row(self, line, format_date, format_number): 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.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)), line.description_used 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): __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', -1)), ('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): __name__ = 'account.fr.fec.result' file = fields.Binary('File', readonly=True, filename='filename') filename = fields.Char('File Name', readonly=True)