[s2e] New module for Employee Savings Plans. Support for Esalia, Capeasi, "BNP Paribas Épargne & Retraite Entreprises" and "HSBC Epargne et Retraite en Entreprise"

This commit is contained in:
Kitof 2015-02-26 11:48:24 +01:00 committed by Florent Fourcot
commit 170db43de1
5 changed files with 351 additions and 0 deletions

23
modules/s2e/__init__.py Normal file
View file

@ -0,0 +1,23 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2015 Christophe Lampin
#
# 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 <http://www.gnu.org/licenses/>.
from .module import S2eModule
__all__ = ['S2eModule']

127
modules/s2e/browser.py Normal file
View file

@ -0,0 +1,127 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2015 Christophe Lampin
#
# 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 <http://www.gnu.org/licenses/>.
from datetime import datetime
from decimal import Decimal
from weboob.browser import LoginBrowser, URL, need_login
from weboob.browser.profiles import Android
from weboob.exceptions import BrowserIncorrectPassword
from weboob.capabilities.bank import Account, Transaction
from .pages import LoginPage, CalcPage, ProfilPage, AccountsPage, HistoryPage, I18nPage
__all__ = ['Esalia']
class S2eBrowser(LoginBrowser):
PROFILE = Android()
CTCC = ""
LANG = "FR"
sessionId = None
loginp = URL('/$', LoginPage)
calcp = URL('/s2e_services/restServices/calculetteService/grillemdp\?uuid=(?P<uuid>)', CalcPage)
profilp = URL('/s2e_services/restServices/authentification/loginS', ProfilPage)
accountsp = URL('/s2e_services/restServices/situationCompte', AccountsPage)
historyp = URL('/s2e_services/restServices/listeOperation', HistoryPage)
i18np = URL('/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)
def __init__(self, url, username, password, *args, **kwargs):
super(S2eBrowser, self).__init__(username, password, *args, **kwargs)
self.BASEURL = "https://" + url
def do_login(self):
self.logger.debug('call Browser.do_login')
self.loginp.stay_or_go()
self.page.login(self.username, self.password)
if self.sessionId is None:
raise BrowserIncorrectPassword()
@need_login
def get_accounts_list(self):
data = {'clang': self.LANG,
'ctcc': self.CTCC,
'login': self.username,
'session': self.sessionId}
for k, fond in self.accountsp.open(data=data).get_list().items():
a = Account()
a.id = k
a.type = Account.TYPE_LOAN
a.balance = Decimal(fond["montantValeurEuro"]).quantize(Decimal('.01'))
a.label = fond["libelleSupport"]
a.currency = u"EUR" # Don't find any possbility to get that from configuration.
yield a
@need_login
def iter_history(self, account):
# Load i18n for type translation
self.i18np.open(lang1=self.LANG, lang2=self.LANG).load_i18n()
# For now detail for each account is not available. History is global for all accounts and very simplist
data = {'clang': self.LANG,
'ctcc': self.CTCC,
'login': self.username,
'session': self.sessionId}
for trans in self.historyp.open(data=data).get_transactions():
t = Transaction()
t.id = trans["referenceOperationIndividuelle"]
t.date = datetime.strptime(trans["dateHeureSaisie"], "%d/%m/%Y")
t.rdate = datetime.strptime(trans["dateHeureSaisie"], "%d/%m/%Y")
t.type = Transaction.TYPE_DEPOSIT if trans["montantNetEuro"] > 0 else Transaction.TYPE_PAYBACK
t.raw = trans["typeOperation"]
t.label = self.i18n["OPERATION_TYPE_" + trans["casDeGestion"]]
t.amount = Decimal(trans["montantNetEuro"]).quantize(Decimal('.01'))
yield t
class Esalia(S2eBrowser):
CTCC = "SG"
loginp = URL('/Esalia/$', LoginPage)
i18np = URL('/Esalia/SG/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)
class Capeasi(S2eBrowser):
CTCC = "AXA"
loginp = URL('/AXA/$', LoginPage)
i18np = URL('/AXA/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)
class EREHSBC(S2eBrowser):
CTCC = "HSBC"
loginp = URL('/ERE-HSBC/$', LoginPage)
i18np = URL('/ERE-HSBC/HSBC/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)
class CreditNord(S2eBrowser):
CTCC = "" # FIXME : Not Available Yet
loginp = URL('//$', LoginPage)
# Hack : Lang.json of BNPERE is only available in app. Get it from Esalia
i18np = URL('https://m.esalia.com/Esalia/SG/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)
class BNPPERE(S2eBrowser):
CTCC = "BNP"
loginp = URL('/$', LoginPage)
# Hack : Lang.json of BNPERE is only available in app. Get it from Esalia
i18np = URL('https://m.esalia.com/Esalia/SG/(?P<lang1>.*)/LANG/(?P<lang2>.*).json', I18nPage)

73
modules/s2e/module.py Normal file
View file

@ -0,0 +1,73 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2015 Christophe Lampin
# 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 <http://www.gnu.org/licenses/>.
from weboob.capabilities.base import find_object
from weboob.capabilities.bank import CapBank, AccountNotFound
from weboob.tools.backend import Module, BackendConfig
from weboob.tools.value import Value, ValueBackendPassword
from weboob.tools.ordereddict import OrderedDict
from .browser import Esalia, Capeasi, EREHSBC, BNPPERE
__all__ = ['S2eModule']
class S2eModule(Module, CapBank):
NAME = 's2e'
MAINTAINER = u'Christophe Lampin'
EMAIL = 'weboob@lampin.net'
VERSION = '1.1'
LICENSE = 'AGPLv3+'
DESCRIPTION = u'S2e module for Employee Savings Plans. Support for Esalia, Capeasi, "BNP Paribas Épargne & Retraite Entreprises" and "HSBC Epargne et Retraite en Entreprise"'
website_choices = OrderedDict([(k, u'%s (%s)' % (v, k)) for k, v in sorted({
'm.esalia.com': u'Esalia', # Good Url. Tested
'mobile.capeasi.com': u'Capeasi', # Good Url. Not fully tested
'mobi.ere.hsbc.fr': u'ERE HSBC', # Good Url. Not fully tested
'smartphone.s2e-net.com': u'BNPP ERE', # Url To Confirm. Not tested
# 'smartphone.s2e-net.com': u'Groupe Crédit du Nord', # Mobile version not available yet.
}.iteritems(), key=lambda k_v: (k_v[1], k_v[0]))])
BROWSERS = {
'm.esalia.com': Esalia,
'mobile.capeasi.com': Capeasi,
'mobi.ere.hsbc.fr': EREHSBC,
'smartphone.s2e-net.com': BNPPERE,
# 'smartphone.s2e-net.com': CreditNord, # Mobile version not available yet.
}
CONFIG = BackendConfig(Value('website', label='Banque', choices=website_choices, default='smartphone.s2e-net.com'),
ValueBackendPassword('login', label='Identifiant', masked=False),
ValueBackendPassword('password', label='Code secret', regexp='^(\d{6}|)$'))
def create_default_browser(self):
self.BROWSER = self.BROWSERS[self.config['website'].get()]
return self.create_browser(self.config['website'].get(),
self.config['login'].get(),
self.config['password'].get())
def iter_accounts(self):
return self.browser.get_accounts_list()
def get_account(self, _id):
return find_object(self.browser.get_accounts_list(), id=_id, error=AccountNotFound)
def iter_history(self, account):
return self.browser.iter_history(account)

97
modules/s2e/pages.py Normal file
View file

@ -0,0 +1,97 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2015 Christophe Lampiné
#
# 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 <http://www.gnu.org/licenses/>.
import random
from weboob.browser.pages import HTMLPage, LoggedPage, JsonPage
class LoginPage(HTMLPage):
def generate_uuid(self):
chars = list('0123456789abcdef')
uuid = [None]*36
rnd = random.random
for i in (8, 13, 18, 23):
uuid[i] = '-'
uuid[14] = '4' # version 4
for i in range(36):
if uuid[i] is None:
r = 0 | int(rnd()*16)
idx = (((r & 0x3) | 0x8) if (i == 19) else (r & 0xf))
uuid[i] = chars[idx]
i += 1
return ''.join(uuid)
def login(self, login, password):
uuid = self.generate_uuid()
data = self.browser.calcp.open(uuid=uuid).get_data(login, password)
self.browser.profilp.open(data=data).store_sessionId()
class CalcPage(JsonPage):
def get_data(self, login, password):
convert_data = {}
for num_data in self.doc['grilleMdp']:
convert_data[num_data["nom"]] = num_data["valeur"]
encrypt_pass = ""
for char in password:
encrypt_pass += (convert_data[int(char)] + ":")
data = {'clang': self.browser.LANG,
'conversationId': self.doc["conversationId"],
'ctcc': self.browser.CTCC,
'login': login,
'password': encrypt_pass}
return data
class ProfilPage(JsonPage):
def store_sessionId(self):
self.browser.sessionId = self.doc['session']
class AccountsPage(LoggedPage, JsonPage):
def get_list(self):
accounts = {}
for entreprise in self.doc["listeEntreprise"]:
for dispositif in entreprise["listeDispositf"]: # Ceci n'est pas une erreur de frappe ;)
for fonds in dispositif["listeFonds"]:
if fonds["montantValeurEuro"] == 0:
continue
fonds["codeLong"] = entreprise["codeEntreprise"] + dispositif["codeDispositif"] + fonds["codeSupport"]
if fonds["codeLong"] in accounts:
accounts[fonds["codeLong"]]["montantValeurEuro"] += fonds["montantValeurEuro"]
else:
accounts[fonds["codeLong"]] = fonds
return accounts
class HistoryPage(LoggedPage, JsonPage):
def get_transactions(self):
for operation in self.doc["listeOperations"]:
yield operation
class I18nPage(JsonPage):
def load_i18n(self):
self.browser.i18n = self.doc["i18n"]

31
modules/s2e/test.py Normal file
View file

@ -0,0 +1,31 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2015 Christophe Lampin
#
# 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 <http://www.gnu.org/licenses/>.
from weboob.tools.test import BackendTest
class S2eTest(BackendTest):
MODULE = 's2e'
def test_bank(self):
l = list(self.backend.iter_accounts())
if len(l) > 0:
a = l[0]
list(self.backend.iter_history(a))