support new website of bred
This commit is contained in:
parent
fe254f3279
commit
a9b2f5fe66
6 changed files with 140 additions and 25 deletions
3
modules/bred/bred/__init__.py
Normal file
3
modules/bred/bred/__init__.py
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
from .browser import BredBrowser
|
||||||
|
|
||||||
|
__all__ = ['BredBrowser']
|
||||||
114
modules/bred/bred/browser.py
Normal file
114
modules/bred/bred/browser.py
Normal 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
|
||||||
3
modules/bred/dispobank/__init__.py
Normal file
3
modules/bred/dispobank/__init__.py
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
from .browser import DispoBankBrowser
|
||||||
|
|
||||||
|
__all__ = ['DispoBankBrowser']
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
# Copyright(C) 2012 Romain Bignon
|
# Copyright(C) 2012-2014 Romain Bignon
|
||||||
#
|
#
|
||||||
# This file is part of weboob.
|
# 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
|
from .pages import LoginPage, LoginResultPage, AccountsPage, EmptyPage, TransactionsPage
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['BredBrowser']
|
__all__ = ['DispoBankBrowser']
|
||||||
|
|
||||||
|
|
||||||
class BredBrowser(Browser):
|
class DispoBankBrowser(Browser):
|
||||||
PROTOCOL = 'https'
|
PROTOCOL = 'https'
|
||||||
DOMAIN = 'www.bred.fr'
|
DOMAIN = 'www.dispobank.fr'
|
||||||
CERTHASH = ['9b77dab9c84e1dc9e0798de561a6541ff15f038f60b36ca74c29be1def6c19a3', '375f1fed165d34aacaaf71674ab14ca6c1b38404cf748278714fde3c58385ff0', '0853a056453b56aea6a29085ef3f3721b18db2052aa8e84220720d44e0eb22af']
|
CERTHASH = ['9b77dab9c84e1dc9e0798de561a6541ff15f038f60b36ca74c29be1def6c19a3', '375f1fed165d34aacaaf71674ab14ca6c1b38404cf748278714fde3c58385ff0', '0853a056453b56aea6a29085ef3f3721b18db2052aa8e84220720d44e0eb22af']
|
||||||
ENCODING = 'iso-8859-15'
|
ENCODING = 'iso-8859-15'
|
||||||
PAGES = {r'https://www.\w+.fr/mylittleform.*': LoginPage,
|
PAGES = {r'https://www.\w+.fr/mylittleform.*': LoginPage,
|
||||||
|
|
@ -50,10 +50,9 @@ class BredBrowser(Browser):
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, website, accnum, *args, **kwargs):
|
def __init__(self, accnum, *args, **kwargs):
|
||||||
self.accnum = accnum.replace(' ','').zfill(11)
|
self.accnum = accnum
|
||||||
self.DOMAIN = 'www.%s.fr' % website
|
self.website = 'dispobank'
|
||||||
self.website = website
|
|
||||||
Browser.__init__(self, *args, **kwargs)
|
Browser.__init__(self, *args, **kwargs)
|
||||||
|
|
||||||
def is_logged(self):
|
def is_logged(self):
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
# Copyright(C) 2012-2013 Romain Bignon
|
# Copyright(C) 2012-2014 Romain Bignon
|
||||||
#
|
#
|
||||||
# This file is part of weboob.
|
# This file is part of weboob.
|
||||||
#
|
#
|
||||||
|
|
@ -19,10 +19,12 @@
|
||||||
|
|
||||||
|
|
||||||
from weboob.capabilities.bank import CapBank, AccountNotFound
|
from weboob.capabilities.bank import CapBank, AccountNotFound
|
||||||
|
from weboob.capabilities.base import find_object
|
||||||
from weboob.tools.backend import Module, BackendConfig
|
from weboob.tools.backend import Module, BackendConfig
|
||||||
from weboob.tools.value import ValueBackendPassword, Value
|
from weboob.tools.value import ValueBackendPassword, Value
|
||||||
|
|
||||||
from .browser import BredBrowser
|
from .bred import BredBrowser
|
||||||
|
from .dispobank import DispoBankBrowser
|
||||||
|
|
||||||
|
|
||||||
__all__ = ['BredModule']
|
__all__ = ['BredModule']
|
||||||
|
|
@ -41,28 +43,22 @@ class BredModule(Module, CapBank):
|
||||||
choices={'bred': 'BRED', 'dispobank': 'DispoBank'}),
|
choices={'bred': 'BRED', 'dispobank': 'DispoBank'}),
|
||||||
Value('accnum', label=u'Numéro du compte bancaire (optionnel)', default='', masked=False)
|
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):
|
def create_default_browser(self):
|
||||||
return self.create_browser(self.config['website'].get(),
|
self.BROWSER = self.BROWSERS[self.config['website'].get()]
|
||||||
self.config['accnum'].get(),
|
return self.create_browser(self.config['accnum'].get().replace(' ','').zfill(11),
|
||||||
self.config['login'].get(),
|
self.config['login'].get(),
|
||||||
self.config['password'].get())
|
self.config['password'].get())
|
||||||
|
|
||||||
def iter_accounts(self):
|
def iter_accounts(self):
|
||||||
with self.browser:
|
return self.browser.get_accounts_list()
|
||||||
for account in self.browser.get_accounts_list():
|
|
||||||
yield account
|
|
||||||
|
|
||||||
def get_account(self, _id):
|
def get_account(self, _id):
|
||||||
with self.browser:
|
return find_object(self.browser.get_accounts_list(), id=_id, error=AccountNotFound)
|
||||||
account = self.browser.get_account(_id)
|
|
||||||
|
|
||||||
if account:
|
|
||||||
return account
|
|
||||||
else:
|
|
||||||
raise AccountNotFound()
|
|
||||||
|
|
||||||
def iter_history(self, account):
|
def iter_history(self, account):
|
||||||
with self.browser:
|
|
||||||
return self.browser.get_history(account)
|
return self.browser.get_history(account)
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue