# -*- coding: utf-8 -*- # Copyright(C) 2010-2011 Jocelyn Jaubert # # 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 weboob.tools.browser import BaseBrowser, BrowserIncorrectPassword, BrowserUnavailable from .pages.accounts_list import AccountsList, AccountHistory, CardsList from .pages.login import LoginPage, BadLoginPage __all__ = ['SocieteGenerale'] class SocieteGenerale(BaseBrowser): DOMAIN_LOGIN = 'particuliers.societegenerale.fr' CERTHASH_LOGIN = ['72b78ce0b8ffc63a6dcbf8fc375a1ab5502d5dfefcac1d00901a73f5a94e9ed5', '629873f98004aef6c42d15714ff9689fdfce04746483122417f432cd693f5007'] DOMAIN = 'particuliers.secure.societegenerale.fr' CERTHASH = '4499ca391d0d690050d80e625fd0b16e83476fd565d8e43315c7a9c025f02b88' PROTOCOL = 'https' ENCODING = None # refer to the HTML encoding PAGES = { 'https://particuliers.societegenerale.fr/.*': LoginPage, 'https://.*.societegenerale.fr//acces/authlgn.html': BadLoginPage, 'https://.*.societegenerale.fr/error403.html': BadLoginPage, '.*restitution/cns_listeprestation.html': AccountsList, '.*restitution/cns_listeCartes.*.html.*': CardsList, '.*restitution/cns_detail.*\.html.*': AccountHistory, 'https://.*.societegenerale.fr/lgn/url.html.*':AccountHistory, } def __init__(self, *args, **kwargs): self.lowsslcheck(self.DOMAIN_LOGIN, self.CERTHASH_LOGIN) BaseBrowser.__init__(self, *args, **kwargs) def home(self): self.location('https://' + self.DOMAIN_LOGIN + '/index.html') def is_logged(self): if not self.page or self.is_on_page(LoginPage): return False error = self.page.get_error() if error is None: return True return False def login(self): assert isinstance(self.username, basestring) assert isinstance(self.password, basestring) assert self.password.isdigit() if not self.is_on_page(LoginPage): self.location('https://' + self.DOMAIN_LOGIN + '/index.html', no_login=True) self.page.login(self.username, self.password) if self.is_on_page(LoginPage): raise BrowserIncorrectPassword() if self.is_on_page(BadLoginPage): error = self.page.get_error() if error is None: raise BrowserIncorrectPassword() elif error.startswith('Votre session a'): raise BrowserUnavailable('Session has expired') elif error.startswith('Le service est momentan'): raise BrowserUnavailable(error) else: raise BrowserIncorrectPassword(error) def get_accounts_list(self): if not self.is_on_page(AccountsList): self.location('/restitution/cns_listeprestation.html') return self.page.get_list() def get_account(self, id): assert isinstance(id, basestring) if not self.is_on_page(AccountsList): self.location('/restitution/cns_listeprestation.html') for a in self.page.get_list(): if a.id == id: return a return None def iter_history(self, account): self.location(account._link_id) transactions = [] if self.is_on_page(CardsList): for card_link in self.page.iter_cards(): self.location(card_link) transactions += list(self.page.iter_transactions(coming=True)) elif self.is_on_page(AccountHistory): transactions += list(self.page.iter_transactions(coming=(account.type == account.TYPE_CARD))) else: self.logger.warning('This account is not supported') def key(tr): # Can't compare datetime and date, so cast them. try: return tr.rdate.date() except AttributeError: return tr.rdate transactions.sort(key=key, reverse=True) return iter(transactions)