First implementation of freemobile website

This commit is contained in:
Florent 2012-02-24 13:11:54 +01:00 committed by Romain Bignon
commit 1484251c65
7 changed files with 439 additions and 0 deletions

View 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']

View 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

View 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

View 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']

View 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

View 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

View 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)