diff --git a/modules/cic/browser.py b/modules/cic/browser.py
index 88edad20..3e9fbdfa 100644
--- a/modules/cic/browser.py
+++ b/modules/cic/browser.py
@@ -18,105 +18,104 @@
# along with weboob. If not, see .
-from urlparse import urlsplit, parse_qsl, urlparse
-from datetime import datetime, timedelta
+try:
+ from urlparse import urlsplit, parse_qsl, urlparse
+except ImportError:
+ from urllib.parse import urlsplit, parse_qsl, urlparse
-from weboob.deprecated.browser import Browser, BrowserIncorrectPassword
+from datetime import datetime, timedelta
+from random import randint
+
+from weboob.tools.compat import basestring
+from weboob.browser.browsers import LoginBrowser, need_login
+from weboob.browser.profiles import Wget
+from weboob.browser.url import URL
+from weboob.exceptions import BrowserIncorrectPassword
from weboob.capabilities.bank import Transfer, TransferError
-from .pages import LoginPage, LoginErrorPage, AccountsPage, UserSpacePage, EmptyPage, \
- OperationsPage, CardPage, ComingPage, NoOperationsPage, InfoPage, \
- TransfertPage, ChangePasswordPage, VerifCodePage
+from .pages import LoginPage, LoginErrorPage, AccountsPage, UserSpacePage, \
+ OperationsPage, CardPage, ComingPage, NoOperationsPage, \
+ TransfertPage, ChangePasswordPage, VerifCodePage, EmptyPage
__all__ = ['CICBrowser']
-# Browser
-class CICBrowser(Browser):
- PROTOCOL = 'https'
- DOMAIN = 'www.cic.fr'
- CERTHASH = '9f41522275058310a6fb348504daeadd16ae852a686a91383b10ad045da76d29'
- ENCODING = 'iso-8859-1'
- USER_AGENT = Browser.USER_AGENTS['wget']
- PAGES = {'https://www.cic.fr/.*/fr/banques/particuliers/index.html': LoginPage,
- 'https://www.cic.fr/.*/fr/identification/default.cgi': LoginErrorPage,
- 'https://www.cic.fr/.*/fr/banque/situation_financiere.cgi': AccountsPage,
- 'https://www.cic.fr/.*/fr/banque/situation_financiere.html': AccountsPage,
- 'https://www.cic.fr/.*/fr/banque/espace_personnel.aspx': UserSpacePage,
- 'https://www.cic.fr/.*/fr/banque/mouvements.cgi.*': OperationsPage,
- 'https://www.cic.fr/.*/fr/banque/mouvements.html.*': OperationsPage,
- 'https://www.cic.fr/.*/fr/banque/mvts_instance.cgi.*': ComingPage,
- 'https://www.cic.fr/.*/fr/banque/nr/nr_devbooster.aspx.*': OperationsPage,
- 'https://www.cic.fr/.*/fr/banque/operations_carte\.cgi.*': CardPage,
- 'https://www.cic.fr/.*/fr/banque/CR/arrivee\.asp.*': NoOperationsPage,
- 'https://www.cic.fr/.*/fr/banque/BAD.*': InfoPage,
- 'https://www.cic.fr/.*/fr/banque/.*Vir.*': TransfertPage,
- 'https://www.cic.fr/.*/fr/validation/change_password.cgi': ChangePasswordPage,
- 'https://www.cic.fr/.*/fr/validation/verif_code.cgi.*': VerifCodePage,
- 'https://www.cic.fr/.*/fr/': EmptyPage,
- 'https://www.cic.fr/.*/fr/banques/index.html': EmptyPage,
- 'https://www.cic.fr/.*/fr/banque/paci_beware_of_phishing.html.*': EmptyPage,
- 'https://www.cic.fr/.*/fr/validation/(?!change_password|verif_code).*': EmptyPage,
- }
+class CICBrowser(LoginBrowser):
+ PROFILE = Wget()
+ BASEURL = 'https://www.cic.fr'
+
+ login = URL('/sb/fr/banques/particuliers/index.html',
+ '/(?P.*)/fr/$',
+ '/(?P.*)/fr/banques/accueil.html',
+ '/(?P.*)/fr/banques/particuliers/index.html',
+ LoginPage)
+ login_error = URL('/(?P.*)/fr/identification/default.cgi', LoginErrorPage)
+ accounts = URL('/(?P.*)/fr/banque/situation_financiere.cgi',
+ '/(?P.*)/fr/banque/situation_financiere.html',
+ AccountsPage)
+ user_space = URL('/(?P.*)/fr/banque/espace_personnel.aspx', UserSpacePage)
+ operations = URL('/(?P.*)/fr/banque/mouvements.cgi.*',
+ '/(?P.*)/fr/banque/mouvements.html.*',
+ '/(?P.*)/fr/banque/nr/nr_devbooster.aspx.*',
+ OperationsPage)
+ coming = URL('/(?P.*)/fr/banque/mvts_instance.cgi.*', ComingPage)
+ card = URL('/(?P.*)/fr/banque/operations_carte.cgi.*', CardPage)
+ noop = URL('/(?P.*)/fr/banque/CR/arrivee.asp.*', NoOperationsPage)
+ info = URL('/(?P.*)/fr/banque/BAD.*', EmptyPage)
+ transfert = URL('/(?P.*)/fr/banque/virements/vplw_vi.html', EmptyPage)
+ transfert_2 = URL('/(?P.*)/fr/banque/virements/vplw_cmweb.aspx.*', TransfertPage)
+ change_pass = URL('/(?P.*)/fr/validation/change_password.cgi', ChangePasswordPage)
+ verify_pass = URL('/(?P.*)/fr/validation/verif_code.cgi.*', VerifCodePage)
+ empty = URL('/(?P.*)/fr/banques/index.html',
+ '/(?P.*)/fr/banque/paci_beware_of_phishing.*',
+ '/(?P.*)/fr/validation/(?!change_password|verif_code).*',
+ '/(?P.*)/fr/banque/paci_engine/static_content_manager.aspx',
+ '/(?P.*)/fr/banque/DELG_Gestion.*',
+ EmptyPage)
currentSubBank = None
- def is_logged(self):
- return not self.is_on_page(LoginPage) and not self.is_on_page(LoginErrorPage)
+ __states__ = ['currentSubBank']
- def home(self):
- return self.location('https://www.cic.fr/sb/fr/banques/particuliers/index.html')
+ def do_login(self):
+ self.login.go()
- def login(self):
- assert isinstance(self.username, basestring)
- assert isinstance(self.password, basestring)
+ if not self.page.logged:
+ self.page.login(self.username, self.password)
- if not self.is_on_page(LoginPage):
- self.location('https://www.cic.fr/', no_login=True)
+ if not self.page.logged or self.login_error.is_here():
+ raise BrowserIncorrectPassword()
- self.page.login(self.username, self.password)
-
- if not self.is_logged() or self.is_on_page(LoginErrorPage):
- raise BrowserIncorrectPassword()
self.getCurrentSubBank()
+ @need_login
def get_accounts_list(self):
- if not self.is_on_page(AccountsPage):
- self.location('https://www.cic.fr/%s/fr/banque/situation_financiere.cgi' % self.currentSubBank)
- return self.page.get_list()
+ return self.accounts.stay_or_go(subbank=self.currentSubBank).iter_accounts()
def get_account(self, id):
assert isinstance(id, basestring)
- l = self.get_accounts_list()
- for a in l:
+ for a in self.get_accounts_list():
if a.id == id:
return a
- return None
-
def getCurrentSubBank(self):
# the account list and history urls depend on the sub bank of the user
- url = urlparse(self.geturl())
+ url = urlparse(self.url)
self.currentSubBank = url.path.lstrip('/').split('/')[0]
def list_operations(self, page_url):
if page_url.startswith('/') or page_url.startswith('https'):
self.location(page_url)
else:
- self.location('https://%s/%s/fr/banque/%s' % (self.DOMAIN, self.currentSubBank, page_url))
+ self.location('%s/%s/fr/banque/%s' % (self.BASEURL, self.currentSubBank, page_url))
- go_next = True
- while go_next:
- if not self.is_on_page(OperationsPage):
- return
+ if not self.operations.is_here():
+ return iter([])
- for op in self.page.get_history():
- yield op
-
- go_next = self.page.go_next()
+ return self.pagination(lambda: self.page.get_history())
def get_history(self, account):
transactions = []
@@ -129,7 +128,7 @@ class CICBrowser(Browser):
elif last_debit is None:
last_debit = (tr.date - timedelta(days=10)).month
- coming_link = self.page.get_coming_link() if self.is_on_page(OperationsPage) else None
+ coming_link = self.page.get_coming_link() if self.operations.is_here() else None
if coming_link is not None:
for tr in self.list_operations(coming_link):
transactions.append(tr)
@@ -154,44 +153,53 @@ class CICBrowser(Browser):
def transfer(self, account, to, amount, reason=None):
# access the transfer page
- transfert_url = 'WI_VPLV_VirUniSaiCpt.asp?RAZ=ALL&Cat=6&PERM=N&CHX=A'
- self.location('https://%s/%s/fr/banque/%s' % (self.DOMAIN, self.currentSubBank, transfert_url))
+ self.transfert.go(subbank=self.currentSubBank)
# fill the form
- self.select_form(name='FormVirUniSaiCpt')
- self['IDB'] = [account[-1]]
- self['ICR'] = [to[-1]]
- self['MTTVIR'] = '%s' % str(amount).replace('.', ',')
+ form = self.page.get_form(xpath="//form[@id='P:F']")
+ try:
+ form['data_input_indiceCompteADebiter'] = self.page.get_from_account_index(account)
+ form['data_input_indiceCompteACrediter'] = self.page.get_to_account_index(to)
+ except ValueError as e:
+ raise TransferError(e.message)
+ form['[t:dbt%3adouble;]data_input_montant_value_0_'] = '%s' % str(amount).replace('.', ',')
if reason is not None:
- self['LIBDBT'] = reason
- self['LIBCRT'] = reason
- self.submit()
+ form['[t:dbt%3astring;x(27)]data_input_libelleCompteDebite'] = reason
+ form['[t:dbt%3astring;x(31)]data_input_motifCompteCredite'] = reason
+ del form['_FID_GoCancel']
+ del form['_FID_DoValidate']
+ form['_FID_DoValidate.x'] = str(randint(3, 125))
+ form['_FID_DoValidate.y'] = str(randint(3, 22))
+ form.submit()
# look for known errors
- content = unicode(self.response().get_data(), self.ENCODING)
- insufficient_amount_message = u'Montant insuffisant.'
- maximum_allowed_balance_message = u'Solde maximum autorisé dépassé.'
+ content = self.page.get_unicode_content()
+ insufficient_amount_message = u'Le montant du virement doit être positif, veuillez le modifier'
+ maximum_allowed_balance_message = u'Montant maximum autorisé au débit pour ce compte'
- if content.find(insufficient_amount_message) != -1:
+ if insufficient_amount_message in content:
raise TransferError('The amount you tried to transfer is too low.')
- if content.find(maximum_allowed_balance_message) != -1:
+ if maximum_allowed_balance_message in content:
raise TransferError('The maximum allowed balance for the target account has been / would be reached.')
# look for the known "all right" message
- ready_for_transfer_message = u'Confirmez un virement entre vos comptes'
- if not content.find(ready_for_transfer_message):
+ ready_for_transfer_message = u'Confirmer un virement entre vos comptes'
+ if ready_for_transfer_message not in content:
raise TransferError('The expected message "%s" was not found.' % ready_for_transfer_message)
# submit the confirmation form
- self.select_form(name='FormVirUniCnf')
+ form = self.page.get_form(xpath="//form[@id='P:F']")
+ del form['_FID_DoConfirm']
+ form['_FID_DoConfirm.x'] = str(randint(3, 125))
+ form['_FID_DoConfirm.y'] = str(randint(3, 22))
submit_date = datetime.now()
- self.submit()
+ form.submit()
# look for the known "everything went well" message
- content = unicode(self.response().get_data(), self.ENCODING)
- transfer_ok_message = u'Votre virement a été exécuté ce jour'
- if not content.find(transfer_ok_message):
+ content = self.page.get_unicode_content()
+ transfer_ok_message = u'Votre virement a été exécuté'
+ if transfer_ok_message not in content:
raise TransferError('The expected message "%s" was not found.' % transfer_ok_message)
# We now have to return a Transfer object
diff --git a/modules/cic/module.py b/modules/cic/module.py
index 252a2c03..c72a59ec 100644
--- a/modules/cic/module.py
+++ b/modules/cic/module.py
@@ -1,6 +1,7 @@
# -*- coding: utf-8 -*-
# Copyright(C) 2010-2011 Julien Veyssier
+# Copyright(C) 2012-2013 Romain Bignon
#
# This file is part of weboob.
#
@@ -33,8 +34,8 @@ __all__ = ['CICModule']
class CICModule(Module, CapBank):
NAME = 'cic'
- MAINTAINER = u'Romain Bignon'
- EMAIL = 'romain@weboob.org'
+ MAINTAINER = u'Julien Veyssier'
+ EMAIL = 'julien.veyssier@aiur.fr'
VERSION = '1.1'
DESCRIPTION = u'CIC'
LICENSE = 'AGPLv3+'
@@ -57,16 +58,14 @@ class CICModule(Module, CapBank):
raise AccountNotFound()
def iter_coming(self, account):
- with self.browser:
- for tr in self.browser.get_history(account):
- if tr._is_coming:
- yield tr
+ for tr in self.browser.get_history(account):
+ if tr._is_coming:
+ yield tr
def iter_history(self, account):
- with self.browser:
- for tr in self.browser.get_history(account):
- if not tr._is_coming:
- yield tr
+ for tr in self.browser.get_history(account):
+ if not tr._is_coming:
+ yield tr
def iter_transfer_recipients(self, ignored):
for account in self.browser.get_accounts_list():
@@ -88,5 +87,4 @@ class CICModule(Module, CapBank):
except (AssertionError, ValueError):
raise AccountNotFound()
- with self.browser:
- return self.browser.transfer(account, to, amount, reason)
+ return self.browser.transfer(account, to, amount, reason)
diff --git a/modules/cic/pages.py b/modules/cic/pages.py
index 3a0d4d7e..87c3840c 100644
--- a/modules/cic/pages.py
+++ b/modules/cic/pages.py
@@ -18,58 +18,85 @@
# along with weboob. If not, see .
-import urllib
-from urlparse import urlparse, parse_qs
-from decimal import Decimal
+try:
+ from urlparse import urlparse, parse_qs
+except ImportError:
+ from urllib.parse import urlparse, parse_qs
+
+from decimal import Decimal, InvalidOperation
import re
from dateutil.relativedelta import relativedelta
-from weboob.deprecated.browser import Page, BrowserIncorrectPassword, BrokenPageError
-from weboob.tools.ordereddict import OrderedDict
+from weboob.browser.pages import HTMLPage, FormNotFound, LoggedPage
+from weboob.browser.elements import ListElement, ItemElement, SkipItem, method
+from weboob.browser.filters.standard import Filter, Env, CleanText, CleanDecimal, Field, TableCell
+from weboob.browser.filters.html import Link
+from weboob.exceptions import BrowserIncorrectPassword, ParseError
+from weboob.capabilities import NotAvailable
from weboob.capabilities.bank import Account
from weboob.tools.capabilities.bank.transactions import FrenchTransaction
from weboob.tools.date import parse_french_date
-class LoginPage(Page):
+class LoginPage(HTMLPage):
+ REFRESH_MAX = 10.0
+
def login(self, login, passwd):
- self.browser.select_form(name='ident')
- self.browser['_cm_user'] = login.encode(self.browser.ENCODING)
- self.browser['_cm_pwd'] = passwd.encode(self.browser.ENCODING)
- self.browser.submit(nologin=True)
+ form = self.get_form(name='ident')
+ form['_cm_user'] = login
+ form['_cm_pwd'] = passwd
+ form.submit()
+
+ @property
+ def logged(self):
+ return self.doc.xpath('//div[@id="e_identification_ok"]')
-class LoginErrorPage(Page):
+class LoginErrorPage(HTMLPage):
pass
-class ChangePasswordPage(Page):
- def on_loaded(self):
+class EmptyPage(LoggedPage, HTMLPage):
+ REFRESH_MAX = 10.0
+
+
+class UserSpacePage(LoggedPage, HTMLPage):
+ pass
+
+
+class ChangePasswordPage(LoggedPage, HTMLPage):
+ def on_load(self):
raise BrowserIncorrectPassword('Please change your password')
-class VerifCodePage(Page):
- def on_loaded(self):
+class VerifCodePage(LoggedPage, HTMLPage):
+ def on_load(self):
raise BrowserIncorrectPassword('Unable to login: website asks a code from a card')
-class InfoPage(Page):
- pass
+class TransfertPage(LoggedPage, HTMLPage):
+ def get_account_index(self, direction, account):
+ for div in self.doc.getroot().cssselect(".dw_dli_contents"):
+ inp = div.cssselect("input")[0]
+ if inp.name != direction:
+ continue
+ acct = div.cssselect("span.doux")[0].text.replace(" ", "")
+ if account.endswith(acct):
+ return inp.attrib['value']
+ else:
+ raise ValueError("account %s not found" % account)
+
+ def get_from_account_index(self, account):
+ return self.get_account_index('data_input_indiceCompteADebiter', account)
+
+ def get_to_account_index(self, account):
+ return self.get_account_index('data_input_indiceCompteACrediter', account)
+
+ def get_unicode_content(self):
+ return self.content.decode(self.detect_encoding())
-class EmptyPage(Page):
- pass
-
-
-class TransfertPage(Page):
- pass
-
-
-class UserSpacePage(Page):
- pass
-
-
-class AccountsPage(Page):
+class AccountsPage(LoggedPage, HTMLPage):
TYPES = {'C/C': Account.TYPE_CHECKING,
'Livret': Account.TYPE_SAVINGS,
'Pret': Account.TYPE_LOAN,
@@ -78,94 +105,88 @@ class AccountsPage(Page):
'Compte Epargne': Account.TYPE_SAVINGS,
}
- def get_list(self):
- accounts = OrderedDict()
+ @method
+ class iter_accounts(ListElement):
+ item_xpath = '//tr'
+ flush_at_end = True
- for tr in self.document.getiterator('tr'):
- if not tr.getchildren():
- continue
- first_td = tr.getchildren()[0]
- if (first_td.attrib.get('class', '') == 'i g' \
- or first_td.attrib.get('class', '') == 'p g' \
- or 'i _c1' in first_td.attrib.get('class', '') \
- or 'p _c1' in first_td.attrib.get('class', '')) \
- and first_td.find('a') is not None:
+ class item(ItemElement):
+ klass = Account
- a = first_td.find('a')
- link = a.get('href', '')
+ def condition(self):
+ if len(self.el.xpath('./td')) < 2:
+ return False
+
+ first_td = self.el.xpath('./td')[0]
+ return (("i" in first_td.attrib.get('class', '') or "p" in first_td.attrib.get('class', ''))
+ and first_td.find('a') is not None)
+
+ class Label(Filter):
+ def filter(self, text):
+ return text.lstrip(' 0123456789').title()
+
+ class Type(Filter):
+ def filter(self, label):
+ for pattern, actype in AccountsPage.TYPES.iteritems():
+ if label.startswith(pattern):
+ return actype
+ return Account.TYPE_UNKNOWN
+
+ obj_id = Env('id')
+ obj_label = Label(CleanText('./td[1]/a/node()[not(contains(@class, "doux"))]'))
+ obj_coming = Env('coming')
+ obj_balance = Env('balance')
+ obj_currency = FrenchTransaction.Currency('./td[2] | ./td[3]')
+ obj__link_id = Link('./td[1]/a')
+ obj__card_links = []
+ obj_type = Type(Field('label'))
+
+ def parse(self, el):
+ link = el.xpath('./td[1]/a')[0].get('href', '')
if link.startswith('POR_SyntheseLst'):
- continue
+ raise SkipItem()
url = urlparse(link)
p = parse_qs(url.query)
- if 'rib' not in p:
- continue
+ if 'rib' not in p and 'webid' not in p:
+ raise SkipItem()
- for i in (2,1):
- if tr.getchildren()[i].text is None:
- if not tr.getchildren()[i].getchildren():
- continue
- amout = tr.getchildren()[i].getchildren()[0].text
+ for td in el.xpath('./td[2] | ./td[3]'):
+ try:
+ balance = CleanDecimal('.', replace_dots=True)(td)
+ except InvalidOperation:
+ continue
else:
- amout = tr.getchildren()[i].text
- balance = FrenchTransaction.clean_amount(amout)
- currency = Account.get_currency(amout)
- if len(balance) > 0:
break
- balance = Decimal(balance)
+ else:
+ raise ParseError('Unable to find balance for account %s' % CleanText('./td[1]/a')(el))
- id = p['rib'][0]
- if id in accounts:
- account = accounts[id]
+ id = p['rib'][0] if 'rib' in p else p['webid'][0]
+
+ # Handle cards
+ if id in self.parent.objects:
+ account = self.parent.objects[id]
if not account.coming:
account.coming = Decimal('0.0')
account.coming += balance
account._card_links.append(link)
- continue
+ raise SkipItem()
- account = Account()
- account.id = id
+ self.env['id'] = id
- if len(a.getchildren()) > 0:
- account.label = u' '.join([unicode(c.text) for c in a.getchildren()])
- else:
- account.label = unicode(a.text)
- account.label.strip().lstrip(' 0123456789').title()
-
- for pattern, actype in self.TYPES.iteritems():
- if account.label.startswith(pattern):
- account.type = actype
-
- account._link_id = link
- account._card_links = []
-
- # Find accounting amount
- page = self.browser.get_document(self.browser.openurl(link))
- coming = self.find_amount(page, u"Opérations à venir")
- accounting = self.find_amount(page, u"Solde comptable")
+ # Handle real balances
+ page = self.page.browser.open(link).page
+ coming = page.find_amount(u"Opérations à venir") if page else None
+ accounting = page.find_amount(u"Solde comptable") if page else None
if accounting is not None and accounting + (coming or Decimal('0')) != balance:
- self.logger.warning('%s + %s != %s' % (accounting, coming, balance))
+ self.page.logger.warning('%s + %s != %s' % (accounting, coming, balance))
if accounting is not None:
balance = accounting
- if coming is not None:
- account.coming = coming
- account.balance = balance
- account.currency = currency
-
- accounts[account.id] = account
-
- return accounts.itervalues()
-
- def find_amount(self, page, title):
- try:
- td = page.xpath(u'//th[contains(text(), "%s")]/../td' % title)[0]
- except IndexError:
- return None
- else:
- return Decimal(FrenchTransaction.clean_amount(td.text))
+ self.env['balance'] = balance
+ self.env['coming'] = coming or NotAvailable
class Transaction(FrenchTransaction):
@@ -176,153 +197,122 @@ class Transaction(FrenchTransaction):
(re.compile('^RETRAIT DAB (?P\d{2})(?P\d{2}) (?P.*) CARTE [\*\d]+'),
FrenchTransaction.TYPE_WITHDRAWAL),
(re.compile('^CHEQUE( (?P.*))?$'), FrenchTransaction.TYPE_CHECK),
- (re.compile('^(F )?COTIS\.? (?P.*)'),FrenchTransaction.TYPE_BANK),
- (re.compile('^(REMISE|REM CHQ) (?P.*)'),FrenchTransaction.TYPE_DEPOSIT),
+ (re.compile('^(F )?COTIS\.? (?P.*)'), FrenchTransaction.TYPE_BANK),
+ (re.compile('^(REMISE|REM CHQ) (?P.*)'), FrenchTransaction.TYPE_DEPOSIT),
]
_is_coming = False
-class OperationsPage(Page):
- def get_history(self):
- seen = set()
- for tr in self.document.getiterator('tr'):
- # columns can be:
- # - date | value | operation | debit | credit | contre-valeur
- # - date | value | operation | debit | credit
- # - date | operation | debit | credit
- # That's why we skip any extra columns, and take operation, debit
- # and credit from last instead of first indexes.
- tds = tr.getchildren()[:5]
- if len(tds) < 4:
- continue
+class Pagination(object):
+ def next_page(self):
+ try:
+ form = self.page.get_form('//form[@id="paginationForm"]')
+ except FormNotFound:
+ return
- if tds[0].attrib.get('class', '') == 'i g' or \
- tds[0].attrib.get('class', '') == 'p g' or \
- tds[0].attrib.get('class', '').endswith('_c1 c _c1') or \
- tds[0].attrib.get('class', '').endswith('_c1 i _c1'):
- operation = Transaction(0)
-
- parts = [txt.strip() for txt in tds[-3].itertext() if len(txt.strip()) > 0]
-
- # To simplify categorization of CB, reverse order of parts to separate
- # location and institution.
- if parts[0].startswith('PAIEMENT CB'):
- parts.reverse()
-
- date = tds[0].text
- vdate = tds[1].text if len(tds) >= 5 else None
- raw = u' '.join(parts)
-
- operation.parse(date=date, vdate=vdate, raw=raw)
-
- credit = self.parser.tocleanstring(tds[-1])
- debit = self.parser.tocleanstring(tds[-2])
- operation.set_amount(credit, debit)
- operation.id = operation.unique_id(seen)
- yield operation
-
- def go_next(self):
- form = self.document.xpath('//form[@id="paginationForm"]')
- if len(form) == 0:
- return False
-
- form = form[0]
-
- text = self.parser.tocleanstring(form)
+ text = CleanText.clean(form.el)
m = re.search(u'(\d+) / (\d+)', text or '', flags=re.MULTILINE)
if not m:
- return False
+ return
cur = int(m.group(1))
last = int(m.group(2))
if cur == last:
- return False
+ return
- inputs = {}
- for elm in form.xpath('.//input[@type="input"]'):
- key = elm.attrib['name']
- value = elm.attrib['value']
- inputs[key] = value
+ form['page'] = str(cur + 1)
+ return form.request
- inputs['page'] = str(cur + 1)
- self.browser.location(form.attrib['action'], urllib.urlencode(inputs))
+class OperationsPage(LoggedPage, HTMLPage):
+ @method
+ class get_history(Pagination, Transaction.TransactionsElement):
+ head_xpath = '//table[@class="liste"]//thead//tr/th'
+ item_xpath = '//table[@class="liste"]//tbody/tr'
- return True
+ class item(Transaction.TransactionElement):
+ condition = lambda self: len(self.el.xpath('./td')) >= 4 and len(self.el.xpath('./td[@class="i g" or @class="p g" or contains(@class, "_c1")]')) > 0
+
+ class OwnRaw(Filter):
+ def __call__(self, item):
+ parts = [txt.strip() for txt in TableCell('raw')(item)[0].itertext() if len(txt.strip()) > 0]
+
+ # To simplify categorization of CB, reverse order of parts to separate
+ # location and institution.
+ if parts[0].startswith('PAIEMENT CB'):
+ parts.reverse()
+
+ return u' '.join(parts)
+
+ obj_raw = Transaction.Raw(OwnRaw())
+
+ def find_amount(self, title):
+ try:
+ td = self.doc.xpath(u'//th[contains(text(), "%s")]/../td' % title)[0]
+ except IndexError:
+ return None
+ else:
+ return Decimal(FrenchTransaction.clean_amount(td.text))
def get_coming_link(self):
try:
- a = self.parser.select(self.document, u'//a[contains(text(), "Opérations à venir")]', 1, 'xpath')
- except BrokenPageError:
+ a = self.doc.xpath(u'//a[contains(text(), "Opérations à venir")]')[0]
+ except IndexError:
return None
else:
return a.attrib['href']
-class ComingPage(OperationsPage):
- def get_history(self):
- index = 0
- for tr in self.document.xpath('//table[@class="liste"]/tbody/tr'):
- tds = tr.findall('td')
- if len(tds) < 3:
- continue
+class ComingPage(OperationsPage, LoggedPage):
+ @method
+ class get_history(Pagination, Transaction.TransactionsElement):
+ head_xpath = '//table[@class="liste"]//thead//tr/th/text()'
+ item_xpath = '//table[@class="liste"]//tbody/tr'
- tr = Transaction(index)
+ col_date = u"Date de l'annonce"
- date = self.parser.tocleanstring(tds[0])
- raw = self.parser.tocleanstring(tds[1])
- amount = self.parser.tocleanstring(tds[-1])
-
- tr.parse(date=date, raw=raw)
- tr.set_amount(amount)
- tr._is_coming = True
- yield tr
+ class item(Transaction.TransactionElement):
+ obj__is_coming = True
-class CardPage(OperationsPage):
- def get_history(self):
- index = 0
+class CardPage(OperationsPage, LoggedPage):
+ @method
+ class get_history(Pagination, ListElement):
+ class list_cards(ListElement):
+ item_xpath = '//table[@class="liste"]/tbody/tr/td/a'
- # Check if this is a multi-cards page
- pages = []
- for a in self.document.xpath('//table[@class="liste"]/tbody/tr/td/a'):
- card_link = a.get('href')
- history_url = 'https://%s/%s/fr/banque/%s' % (self.browser.DOMAIN, self.browser.currentSubBank, card_link)
- page = self.browser.get_document(self.browser.openurl(history_url))
- pages.append(page)
+ class item(ItemElement):
+ def __iter__(self):
+ card_link = self.el.get('href')
+ history_url = '%s/%s/fr/banque/%s' % (self.page.browser.BASEURL, self.page.browser.currentSubBank, card_link)
+ page = self.page.browser.location(history_url).page
- if len(pages) == 0:
- # If not, add this page as transactions list
- pages.append(self.document)
+ for op in page.get_history():
+ yield op
- for page in pages:
- label = self.parser.tocleanstring(self.parser.select(page.getroot(), 'div.lister p.c', 1))
- label = re.findall('(\d+ [^ ]+ \d+)', label)[-1]
- # use the trick of relativedelta to get the last day of month.
- debit_date = parse_french_date(label) + relativedelta(day=31)
+ class list_history(Transaction.TransactionsElement):
+ head_xpath = '//table[@class="liste"]//thead/tr/th'
+ item_xpath = '//table[@class="liste"]/tbody/tr'
- for tr in page.xpath('//table[@class="liste"]/tbody/tr'):
- tds = tr.findall('td')[:4]
- if len(tds) < 4:
- continue
+ def parse(self, el):
+ label = CleanText('//div[contains(@class, "lister")]//p[@class="c"]')(el)
+ if not label:
+ return
+ label = re.findall('(\d+ [^ ]+ \d+)', label)[-1]
+ # use the trick of relativedelta to get the last day of month.
+ self.env['debit_date'] = parse_french_date(label) + relativedelta(day=31)
- tr = Transaction(index)
+ class item(Transaction.TransactionElement):
+ condition = lambda self: len(self.el.xpath('./td')) >= 4
- parts = [txt.strip() for txt in list(tds[-3].itertext()) + list(tds[-2].itertext()) if len(txt.strip()) > 0]
-
- tr.parse(date=tds[0].text.strip(' \xa0'),
- raw=u' '.join(parts))
- tr.date = debit_date
- tr.type = tr.TYPE_CARD
-
- # Don't take all of the content (with tocleanstring for example),
- # because there is a span.aide.
- tr.set_amount(tds[-1].text)
- yield tr
+ obj_raw = Transaction.Raw('./td[last()-2] | ./td[last()-1]')
+ obj_type = Transaction.TYPE_CARD
+ obj_date = Env('debit_date')
+ obj_rdate = Transaction.Date(TableCell('date'))
-class NoOperationsPage(OperationsPage):
+class NoOperationsPage(OperationsPage, LoggedPage):
def get_history(self):
return iter([])