added fortuneo module

This commit is contained in:
sputnick 2012-04-16 01:00:47 +02:00 committed by Romain Bignon
commit 38f2f57517
7 changed files with 471 additions and 0 deletions

View file

View file

@ -0,0 +1,149 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2012 Gilles-Alexandre Quenot
#
# 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 urlparse import parse_qs, urlparse
from lxml.etree import XML
from cStringIO import StringIO
from decimal import Decimal
import re
from weboob.capabilities.bank import Account
from weboob.tools.capabilities.bank.transactions import FrenchTransaction
from weboob.tools.browser import BasePage, BrokenPageError
__all__ = ['AccountsList', 'AccountHistory']
class AccountsList(BasePage):
def on_loaded(self):
pass
def get_list(self):
l = []
#for el in self.document.xpath('//table[@id="tableauComptesTitEtCotit"]/tbody/'):
#l.append('test')
#l.append('Livret +')
#l.append('20')
#l.append('https://www.fortuneo.fr/fr/prive/mes-comptes/livret/caracteristiques-mon-compte/?COMPTE_ACTIF=FT00654224521421145')
account.label = "test"
account.id = "Livret +"
account.balance = "20"
account._link_id = "https://www.fortuneo.fr/fr/prive/mes-comptes/livret/caracteristiques-mon-compte/?COMPTE_ACTIF=FT00654224521421145"
l.append(account)
#for tr in self.document.getiterator('tr'):
# if 'LGNTableRow' in tr.attrib.get('class', '').split():
# account = Account()
# for td in tr.getiterator('td'):
# if td.attrib.get('headers', '') == 'TypeCompte':
# a = td.find('a')
# account.label = unicode(a.find("span").text)
# account._link_id = a.get('href', '')
# elif td.attrib.get('headers', '') == 'NumeroCompte':
# id = td.text
# id = id.replace(u'\xa0','')
# account.id = id
# elif td.attrib.get('headers', '') == 'Libelle':
# pass
# elif td.attrib.get('headers', '') == 'Solde':
# balance = td.find('div').text
# if balance != None:
# balance = balance.replace(u'\xa0','').replace(',','.')
# account.balance = Decimal(balance)
# else:
# account.balance = Decimal(0)
# l.append(account)
return l
class Transaction(FrenchTransaction):
print "DEBUG a implementer"
pass
#PATTERNS = [(re.compile(r'^CARTE \w+ RETRAIT DAB.* (?P<dd>\d{2})/(?P<mm>\d{2}) (?P<HH>\d+)H(?P<MM>\d+) (?P<text>.*)'),
# FrenchTransaction.TYPE_WITHDRAWAL),
# (re.compile(r'^(?P<category>CARTE) \w+ (?P<dd>\d{2})/(?P<mm>\d{2}) (?P<text>.*)'),
# FrenchTransaction.TYPE_CARD),
# (re.compile(r'^(?P<category>(COTISATION|PRELEVEMENT|TELEREGLEMENT|TIP)) (?P<text>.*)'),
# FrenchTransaction.TYPE_ORDER),
# (re.compile(r'^(?P<category>VIR(EMEN)?T? \w+) (?P<text>.*)'),
# FrenchTransaction.TYPE_TRANSFER),
# (re.compile(r'^(CHEQUE) (?P<text>.*)'), FrenchTransaction.TYPE_CHECK),
# (re.compile(r'^(FRAIS) (?P<text>.*)'), FrenchTransaction.TYPE_BANK),
# (re.compile(r'^(?P<category>ECHEANCEPRET)(?P<text>.*)'),
# FrenchTransaction.TYPE_LOAN_PAYMENT),
# (re.compile(r'^(?P<category>REMISE CHEQUES)(?P<text>.*)'),
# FrenchTransaction.TYPE_DEPOSIT),
# ]
class AccountHistory(BasePage):
def get_part_url(self):
print "DEBUG a implementer"
pass
#for script in self.document.getiterator('script'):
# if script.text is None:
# continue
# m = re.search('var listeEcrCavXmlUrl="(.*)";', script.text)
# if m:
# return m.group(1)
#raise BrokenPageError('Unable to find link to history part')
def iter_transactions(self):
print "DEBUG a implementer"
pass
#url = self.get_part_url()
#while 1:
# d = XML(self.browser.readurl(url))
# el = d.xpath('//dataBody')[0]
# s = StringIO(el.text)
# doc = self.browser.get_document(s)
# for tr in self._iter_transactions(doc):
# yield tr
# el = d.xpath('//dataHeader')[0]
# if int(el.find('suite').text) != 1:
# return
# url = urlparse(url)
# p = parse_qs(url.query)
# url = self.browser.buildurl(url.path, n10_nrowcolor=0,
# operationNumberPG=el.find('operationNumber').text,
# operationTypePG=el.find('operationType').text,
# pageNumberPG=el.find('pageNumber').text,
# sign=p['sign'][0],
# src=p['src'][0])
def _iter_transactions(self, doc):
print "DEBUG a implementer"
pass
#for i, tr in enumerate(self.parser.select(doc.getroot(), 'tr')):
# t = Transaction(i)
# t.parse(date=tr.xpath('./td[@headers="Date"]')[0].text,
# raw=tr.attrib['title'].strip())
# t.set_amount(*reversed([el.text for el in tr.xpath('./td[@class="right"]')]))
# t._coming = tr.xpath('./td[@headers="AVenir"]')[0].text
# yield t

View file

@ -0,0 +1,99 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2012 Gilles-Alexandre Quenot
#
# 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 logging import error
from weboob.tools.browser import BasePage, BrowserUnavailable
#from lxml import etree
__all__ = ['LoginPage']
def dump(obj):
for attr in dir(obj):
print "obj.%s = %s" % (attr, getattr(obj, attr))
class LoginPage(BasePage):
def login(self, login, passwd):
#print "DEBUG BasePage=", BasePage.url
#dump(BasePage)
self.browser.select_form(nr=3)
#self.browser['locale'] = 'fr'
self.browser['login'] = login
self.browser['passwd'] = passwd
#self.browser['idDyn'] = 'false'
self.browser.submit()
#print "DEBUG ", self.page
#class LoginPage(BasePage):
# def on_loaded(self):
# pass
# #for td in self.document.getroot().cssselect('td.LibelleErreur'):
# # if td.text is None:
# # continue
# # msg = td.text.strip()
# # if 'indisponible' in msg:
# # raise BrowserUnavailable(msg)
#
# def login(self, login, password):
# DOMAIN_LOGIN = self.browser.DOMAIN_LOGIN
# DOMAIN = self.browser.DOMAIN
#
# url_login = 'https://' + DOMAIN_LOGIN + '/index.html'
#
# base_url = 'https://' + DOMAIN
# url = base_url + '/cvcsgenclavier?mode=jsom&estSession=0'
# headers = {
# 'Referer': url_login
# }
# request = self.browser.request_class(url, None, headers)
# infos_data = self.browser.readurl(request)
# infos_xml = etree.XML(infos_data)
# infos = {}
# for el in ("cryptogramme", "nblignes", "nbcolonnes"):
# infos[el] = infos_xml.find(el).text
#
# infos["grille"] = ""
# for g in infos_xml.findall("grille"):
# infos["grille"] += g.text + ","
# infos["keyCodes"] = infos["grille"].split(",")
#
# url = base_url + '/cvcsgenimage?modeClavier=0&cryptogramme=' + infos["cryptogramme"]
# img = Captcha(self.browser.openurl(url), infos)
#
# try:
# img.build_tiles()
# except TileError, err:
# error("Error: %s" % err)
# if err.tile:
# err.tile.display()
#
# self.browser.openurl(url_login)
# self.browser.select_form('authentification')
# self.browser.set_all_readonly(False)
#
# self.browser['codcli'] = login
# self.browser['codsec'] = img.get_codes(password)
# self.browser['cryptocvcs'] = infos["cryptogramme"]
# self.browser.submit()
class BadLoginPage(BasePage):
pass