# -*- coding: utf-8 -*- # Copyright(C) 2010-2011 Julien Veyssier # # 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 . try: from urlparse import urlsplit, parse_qsl, urlparse except ImportError: from urllib.parse import urlsplit, parse_qsl, urlparse from datetime import datetime, timedelta from random import randint from weboob.tools.compat import basestring from weboob.browser.browsers import LoginBrowser, need_login from weboob.browser.profiles import Wget from weboob.browser.url import URL from weboob.exceptions import BrowserIncorrectPassword from weboob.capabilities.bank import Transfer, TransferError from .pages import LoginPage, LoginErrorPage, AccountsPage, UserSpacePage, \ OperationsPage, CardPage, ComingPage, NoOperationsPage, \ TransfertPage, ChangePasswordPage, VerifCodePage, EmptyPage __all__ = ['CICBrowser'] class CICBrowser(LoginBrowser): PROFILE = Wget() BASEURL = 'https://www.cic.fr' login = URL('/sb/fr/banques/particuliers/index.html', '/(?P.*)/fr/$', '/(?P.*)/fr/banques/accueil.html', '/(?P.*)/fr/banques/particuliers/index.html', LoginPage) login_error = URL('/(?P.*)/fr/identification/default.cgi', LoginErrorPage) accounts = URL('/(?P.*)/fr/banque/situation_financiere.cgi', '/(?P.*)/fr/banque/situation_financiere.html', AccountsPage) user_space = URL('/(?P.*)/fr/banque/espace_personnel.aspx', UserSpacePage) operations = URL('/(?P.*)/fr/banque/mouvements.cgi.*', '/(?P.*)/fr/banque/mouvements.html.*', '/(?P.*)/fr/banque/nr/nr_devbooster.aspx.*', OperationsPage) coming = URL('/(?P.*)/fr/banque/mvts_instance.cgi.*', ComingPage) card = URL('/(?P.*)/fr/banque/operations_carte.cgi.*', CardPage) noop = URL('/(?P.*)/fr/banque/CR/arrivee.asp.*', NoOperationsPage) info = URL('/(?P.*)/fr/banque/BAD.*', EmptyPage) transfert = URL('/(?P.*)/fr/banque/virements/vplw_vi.html', EmptyPage) transfert_2 = URL('/(?P.*)/fr/banque/virements/vplw_cmweb.aspx.*', TransfertPage) change_pass = URL('/(?P.*)/fr/validation/change_password.cgi', ChangePasswordPage) verify_pass = URL('/(?P.*)/fr/validation/verif_code.cgi.*', VerifCodePage) empty = URL('/(?P.*)/fr/banques/index.html', '/(?P.*)/fr/banque/paci_beware_of_phishing.*', '/(?P.*)/fr/validation/(?!change_password|verif_code).*', '/(?P.*)/fr/banque/paci_engine/static_content_manager.aspx', '/(?P.*)/fr/banque/DELG_Gestion.*', EmptyPage) currentSubBank = None __states__ = ['currentSubBank'] def do_login(self): self.login.go() if not self.page.logged: self.page.login(self.username, self.password) if not self.page.logged or self.login_error.is_here(): raise BrowserIncorrectPassword() self.getCurrentSubBank() @need_login def get_accounts_list(self): return self.accounts.stay_or_go(subbank=self.currentSubBank).iter_accounts() def get_account(self, id): assert isinstance(id, basestring) for a in self.get_accounts_list(): if a.id == id: return a def getCurrentSubBank(self): # the account list and history urls depend on the sub bank of the user url = urlparse(self.url) self.currentSubBank = url.path.lstrip('/').split('/')[0] def list_operations(self, page_url): if page_url.startswith('/') or page_url.startswith('https'): self.location(page_url) else: self.location('%s/%s/fr/banque/%s' % (self.BASEURL, self.currentSubBank, page_url)) if not self.operations.is_here(): return iter([]) return self.pagination(lambda: self.page.get_history()) def get_history(self, account): transactions = [] last_debit = None for tr in self.list_operations(account._link_id): # to prevent redundancy with card transactions, we do not # store 'RELEVE CARTE' transaction. if tr.raw != 'RELEVE CARTE': transactions.append(tr) elif last_debit is None: last_debit = (tr.date - timedelta(days=10)).month coming_link = self.page.get_coming_link() if self.operations.is_here() else None if coming_link is not None: for tr in self.list_operations(coming_link): transactions.append(tr) month = 0 for card_link in account._card_links: v = urlsplit(card_link) args = dict(parse_qsl(v.query)) # useful with 12 -> 1 if int(args['mois']) < month: month = month + 1 else: month = int(args['mois']) for tr in self.list_operations(card_link): if month > last_debit: tr._is_coming = True transactions.append(tr) transactions.sort(key=lambda tr: tr.rdate, reverse=True) return transactions def transfer(self, account, to, amount, reason=None): # access the transfer page self.transfert.go(subbank=self.currentSubBank) # fill the form form = self.page.get_form(xpath="//form[@id='P:F']") try: form['data_input_indiceCompteADebiter'] = self.page.get_from_account_index(account) form['data_input_indiceCompteACrediter'] = self.page.get_to_account_index(to) except ValueError as e: raise TransferError(e.message) form['[t:dbt%3adouble;]data_input_montant_value_0_'] = '%s' % str(amount).replace('.', ',') if reason is not None: form['[t:dbt%3astring;x(27)]data_input_libelleCompteDebite'] = reason form['[t:dbt%3astring;x(31)]data_input_motifCompteCredite'] = reason del form['_FID_GoCancel'] del form['_FID_DoValidate'] form['_FID_DoValidate.x'] = str(randint(3, 125)) form['_FID_DoValidate.y'] = str(randint(3, 22)) form.submit() # look for known errors content = self.page.get_unicode_content() insufficient_amount_message = u'Le montant du virement doit être positif, veuillez le modifier' maximum_allowed_balance_message = u'Montant maximum autorisé au débit pour ce compte' if insufficient_amount_message in content: raise TransferError('The amount you tried to transfer is too low.') if maximum_allowed_balance_message in content: raise TransferError('The maximum allowed balance for the target account has been / would be reached.') # look for the known "all right" message ready_for_transfer_message = u'Confirmer un virement entre vos comptes' if ready_for_transfer_message not in content: raise TransferError('The expected message "%s" was not found.' % ready_for_transfer_message) # submit the confirmation form form = self.page.get_form(xpath="//form[@id='P:F']") del form['_FID_DoConfirm'] form['_FID_DoConfirm.x'] = str(randint(3, 125)) form['_FID_DoConfirm.y'] = str(randint(3, 22)) submit_date = datetime.now() form.submit() # look for the known "everything went well" message content = self.page.get_unicode_content() transfer_ok_message = u'Votre virement a été exécuté' if transfer_ok_message not in content: raise TransferError('The expected message "%s" was not found.' % transfer_ok_message) # We now have to return a Transfer object transfer = Transfer(submit_date.strftime('%Y%m%d%H%M%S')) transfer.amount = amount transfer.origin = account transfer.recipient = to transfer.date = submit_date return transfer