# -*- coding: utf-8 -*- # Copyright(C) 2009-2011 Romain Bignon, Christophe Benz # # This file is part of weboob. # # weboob is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # weboob is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with weboob. If not, see . from __future__ import print_function import datetime, uuid from dateutil.relativedelta import relativedelta from dateutil.parser import parse as parse_date from decimal import Decimal, InvalidOperation from weboob.capabilities.base import empty from weboob.capabilities.bank import CapBank, Account, Transaction from weboob.tools.application.repl import ReplApplication, defaultcount from weboob.tools.application.formatters.iformatter import IFormatter, PrettyFormatter __all__ = ['Boobank'] class OfxFormatter(IFormatter): MANDATORY_FIELDS = ('id', 'date', 'raw', 'amount', 'category') TYPES_ACCTS = ['', 'CHECKING', 'SAVINGS', 'DEPOSIT', 'LOAN', 'MARKET', 'JOINT', 'CARD'] TYPES_TRANS = ['', 'DIRECTDEP', 'PAYMENT', 'CHECK', 'DEP', 'OTHER', 'ATM', 'POS', 'INT', 'FEE'] TYPES_CURRS = ['', 'EUR', 'CHF', 'USD'] balance = Decimal(0) coming = Decimal(0) def start_format(self, **kwargs): account = kwargs['account'] self.balance = account.balance self.coming = account.coming self.output(u'OFXHEADER:100') self.output(u'DATA:OFXSGML') self.output(u'VERSION:102') self.output(u'SECURITY:NONE') self.output(u'ENCODING:USASCII') self.output(u'CHARSET:1252') self.output(u'COMPRESSION:NONE') self.output(u'OLDFILEUID:NONE') self.output(u'NEWFILEUID:%s\n' % uuid.uuid1()) self.output(u'0INFO') self.output(u'%s113942ENG' % datetime.date.today().strftime('%Y%m%d')) self.output(u'%s' % uuid.uuid1()) self.output(u'0INFOnull') self.output(u'%s' % (account.currency or 'EUR')) self.output(u'null') self.output(u'null') self.output(u'%s' % account.id) try: account_type = self.TYPES_ACCTS[account.type] except IndexError: account_type = '' self.output(u'%s' % (account_type or 'CHECKING')) self.output(u'null') self.output(u'') self.output(u'%s' % datetime.date.today().strftime('%Y%m%d')) self.output(u'%s' % datetime.date.today().strftime('%Y%m%d')) def format_obj(self, obj, alias): if obj.type != 0: result = u'%s\n' % self.TYPES_TRANS[obj.type] else: result = u'%s\n' % ('DEBIT' if obj.amount < 0 else 'CREDIT') result += u'%s\n' % obj.date.strftime('%Y%m%d') result += u'%s\n' % obj.amount result += u'%s\n' % obj.unique_id() if hasattr(obj, 'label') and not empty(obj.label): result += u'%s' % obj.label.replace('&', '&') else: result += u'%s' % obj.raw.replace('&', '&') return result def flush(self): self.output(u'') self.output(u'%s' % self.balance) self.output(u'%s' % datetime.date.today().strftime('%Y%m%d')) try: self.output(u'%s' % (self.balance + self.coming)) except TypeError: self.output(u'%s' % self.balance) self.output(u'%s' % datetime.date.today().strftime('%Y%m%d')) self.output(u'') class QifFormatter(IFormatter): MANDATORY_FIELDS = ('id', 'date', 'raw', 'amount') def start_format(self, **kwargs): self.output(u'!Type:Bank') def format_obj(self, obj, alias): result = u'D%s\n' % obj.date.strftime('%d/%m/%y') result += u'T%s\n' % obj.amount if hasattr(obj, 'category') and not empty(obj.category): result += u'N%s\n' % obj.category result += u'M%s\n' % obj.raw result += u'^' return result class PrettyQifFormatter(QifFormatter): MANDATORY_FIELDS = ('id', 'date', 'raw', 'amount', 'category') def start_format(self, **kwargs): self.output(u'!Type:Bank') def format_obj(self, obj, alias): if hasattr(obj, 'rdate') and not empty(obj.rdate): result = u'D%s\n' % obj.rdate.strftime('%d/%m/%y') else: result = u'D%s\n' % obj.date.strftime('%d/%m/%y') result += u'T%s\n' % obj.amount if hasattr(obj, 'category') and not empty(obj.category): result += u'N%s\n' % obj.category if hasattr(obj, 'label') and not empty(obj.label): result += u'M%s\n' % obj.label else: result += u'M%s\n' % obj.raw result += u'^' return result class TransactionsFormatter(IFormatter): MANDATORY_FIELDS = ('date', 'label', 'amount') TYPES = ['', 'Transfer', 'Order', 'Check', 'Deposit', 'Payback', 'Withdrawal', 'Card', 'Loan', 'Bank'] def start_format(self, **kwargs): self.output(' Date Category Label Amount ') self.output('------------+------------+---------------------------------------------------+-----------') def format_obj(self, obj, alias): if hasattr(obj, 'category') and obj.category: _type = obj.category else: try: _type = self.TYPES[obj.type] except (IndexError, AttributeError): _type = '' label = obj.label if not label and hasattr(obj, 'raw'): label = obj.raw date = obj.date.strftime('%Y-%m-%d') if not empty(obj.date) else '' amount = obj.amount or Decimal('0') return ' %s %s %s %s' % (self.colored('%-10s' % date, 'blue'), self.colored('%-12s' % _type[:12], 'magenta'), self.colored('%-50s' % label[:50], 'yellow'), self.colored('%10.2f' % amount, 'green' if amount >= 0 else 'red')) class TransferFormatter(IFormatter): MANDATORY_FIELDS = ('id', 'date', 'origin', 'recipient', 'amount') DISPLAYED_FIELDS = ('reason', ) def format_obj(self, obj, alias): result = u'------- Transfer %s -------\n' % obj.fullid result += u'Date: %s\n' % obj.date result += u'Origin: %s\n' % obj.origin result += u'Recipient: %s\n' % obj.recipient result += u'Amount: %.2f\n' % obj.amount if obj.reason: result += u'Reason: %s\n' % obj.reason return result class InvestmentFormatter(IFormatter): MANDATORY_FIELDS = ('label', 'quantity', 'unitvalue') DISPLAYED_FIELDS = ('code', 'diff') tot_valuation = Decimal(0) tot_diff = Decimal(0) def start_format(self, **kwargs): self.output(' Label Code Quantity Unit Value Valuation diff ') self.output('-------------------------------+--------+----------+-----------+-----------+--------') def format_obj(self, obj, alias): label = obj.label if not empty(obj.diff): diff = obj.diff else: diff = obj.valuation - (obj.quantity * obj.unitprice) self.tot_diff += diff self.tot_valuation += obj.valuation return u' %s %s %s %s %s %s' % \ (self.colored('%-30s' % label[:30], 'red'), self.colored('%-10s' % obj.code[:8], 'yellow') if not empty(obj.code) else ' ' * 10, self.colored('%6d' % obj.quantity, 'yellow'), self.colored('%11.2f' % obj.unitvalue, 'yellow'), self.colored('%11.2f' % obj.valuation, 'yellow'), self.colored('%8.2f' % diff, 'green' if diff >= 0 else 'red') ) def flush(self): self.output('-------------------------------+--------+----------+-----------+-----------+--------') self.output(u' Total %s %s' % (self.colored('%8.2f' % self.tot_valuation, 'yellow'), self.colored('%8.2f' % self.tot_diff, 'green' if self.tot_diff >=0 else 'red') )) self.tot_valuation = Decimal(0) self.tot_diff = Decimal(0) class RecipientListFormatter(PrettyFormatter): MANDATORY_FIELDS = ('id', 'label') def start_format(self, **kwargs): self.output('Available recipients:') def get_title(self, obj): return obj.label class AccountListFormatter(IFormatter): MANDATORY_FIELDS = ('id', 'label', 'balance', 'coming') tot_balance = Decimal(0) tot_coming = Decimal(0) def start_format(self, **kwargs): self.output(' %s Account Balance Coming ' % ((' ' * 15) if not self.interactive else '')) self.output('------------------------------------------%s+----------+----------' % (('-' * 15) if not self.interactive else '')) def format_obj(self, obj, alias): if alias is not None: id = '%s (%s)' % (self.colored('%3s' % ('#' + alias), 'red', 'bold'), self.colored(obj.backend, 'blue', 'bold')) clean = '#%s (%s)' % (alias, obj.backend) if len(clean) < 15: id += (' ' * (15 - len(clean))) else: id = self.colored('%30s' % obj.fullid, 'red', 'bold') balance = obj.balance or Decimal('0') coming = obj.coming or Decimal('0') result = u'%s %s %s %s' % (id, self.colored('%-25s' % obj.label[:25], 'yellow'), self.colored('%9.2f' % obj.balance, 'green' if balance >= 0 else 'red') if not empty(obj.balance) else ' ' * 9, self.colored('%9.2f' % obj.coming, 'green' if coming >= 0 else 'red') if not empty(obj.coming) else '') self.tot_balance += balance self.tot_coming += coming return result def flush(self): self.output(u'------------------------------------------%s+----------+----------' % (('-' * 15) if not self.interactive else '')) self.output(u'%s Total %s %s' % ( (' ' * 15) if not self.interactive else '', self.colored('%8.2f' % self.tot_balance, 'green' if self.tot_balance >= 0 else 'red'), self.colored('%8.2f' % self.tot_coming, 'green' if self.tot_coming >= 0 else 'red')) ) self.tot_balance = Decimal(0) self.tot_coming = Decimal(0) class Boobank(ReplApplication): APPNAME = 'boobank' VERSION = '1.0' COPYRIGHT = 'Copyright(C) 2010-2011 Romain Bignon, Christophe Benz' CAPS = CapBank DESCRIPTION = "Console application allowing to list your bank accounts and get their balance, " \ "display accounts history and coming bank operations, and transfer money from an account to " \ "another (if available)." SHORT_DESCRIPTION = "manage bank accounts" EXTRA_FORMATTERS = {'account_list': AccountListFormatter, 'recipient_list': RecipientListFormatter, 'transfer': TransferFormatter, 'qif': QifFormatter, 'pretty_qif': PrettyQifFormatter, 'ofx': OfxFormatter, 'ops_list': TransactionsFormatter, 'investment_list': InvestmentFormatter, } DEFAULT_FORMATTER = 'table' COMMANDS_FORMATTERS = {'ls': 'account_list', 'list': 'account_list', 'transfer': 'transfer', 'history': 'ops_list', 'coming': 'ops_list', 'investment': 'investment_list', } COLLECTION_OBJECTS = (Account, Transaction, ) def _complete_account(self, exclude=None): if exclude: exclude = '%s@%s' % self.parse_id(exclude) return [s for s in self._complete_object() if s != exclude] def do_list(self, line): """ list [-U] List accounts. Use -U to disable sorting of results. """ return self.do_ls(line) def show_history(self, command, line): id, end_date = self.parse_command_args(line, 2, 1) account = self.get_object(id, 'get_account', []) if not account: print('Error: account "%s" not found (Hint: try the command "list")' % id, file=self.stderr) return 2 if end_date is not None: try: end_date = parse_date(end_date) except ValueError: print('"%s" is an incorrect date format (for example "%s")' % \ (end_date, (datetime.date.today() - relativedelta(months=1)).strftime('%Y-%m-%d')), file=self.stderr) return 3 old_count = self.options.count self.options.count = None self.start_format(account=account) for backend, transaction in self.do(command, account, backends=account.backend): if end_date is not None and transaction.date < end_date: break self.format(transaction) if end_date is not None: self.options.count = old_count def complete_history(self, text, line, *ignored): args = line.split(' ') if len(args) == 2: return self._complete_account() @defaultcount(10) def do_history(self, line): """ history ID [END_DATE] Display history of transactions. If END_DATE is supplied, list all transactions until this date. """ return self.show_history('iter_history', line) def complete_coming(self, text, line, *ignored): args = line.split(' ') if len(args) == 2: return self._complete_account() @defaultcount(10) def do_coming(self, line): """ coming ID [END_DATE] Display future transactions. If END_DATE is supplied, show all transactions until this date. """ return self.show_history('iter_coming', line) def complete_transfer(self, text, line, *ignored): args = line.split(' ') if len(args) == 2: return self._complete_account() if len(args) == 3: return self._complete_account(args[1]) def do_transfer(self, line): """ transfer ACCOUNT [RECIPIENT AMOUNT [REASON]] Make a transfer beetwen two account - ACCOUNT the source account - RECIPIENT the recipient - AMOUNT amount to transfer - REASON reason of transfer If you give only the ACCOUNT parameter, it lists all the available recipients for this account. """ id_from, id_to, amount, reason = self.parse_command_args(line, 4, 1) account = self.get_object(id_from, 'get_account', []) if not account: print('Error: account %s not found' % id_from, file=self.stderr) return 1 if not id_to: self.objects = [] self.set_formatter('recipient_list') self.set_formatter_header(u'Available recipients') self.start_format() for backend, recipient in self.do('iter_transfer_recipients', account.id, backends=account.backend): self.cached_format(recipient) return 0 id_to, backend_name_to = self.parse_id(id_to) if account.backend != backend_name_to: print("Transfer between different backends is not implemented", file=self.stderr) return 4 try: amount = Decimal(amount) except (TypeError, ValueError, InvalidOperation): print('Error: please give a decimal amount to transfer', file=self.stderr) return 2 if self.interactive: # Try to find the recipient label. It can be missing from # recipients list, for example for banks which allow transfers to # arbitrary recipients. to = id_to for backend, recipient in self.do('iter_transfer_recipients', account.id, backends=account.backend): if recipient.id == id_to: to = recipient.label break print('Amount: %s%s' % (amount, account.currency_text)) print('From: %s' % account.label) print('To: %s' % to) print('Reason: %s' % (reason or '')) if not self.ask('Are you sure to do this transfer?', default=True): return self.start_format() for backend, transfer in self.do('transfer', account.id, id_to, amount, reason, backends=account.backend): self.format(transfer) def do_investment(self, id): """ investment ID Display investments of an account. """ account = self.get_object(id, 'get_account', []) if not account: print('Error: account "%s" not found (Hint: try the command "list")' % id, file=self.stderr) return 2 self.start_format() for backend, investment in self.do('iter_investment', account, backends=account.backend): self.format(investment)