# -*- coding: utf-8 -*-
# Copyright(C) 2010-2011 Romain Bignon, Pierre Mazière
#
# This file is part of a weboob module.
#
# This weboob module is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This weboob module 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this weboob module. If not, see .
from __future__ import unicode_literals, division
import re
import requests
import base64
import math
import random
from decimal import Decimal
from io import BytesIO
from datetime import datetime, timedelta
from weboob.capabilities.base import empty, find_object, NotAvailable
from weboob.capabilities.bank import (
Account, Investment, Recipient, TransferError, TransferBankError, Transfer,
)
from weboob.capabilities.bill import Document, Subscription, DocumentTypes
from weboob.capabilities.profile import Person, ProfileMissing
from weboob.capabilities.contact import Advisor
from weboob.browser.elements import method, ListElement, TableElement, ItemElement, DictElement
from weboob.browser.exceptions import ServerError
from weboob.browser.pages import LoggedPage, HTMLPage, JsonPage, FormNotFound, pagination
from weboob.browser.filters.html import Attr, Link, TableCell, AttributeNotFound, AbsoluteLink
from weboob.browser.filters.standard import (
CleanText, Field, Regexp, Format, Date, CleanDecimal, Map, AsyncLoad, Async, Env, Slugify,
BrowserURL, Eval, Currency,
)
from weboob.browser.filters.json import Dict
from weboob.exceptions import BrowserUnavailable, BrowserIncorrectPassword, ActionNeeded, ParseError
from weboob.tools.capabilities.bank.transactions import FrenchTransaction
from weboob.tools.captcha.virtkeyboard import MappedVirtKeyboard, VirtKeyboardError
from weboob.tools.compat import unicode, urlparse, parse_qs, urljoin
from weboob.tools.html import html2text
from weboob.tools.date import parse_french_date
from weboob.tools.capabilities.bank.investments import is_isin_valid
def MyDecimal(*args, **kwargs):
kwargs.update(replace_dots=True, default=Decimal(0))
return CleanDecimal(*args, **kwargs)
def myXOR(value, seed):
s = ''
for i in range(len(value)):
s += chr(seed ^ ord(value[i]))
return s
class LCLBasePage(HTMLPage):
def get_from_js(self, pattern, end, is_list=False):
"""
find a pattern in any javascript text
"""
value = None
for script in self.doc.xpath('//script'):
txt = script.text
if txt is None:
continue
start = txt.find(pattern)
if start < 0:
continue
while True:
if value is None:
value = ''
else:
value += ','
value += txt[start + len(pattern):start + txt[start + len(pattern):].find(end) + len(pattern)]
if not is_list:
break
txt = txt[start + len(pattern) + txt[start + len(pattern):].find(end):]
start = txt.find(pattern)
if start < 0:
break
return value
class LCLVirtKeyboard(MappedVirtKeyboard):
symbols = {
'0': '9da2724133f2221482013151735f033c',
'1': '873ab0087447610841ae1332221be37b',
'2': '93ce6c330393ff5980949d7b6c800f77',
'3': 'b2d70c69693784e1bf1f0973d81223c0',
'4': '498c8f5d885611938f94f1c746c32978',
'5': '359bcd60a9b8565917a7bf34522052c3',
'6': 'aba912172f21f78cd6da437cfc4cdbd0',
'7': 'f710190d6b947869879ec02d8e851dfa',
'8': 'b42cc25e1539a15f767aa7a641f3bfec',
'9': 'cc60e5894a9d8e12ee0c2c104c1d5490'
}
url = "/outil/UAUT/Clavier/creationClavier?random="
color = (255, 255, 255, 255)
def __init__(self, basepage):
img = basepage.doc.find("//img[@id='idImageClavier']")
random.seed()
self.url += "%s" % str(int(math.floor(int(random.random() * 1000000000000000000000))))
super(LCLVirtKeyboard, self).__init__(
BytesIO(basepage.browser.open(self.url).content),
basepage.doc, img, self.color, "id")
self.check_symbols(self.symbols, basepage.browser.responses_dirname)
def get_symbol_code(self, md5sum):
code = MappedVirtKeyboard.get_symbol_code(self, md5sum)
return code[-2:]
def get_string_code(self, string):
code = ''
for c in string:
code += self.get_symbol_code(self.symbols[c])
return code
class LoginPage(HTMLPage):
def on_load(self):
try:
form = self.get_form(xpath='//form[@id="setInfosCGS" or @name="form"]')
except FormNotFound:
return
form.submit()
def login(self, login, passwd):
try:
vk = LCLVirtKeyboard(self)
except VirtKeyboardError as err:
self.logger.exception(err)
return False
password = vk.get_string_code(passwd)
seed = -1
s = "var aleatoire = "
for script in self.doc.findall("//script"):
if script.text is None or len(script.text) == 0:
continue
offset = script.text.find(s)
if offset != -1:
seed = int(script.text[offset + len(s) + 1:offset + len(s) + 2])
break
if seed == -1:
raise ParseError("Variable 'aleatoire' not found")
form = self.get_form('//form[@id="formAuthenticate"]')
form['identifiant'] = login
form['postClavierXor'] = base64.b64encode(
myXOR(password, seed).encode("utf-8")
)
try:
form['identifiantRouting'] = self.browser.IDENTIFIANT_ROUTING
except AttributeError:
pass
try:
form.submit()
except BrowserUnavailable:
# Login is not valid
return False
return True
def check_error(self):
errors = self.doc.xpath(u'//*[@class="erreur" or @class="messError"]')
if not errors or self.doc.xpath('//a[@href="/outil/UWHO/Accueil/"]'):
return
for error in errors:
error_text = CleanText(error.xpath('./div/text()'))(self.doc)
if 'Suite à la saisie de plusieurs identifiant / code erronés' in error_text:
raise ActionNeeded(error_text)
if 'Votre identifiant ou votre code personnel est incorrect' in error_text:
raise BrowserIncorrectPassword(error_text)
raise BrowserIncorrectPassword()
class ContractsPage(LoginPage):
def on_load(self):
# after login we are redirect in ContractsPage even if there is an error at login
# I let the error check code here to simplify
# a better solution will be to put error check on browser.py and error parsing in pages.py
self.check_error()
# To avoid skipping contract page the first time we see it,
# and to be able to get the contracts list from it
if self.browser.parsed_contracts:
self.select_contract()
def get_contracts_list(self):
return self.doc.xpath('//input[@name="contratId"]/@value')
def select_contract(self, id_contract=None):
link = self.doc.xpath('//a[contains(text(), "Votre situation globale")]')
if not id_contract and len(link):
self.browser.location(link[0].attrib['href'])
else:
form = self.get_form(nr=0)
if 'contratId' in form:
if id_contract:
form['contratId'] = id_contract
self.browser.current_contract = form['contratId']
form.submit()
class ContractsChoicePage(ContractsPage):
def on_load(self):
self.check_error()
if not self.logged and not self.browser.current_contract:
self.select_contract()
class AccountsPage(LoggedPage, HTMLPage):
def on_load(self):
warn = self.doc.xpath('//div[@id="attTxt"]')
if len(warn) > 0:
raise BrowserIncorrectPassword(warn[0].text)
def get_name(self):
return CleanText('//li[@id="nomClient"]/p')(self.doc)
@method
class get_list(ListElement):
# XXX Ugly Hack to replace account by second occurrence.
# LCL pro website sometimes display the same account twice and only second link is valid to fetch transactions.
def store(self, obj):
assert obj.id
if obj.id in self.objects:
self.logger.warning('There are two objects with the same ID! %s' % obj.id)
self.objects[obj.id] = obj
return obj
item_xpath = '//tr[contains(@onclick, "redirect")]'
flush_at_end = True
class account(ItemElement):
klass = Account
def condition(self):
return '/outil/UWLM/ListeMouvement' in self.el.attrib['onclick']
NATURE2TYPE = {
'001': Account.TYPE_SAVINGS,
'004': Account.TYPE_CHECKING,
'005': Account.TYPE_CHECKING,
'006': Account.TYPE_CHECKING,
'007': Account.TYPE_SAVINGS,
'012': Account.TYPE_SAVINGS,
'023': Account.TYPE_CHECKING,
'036': Account.TYPE_SAVINGS,
'046': Account.TYPE_SAVINGS,
'047': Account.TYPE_SAVINGS,
'049': Account.TYPE_SAVINGS,
'058': Account.TYPE_CHECKING,
'068': Account.TYPE_PEA,
'069': Account.TYPE_SAVINGS,
}
obj__link_id = Format('%s&mode=190', Regexp(CleanText('./@onclick'), "'(.*)'"))
obj__agence = Regexp(Field('_link_id'), r'.*agence=(\w+)')
obj__compte = Regexp(Field('_link_id'), r'compte=(\w+)')
obj_id = Format('%s%s', Field('_agence'), Field('_compte'))
obj__transfer_id = Format('%s0000%s', Field('_agence'), Field('_compte'))
obj_label = CleanText('.//div[@class="libelleCompte"]')
obj_balance = MyDecimal('.//td[has-class("right")]', replace_dots=True)
obj_currency = FrenchTransaction.Currency('.//td[has-class("right")]')
obj_type = Map(Regexp(Field('_link_id'), r'.*nature=(\w+)'), NATURE2TYPE, default=Account.TYPE_UNKNOWN)
obj__market_link = None
obj_number = Field('id')
def get_deferred_cards(self):
trs = self.doc.xpath('//tr[contains(@onclick, "EncoursCB")]')
links = []
for tr in trs:
parent_id = Regexp(CleanText('./@onclick'), r'.*AGENCE=(\w+).*COMPTE=(\w+).*CLE=(\w+)', r'\1\2\3')(tr)
link = Regexp(CleanText('./@onclick'), "'(.*)'")(tr)
links.append((parent_id, link))
return links
@method
class get_advisor(ItemElement):
klass = Advisor
obj_name = CleanText('//div[@id="contacterMaBqMenu"]//p[@id="itemNomContactMaBq"]/span')
obj_email = obj_mobile = obj_fax = NotAvailable
obj_phone = Regexp(CleanText('//div[@id="contacterMaBqMenu"]//p[contains(text(), "Tel")]', replace=[(' ', '')]), '([\s\d]+)', default=NotAvailable)
obj_agency = CleanText('//div[@id="sousContentAgence"]//p[@class="itemSousTitreMenuMaBq"][1]')
def obj_address(self):
address = CleanText('//div[@id="sousContentAgence"]//p[@class="itemSousTitreMenuMaBq"][2]', default=None)(self)
city = CleanText('//div[@id="sousContentAgence"]//p[@class="itemSousTitreMenuMaBq"][3]', default=None)(self)
return "%s %s" % (address, city) if address and city else NotAvailable
class LoansPage(LoggedPage, HTMLPage):
@method
class get_list(TableElement):
item_xpath = '//table[.//th[contains(text(), "Emprunteur")]]/tbody/tr[td[3]]'
head_xpath = '//table[.//th[contains(text(), "Emprunteur")]]/thead/tr/th'
flush_at_end = True
col_id = re.compile('Emprunteur')
col_balance = [u'Capital restant dû', re.compile('Sommes totales restant dues'), re.compile('Montant disponible')]
class account(ItemElement):
klass = Account
obj_balance = CleanDecimal(TableCell('balance'), replace_dots=True, sign=lambda x: -1)
obj_currency = FrenchTransaction.Currency(TableCell('balance'))
obj_type = Account.TYPE_LOAN
obj_id = Env('id')
obj__transfer_id = None
obj_number = Regexp(CleanText(TableCell('id'), replace=[(' ', ''), ('-', '')]), r'(\d{11}[A-Z])')
def obj_label(self):
has_type = CleanText('./ancestor::table[.//th[contains(text(), "Type")]]', default=None)(self)
return CleanText('./td[2]')(self) if has_type else CleanText('./ancestor::table/preceding-sibling::div[1]')(self).split(' - ')[0]
def parse(self, el):
label = Field('label')(self)
trs = self.xpath('//td[contains(text(), $label)]/ancestor::tr[1] | ./ancestor::table[1]/tbody/tr', label=label)
i = [i for i in range(len(trs)) if el == trs[i]]
i = i[0] if i else 0
label = label.replace(' ', '')
self.env['id'] = "%s%s%s" % (Regexp(CleanText(TableCell('id')), r'(\w+)\s-\s(\w+)', r'\1\2')(self), label.replace(' ', ''), i)
class LoansProPage(LoggedPage, HTMLPage):
@method
class get_list(TableElement):
item_xpath = '//table[.//th[contains(text(), "Emprunteur")]]/tbody/tr[td[3]]'
head_xpath = '//table[.//th[contains(text(), "Emprunteur")]]/thead/tr/th'
flush_at_end = True
col_id = re.compile('Emprunteur')
col_balance = [u'Capital restant dû', re.compile('Sommes totales restant dues')]
class account(ItemElement):
klass = Account
obj_balance = CleanDecimal(TableCell('balance'), replace_dots=True, sign=lambda x: -1)
obj_currency = FrenchTransaction.Currency(TableCell('balance'))
obj_type = Account.TYPE_LOAN
obj_id = Env('id')
obj__transfer_id = None
obj_number = Regexp(CleanText(TableCell('id'), replace=[(' ', ''), ('-', '')]), r'(\d{11}[A-Z])')
def obj_label(self):
has_type = CleanText('./ancestor::table[.//th[contains(text(), "Nature libell")]]', default=None)(self)
return CleanText('./td[3]')(self) if has_type else CleanText('./ancestor::table/preceding-sibling::div[1]')(self).split(' - ')[0]
def parse(self, el):
label = Field('label')(self)
trs = self.xpath('//td[contains(text(), $label)]/ancestor::tr[1] | ./ancestor::table[1]/tbody/tr', label=label)
i = [i for i in range(len(trs)) if el == trs[i]]
i = i[0] if i else 0
label = label.replace(' ', '')
self.env['id'] = "%s%s%s" % (Regexp(CleanText(TableCell('id')), r'(\w+)\s-\s(\w+)', r'\1\2')(self), label.replace(' ', ''), i)
class Transaction(FrenchTransaction):
PATTERNS = [
(re.compile('^(?PCB) (?PRETRAIT) DU (?P\d+)/(?P\d+)'),
FrenchTransaction.TYPE_WITHDRAWAL),
(re.compile('^(?P(PRLV|PE)( SEPA)?) (?P.*)'),
FrenchTransaction.TYPE_ORDER),
(re.compile('^(?PCHQ\.) (?P.*)'),
FrenchTransaction.TYPE_CHECK),
(re.compile('^(?PRELEVE CB) AU (\d+)/(\d+)/(\d+)'),
FrenchTransaction.TYPE_CARD),
(re.compile('^(?PCB) (?P.*) (?P\d+)/(?P\d+)/(?P\d+)'),
FrenchTransaction.TYPE_CARD),
(re.compile('^(?P(PRELEVEMENT|TELEREGLEMENT|TIP)) (?P.*)'),
FrenchTransaction.TYPE_ORDER),
(re.compile('^(?P(ECHEANCE\s*)?PRET)(?P.*)'), FrenchTransaction.TYPE_LOAN_PAYMENT),
(re.compile('^(?P(EVI|VIR(EM(EN)?)?T?)(.PERMANENT)? ((RECU|FAVEUR) TIERS|SEPA RECU)?)( /FRM)?(?P.*)'),
FrenchTransaction.TYPE_TRANSFER),
(re.compile('^(?PREMBOURST)(?P.*)'), FrenchTransaction.TYPE_PAYBACK),
(re.compile('^(?PCOM(MISSIONS?)?)(?P.*)'), FrenchTransaction.TYPE_BANK),
(re.compile('^(?P(?PREMUNERATION).*)'), FrenchTransaction.TYPE_BANK),
(re.compile('^(?P(?PABON.*?)\s*.*)'), FrenchTransaction.TYPE_BANK),
(re.compile('^(?P(?PRESULTAT .*?)\s*.*)'), FrenchTransaction.TYPE_BANK),
(re.compile('^(?P(?PTRAIT\..*?)\s*.*)'), FrenchTransaction.TYPE_BANK),
(re.compile('^(?PREM CHQ) (?P.*)'), FrenchTransaction.TYPE_DEPOSIT),
(re.compile('^VIREMENT.*'), FrenchTransaction.TYPE_TRANSFER),
(re.compile('.*(PRELEVEMENTS|PRELVT|TIP).*'), FrenchTransaction.TYPE_ORDER),
(re.compile('.*CHEQUE.*'), FrenchTransaction.TYPE_CHECK),
(re.compile('.*ESPECES.*'), FrenchTransaction.TYPE_DEPOSIT),
(re.compile('.*(CARTE|CB).*'), FrenchTransaction.TYPE_CARD),
(re.compile('.*(AGIOS|ANNULATIONS|IMPAYES|CREDIT).*'), FrenchTransaction.TYPE_BANK),
(re.compile('.*(FRAIS DE TENUE DE COMPTE).*'), FrenchTransaction.TYPE_BANK),
(re.compile(r'.*\b(RETRAIT)\b.*'), FrenchTransaction.TYPE_WITHDRAWAL),
]
class Pagination(object):
def next_page(self):
links = self.page.doc.xpath('//div[@class="pagination"] /a')
if len(links) == 0:
return
for link in links:
if link.xpath('./span')[0].text == 'Page suivante':
return link.attrib.get('href')
return
class AccountHistoryPage(LoggedPage, HTMLPage):
class _get_operations(Pagination, Transaction.TransactionsElement):
item_xpath = '//table[has-class("tagTab") and (not(@style) or @style="")]/tr'
head_xpath = '//table[has-class("tagTab") and (not(@style) or @style="")]/tr/th'
col_raw = [u'Vos opérations', u'Libellé']
class item(Transaction.TransactionElement):
load_details = Attr('.', 'href', default=None) & AsyncLoad
def obj_type(self):
type = Async('details', CleanText(u'//td[contains(text(), "Nature de l\'opération")]/following-sibling::*[1]'))(self)
if not type:
return Transaction.TYPE_UNKNOWN
for pattern, _type in Transaction.PATTERNS:
match = pattern.match(type)
if match:
return _type
return Transaction.TYPE_UNKNOWN
def condition(self):
return (self.parent.get_colnum('date') is not None
and len(self.el.findall('td')) >= 3
and self.el.get('class')
and 'tableTr' not in self.el.get('class'))
def validate(self, obj):
if obj.category == 'RELEVE CB':
obj.type = Transaction.TYPE_CARD_SUMMARY
raw = Async('details', CleanText(u'//td[contains(text(), "Libellé")]/following-sibling::*[1]|//td[contains(text(), "Nom du donneur")]/following-sibling::*[1]', default=obj.raw))(self)
if raw:
if obj.raw in raw or raw in obj.raw or ' ' not in obj.raw:
obj.raw = raw
obj.label = raw
else:
obj.label = '%s %s' % (obj.raw, raw)
obj.raw = '%s %s' % (obj.raw, raw)
m = re.search(r'\d+,\d+COM (\d+,\d+)', raw)
if m:
obj.commission = -CleanDecimal(replace_dots=True).filter(m.group(1))
elif not obj.raw:
# Empty transaction label
obj.raw = obj.label = Async('details', CleanText(u'//td[contains(text(), "Nature de l\'opération")]/following-sibling::*[1]'))(self)
if not obj.date:
obj.date = Async('details', Date(CleanText(u'//td[contains(text(), "Date de l\'opération")]/following-sibling::*[1]', default=u''), dayfirst=True, default=NotAvailable))(self)
obj.rdate = obj.date
obj.vdate = Async('details', Date(CleanText(u'//td[contains(text(), "Date de valeur")]/following-sibling::*[1]', default=u''), dayfirst=True, default=NotAvailable))(self)
obj.amount = Async('details', CleanDecimal(u'//td[contains(text(), "Montant")]/following-sibling::*[1]', replace_dots=True, default=NotAvailable))(self)
# ugly hack to fix broken html
# sometimes transactions have really an amount of 0...
if not obj.amount and CleanDecimal(TableCell('credit'), default=None)(self) is None:
obj.amount = Async('details', CleanDecimal(u'//td[contains(text(), "Montant")]/following-sibling::*[1]', replace_dots=True, default=NotAvailable))(self)
return True
@pagination
def get_operations(self):
return self._get_operations(self)()
class CardsPage(LoggedPage, HTMLPage):
def deferred_date(self):
deferred_date = Regexp(CleanText('//div[@class="date"][contains(text(), "Carte")]'), r'le ([^:]+)', default=None)(self.doc)
assert deferred_date, 'Cannot find deferred_date'
return parse_french_date(deferred_date).date()
def get_card_summary(self):
amount = CleanDecimal.French('//div[@class="montantEncours"]')(self.doc)
if amount:
t = Transaction()
t.date = t.rdate = self.deferred_date()
t.type = Transaction.TYPE_CARD_SUMMARY
t.label = t.raw = CleanText('//div[@class="date"][contains(text(), "Carte")]')(self.doc)
t.amount = abs(amount)
return t
def format_url(self, url):
cb_type = re.match(r'.*(UWCBEncours.*)/.*', url).group(1)
return '/outil/UWCB/%s/listeOperations' % cb_type
@method
class iter_multi_cards(TableElement):
head_xpath = '//table[@class="tagTab"]/tr/th'
item_xpath = '//table[@class="tagTab"]//tr[position()>1]'
col_label = re.compile('Type')
col_number = re.compile('Numéro')
col_owner = re.compile('Titulaire')
col_coming = re.compile('Montant')
class Item(ItemElement):
klass = Account
obj_type = Account.TYPE_CARD
obj_balance = Decimal(0)
obj_parent = Env('parent_account')
obj_coming = CleanDecimal.French(TableCell('coming'))
obj_currency = Currency(TableCell('coming'))
obj__transfer_id = None
obj__cards_list = CleanText(Env('cards_list'))
def obj__transactions_link(self):
link = Attr('.', 'onclick')(self)
url = re.match('.*\'(.*)\'\\.*', link).group(1)
return self.page.format_url(url)
def obj_number(self):
card_number = re.match('((XXXX ){3}X ([0-9]{3}))', CleanText(TableCell('number'))(self))
return card_number.group(1)[0:16] + card_number.group(1)[-3:]
def obj_label(self):
return '%s %s %s' % (
CleanText(TableCell('label'))(self),
CleanText(TableCell('owner'))(self),
Field('number')(self),
)
def obj_id(self):
card_number = re.match('((XXXX ){3}X([0-9]{3}))', CleanText(Field('number'))(self))
return '%s-%s' % (Env('parent_account')(self).id, card_number.group(3))
def get_single_card(self, parent_account):
account = Account()
card_info = CleanText('//select[@id="selectCard"]/option/text()')(self.doc)
# ex: VISA INFINITE DD M FIRSTNAME LASTNAME N°XXXX XXXX XXXX X103
regex = '(.*)N°((XXXX ){3}X([0-9]{3})).*'
card_infos = re.match(regex, card_info)
coming = CleanDecimal.French('//div[@class="montantEncours"]/text()')(self.doc)
account.id = '%s-%s' % (parent_account.id, card_infos.group(4))
account.type = Account.TYPE_CARD
account.parent = parent_account
account.balance = Decimal('0')
account.coming = coming
account.number = card_infos.group(2)
account.label = card_info
account.currency = parent_account.currency
account._transactions_link = self.format_url(self.url)
account._transfer_id = None
# We need to store this url. It will be useful later to get the transactions.
account._cards_list = self.url
return account
def get_child_cards(self, parent_account):
# There is a selector with only one entry when there is only one card
# But not when there are multiple card.
if self.doc.xpath('//select[@id="selectCard"]'):
return [self.get_single_card(parent_account)]
return list(self.iter_multi_cards(parent_account=parent_account, cards_list=self.url))
@method
class iter_transactions(TableElement):
item_xpath = '//tr[contains(@class, "ligne")]'
head_xpath = '//th'
col_date = re.compile('Date')
col_label = re.compile('Libellé')
col_amount = re.compile('Montant')
class item(ItemElement):
klass = Transaction
obj_rdate = obj_bdate = Date(CleanText(TableCell('date')), dayfirst=True)
obj_type = Transaction.TYPE_DEFERRED_CARD
obj_raw = obj_label = CleanText(TableCell('label'))
obj_amount = CleanDecimal.French(TableCell('amount'))
def obj_date(self):
return self.page.deferred_date()
def condition(self):
if Field('date')(self) < Field('rdate')(self):
self.logger.error(
'skipping transaction with rdate(%s) > date(%s) for label(%s)',
Field('rdate')(self), Field('date')(self), Field('label')(self)
)
return False
return True
class BoursePage(LoggedPage, HTMLPage):
ENCODING = 'latin-1'
REFRESH_MAX = 0
TYPES = {
'plan épargne en actions': Account.TYPE_PEA,
"plan d'épargne en actions": Account.TYPE_PEA,
'plan épargne en actions bourse': Account.TYPE_PEA,
"plan d'épargne en actions bourse": Account.TYPE_PEA,
'pea pme bourse': Account.TYPE_PEA,
'pea pme': Account.TYPE_PEA,
}
def on_load(self):
"""
Sometimes we are directed towards a prior html page before accessing Bourse Page.
Submit the form to access the page that contains the Bourse Page's session cookie.
"""
try:
form = self.get_form(id='form')
except FormNotFound: # already on the targetted page
pass
else:
form.submit()
super(BoursePage, self).on_load()
def open_iframe(self):
# should be done always (in on_load)?
for iframe in self.doc.xpath('//iframe[@id="mainIframe"]'):
self.browser.location(iframe.attrib['src'])
break
def password_required(self):
return CleanText(u'//b[contains(text(), "Afin de sécuriser vos transactions, nous vous invitons à créer un mot de passe trading")]')(self.doc)
def get_next(self):
if 'onload' in self.doc.xpath('.//body')[0].attrib:
return re.search('"(.*?)"', self.doc.xpath('.//body')[0].attrib['onload']).group(1)
def get_fullhistory(self):
form = self.get_form(id="historyFilter")
form['cashFilter'] = "ALL"
# We can't go above 2 years
form['beginDayfilter'] = (datetime.strptime(form['endDayfilter'], '%d/%m/%Y') - timedelta(days=730)).strftime('%d/%m/%Y')
form.submit()
@method
class get_list(TableElement):
item_xpath = '//table[has-class("tableau_comptes_details")]//tr[td and not(parent::tfoot)]'
head_xpath = '//table[has-class("tableau_comptes_details")]/thead/tr/th'
col_label = 'Comptes'
col_titres = re.compile('Valorisation')
col_especes = re.compile('Solde espèces')
class item(ItemElement):
klass = Account
load_details = Field('_market_link') & AsyncLoad
obj__especes = CleanDecimal(TableCell('especes'), replace_dots=True, default=0)
obj__titres = CleanDecimal(TableCell('titres'), replace_dots=True, default=0)
obj_valuation_diff = Async('details') & CleanDecimal(
'//td[contains(text(), "value latente")]/following-sibling::td[1]',
replace_dots=True,
)
obj__market_link = Regexp(Attr(TableCell('label'), 'onclick'), "'(.*?)'")
obj__link_id = Async('details') & Link(u'//a[text()="Historique"]')
obj__transfer_id = None
obj_balance = Field('_titres')
obj_currency = Currency(CleanText(TableCell('titres')))
def obj_number(self):
number = CleanText((TableCell('label')(self)[0]).xpath('./div[not(b)]'))(self).replace(' - ', '')
m = re.search(r'(\d{11,})[A-Z]', number)
if m:
number = m.group(0)
return number
def obj_id(self):
return "%sbourse" % Field('number')(self)
def obj_label(self):
return "%s Bourse" % CleanText((TableCell('label')(self)[0]).xpath('./div[b]'))(self)
def obj_type(self):
_label = ' '.join(Field('label')(self).split()[:-1]).lower()
for key in self.page.TYPES:
if key in _label:
return self.page.TYPES.get(key)
return Account.TYPE_MARKET
def get_logout_link(self):
return Link('//a[@class="link-underline" and contains(text(), "espace client")]')(self.doc)
@method
class iter_investment(ListElement):
item_xpath = '//table[@id="tableValeurs"]/tbody/tr[@id and count(descendant::td) > 1]'
class item(ItemElement):
klass = Investment
obj_label = CleanText('.//td[2]/div/a')
obj_code = CleanText('.//td[2]/div/br/following-sibling::text()') & Regexp(pattern='^([^ ]+).*', default=NotAvailable)
obj_quantity = MyDecimal('.//td[3]/span')
obj_diff = MyDecimal('.//td[7]/span')
obj_valuation = MyDecimal('.//td[5]')
def obj_code_type(self):
code = Field('code')(self)
if code and is_isin_valid(code):
return Investment.CODE_TYPE_ISIN
return NotAvailable
def obj_unitvalue(self):
if "%" in CleanText('.//td[4]')(self) and "%" in CleanText('.//td[6]')(self):
return NotAvailable
return MyDecimal('.//td[4]/text()')(self)
def obj_unitprice(self):
if "%" in CleanText('.//td[4]')(self) and "%" in CleanText('.//td[6]')(self):
return NotAvailable
return MyDecimal('.//td[6]')(self)
@pagination
@method
class iter_history(TableElement):
item_xpath = '//table[@id="historyTable" and thead]/tbody/tr'
head_xpath = '//table[@id="historyTable" and thead]/thead/tr/th'
col_date = 'Date'
col_label = u'Opération'
col_quantity = u'Qté'
col_code = u'Libellé'
col_amount = 'Montant'
def next_page(self):
form = self.page.get_form(id="historyFilter")
form['PAGE'] = int(form['PAGE']) + 1
return requests.Request("POST", form.url, data=dict(form)) \
if self.page.doc.xpath('//*[@data-page = $page]', page=form['PAGE']) else None
class item(ItemElement):
klass = Transaction
obj_date = Date(CleanText(TableCell('date')), dayfirst=True)
obj_type = Transaction.TYPE_BANK
obj_amount = CleanDecimal(TableCell('amount'), replace_dots=True)
obj_investments = Env('investments')
def obj_label(self):
return TableCell('label')(self)[0].xpath('./text()')[0].strip()
def parse(self, el):
i = None
if CleanText(TableCell('code'))(self):
i = Investment()
i.label = Field('label')(self)
i.code = unicode(TableCell('code')(self)[0].xpath('./text()[last()]')[0]).strip()
i.quantity = MyDecimal(TableCell('quantity'))(self)
i.valuation = Field('amount')(self)
i.vdate = Field('date')(self)
self.env['investments'] = [i] if i else []
class DiscPage(LoggedPage, HTMLPage):
def on_load(self):
try:
# when life insurance access is restricted, a complete lcl logout form is present, don't use it
# and sometimes there's just no form
form = self.get_form(xpath='//form[not(@id="formLogout")]')
form.submit()
except FormNotFound:
# Sometime no form is present, just a redirection
self.logger.debug('no form on this page')
super(DiscPage, self).on_load()
class NoPermissionPage(LoggedPage, HTMLPage):
def get_error_msg(self):
error_msg = CleanText(
'//div[@id="divContenu"]//div[@id="attTxt" and contains(text(), "vous n\'avez pas accès à cette opération")]'
)(self.doc)
return error_msg
class AVPage(LoggedPage, HTMLPage):
def get_routage_url(self):
for account in self.doc.xpath('//table[@class]/tbody/tr'):
if account.xpath('.//td[has-class("nomContrat")]//a[has-class("routageCAR")]'):
return Link('.//td[has-class("nomContrat")]//a[has-class("routageCAR")]')(account)
def is_website_life_insurance(self):
# no need specific account to go on life insurance external website
# because we just need to go on life insurance external website
return bool(self.get_routage_url())
def get_calie_life_insurances_first_index(self):
# indices are associated to calie life insurances to make requests to them
# if only one life insurance, this request directly leads to details on CaliePage
# otherwise, any index will lead to CalieContractsPage,
# so we stop at the first index
for account in self.doc.xpath('//table[@class]/tbody/tr'):
if account.xpath('.//td[has-class("nomContrat")]//a[contains(@class, "redirect")][@href="#"]'):
index = Attr(account.xpath('.//td[has-class("nomContrat")]//a[contains(@class, "redirect")][@href="#"]'), 'id')(self)
return index
@method
class get_popup_life_insurance(ListElement):
item_xpath = '//table[@class]/tbody/tr'
class item(ItemElement):
klass = Account
def condition(self):
if self.obj_balance(self) == 0 and not self.el.xpath('.//td[has-class("nomContrat")]//a'):
self.logger.warning("ignoring an AV account because there's no link for it")
return False
# there is life insurance detail page link but check if it's a popup
return self.el.xpath('.//td[has-class("nomContrat")]//a[has-class("clickPopupDetail")]')
obj__owner = CleanText('.//td[2]')
obj_label = Format(u'%s %s', CleanText('.//td/text()[following-sibling::br]'), obj__owner)
obj_balance = CleanDecimal('.//td[last()]', replace_dots=True)
obj_type = Account.TYPE_LIFE_INSURANCE
obj_currency = 'EUR'
obj__link_id = None
obj__market_link = None
obj__coming_links = []
obj__transfer_id = None
obj_number = Field('id')
obj__external_website = False
obj__is_calie_account = False
def obj_id(self):
_id = CleanText('.//td/@id')(self)
# in old code, we use _id, it seems that is not used anymore
# but check if it's the case for all users
assert not _id, '_id is still used to retrieve life insurance'
try:
self.page.browser.assurancevie.go()
ac_details_page = self.page.browser.open(Link('.//td[has-class("nomContrat")]//a')(self)).page
return CleanText('(//tr[3])/td[2]')(ac_details_page.doc)
except ServerError:
self.logger.debug("link didn't work, trying with the form instead")
# the above server error can cause the form to fail, so we may have to go back on the accounts list before submitting
self.page.browser.open(self.page.url)
# redirection to lifeinsurances accounts and comeback on Lcl original website
page = self.obj__form().submit().page
# Getting the account details from the JSON containing the account information:
details_page = self.page.browser.open(BrowserURL('av_investments')(self)).page
account_id = Dict('situationAdministrativeEpargne/idcntcar')(details_page.doc)
page.come_back()
return account_id
def obj__form(self):
# maybe deprecated
form_id = Attr('.//td[has-class("nomContrat")]//a', 'id', default=None)(self)
if form_id:
if '-' in form_id:
id_contrat = re.search(r'^(.*?)-', form_id).group(1)
producteur = re.search(r'-(.*?)$', form_id).group(1)
else:
id_contrat = form_id
producteur = None
else:
if len(self.xpath('.//td[has-class("nomContrat")]/a[has-class("clickPopupDetail")]')):
# making a form of this link sometimes makes the site return an empty response...
# the link is a link to some info, not full AV website
# it's probably an indication the account is restricted anyway, so avoid it
self.logger.debug("account is probably restricted, don't try its form")
return None
# sometimes information are not in id but in href
url = Attr('.//td[has-class("nomContrat")]//a', 'href', default=None)(self)
parsed_url = urlparse(url)
params = parse_qs(parsed_url.query)
id_contrat = params['ID_CONTRAT'][0]
producteur = params['PRODUCTEUR'][0]
if self.xpath('//form[@id="formRedirectPart"]'):
form = self.page.get_form('//form[@id="formRedirectPart"]')
else:
form = self.page.get_form('//form[@id="formRoutage"]')
form['PRODUCTEUR'] = producteur
form['ID_CONTRAT'] = id_contrat
return form
class CalieContractsPage(LoggedPage, HTMLPage):
@method
class iter_calie_life_insurance(TableElement):
head_xpath = '//table[contains(@id, "MainTable")]//tr[contains(@id, "HeadersRow")]//td[text()]'
item_xpath = '//table[contains(@id, "MainTable")]//tr[contains(@id, "DataRow")]'
col_number = 'Numéro contrat' # internal contrat number
class item(ItemElement):
klass = Account
# internal contrat number, to be replaced by external number in CaliePage.fill_account()
# obj_id is needed here though, to avoid dupicate account errors
obj_id = CleanText(TableCell('number'))
obj_url = AbsoluteLink('.//a') # need AbsoluteLink since we moved out of basurl domain
class SendTokenPage(LoggedPage, LCLBasePage):
def on_load(self):
form = self.get_form('//form')
return form.submit()
class Form2Page(LoggedPage, LCLBasePage):
def assurancevie_hist_not_available(self):
msg = "Ne détenant pas de compte dépôt chez LCL, l'accès à ce service vous est indisponible"
return msg in CleanText('//div[@id="attTxt"]')(self.doc)
def on_load(self):
if self.assurancevie_hist_not_available():
return
error = CleanText('//div[@id="attTxt"]/text()[1]')(self.doc)
if "L’accès au service est momentanément indisponible" in error:
raise BrowserUnavailable(error)
form = self.get_form()
return form.submit()
class CalieTableElement(TableElement):
# We need to set the first column to 1 otherwise
# there is a shift between column titles and contents
def get_colnum(self, name):
return super(CalieTableElement, self).get_colnum(name) + 1
class CaliePage(LoggedPage, HTMLPage):
def check_error(self):
message = CleanText('//div[contains(@class, "disclaimer-div")]//text()[contains(., "utilisation vaut acceptation")]')(self.doc)
if self.doc.xpath('//button[@id="acceptDisclaimerButton"]') and message:
raise ActionNeeded(message)
@method
class iter_investment(CalieTableElement):
# Careful, contains many nested
# Two first lines are titles, two last are investment sum-ups
item_xpath = '//table[@class="dxgvTable dxgvRBB"]//tr[contains(@class, "DataRow")]'
head_xpath = '//table[contains(@id, "MainTable")]//tr[contains(@id, "HeadersRow")]//td[text()]'
col_label = 'Support'
col_vdate = 'Date de valeur'
col_original_valuation = 'Valeur dans la devise du support'
col_valuation = 'Valeur dans la devise du support (EUR)'
col_unitvalue = 'Valeur unitaire'
col_quantity = 'Parts'
col_diff_ratio = 'Performance'
col_portfolio_share = 'Répartition (%)'
class item(ItemElement):
klass = Investment
obj_label = CleanText(TableCell('label'))
obj_original_valuation = CleanDecimal(TableCell('original_valuation'), replace_dots=True)
obj_valuation = CleanDecimal(TableCell('valuation'), replace_dots=True)
obj_vdate = Date(CleanText(TableCell('vdate')), dayfirst=True)
obj_unitvalue = CleanDecimal(TableCell('unitvalue'), replace_dots=True, default=NotAvailable) # displayed with format '123.456,78 EUR'
obj_quantity = CleanDecimal(TableCell('quantity'), replace_dots=True, default=NotAvailable) # displayed with format '1.234,5678 u.'
obj_portfolio_share = Eval(lambda x: x / 100, CleanDecimal(TableCell('portfolio_share')))
def obj_diff_ratio(self):
_diff_ratio = CleanDecimal(TableCell('diff_ratio'), default=NotAvailable)(self)
if not empty(_diff_ratio):
return Eval(lambda x: x / 100, _diff_ratio)(self)
return NotAvailable
# Unfortunately on the Calie space the links to the
# invest details return Forbidden even on the website
obj_code = NotAvailable
obj_code_type = NotAvailable
@method
class fill_account(ItemElement):
obj_number = obj_id = Regexp(CleanText('.'), r'Numéro externe (.{10})')
obj_label = Format(
'%s %s',
Regexp(CleanText('.'), r'Produit (.*) Statut'),
Field('id')
)
obj_balance = CleanDecimal('//tr[contains(@id, "FooterRow")]', replace_dots=True)
obj_type = Account.TYPE_LIFE_INSURANCE
obj_currency = 'EUR'
obj__external_website = True
obj__is_calie_account = True
obj__transfer_id = None
def obj__history_url(self):
relative_url = Regexp(Attr('//a[contains(text(), "Opérations")]', 'onclick'), r'href=\'(.*)\'')(self)
return urljoin(self.page.url, relative_url)
class AVDetailPage(LoggedPage, LCLBasePage):
def come_back(self):
session = self.get_from_js('idSessionSag = "', '"')
params = {}
params['sessionSAG'] = session
params['stbpg'] = 'pagePU'
params['act'] = ''
params['typeaction'] = 'reroutage_retour'
params['site'] = 'LCLI'
params['stbzn'] = 'bnc'
return self.browser.location('https://assurance-vie-et-prevoyance.secure.lcl.fr/filiale/entreeBam', params=params)
class AVListPage(LoggedPage, JsonPage):
@method
class iter_life_insurance(DictElement):
item_xpath = 'syntheseContrats'
class item(ItemElement):
def condition(self):
activity = Dict('lcstacntgen')(self)
account_type = Dict('lcgampdt')(self)
# We ignore accounts without activities or when the activity is 'Closed',
# they are inactive and closed and they don't appear on the website.
return bool(
activity and account_type
and activity.lower() == 'actif'
and account_type.lower() == 'epargne'
)
klass = Account
obj_id = obj_number = Dict('idcntcar')
obj_balance = CleanDecimal(Dict('mtvalcnt'))
obj_label = Dict('lnpdt')
obj_type = Account.TYPE_LIFE_INSURANCE
obj_currency = 'EUR'
obj__external_website = True
obj__form = None
obj__link_id = None
obj__market_link = None
obj__coming_links = []
obj__transfer_id = None
obj__is_calie_account = False
class AVHistoryPage(LoggedPage, JsonPage):
@method
class iter_history(DictElement):
item_xpath = 'listeOperations'
class item(ItemElement):
klass = Transaction
obj_label = CleanText(Dict('lcope'))
obj_amount = CleanDecimal(Dict('mtope'))
obj_type = Transaction.TYPE_BANK
obj_investments = NotAvailable
# The 'idope' key contains a string such as "70_ABC666ABC 2018-03-182018-03-16-20.55.27.960852"
# 70= N° transaction, 6660666= N° account, 2018-03-18= date and 2018-03-16=rdate.
# We thus use "70_ABC666ABC" for the transaction ID.
obj_id = Regexp(CleanText(Dict('idope')), '(\d+_[\dA-Z]+)')
def obj__dates(self):
raw = CleanText(Dict('idope'))(self)
m = re.findall('\d{4}-\d{2}-\d{2}', raw)
# We must verify that the two dates are correctly fetched
assert len(m) == 2
return m
def obj_date(self):
return Date().filter(Field('_dates')(self)[0])
def obj_rdate(self):
return Date().filter(Field('_dates')(self)[1])
class AVInvestmentsPage(LoggedPage, JsonPage):
def update_life_insurance_account(self, life_insurance):
life_insurance._owner = Format(
'%s %s',
Dict('situationAdministrativeEpargne/lppeoscp'),
Dict('situationAdministrativeEpargne/lnpeoscp'),
)(self.doc)
life_insurance.label = '%s %s' % (Dict('situationAdministrativeEpargne/lcofc')(self.doc), life_insurance._owner)
life_insurance.valuation_diff = CleanDecimal(Dict('situationFinanciereEpargne/mtpmvcnt'), default=NotAvailable)(self.doc)
return life_insurance
@method
class iter_investment(DictElement):
item_xpath = 'listeSupports/support'
class item(ItemElement):
klass = Investment
obj_label = CleanText(Dict('lcspt'))
obj_valuation = CleanDecimal(Dict('mtvalspt'))
obj_code = CleanText(Dict('cdsptisn'), default=NotAvailable)
obj_unitvalue = CleanDecimal(Dict('mtliqpaaspt'), default=NotAvailable)
obj_quantity = CleanDecimal(Dict('qtpaaspt'), default=NotAvailable)
obj_diff = CleanDecimal(Dict('mtpmvspt'), default=NotAvailable)
obj_vdate = Date(Dict('dvspt'), default=NotAvailable)
def obj_portfolio_share(self):
ptf = CleanDecimal(Dict('txrpaspt'), default=NotAvailable)(self)
ptf /= 100
return ptf
def obj_code_type(self):
if is_isin_valid(Field('code')(self)):
return Investment.CODE_TYPE_ISIN
return NotAvailable
class RibPage(LoggedPage, LCLBasePage):
def get_iban(self):
if (self.doc.xpath('//div[contains(@class, "rib_cadre")]//div[contains(@class, "rib_internat")]')):
return CleanText('//div[contains(@class, "rib_cadre") and not(contains(@class, "hidden"))]//div[contains(@class, "rib_internat")]//p//strong/text()[1]', replace=[(' ', '')])(self.doc)
def check_iban_by_account(self, account_id):
iban_account_id = CleanText().filter(self.doc.xpath('(//td[@class[contains(., "guichet-")]]/following-sibling::*)[1]/strong'))
iban_guichet_id = CleanText().filter(self.doc.xpath('(//td[@class[contains(., "guichet-")]]/strong)[1]'))
iban_account = "%s%s" % (iban_guichet_id, iban_account_id[4:])
if account_id == iban_account:
return CleanText('//div[contains(@class, "rib_cadre") and not(contains(@class, "hidden"))]//div[contains(@class, "rib_internat")]//p//strong/text()[1]', replace=[(' ', '')])(self.doc)
return None
def has_iban_choice(self):
return False if self.doc.xpath('(//strong[contains(., "RELEVE D\'IDENTITE BANCAIRE")])[1]') else True
class HomePage(LoggedPage, HTMLPage):
pass
class TransferPage(LoggedPage, HTMLPage):
def on_load(self):
# This aims to track input errors.
script_error = CleanText(u"//script[contains(text(), 'if (\"true\"===\"true\")')]")(self.doc)
if script_error:
raise TransferBankError(message=CleanText().filter(html2text(re.search(u'\.html\("(.*?)"\)', script_error).group(1))))
def can_transfer(self, account_transfer_id):
for div in self.doc.xpath('//div[input[@id="indexCompteEmetteur"]]//div[@class="infoCompte" and not(@title)]'):
if account_transfer_id in CleanText('.', replace=[(' ', '')])(div):
return True
return False
def get_account_index(self, xpath, account_id):
for option in self.doc.xpath('//select[@id=$id]/option', id=xpath):
if account_id in CleanText('.', replace=[(' ', '')])(option):
return option.attrib['value']
else:
raise TransferError("account %s not found" % account_id)
def choose_recip(self, recipient):
form = self.get_form(id='formulaire')
form['indexCompteDestinataire'] = self.get_value(recipient._transfer_id, 'recipient')
form.submit()
def transfer(self, amount, reason):
form = self.get_form(id='formulaire')
form['libMontant'] = amount
form['motifVirement'] = reason
form.submit()
def deferred_transfer(self, amount, reason, exec_date):
form = self.get_form(id='formulaire')
form['libMontant'] = amount
form['motifVirement'] = reason
form['libDateProg'] = exec_date.strftime('%d/%m/%Y')
form['dateFinVirement'] = 'N'
form['frequenceVirement'] = 'UD'
form['premierJourVirementProg'] = '01'
form['typeVirement'] = 'Programme'
form.submit()
def get_id_from_response(self, acc):
id_xpath = '//div[@id="contenuPageVirement"]//div[@class="infoCompte" and not(@title)]'
acc_ids = [CleanText('.')(acc_id) for acc_id in self.doc.xpath(id_xpath)]
# there should have 2 ids, one for account and one for recipient
assert len(acc_ids) == 2
for index, acc_id in enumerate(acc_ids):
_id = acc_id.split(' ')
if len(_id) == 2:
# to match with weboob account id
acc_ids[index] = _id[0] + _id[1][4:]
if acc == 'account':
return acc_ids[0]
return acc_ids[1]
def handle_response(self, account, recipient):
transfer = Transfer()
transfer._account = account
transfer.account_id = self.get_id_from_response('account')
transfer.account_iban = account.iban
transfer.account_label = account.label
transfer.account_balance = account.balance
assert account._transfer_id in CleanText(
u'//div[div[@class="libelleChoix" and contains(text(), "Compte émetteur")]] \
//div[@class="infoCompte" and not(@title)]', replace=[(' ', '')]
)(self.doc)
transfer._recipient = recipient
transfer.recipient_id = self.get_id_from_response('recipient')
transfer.recipient_iban = recipient.iban
transfer.recipient_label = recipient.label
assert recipient._transfer_id in CleanText(
u'//div[div[@class="libelleChoix" and contains(text(), "Compte destinataire")]] \
//div[@class="infoCompte" and not(@title)]', replace=[(' ', '')]
)(self.doc)
transfer.currency = FrenchTransaction.Currency('//div[@class="topBox"]/div[@class="montant"]')(self.doc)
transfer.amount = CleanDecimal('//div[@class="topBox"]/div[@class="montant"]', replace_dots=True)(self.doc)
transfer.exec_date = Date(
Regexp(CleanText('//div[@class="topBox"]/div[@class="date"]'), r'(\d{2}\/\d{2}\/\d{4})'),
dayfirst=True
)(self.doc)
# skip html comment with filtering on text() content
transfer.label = CleanText('//div[@class="motif"]/text()[contains(., "Motif : ")]',
replace=[('Motif : ', '')])(self.doc)
return transfer
def confirm(self):
form = self.get_form(id='formulaire')
form.submit()
def get_value(self, _id, value_type):
for div in self.doc.xpath('//div[@onclick]'):
if _id in CleanText('.//div[not(@title)]', replace=[(' ', '')])(div):
return Regexp(Attr('.', 'onclick'), '(\d+)')(div)
raise TransferError('Could not find %s account.' % value_type)
def choose_origin(self, account_transfer_id):
form = self.get_form()
form['indexCompteEmetteur'] = self.get_value(account_transfer_id, 'origin')
form.submit()
@method
class iter_recipients(ListElement):
item_xpath = '//div[@id="listeDestinataires"]//div[@class="pointeur cardCompte"]'
class Item(ItemElement):
klass = Recipient
def condition(self):
return len(self.el.xpath('./div')) > 1
obj_id = CleanText('./div[@class="infoCompte" and not(@title)]', replace=[(' 0000', '')])
obj__transfer_id = CleanText('./div[@class="infoCompte" and not(@title)]', replace=[(' ', '')])
obj_label = CleanText('./div[1]')
obj_bank_name = Env('bank_name')
obj_category = Env('category')
obj_iban = Env('iban')
def obj_enabled_at(self):
return datetime.now().replace(microsecond=0)
def validate(self, obj):
return Field('id')(self) != self.env['account_transfer_id']
def parse(self, el):
if bool(CleanText('./div[@id="soldeEurosCompte"]')(self)):
self.env['category'] = u'Interne'
account = find_object(self.page.browser.get_accounts_list(), id=self.obj_id(self))
self.env['iban'] = account.iban if account else NotAvailable
self.env['bank_name'] = u'LCL'
else:
self.env['category'] = u'Externe'
self.env['iban'] = self.obj_id(self)
self.env['bank_name'] = NotAvailable
def check_error(self):
transfer_confirmation_msg = CleanText('//div[@class="alertConfirmationVirement"]')(self.doc)
assert transfer_confirmation_msg, 'Transfer confirmation message is not found.'
class AddRecipientPage(LoggedPage, HTMLPage):
def validate(self, iban, label):
form = self.get_form(id='mainform')
form['PAYS_IBAN'] = iban[:2]
form['LIBELLE'] = label
form['COMPTE_IBAN'] = iban[2:]
form.submit()
class CheckValuesPage(LoggedPage, HTMLPage):
def check_values(self, iban, label):
values = CleanText('//form[contains(@action, "/outil")]')(self.doc)
assert iban in values, 'Iban (%s) not found in values: %s' % (iban, values)
assert label in values, 'Recipient label (%s) not found in values: %s' % (label, values)
class DocumentsPage(LoggedPage, HTMLPage):
def do_search_request(self):
form = self.get_form(id="rechercherForm")
form['listePeriode'] = "PERIODE1"
form['listeFamille'] = "ALL"
form['debutRec'] = None
form['finRec'] = None
form['typeDocFamHidden'] = "ALL"
form['typeDocSFamHidden'] = None
form.submit()
@method
class get_list(TableElement):
head_xpath = '//table[@class="dematTab"]/thead/tr/th'
item_xpath = u'//table[@class="dematTab"]/tbody/tr[./td[@class="dematTab-firstCell"]]'
ignore_duplicate = True
col_label = 'Nature de document'
col_id = 'Type de document'
col_url = 'Visualiser'
col_date = 'Date'
class item(ItemElement):
klass = Document
obj_id = Slugify(Format('%s_%s', CleanText(TableCell('id')), CleanText(TableCell('date'))))
obj_label = Format('%s %s', CleanText(TableCell('label')), CleanText(TableCell('date')))
obj_date = Date(CleanText(TableCell('date')), dayfirst=True)
obj_format = "pdf"
def obj_url(self):
return Link(TableCell('url')(self)[0].xpath('./a'))(self)
def obj_type(self):
if 'Relevé' in Field('label')(self):
return DocumentTypes.STATEMENT
elif 'Bourse' in Field('label')(self):
return DocumentTypes.REPORT
elif ('information' in Field('id')(self)) or ('avis' in Field('id')(self)):
return DocumentTypes.NOTICE
else:
return DocumentTypes.OTHER
class ClientPage(LoggedPage, HTMLPage):
@method
class get_item(ItemElement):
klass = Subscription
obj_id = CleanText('//li[@id="nomClient"]', replace=[('M', ''), ('Mme', ''), (' ', '')])
obj_label = CleanText('//li[@id="nomClient"]', replace=[('M', ''), ('Mme', '')])
obj_subscriber = CleanText('//li[@id="nomClient"]', replace=[('M', ''), ('Mme', '')])
class RecipConfirmPage(CheckValuesPage):
pass
class RecipientPage(LoggedPage, HTMLPage):
pass
class SmsPage(LoggedPage, HTMLPage):
pass
class RecipRecapPage(CheckValuesPage):
pass
class ProfilePage(LoggedPage, HTMLPage):
def get_profile(self, name):
error_xpath = '//div[contains(text(), "Nous vous invitons à prendre contact avec votre conseiller")]'
if self.doc.xpath(error_xpath):
raise ProfileMissing(CleanText(error_xpath, children=False)(self.doc))
profile = Person()
profile.name = name
try:
profile.email = Attr('//input[@id="textMail"]', 'value', default=NotAvailable)(self.doc)
except AttributeNotFound:
pass
nb = Attr('//input[@id="nbEnfant"]', 'value', default=NotAvailable)(self.doc)
if nb:
profile.children = Decimal(nb)
return profile
class DepositPage(LoggedPage, HTMLPage):
@method
class get_list(TableElement):
head_xpath = '//table/thead/tr/th'
item_xpath = '//table/tbody/tr[not(@class="tableTrSolde")]'
col_owner = 'Titulaire'
col_name = 'Nom du contrat'
col_balance = 'Capital investi'
class item(ItemElement):
klass = Account
obj_type = Account.TYPE_DEPOSIT
obj_label = Format('%s %s', CleanText(TableCell('name')), CleanText(TableCell('owner')))
obj_balance = MyDecimal(TableCell('balance'))
obj_currency = 'EUR'
obj__contract = CleanText(TableCell('name'))
obj__link_index = Regexp(CleanText('.//a/@id'), r'(\d+)')
# So it can be modified later
obj_id = None
obj__transfer_id = None
def set_deposit_account_id(self, account):
account.id = CleanText('//td[contains(text(), "N° contrat")]/following::td[1]//b')(self.doc)