First implementation of freemobile website
This commit is contained in:
parent
72893d88a0
commit
1484251c65
7 changed files with 439 additions and 0 deletions
23
modules/freemobile/__init__.py
Normal file
23
modules/freemobile/__init__.py
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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 .backend import FreeMobileBackend
|
||||||
|
|
||||||
|
__all__ = ['FreeMobileBackend']
|
||||||
73
modules/freemobile/backend.py
Normal file
73
modules/freemobile/backend.py
Normal file
|
|
@ -0,0 +1,73 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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.bill import ICapBill, SubscriptionNotFound
|
||||||
|
from weboob.tools.backend import BaseBackend, BackendConfig
|
||||||
|
from weboob.tools.value import ValueBackendPassword
|
||||||
|
|
||||||
|
from .browser import Freemobile
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = ['FreeMobileBackend']
|
||||||
|
|
||||||
|
|
||||||
|
class FreeMobileBackend(BaseBackend, ICapBill):
|
||||||
|
NAME = 'freemobile'
|
||||||
|
MAINTAINER = 'Florent Fourcot'
|
||||||
|
EMAIL = 'weboob@flo.fourcot.fr'
|
||||||
|
VERSION = '0.b'
|
||||||
|
LICENSE = 'AGPLv3+'
|
||||||
|
DESCRIPTION = 'Free Mobile website'
|
||||||
|
CONFIG = BackendConfig(ValueBackendPassword('login', label='Account ID', masked=False, regexp='^(\d{8}|)$'),
|
||||||
|
ValueBackendPassword('password', label='Password')
|
||||||
|
)
|
||||||
|
BROWSER = Freemobile
|
||||||
|
|
||||||
|
def create_default_browser(self):
|
||||||
|
return self.create_browser(self.config['login'].get(),
|
||||||
|
self.config['password'].get())
|
||||||
|
|
||||||
|
def iter_subscription(self):
|
||||||
|
for subscription in self.browser.get_subscription_list():
|
||||||
|
yield subscription
|
||||||
|
|
||||||
|
def get_subscription(self, _id):
|
||||||
|
if not _id.isdigit():
|
||||||
|
raise SubscriptionNotFound()
|
||||||
|
with self.browser:
|
||||||
|
subscription = self.browser.get_subscription(_id)
|
||||||
|
if subscription:
|
||||||
|
return subscription
|
||||||
|
else:
|
||||||
|
raise SubscriptionNotFound()
|
||||||
|
|
||||||
|
|
||||||
|
def iter_history(self, subscription):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def get_pdf(self, account):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
# The subscription is actually useless, but maybe for the futur...
|
||||||
|
def get_details(self, subscription):
|
||||||
|
with self.browser:
|
||||||
|
for detail in self.browser.get_details():
|
||||||
|
yield detail
|
||||||
|
|
||||||
87
modules/freemobile/browser.py
Normal file
87
modules/freemobile/browser.py
Normal file
|
|
@ -0,0 +1,87 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Romain Bignon
|
||||||
|
#
|
||||||
|
# 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.browser import BaseBrowser, BrowserIncorrectPassword
|
||||||
|
from .pages import HomePage, LoginPage, HistoryPage
|
||||||
|
|
||||||
|
__all__ = ['Freemobile']
|
||||||
|
|
||||||
|
|
||||||
|
class Freemobile(BaseBrowser):
|
||||||
|
DOMAIN = 'mobile.free.fr'
|
||||||
|
PROTOCOL = 'https'
|
||||||
|
ENCODING = None # refer to the HTML encoding
|
||||||
|
PAGES = {'.*moncompte/index.php': LoginPage,
|
||||||
|
'.*page=home': HomePage,
|
||||||
|
'.*page=suiviconso': HistoryPage
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
BaseBrowser.__init__(self, *args, **kwargs)
|
||||||
|
|
||||||
|
def home(self):
|
||||||
|
self.location('https://mobile.free.fr/moncompte/index.php')
|
||||||
|
|
||||||
|
def is_logged(self):
|
||||||
|
return not self.is_on_page(LoginPage)
|
||||||
|
|
||||||
|
def login(self):
|
||||||
|
assert isinstance(self.username, basestring)
|
||||||
|
assert isinstance(self.password, basestring)
|
||||||
|
assert self.username.isdigit()
|
||||||
|
|
||||||
|
if not self.is_on_page(LoginPage):
|
||||||
|
self.location('https://mobile.free.fr/moncompte/index.php')
|
||||||
|
|
||||||
|
self.page.login(self.username, self.password)
|
||||||
|
|
||||||
|
if self.is_on_page(LoginPage):
|
||||||
|
raise BrowserIncorrectPassword()
|
||||||
|
|
||||||
|
def get_subscription_list(self):
|
||||||
|
if not self.is_on_page(HomePage):
|
||||||
|
self.location('/moncompte/index.php?page=home')
|
||||||
|
|
||||||
|
return self.page.get_list()
|
||||||
|
|
||||||
|
def get_subscription(self, id):
|
||||||
|
assert isinstance(id, basestring)
|
||||||
|
|
||||||
|
if not self.is_on_page(HomePage):
|
||||||
|
self.location('/moncompte/index.php?page=home')
|
||||||
|
|
||||||
|
l = self.page.get_list()
|
||||||
|
for a in l:
|
||||||
|
if a.id == id:
|
||||||
|
return a
|
||||||
|
|
||||||
|
return None
|
||||||
|
|
||||||
|
# XXX : not implemented
|
||||||
|
def get_history(self):
|
||||||
|
if not self.is_on_page(HistoryPage):
|
||||||
|
self.location('/moncompte/index.php?page=suiviconso')
|
||||||
|
return self.page.get_calls()
|
||||||
|
|
||||||
|
def get_details(self):
|
||||||
|
if not self.is_on_page(HistoryPage):
|
||||||
|
self.location('/moncompte/index.php?page=suiviconso')
|
||||||
|
test = self.page.get_details()
|
||||||
|
return test
|
||||||
25
modules/freemobile/pages/__init__.py
Normal file
25
modules/freemobile/pages/__init__.py
Normal file
|
|
@ -0,0 +1,25 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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 .homepage import HomePage
|
||||||
|
from .history import HistoryPage
|
||||||
|
from .login import LoginPage
|
||||||
|
|
||||||
|
__all__ = ['LoginPage', 'HomePage', 'HistoryPage']
|
||||||
77
modules/freemobile/pages/history.py
Normal file
77
modules/freemobile/pages/history.py
Normal file
|
|
@ -0,0 +1,77 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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.browser import BasePage
|
||||||
|
from weboob.capabilities.bill import Detail
|
||||||
|
|
||||||
|
__all__ = ['HistoryPage']
|
||||||
|
|
||||||
|
def convert_price(div):
|
||||||
|
try:
|
||||||
|
price = div.find('div[@class="horsForfait"]/p/span').text
|
||||||
|
price = price.encode('utf-8', 'replace').replace('€', '').replace(',', '.')
|
||||||
|
return float(price)
|
||||||
|
except:
|
||||||
|
return 0.
|
||||||
|
|
||||||
|
|
||||||
|
class HistoryPage(BasePage):
|
||||||
|
calls = []
|
||||||
|
details = []
|
||||||
|
|
||||||
|
def on_loaded(self):
|
||||||
|
|
||||||
|
divnat = self.document.xpath('//div[@class="national"]')[0]
|
||||||
|
divs = divnat.xpath('div[@class="detail"]')
|
||||||
|
divvoice = divs.pop(0)
|
||||||
|
|
||||||
|
# Two informations in one div...
|
||||||
|
voice = Detail()
|
||||||
|
voice.label = divvoice.find('div[@class="titreDetail"]/p').text_content()
|
||||||
|
voice.price = convert_price(divvoice)
|
||||||
|
voicenat = divvoice.xpath('div[@class="consoDetail"]/p/span')[0].text
|
||||||
|
voiceint = divvoice.xpath('div[@class="consoDetail"]/p/span')[1].text
|
||||||
|
voice.infos = "Consommation : " + voicenat + " International : " + voiceint
|
||||||
|
self.details.append(voice)
|
||||||
|
|
||||||
|
self.iter_divs(divs)
|
||||||
|
divint = self.document.xpath('//div[@class="international hide"]')[0]
|
||||||
|
self.iter_divs(divint.xpath('div[@class="detail"]'), True)
|
||||||
|
|
||||||
|
|
||||||
|
def iter_divs(self, divs, inter=False):
|
||||||
|
for div in divs:
|
||||||
|
detail = Detail()
|
||||||
|
|
||||||
|
detail.label = div.find('div[@class="titreDetail"]/p').text_content()
|
||||||
|
if inter:
|
||||||
|
detail.label = detail.label + " (international)"
|
||||||
|
detail.infos = div.find('div[@class="consoDetail"]/p').text_content().lstrip()
|
||||||
|
detail.price = convert_price(div)
|
||||||
|
|
||||||
|
self.details.append(detail)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
def get_calls(self):
|
||||||
|
return self.calls
|
||||||
|
|
||||||
|
def get_details(self):
|
||||||
|
return self.details
|
||||||
47
modules/freemobile/pages/homepage.py
Normal file
47
modules/freemobile/pages/homepage.py
Normal file
|
|
@ -0,0 +1,47 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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.bill import Subscription
|
||||||
|
from weboob.tools.browser import BasePage
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = ['HomePage']
|
||||||
|
|
||||||
|
|
||||||
|
class HomePage(BasePage):
|
||||||
|
def on_loaded(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def get_list(self):
|
||||||
|
l = []
|
||||||
|
divabo = self.document.xpath('//div[@class="idAbonne"]')[0]
|
||||||
|
owner = divabo.xpath('p')[0].text.replace(' - ', '')
|
||||||
|
phone = divabo.xpath('p/span')[0].text
|
||||||
|
self.browser.logger.debug('Found ' + owner + ' has subscriber')
|
||||||
|
self.browser.logger.debug('Found ' + phone + ' has phone number')
|
||||||
|
phoneplan = self.document.xpath('//div[@class="forfaitChoisi"]')[0].text
|
||||||
|
self.browser.logger.debug('Found ' + phoneplan + ' has subscription type')
|
||||||
|
|
||||||
|
subscription = Subscription(phone)
|
||||||
|
subscription.label = phone + ' - ' + phoneplan
|
||||||
|
subscription.owner = owner
|
||||||
|
|
||||||
|
l.append(subscription)
|
||||||
|
|
||||||
|
return l
|
||||||
107
modules/freemobile/pages/login.py
Normal file
107
modules/freemobile/pages/login.py
Normal file
|
|
@ -0,0 +1,107 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright(C) 2012 Florent Fourcot
|
||||||
|
#
|
||||||
|
# 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/>.
|
||||||
|
|
||||||
|
|
||||||
|
import Image
|
||||||
|
|
||||||
|
from weboob.tools.browser import BasePage
|
||||||
|
|
||||||
|
__all__ = ['LoginPage']
|
||||||
|
|
||||||
|
class FreeKeyboard(object):
|
||||||
|
symbols={'0':'001111111111110011111111111111111111111111111110000000000011110000000000011111111111111111011111111111111001111111111110',
|
||||||
|
'1':'001110000000000001110000000000001110000000000011111111111111111111111111111111111111111111000000000000000000000000000000',
|
||||||
|
'2':'011110000001111011110000111111111000001111111110000011110011110000111100011111111111000011011111110000011001111000000011',
|
||||||
|
'3':'011100000011110111100000011111111000110000111110000110000011110001110000011111111111111111011111111111110001110001111100',
|
||||||
|
'4':'000000011111000000001111111000000111110011000011110000011000111111111111111111111111111111111111111111111000000000011000',
|
||||||
|
'5':'111111110011110111111110011111111001110000111111001100000011111001100000011111001111111111111001111111111010000111111110',
|
||||||
|
'6':'001111111111110011111111111111111111111111111110001100000011110001100000011111001111111111111101111111111011100111111110',
|
||||||
|
'7':'111000000000000111000000000000111000000011111111000011111111111011111111111111111111000000111111000000000111100000000000',
|
||||||
|
'8':'001110001111110011111111111111111111111111111110000110000011110000110000011111111111111111011111111111111001111001111110',
|
||||||
|
'9':'001111111000110011111111100111111111111100111110000001100011110000001100011111111111111111011111111111111001111111111110'
|
||||||
|
}
|
||||||
|
fingerprints = []
|
||||||
|
|
||||||
|
def __init__(self,basepage):
|
||||||
|
for htmlimg in basepage.document.xpath('//img[@class="ident_chiffre_img pointer"]'):
|
||||||
|
url = htmlimg.attrib.get("src")
|
||||||
|
fichier = basepage.browser.openurl(url)
|
||||||
|
image = Image.open(fichier)
|
||||||
|
matrix = image.load()
|
||||||
|
s = ""
|
||||||
|
# The digit is only displayed in the center of image
|
||||||
|
for x in range(15, 23):
|
||||||
|
for y in range(12, 27):
|
||||||
|
(r, g, b) = matrix[x,y]
|
||||||
|
# If the pixel is "red" enough
|
||||||
|
if (g*g + b*b) < r*r:
|
||||||
|
s += "1"
|
||||||
|
else:
|
||||||
|
s += "0"
|
||||||
|
|
||||||
|
self.fingerprints.append(s)
|
||||||
|
|
||||||
|
def get_symbol_code(self,digit):
|
||||||
|
fingerprint = self.symbols[digit]
|
||||||
|
i = 0
|
||||||
|
for string in self.fingerprints:
|
||||||
|
if string.__eq__(fingerprint):
|
||||||
|
return i
|
||||||
|
i += 1
|
||||||
|
# Image contains some noise, and the match is not alaways perfect
|
||||||
|
# (this is why we can't use md5 hashs)
|
||||||
|
# But if we can't find the perfect one, we can take one with smalls errors
|
||||||
|
i = 0
|
||||||
|
for string in self.fingerprints:
|
||||||
|
j = 0
|
||||||
|
match = 0
|
||||||
|
for bit in string:
|
||||||
|
if bit == fingerprint[j]:
|
||||||
|
match += 1
|
||||||
|
j += 1
|
||||||
|
if match > 115:
|
||||||
|
return i
|
||||||
|
i += 1
|
||||||
|
|
||||||
|
# TODO : exception
|
||||||
|
|
||||||
|
def get_string_code(self,string):
|
||||||
|
code=''
|
||||||
|
for c in string:
|
||||||
|
codesymbol = self.get_symbol_code(c)
|
||||||
|
code+=str(codesymbol)
|
||||||
|
return code
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class LoginPage(BasePage):
|
||||||
|
def on_loaded(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def login(self, login, password):
|
||||||
|
vk = FreeKeyboard(self)
|
||||||
|
|
||||||
|
# Fucking form without name...
|
||||||
|
self.browser.select_form(nr=0)
|
||||||
|
self.browser.set_all_readonly(False)
|
||||||
|
self.browser['login_abo'] = vk.get_string_code(login)
|
||||||
|
self.browser['pwd_abo'] = password
|
||||||
|
self.browser.submit(nologin=True)
|
||||||
|
|
||||||
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue