162 lines
6.8 KiB
Python
162 lines
6.8 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright(C) 2015 Baptiste Delpey
|
|
#
|
|
# 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 StringIO import StringIO
|
|
import hashlib
|
|
from decimal import Decimal
|
|
from datetime import datetime
|
|
|
|
from weboob.capabilities.bank import Account
|
|
from weboob.exceptions import BrowserIncorrectPassword
|
|
from weboob.tools.capabilities.bank.transactions import FrenchTransaction
|
|
from weboob.browser.pages import HTMLPage, JsonPage, LoggedPage
|
|
from weboob.tools.captcha.virtkeyboard import MappedVirtKeyboard, VirtKeyboardError
|
|
|
|
|
|
class BNPVirtKeyboard(MappedVirtKeyboard):
|
|
symbols = {'0': 'ff069462836e30a39c911034048f5bb3',
|
|
'1': '7969f04e4e82eaefa2ce7a9a23c26178',
|
|
'2': '1e6020f97ca1c3ce3da4f39ded15d67d',
|
|
'3': 'f84284b40aea93c24814e23e14e76cc8',
|
|
'4': '88bab262d4b344c0ef8f06ddd01adbcf',
|
|
'5': '0a270764fc5d8334bcb55053432b26cb',
|
|
'6': 'e6a4444a6c752cd3e655f2883e530080',
|
|
'7': '933d4ca5df6b2b3df2dea00a21a3fed6',
|
|
'8': ['f28b918777d21a5fde2bffb9899e2138', 'a97e6e27159084d50f8ef00548b70252'],
|
|
'9': 'be751b77af0d998ab4c2cfd38455b2a6',
|
|
}
|
|
|
|
color=(0,0,0)
|
|
|
|
def __init__(self, basepage):
|
|
img = basepage.doc.xpath('//img[@id="gridpass_img"]')[0]
|
|
imgdata = basepage.browser.open(img.attrib['src']).content
|
|
MappedVirtKeyboard.__init__(self, StringIO(imgdata), basepage.doc, img, self.color, convert='RGB')
|
|
self.check_symbols(self.symbols, basepage.browser.responses_dirname)
|
|
|
|
def get_symbol_code(self, md5sum):
|
|
code = MappedVirtKeyboard.get_symbol_code(self, md5sum)
|
|
code = code.split("'")[3]
|
|
assert code.isdigit()
|
|
return code
|
|
|
|
def check_color(self, pixel):
|
|
for p in pixel:
|
|
if p >= 200:
|
|
return False
|
|
return True
|
|
|
|
def checksum(self, coords):
|
|
"""Copy of parent checksum(), but cropping (removes empty lines)"""
|
|
x1, y1, x2, y2 = coords
|
|
s = ''
|
|
for y in range(y1, min(y2 + 1, self.height)):
|
|
for x in range(x1, min(x2 + 1, self.width)):
|
|
if self.check_color(self.pixar[x, y]):
|
|
s += " "
|
|
else:
|
|
s += "O"
|
|
s += "\n"
|
|
s = '\n'.join([l for l in s.splitlines() if l.strip()])
|
|
return hashlib.md5(s).hexdigest()
|
|
|
|
|
|
class LoginPage(HTMLPage):
|
|
def login(self, login, password):
|
|
try:
|
|
vk = BNPVirtKeyboard(self)
|
|
except VirtKeyboardError as err:
|
|
self.logger.error("Error: %s" % err)
|
|
return False
|
|
|
|
form = self.get_form(name='loginPwdForm')
|
|
form['txtAuthentMode'] = 'PASSWORD'
|
|
form['txtPwdUserId'] = login
|
|
form['gridpass_hidden_input'] = vk.get_string_code(password)
|
|
form.submit()
|
|
|
|
def on_load(self):
|
|
if self.doc.xpath('//p[contains(text(), "Your identification is wrong.")]'):
|
|
raise BrowserIncorrectPassword("Your identification is wrong.")
|
|
|
|
|
|
class AccountsPage(LoggedPage, JsonPage):
|
|
FAMILY_TO_TYPE = {
|
|
u"Compte chèque": Account.TYPE_CHECKING,
|
|
}
|
|
|
|
def iter_accounts(self):
|
|
for f in self.path('tableauSoldes.listeGroupes'):
|
|
for g in f:
|
|
for a in g.get('listeComptes'):
|
|
yield Account.from_dict({
|
|
'id': a.get('numeroCompte'),
|
|
'iban': a.get('numeroCompte'),
|
|
'type': self.FAMILY_TO_TYPE.get(a.get('libelleType')) or Account.TYPE_UNKNOWN,
|
|
'label': '%s %s' % (a.get('libelleType'), a.get('libelleTitulaire')),
|
|
'currency': a.get('deviseTenue'),
|
|
'balance': Decimal(a.get('soldeComptable')) / 100,
|
|
'coming': Decimal(a.get('soldePrevisionnel')) / 100,
|
|
})
|
|
|
|
|
|
class HistoryPage(LoggedPage, JsonPage):
|
|
CODE_TO_TYPE = {
|
|
"AUTOP": FrenchTransaction.TYPE_UNKNOWN, # Autres opérations,
|
|
"BOURS": FrenchTransaction.TYPE_BANK, # Bourse / Titres,
|
|
"CARTE": FrenchTransaction.TYPE_CARD, # Cartes,
|
|
"CHEQU": FrenchTransaction.TYPE_CHECK, # Chèques,
|
|
"CREDD": FrenchTransaction.TYPE_UNKNOWN, # Crédits documentaires,
|
|
"CREDI": FrenchTransaction.TYPE_UNKNOWN, # Crédits,
|
|
"EFFET": FrenchTransaction.TYPE_UNKNOWN, # Effets,
|
|
"ESPEC": FrenchTransaction.TYPE_UNKNOWN, # Espèces,
|
|
"FACCB": FrenchTransaction.TYPE_UNKNOWN, # Factures / Retraits cartes,
|
|
"ICHEQ": FrenchTransaction.TYPE_UNKNOWN, # Impayés chèques,
|
|
"IEFFE": FrenchTransaction.TYPE_UNKNOWN, # Impayés et incidents effets,
|
|
"IMPAY": FrenchTransaction.TYPE_UNKNOWN, # Impayés et rejets,
|
|
"IPRLV": FrenchTransaction.TYPE_UNKNOWN, # Impayés prélèvements, TIP et télérèglements,
|
|
"PRLVT": FrenchTransaction.TYPE_UNKNOWN, # Prélèvements, TIP et télérèglements,
|
|
"REMCB": FrenchTransaction.TYPE_UNKNOWN, # Remises cartes,
|
|
"RJVIR": FrenchTransaction.TYPE_ORDER, # Rejets de virements,
|
|
"VIREM": FrenchTransaction.TYPE_ORDER, # Virements,
|
|
"VIRIT": FrenchTransaction.TYPE_ORDER, # Virements internationaux,
|
|
"VIRSP": FrenchTransaction.TYPE_ORDER, # Virements européens,
|
|
"VIRTR": FrenchTransaction.TYPE_ORDER, # Virements de trésorerie,
|
|
"VIRXX": FrenchTransaction.TYPE_ORDER, # Autres virements
|
|
}
|
|
|
|
def one(self, path, context=None):
|
|
try:
|
|
return list(self.path(path, context))[0]
|
|
except IndexError:
|
|
return None
|
|
|
|
def iter_history(self):
|
|
for op in self.get('mouvementsBDDF'):
|
|
codeFamille = self.one('nature.codefamille', op)
|
|
tr = FrenchTransaction.from_dict({
|
|
'id': op.get('id'),
|
|
'type': self.CODE_TO_TYPE.get(codeFamille) or FrenchTransaction.TYPE_UNKNOWN,
|
|
'category': self.one('nature.libelle', op),
|
|
'raw': ' '.join(op.get('libelle').split()) or op.get('nature')['libelle'],
|
|
'date': datetime.fromtimestamp(op.get('dateOperation') / 1000),
|
|
'vdate': datetime.fromtimestamp(op.get('dateValeur') / 1000),
|
|
'amount': Decimal(self.one('montant.montant', op)) / 100,
|
|
})
|
|
yield tr
|