support new website of bred

This commit is contained in:
Romain Bignon 2014-12-07 16:20:51 +01:00
commit a9b2f5fe66
6 changed files with 140 additions and 25 deletions

View file

@ -0,0 +1,3 @@
from .browser import BredBrowser
__all__ = ['BredBrowser']

View file

@ -0,0 +1,114 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2014 Romain Bignon
#
# 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 date
from decimal import Decimal
from weboob.capabilities.bank import Account, Transaction
from weboob.exceptions import BrowserIncorrectPassword
from weboob.browser import DomainBrowser
__all__ = ['BredBrowser']
class BredBrowser(DomainBrowser):
BASEURL = 'https://www.bred.fr'
def __init__(self, accnum, login, password, *args, **kwargs):
super(BredBrowser, self).__init__(*args, **kwargs)
self.login = login
self.password = password
self.accnum = accnum
self.do_login(login, password)
def do_login(self, login, password):
self.location('/transactionnel/Authentication', data={'identifiant': login, 'password': password})
if 'erreur-pwd' in self.url:
raise BrowserIncorrectPassword()
ACCOUNT_TYPES = {'000': Account.TYPE_CHECKING,
'999': Account.TYPE_MARKET,
'011': Account.TYPE_CARD,
'023': Account.TYPE_SAVINGS,
'078': Account.TYPE_SAVINGS,
'080': Account.TYPE_SAVINGS,
'027': Account.TYPE_SAVINGS,
'037': Account.TYPE_SAVINGS,
}
def get_accounts_list(self):
r = self.open('/transactionnel/services/rest/Account/accounts')
for content in r.json()['content']:
if self.accnum != '00000000000' and content['numero'] != self.accnum:
continue
for poste in content['postes']:
a = Account()
a._number = content['numeroLong']
a._nature = poste['codeNature']
a._consultable = poste['consultable']
a.id = '%s.%s' % (a._number, a._nature)
a.type = self.ACCOUNT_TYPES.get(poste['codeNature'], Account.TYPE_UNKNOWN)
if poste['postePortefeuille']:
a.label = u'Portefeuille Titres'
a.balance = Decimal(poste['montantTitres']['valeur'])
a.currency = poste['montantTitres']['monnaie']['code']
yield a
if not 'libelle' in poste:
continue
a.label = ' '.join([content['intitule'].strip(), poste['libelle'].strip()])
a.balance = Decimal(poste['solde']['valeur'])
a.currency = poste['solde']['monnaie']['code']
yield a
def get_history(self, account):
if not account._consultable:
raise NotImplementedError()
offset = 0
next_page = True
while next_page:
r = self.open('/transactionnel/services/applications/operations/get/%(number)s/%(nature)s/00/%(currency)s/%(startDate)s/%(endDate)s/%(offset)s/%(limit)s' %
{'number': account._number,
'nature': account._nature,
'currency': account.currency,
'startDate': '2000-01-01',
'endDate': date.today().strftime('%Y-%m-%d'),
'offset': offset,
'limit': 50
})
next_page = False
offset += 50
for op in r.json()['content']['operations']:
next_page = True
t = Transaction()
t.id = op['id']
t.amount = Decimal(str(op['montant']))
t.date = date.fromtimestamp(op.get('dateDebit', op.get('dateOperation'))/1000)
t.rdate = date.fromtimestamp(op.get('dateOperation', op.get('dateDebit'))/1000)
t.vdate = date.fromtimestamp(op.get('dateValeur', op.get('dateDebit', op.get('dateOperation')))/1000)
t.category = op['categorie']
t.label = op['libelle']
t.raw = ' '.join([op['libelle']] + op['details'])
yield t

View file

@ -0,0 +1,3 @@
from .browser import DispoBankBrowser
__all__ = ['DispoBankBrowser']

View file

@ -1,6 +1,6 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2012 Romain Bignon
# Copyright(C) 2012-2014 Romain Bignon
#
# This file is part of weboob.
#
@ -25,12 +25,12 @@ from weboob.deprecated.browser import Browser, BrowserIncorrectPassword
from .pages import LoginPage, LoginResultPage, AccountsPage, EmptyPage, TransactionsPage
__all__ = ['BredBrowser']
__all__ = ['DispoBankBrowser']
class BredBrowser(Browser):
class DispoBankBrowser(Browser):
PROTOCOL = 'https'
DOMAIN = 'www.bred.fr'
DOMAIN = 'www.dispobank.fr'
CERTHASH = ['9b77dab9c84e1dc9e0798de561a6541ff15f038f60b36ca74c29be1def6c19a3', '375f1fed165d34aacaaf71674ab14ca6c1b38404cf748278714fde3c58385ff0', '0853a056453b56aea6a29085ef3f3721b18db2052aa8e84220720d44e0eb22af']
ENCODING = 'iso-8859-15'
PAGES = {r'https://www.\w+.fr/mylittleform.*': LoginPage,
@ -50,10 +50,9 @@ class BredBrowser(Browser):
}
}
def __init__(self, website, accnum, *args, **kwargs):
self.accnum = accnum.replace(' ','').zfill(11)
self.DOMAIN = 'www.%s.fr' % website
self.website = website
def __init__(self, accnum, *args, **kwargs):
self.accnum = accnum
self.website = 'dispobank'
Browser.__init__(self, *args, **kwargs)
def is_logged(self):

View file

@ -1,6 +1,6 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2012-2013 Romain Bignon
# Copyright(C) 2012-2014 Romain Bignon
#
# This file is part of weboob.
#
@ -19,10 +19,12 @@
from weboob.capabilities.bank import CapBank, AccountNotFound
from weboob.capabilities.base import find_object
from weboob.tools.backend import Module, BackendConfig
from weboob.tools.value import ValueBackendPassword, Value
from .browser import BredBrowser
from .bred import BredBrowser
from .dispobank import DispoBankBrowser
__all__ = ['BredModule']
@ -41,28 +43,22 @@ class BredModule(Module, CapBank):
choices={'bred': 'BRED', 'dispobank': 'DispoBank'}),
Value('accnum', label=u'Numéro du compte bancaire (optionnel)', default='', masked=False)
)
BROWSER = BredBrowser
BROWSERS = {'bred': BredBrowser,
'dispobank': DispoBankBrowser,
}
def create_default_browser(self):
return self.create_browser(self.config['website'].get(),
self.config['accnum'].get(),
self.BROWSER = self.BROWSERS[self.config['website'].get()]
return self.create_browser(self.config['accnum'].get().replace(' ','').zfill(11),
self.config['login'].get(),
self.config['password'].get())
def iter_accounts(self):
with self.browser:
for account in self.browser.get_accounts_list():
yield account
return self.browser.get_accounts_list()
def get_account(self, _id):
with self.browser:
account = self.browser.get_account(_id)
if account:
return account
else:
raise AccountNotFound()
return find_object(self.browser.get_accounts_list(), id=_id, error=AccountNotFound)
def iter_history(self, account):
with self.browser:
return self.browser.get_history(account)