Amazon Store Card banking module. Closes #1698
This commit is contained in:
parent
0f79decdba
commit
2fc837a42f
6 changed files with 398 additions and 0 deletions
23
modules/amazonstorecard/__init__.py
Normal file
23
modules/amazonstorecard/__init__.py
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright(C) 2014-2015 Oleg Plakhotniuk
|
||||
#
|
||||
# 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 .module import AmazonStoreCardModule
|
||||
|
||||
__all__ = ['AmazonStoreCardModule']
|
||||
74
modules/amazonstorecard/browser.py
Normal file
74
modules/amazonstorecard/browser.py
Normal file
|
|
@ -0,0 +1,74 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright(C) 2014-2015 Oleg Plakhotniuk
|
||||
#
|
||||
# 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 weboob.capabilities.bank import AccountNotFound
|
||||
from weboob.browser import LoginBrowser, URL, need_login
|
||||
from weboob.exceptions import BrowserIncorrectPassword
|
||||
|
||||
from .pages import SomePage, LoginPage, RecentPage, StatementsPage, \
|
||||
StatementPage, SummaryPage
|
||||
|
||||
|
||||
__all__ = ['AmazonStoreCard']
|
||||
|
||||
|
||||
class AmazonStoreCard(LoginBrowser):
|
||||
BASEURL = 'https://www.onlinecreditcenter6.com'
|
||||
login = URL('/consumergen2/login.do\?accountType=plcc&clientId=amazon'
|
||||
'&langId=en&subActionId=1000$',
|
||||
'/consumergen2/consumerlogin.do.*$',
|
||||
LoginPage)
|
||||
stmts = URL('/consumergen2/ebill.do$', StatementsPage)
|
||||
recent = URL('/consumergen2/recentActivity.do$', RecentPage)
|
||||
statement = URL('/consumergen2/ebillViewPDF.do.*$', StatementPage)
|
||||
summary = URL('/consumergen2/accountSummary.do$', SummaryPage)
|
||||
unknown = URL('.*', SomePage)
|
||||
|
||||
def __init__(self, config, *args, **kwargs):
|
||||
super(AmazonStoreCard, self).__init__(config['userid'].get(),
|
||||
config['password'].get(), *args, **kwargs)
|
||||
self.config = config
|
||||
|
||||
def do_login(self):
|
||||
self.session.cookies.clear()
|
||||
self.login.go()
|
||||
while self.login.is_here():
|
||||
self.page.proceed(self.config)
|
||||
if not self.page.logged:
|
||||
raise BrowserIncorrectPassword()
|
||||
|
||||
@need_login
|
||||
def get_account(self, id_):
|
||||
a = next(self.iter_accounts())
|
||||
if (a.id != id_):
|
||||
raise AccountNotFound()
|
||||
return a
|
||||
|
||||
@need_login
|
||||
def iter_accounts(self):
|
||||
yield self.summary.go(data=SummaryPage.DATA).account()
|
||||
|
||||
@need_login
|
||||
def iter_history(self, account):
|
||||
for t in self.recent.go(data=RecentPage.DATA).iter_transactions():
|
||||
yield t
|
||||
for s in self.stmts.go(data=StatementsPage.DATA).iter_statements():
|
||||
for t in s.iter_transactions():
|
||||
yield t
|
||||
BIN
modules/amazonstorecard/favicon.png
Normal file
BIN
modules/amazonstorecard/favicon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.8 KiB |
55
modules/amazonstorecard/module.py
Normal file
55
modules/amazonstorecard/module.py
Normal file
|
|
@ -0,0 +1,55 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright(C) 2014-2015 Oleg Plakhotniuk
|
||||
#
|
||||
# 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 weboob.capabilities.bank import CapBank
|
||||
from weboob.tools.backend import Module, BackendConfig
|
||||
from weboob.tools.value import ValueBackendPassword
|
||||
|
||||
from .browser import AmazonStoreCard
|
||||
|
||||
|
||||
__all__ = ['AmazonStoreCardModule']
|
||||
|
||||
|
||||
class AmazonStoreCardModule(Module, CapBank):
|
||||
NAME = 'amazonstorecard'
|
||||
MAINTAINER = u'Oleg Plakhotniuk'
|
||||
EMAIL = 'olegus8@gmail.com'
|
||||
VERSION = '1.1'
|
||||
LICENSE = 'AGPLv3+'
|
||||
DESCRIPTION = u'Amazon Store Card'
|
||||
CONFIG = BackendConfig(
|
||||
ValueBackendPassword('userid', label='User ID', masked=False),
|
||||
ValueBackendPassword('password', label='Password'),
|
||||
ValueBackendPassword('challengeanswer1',
|
||||
label='Challenge answer 1', masked=False))
|
||||
BROWSER = AmazonStoreCard
|
||||
|
||||
def create_default_browser(self):
|
||||
return self.create_browser(config = self.config)
|
||||
|
||||
def iter_accounts(self):
|
||||
return self.browser.iter_accounts()
|
||||
|
||||
def get_account(self, id_):
|
||||
return self.browser.get_account(id_)
|
||||
|
||||
def iter_history(self, account):
|
||||
return self.browser.iter_history(account)
|
||||
212
modules/amazonstorecard/pages.py
Normal file
212
modules/amazonstorecard/pages.py
Normal file
|
|
@ -0,0 +1,212 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright(C) 2014-2015 Oleg Plakhotniuk
|
||||
#
|
||||
# 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 weboob.capabilities.bank import Account, Transaction
|
||||
from weboob.browser.pages import HTMLPage, RawPage, XMLPage
|
||||
from weboob.tools.capabilities.bank.transactions import \
|
||||
AmericanTransaction as AmTr
|
||||
from weboob.tools.date import closest_date
|
||||
from weboob.tools.pdf import decompress_pdf
|
||||
from weboob.tools.tokenizer import ReTokenizer
|
||||
from datetime import datetime, timedelta
|
||||
|
||||
|
||||
class SomePage(HTMLPage):
|
||||
@property
|
||||
def logged(self):
|
||||
return bool(self.doc.xpath('//span[@class="logoutBtn"]'))
|
||||
|
||||
|
||||
class LoginPage(SomePage):
|
||||
INPUTS = ['userId', 'password', 'challengeAnswer1']
|
||||
|
||||
is_here = '//form[@name="consumerLoginForm"]'
|
||||
|
||||
def proceed(self, config):
|
||||
form = self.get_form(name='consumerLoginForm')
|
||||
for inp in (i for i in self.INPUTS if i in form):
|
||||
form[inp] = config[inp.lower()].get()
|
||||
form.submit()
|
||||
return self.browser.page
|
||||
|
||||
|
||||
class SummaryPage(SomePage):
|
||||
DATA = {'subActionId': '1201', 'clientId': 'amazon',
|
||||
'accountType': 'plcc', 'langId': 'en'}
|
||||
|
||||
def account(self):
|
||||
label = u' '.join(u''.join(self.doc.xpath(
|
||||
u'//text()[contains(.,"Account ending in")]')).split())
|
||||
balance = self.doc.xpath(
|
||||
'//span[@id="currentBalance"]/..')[0].text_content()
|
||||
a = Account()
|
||||
a.id = label[-4:]
|
||||
a.label = label
|
||||
a.currency = Account.get_currency(balance)
|
||||
a.balance = -AmTr.decimal_amount(balance)
|
||||
a.type = Account.TYPE_CARD
|
||||
return a
|
||||
|
||||
|
||||
class RecentPage(XMLPage):
|
||||
DATA = {'subActionId': '1300', 'requestType': 'ajaxReq'}
|
||||
|
||||
def iter_transactions(self):
|
||||
for ntrans in reversed(self.doc.xpath('//TRANSACTION')):
|
||||
desc = u' '.join(ntrans.xpath(
|
||||
'TRANSDESCRIPTION/text()')[0].split())
|
||||
t = Transaction()
|
||||
t.date = datetime.strptime(ntrans.xpath(
|
||||
'TRANSACTIONDATE/text()')[0], '%m/%d/%Y')
|
||||
t.rdate = datetime.strptime(ntrans.xpath(
|
||||
'POSTDATE/text()')[0], '%m/%d/%Y')
|
||||
t.type = Transaction.TYPE_UNKNOWN
|
||||
t.raw = desc
|
||||
t.label = desc
|
||||
t.amount = -AmTr.decimal_amount(ntrans.xpath('AMOUNT/text()')[0])
|
||||
yield t
|
||||
|
||||
|
||||
class StatementsPage(SomePage):
|
||||
DATA = {'subActionId': '8168', 'clientId': 'amazon',
|
||||
'accountType': 'plcc', 'langId': 'en'}
|
||||
|
||||
def iter_statements(self):
|
||||
for url in self.doc.xpath('//a[contains(@href,"ebillViewPDF")]/@href'):
|
||||
if url.endswith('inline=false'):
|
||||
self.browser.location(url)
|
||||
yield self.browser.page
|
||||
|
||||
|
||||
class StatementPage(RawPage):
|
||||
LEX = [
|
||||
('charge_amount', r'^\(\$([0-9\.]+)\) Tj$'),
|
||||
('payment_amount', r'^\(\\\(\$([0-9\.]+)\\\)\) Tj$'),
|
||||
('date', r'^\((\d+/\d+)\) Tj$'),
|
||||
('full_date', r'^\((\d+/\d+/\d+)\) Tj$'),
|
||||
('layout_td', r'^([-0-9]+ [-0-9]+) Td$'),
|
||||
('ref', r'^\(([A-Z0-9]{17})\) Tj$'),
|
||||
('text', r'^\((.*)\) Tj$')
|
||||
]
|
||||
|
||||
def __init__(self, *args, **kwArgs):
|
||||
RawPage.__init__(self, *args, **kwArgs)
|
||||
assert self.doc[:4] == '%PDF'
|
||||
self._pdf = decompress_pdf(self.doc)
|
||||
self._tok = ReTokenizer(self._pdf, '\n', self.LEX)
|
||||
|
||||
def iter_transactions(self):
|
||||
return sorted(self.read_transactions(),
|
||||
cmp=lambda t1, t2: cmp(t2.date, t1.date) or
|
||||
cmp(t1.label, t2.label) or
|
||||
cmp(t1.amount, t2.amount))
|
||||
|
||||
def read_transactions(self):
|
||||
# Statement typically cover one month.
|
||||
# Do 60 days, just to be on a safe side.
|
||||
date_to = self.read_closing_date()
|
||||
date_from = date_to - timedelta(days=60)
|
||||
|
||||
pos = 0
|
||||
while not self._tok.tok(pos).is_eof():
|
||||
pos, trans = self.read_transaction(pos, date_from, date_to)
|
||||
if trans:
|
||||
yield trans
|
||||
else:
|
||||
pos += 1
|
||||
|
||||
def read_transaction(self, pos, date_from, date_to):
|
||||
startPos = pos
|
||||
pos, tdate = self.read_date(pos)
|
||||
pos, pdate_layout = self.read_layout_td(pos)
|
||||
pos, pdate = self.read_date(pos)
|
||||
pos, ref_layout = self.read_layout_td(pos)
|
||||
pos, ref = self.read_ref(pos)
|
||||
pos, desc_layout = self.read_layout_td(pos)
|
||||
pos, desc = self.read_text(pos)
|
||||
pos, amount_layout = self.read_layout_td(pos)
|
||||
pos, amount = self.read_amount(pos)
|
||||
if tdate is None or pdate is None \
|
||||
or desc is None or amount is None or amount == 0:
|
||||
return startPos, None
|
||||
else:
|
||||
tdate = closest_date(tdate, date_from, date_to)
|
||||
pdate = closest_date(pdate, date_from, date_to)
|
||||
desc = u' '.join(desc.split())
|
||||
|
||||
trans = Transaction(ref or u'')
|
||||
trans.date = tdate
|
||||
trans.rdate = pdate
|
||||
trans.type = Transaction.TYPE_UNKNOWN
|
||||
trans.raw = desc
|
||||
trans.label = desc
|
||||
trans.amount = amount
|
||||
return pos, trans
|
||||
|
||||
def read_amount(self, pos):
|
||||
pos, ampay = self.read_payment_amount(pos)
|
||||
if ampay is not None:
|
||||
return pos, ampay
|
||||
return self.read_charge_amount(pos)
|
||||
|
||||
def read_charge_amount(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, -AmTr.decimal_amount(t.value())) \
|
||||
if t.is_charge_amount() else (pos, None)
|
||||
|
||||
def read_payment_amount(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, AmTr.decimal_amount(t.value())) \
|
||||
if t.is_payment_amount() else (pos, None)
|
||||
|
||||
def read_closing_date(self):
|
||||
pos = 0
|
||||
while not self._tok.tok(pos).is_eof():
|
||||
pos, text = self.read_text(pos)
|
||||
if text == u'Statement Closing Date':
|
||||
break
|
||||
pos += 1
|
||||
while not self._tok.tok(pos).is_eof():
|
||||
pos, date = self.read_full_date(pos)
|
||||
if date is not None:
|
||||
return date
|
||||
pos += 1
|
||||
|
||||
def read_text(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, unicode(t.value())) \
|
||||
if t.is_text() else (pos, None)
|
||||
|
||||
def read_full_date(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, datetime.strptime(t.value(), '%m/%d/%Y')) \
|
||||
if t.is_full_date() else (pos, None)
|
||||
|
||||
def read_date(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, datetime.strptime(t.value(), '%m/%d')) \
|
||||
if t.is_date() else (pos, None)
|
||||
|
||||
def read_ref(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, t.value()) if t.is_ref() else (pos, None)
|
||||
|
||||
def read_layout_td(self, pos):
|
||||
t = self._tok.tok(pos)
|
||||
return (pos+1, t.value()) if t.is_layout_td() else (pos, None)
|
||||
34
modules/amazonstorecard/test.py
Normal file
34
modules/amazonstorecard/test.py
Normal file
|
|
@ -0,0 +1,34 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
# Copyright(C) 2014-2015 Oleg Plakhotniuk
|
||||
#
|
||||
# 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 weboob.tools.test import BackendTest
|
||||
from itertools import chain
|
||||
|
||||
|
||||
class AmazonStoreCardTest(BackendTest):
|
||||
MODULE = 'amazonstorecard'
|
||||
|
||||
def test_history(self):
|
||||
"""
|
||||
Test that there's at least one transaction in the whole history.
|
||||
"""
|
||||
b = self.backend
|
||||
ts = chain(*[b.iter_history(a) for a in b.iter_accounts()])
|
||||
t = next(ts, None)
|
||||
self.assertNotEqual(t, None)
|
||||
Loading…
Add table
Add a link
Reference in a new issue