diff --git a/modules/oney/__init__.py b/modules/oney/__init__.py
new file mode 100644
index 0000000000000000000000000000000000000000..2404accec5cd2535397abb2853ee6776474da62d
--- /dev/null
+++ b/modules/oney/__init__.py
@@ -0,0 +1,22 @@
+# -*- coding: utf-8 -*-
+
+# Copyright(C) 2014 Budget Insight
+#
+# 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 .
+
+from .backend import OneyBackend
+
+__all__ = ['OneyBackend']
diff --git a/modules/oney/backend.py b/modules/oney/backend.py
new file mode 100644
index 0000000000000000000000000000000000000000..12c4251f66fe93e46669c45f65fe2cb87170d125
--- /dev/null
+++ b/modules/oney/backend.py
@@ -0,0 +1,57 @@
+# -*- coding: utf-8 -*-
+
+# Copyright(C) 2014 Budget Insight
+#
+# 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 .
+
+
+
+from weboob.capabilities.bank import ICapBank, AccountNotFound
+from weboob.capabilities.base import find_object
+from weboob.tools.backend import BaseBackend, BackendConfig
+from weboob.tools.value import ValueBackendPassword
+
+from .browser import OneyBrowser
+
+
+__all__ = ['OneyBackend']
+
+
+class OneyBackend(BaseBackend, ICapBank):
+ NAME = 'oney'
+ MAINTAINER = u'Vincet Paredes'
+ EMAIL = 'vparedes@budget-insight.com'
+ VERSION = '0.j'
+ LICENSE = 'AGPLv3+'
+ DESCRIPTION = 'Oney'
+ CONFIG = BackendConfig(ValueBackendPassword('login', label='Identifiant', masked=False),
+ ValueBackendPassword('password', label='Mot de passe'))
+ BROWSER = OneyBrowser
+
+ def create_default_browser(self):
+ return self.create_browser(self.config['login'].get(),
+ self.config['password'].get())
+
+ def iter_accounts(self):
+ for account in self.browser.get_accounts_list():
+ yield account
+
+ def get_account(self, _id):
+ return find_object(self.browser.get_accounts_list(), id=_id, error=AccountNotFound)
+
+ def iter_history(self, account):
+ for tr in self.browser.iter_history(account):
+ yield tr
diff --git a/modules/oney/browser.py b/modules/oney/browser.py
new file mode 100644
index 0000000000000000000000000000000000000000..1850f42d395b8472f6f5a8bb6904b2b88c222395
--- /dev/null
+++ b/modules/oney/browser.py
@@ -0,0 +1,66 @@
+# -*- coding: utf-8 -*-
+
+# Copyright(C) 2014 Budget Insight
+#
+# 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 .
+
+from weboob.capabilities.bank import Account
+
+from weboob.tools.exceptions import BrowserIncorrectPassword
+from weboob.tools.browser2 import LoginBrowser, URL, need_login
+
+from .pages import LoginPage, IndexPage, OperationsPage
+
+
+__all__ = ['OneyBrowser']
+
+
+class OneyBrowser(LoginBrowser):
+ BASEURL = 'https://www.oney.fr'
+ VERIFY = False
+
+ index = URL(r'/oney/client', IndexPage)
+ login = URL(r'/oney/client', LoginPage)
+ operations = URL(r'/oney/client', OperationsPage)
+
+ def do_login(self):
+ assert isinstance(self.username, basestring)
+ assert isinstance(self.password, basestring)
+ self.login.go()
+
+ self.page.login(self.username, self.password)
+
+ if not self.index.is_here():
+ raise BrowserIncorrectPassword()
+
+ @need_login
+ def get_accounts_list(self):
+ balance = self.index.stay_or_go().get_balance()
+ account = Account()
+ account.id = self.username
+ account.label = u'Carte Oney'
+ account.balance = balance
+ account.currency = u'EUR'
+ return [account]
+
+
+ @need_login
+ def iter_history(self, account):
+ post = {'task': 'Synthese', 'process': 'SyntheseCompte', 'taskid':'Releve'}
+ self.operations.go(data=post)
+
+ return sorted(self.page.iter_transactions(), key=lambda tr: tr.rdate, reverse=True)
+
diff --git a/modules/oney/pages.py b/modules/oney/pages.py
new file mode 100644
index 0000000000000000000000000000000000000000..8e39ae14fbd91c31caf618aba159795b3e87538d
--- /dev/null
+++ b/modules/oney/pages.py
@@ -0,0 +1,154 @@
+# -*- coding: utf-8 -*-
+
+# Copyright(C) 2014 Budget Insight
+#
+# 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 .
+
+import re
+from cStringIO import StringIO
+
+import requests
+
+from weboob.tools.capabilities.bank.transactions import FrenchTransaction
+from weboob.tools.captcha.virtkeyboard import MappedVirtKeyboard, VirtKeyboardError
+
+from weboob.tools.browser2.page import HTMLPage, method, ListElement, ItemElement, LoggedPage, pagination
+from weboob.tools.browser2.filters import Env, CleanDecimal, ParseError
+
+__all__ = ['LoginPage', 'IndexPage', 'OperationsPage']
+
+class Transaction(FrenchTransaction):
+ PATTERNS = [(re.compile(ur'^(?P.*?) - traité le \d+/\d+$'), FrenchTransaction.TYPE_CARD)]
+
+
+class VirtKeyboard(MappedVirtKeyboard):
+ symbols={'0':('069f9dd5e75d8419476b18f1551e59ce','4d3d5662b2a85ab7f0dc33b234eeaa12','9dbc8f4af61329c68cb53f62570ab213',
+ 'c375d0349a6b097ac2708cc12736651a','15997d39cbdceecf8fc3050f86811ded',),
+ '1':('ddc9036ea2ae1bdfbb15616a0e3a0d90',),
+ '2':('27638b6ebedf6a23149990495ef4c1c5','f602ee70136eb2331275a8ac8cd636de','ff1f44c05a5eab4569bef7bde84e5b66',
+ '3b0795e3fc0af85c4838279847cb87f7','ac27be9df781cc8756f999d61f7a46f3',),
+ '3':('45101362592c07bc94f8449a1f4479f1','b24990f89de3454038b7c7940bc1053f','bf0d6bd4f13ea9b57a72f76c6dad0b41',
+ '743762655b13c97908b17ce7b36a1f5a','53f9b643c228e99723c384fe12390a0e','f206adf0be6f3c6613452c19a7b0babe',
+ 'f206adf0be6f3c6613452c19a7b0babe',),
+ '4':('9d5d871b405465218cc892dc015ea6d8','fed023bedd046b9f4d169c6ab12f6d4c','5069a391893fb107fbc39923a9d108ef',),
+ '5':('5ef102b78f5dc642ee98e9bdcf42a02e','496418730424d7f40d2b137d56bcbfe8','139186da206acf5344362ed86da42a7f',
+ 'e080cd4fbda1493034f1444eae484887',),
+ '6':('ab1fff097099fb395fe73470f7afcae0','70c64ac427435d40d2713128e8735b4d','bbb6fc3d0f23fa5104e2ea602ecc2d18',
+ '5d8c50960dd1f50457697fd8a8d5622e',),
+ '7':('30f11190e1772dd0f93740190aaa7608','ef5477640cf97de373e49e13caef8f5c','05dfb006e2668d7dfd210b2fdf74fef8',
+ '8efb472abef04ac9bcd1ba02b49ed6a5',),
+ '8':('8a8258f63f816888b550d704f4c6a068','69cade726c4d6c8e6a72e96df059c8c7','675972437c7733747146a0851bbb5727',
+ '01ce8b70eb0761b7f4047c365faa9cf5','08e6113ad8ac2f74d104c156047819a8','c2278d5c10b9903aff14f1a6516a583b',
+ 'bbba856f115bf8a45ef944a5e41ee436','f365fa7628ef15f172d40f07e54c327a',),
+ '9':('bf8e09357cd69275cbc6fdef42610ea0','212af59d8bc81dff176e02c0f001aa81','a3bc28250187c34a46757f2ab01e436b',
+ '9c0ab75a491e6a64dca57543efe5012c','62bedc16830a5602e26d9a050b13d2df','2b79fff64f55c027d23895baa5d2c66b',
+ '9d5d871b405465218cc892dc015ea6d8',),
+ }
+
+ color=(0,0,0)
+
+ def __init__(self, page):
+ img = page.doc.find("//img[@usemap='#cv']")
+ res = page.browser.open(img.attrib['src'])
+ MappedVirtKeyboard.__init__(self, StringIO(res.content), page.doc, img, self.color, 'href', convert='RGB')
+
+ self.check_symbols(self.symbols, page.browser.responses_dirname)
+
+ def check_color(self, pixel):
+ for p in pixel:
+ if p >= 200:
+ return False
+
+ return True
+
+ def get_symbol_code(self, md5sum_list):
+ for md5sum in md5sum_list:
+ try:
+ code = MappedVirtKeyboard.get_symbol_code(self,md5sum)
+ except VirtKeyboardError:
+ continue
+ else:
+ return ''.join(re.findall(r"'(\d+)'", code)[-2:])
+ raise VirtKeyboardError('Symbol not found')
+
+ def get_string_code(self, string):
+ code = ''
+ for c in string:
+ code += self.get_symbol_code(self.symbols[c])
+ return code
+
+class LoginPage(HTMLPage):
+ is_here ="//form[@id='formulaire-login']"
+
+ def login(self, login, password):
+ vk = VirtKeyboard(self)
+
+ form = self.get_form('//form[@id="formulaire-login"]')
+ code = vk.get_string_code(password)
+ assert len(code)==10, ParseError("Wrong number of character.")
+ form['identifiant'] = login
+ form['codePinpad'] = code
+ form['task'] = 'Login'
+ form['process'] = 'Login'
+ form['eventid'] = 'suivant'
+ form['modeCodeSecret'] = 'pinpad'
+ form['personneIdentifiee'] = 'N'
+ form.submit()
+
+class IndexPage(LoggedPage, HTMLPage):
+ is_here = "//div[@id='situation']"
+
+ def get_balance(self):
+ return -CleanDecimal('.')(self.doc.xpath('//div[@id = "total-sommes-dues"]/p[contains(text(), "sommes dues")]/span[@class = "montant"]')[0])
+
+class OperationsPage(LoggedPage, HTMLPage):
+ is_here = "//div[@id='releve-reserve-credit']"
+
+ @pagination
+ @method
+ class iter_transactions(ListElement):
+ item_xpath = '//div[@id = "releve-reserve-credit"]//table/tbody/tr'
+
+ class credit(ItemElement):
+ klass = Transaction
+ obj_type = Transaction.TYPE_CARD
+ obj_date = Transaction.Date('./td[1]')
+ obj_raw = Transaction.Raw('./td[2]')
+ obj_amount = Env('amount')
+
+ def condition(self):
+ self.env['amount'] = Transaction.Amount('./td[3]')(self.el)
+ return self.env['amount'] > 0
+
+
+ class debit(ItemElement):
+ klass = Transaction
+ obj_type = Transaction.TYPE_CARD
+ obj_date = Transaction.Date('./td[1]')
+ obj_raw = Transaction.Raw('./td[2]')
+ obj_amount = Env('amount')
+
+ def condition(self):
+ self.env['amount'] = Transaction.Amount('', './td[4]')(self.el)
+ return self.env['amount'] < 0
+
+ def next_page(self):
+ options = self.page.doc.xpath('//select[@id="periode"]//option[@selected="selected"]/preceding-sibling::option[1]')
+ if options:
+ data = {'numReleve':options[0].values(),'task':'Releve','process':'Releve','eventid':'select','taskid':'','hrefid':'','hrefext':''}
+ return requests.Request("POST", self.page.url, data=data)
+
+