download status, exception handling

This commit is contained in:
valerio 2017-11-06 00:50:44 +01:00
parent b94920e6e5
commit da975628eb
2 changed files with 173 additions and 136 deletions

View File

@ -1,70 +1,38 @@
#!/usr/bin/env python #!/usr/bin/env python
# TestCalculatorFunctions.py # TestScraper.py
import unittest import unittest
from scraper import Scraper from scraper import Scraper
from pprint import pprint
class KnownValues(unittest.TestCase): class KnownValues(unittest.TestCase):
scraper = Scraper(); scraper = Scraper()
def test_instance(self): def test_instance(self):
self.assertIsInstance(self.scraper, Scraper) self.assertIsInstance(self.scraper, Scraper)
def test_get_stations(self): def test_bytes_to_multiples(self):
self.assertEqual(Scraper.bytes_to_multiples(8302.2049371), '8.3kb')
self.assertEqual(Scraper.bytes_to_multiples(8302.2049371), '8.3kb')
self.assertEqual(Scraper.bytes_to_multiples(10), '10b')
self.assertEqual(Scraper.bytes_to_multiples(10000), '10kb')
self.assertEqual(Scraper.bytes_to_multiples(20000), '20kb')
self.assertEqual(Scraper.bytes_to_multiples(10000000), '10mb')
self.assertEqual(Scraper.bytes_to_multiples(1000000000), '1gb')
self.assertEqual(Scraper.bytes_to_multiples(1250000000), '1.25gb')
self.assertEqual(Scraper.bytes_to_multiples(160000000), '160mb')
fname = 'rickandmorty.conf' def test_get_speed_string(self):
with open(fname) as fopen: self.assertEqual(Scraper.get_speed_string(768000), '768kb/s')
self.assertTrue(True)
def test_parse_stations(self): def test_get_completion(self):
source = '''BARI CENTRALE|S11119 self.assertEqual(Scraper.get_completion(10, 100), '10% - 10b/100b')
BARI TORRE QUETTA|S11004 self.assertEqual(Scraper.get_completion(3, 10), '30% - 3b/10b')
BOLOGNA C.LE|S05043''' self.assertEqual(Scraper.get_completion(24, 289), '8.3% - 24b/289b')
stations = self.scraper.parse_stations(source)
self.assertListEqual(stations, [
{'name': 'BARI CENTRALE', 'code': 'S11119'},
{'name': 'BARI TORRE QUETTA', 'code': 'S11004'},
{'name': 'BOLOGNA C.LE', 'code': 'S05043'},
])
for station in stations:
self.assertTrue('name' in station)
self.assertTrue('code' in station)
def test_parse_station(self): def test_calc_res(self):
station = 'SAN LEONARDO DI CUTRO|S11827' self.assertEqual(Scraper.calc_res('800x600'), 800 * 600)
expected = {'name': 'SAN LEONARDO DI CUTRO', 'code': 'S11827'} self.assertEqual(Scraper.calc_res('1920x1024'), 1920 * 1024)
self.assertDictEqual(self.scraper.parse_station(station), expected)
# def test_can_connect(self):
# scraper = Scraper()
# self.assertEqual(scraper.touch('http://ddg.gg'), 200)
#
# def test_get_page(self):
# scraper = Scraper()
# self.assertEqual(scraper.get_page().status_code, 200)
#
# def test_format_hackerspace(self):
# scraper = Scraper()
# hackerspace = {'name':'pippo'}
# formatted = scraper.format_hackerspace(hackerspace)
# self.assertTrue('url' in formatted)
#
#
# def test_get_hackerspaces(self):
# scraper = Scraper()
# hackerspaces = scraper.get_hackerspaces()
# self.assertGreater(len(hackerspaces), 0)
#
# for hackerspace in hackerspaces:
# self.assertTrue('url' in hackerspace)
#
# def test_convert_text_field_to_hs_url(self):
# scraper = Scraper()
# textfield = '<b><a href="/Freaknet" title="Freaknet">Freaknet</a></b>'
# self.assertEqual(scraper.convert_text_field_to_hs_url(textfield), 'https://wiki.hackerspaces.org/Freaknet')
if __name__ == '__main__': if __name__ == '__main__':

235
scraper.py Normal file → Executable file
View File

@ -1,108 +1,177 @@
import requests, json, ConfigParser, os #!/usr/bin/env python
import requests
import ConfigParser
import time
from bs4 import BeautifulSoup from bs4 import BeautifulSoup
from pprint import pprint
class Scraper(): class Scraper:
# url = 'http://www.viaggiatreno.it/viaggiatrenonew/resteasy/viaggiatreno/partenze/S01480/Tue%20Oct%2011%202017%2008:30:00%20GMT+0200%20(CEST)' name = 'Rick and Morty py'
author = 'NotIsSet'
license = 'GPLv2'
year = '2017'
version = '0.0.1'
def __init__(self): out_buffer = ''
config = ConfigParser.RawConfigParser() out_buffer_time = 0
config.read('rickandmorty.conf') out_buffer_refresh_threshold = 0
# getfloat() raises an exception if the value is not a float @staticmethod
# getint() and getboolean() also do this for their respective types def bytes_to_multiples(b):
self.url = config.get('Main', 'url') divisor = 1
unit = 'b'
pprint(self.url) if b >= 1000000000:
unit = 'gb'
divisor = 1000000000
elif b >= 1000000:
unit = 'mb'
divisor = 1000000
elif b >= 1000:
unit = 'kb'
divisor = 1000
# config.read(['site.cfg', os.path.expanduser('~/.rickandmorty.conf')]) _ret = round((float(b) / divisor), 2)
self.load_page() if _ret - int(_ret) == 0.0:
pass _ret = int(_ret)
_ret = str(_ret) + unit
def calc_res(self, resolution):
if resolution is None:
return 0
vals = resolution.split('x')
if (len(vals) < 2):
return 0
pprint(vals)
_ret = int(vals[0]) * int(vals[1])
return _ret return _ret
def load_page(self): @staticmethod
url = self.url % (1, 1) def get_speed_string(bps):
r = requests.get(url) return Scraper.bytes_to_multiples(bps) + '/s'
# pprint(content)
pprint(url) @staticmethod
def get_completion(written, total):
out_percentage = round(((float(written * 100)) / total), 2)
if out_percentage - int(out_percentage) == 0.0:
out_percentage = int(out_percentage)
return '%s%% - %s/%s' % (
out_percentage,
Scraper.bytes_to_multiples(written),
Scraper.bytes_to_multiples(total)
)
@staticmethod
def calc_res(resolution):
if resolution is None:
return 0
factors = resolution.split('x')
if len(factors) < 2:
return 0
_ret = int(factors[0]) * int(factors[1])
return _ret
@staticmethod
def get_top_resolution(resolutions):
top_res = 0
_ret = resolutions[0][0]
for video in resolutions:
if Scraper.calc_res(video[1]) > top_res:
top_res = Scraper.calc_res(video[1])
_ret = video[0]
return _ret
def buffer_out(self, string, concatenate=True):
current_time = int(round(time.time() * 1000))
out = self.out_buffer + '\n' + string
if concatenate:
self.out_buffer = out
if current_time - self.out_buffer_time > self.out_buffer_refresh_threshold:
self.out_buffer_time = current_time
print '\x1b[2J' + out
def __init__(self):
self.info_header = '\n'.join([
'name: ' + self.name,
'author: ' + self.author,
'license: ' + self.license,
'year: ' + self.year,
'version: ' + self.version,
])
config = ConfigParser.RawConfigParser()
config.read('rickandmorty.conf')
self.url = config.get('Main', 'url')
self.headers = requests.utils.default_headers()
self.headers.update({"User-Agent": "Mozilla/5.0"})
def get_episode_url(self, season, episode):
url = self.url % (season, episode)
r = requests.get(url, headers=self.headers)
# self.buffer_out(url)
soup = BeautifulSoup(r.text, 'html.parser') soup = BeautifulSoup(r.text, 'html.parser')
player = soup.find(id="player") player = soup.find(id="player")
frameUrl = player.get('data-src').strip() if player is None:
return None
r = requests.get(frameUrl) frame_url = player.get('data-src')
# pprint(content) if player is None:
pprint(url) return None
frame_url = frame_url.strip()
# self.buffer_out(frame_url)
r = requests.get(frame_url, headers=self.headers)
soup = BeautifulSoup(r.text, 'html.parser') soup = BeautifulSoup(r.text, 'html.parser')
videoResolutions = [x.get('res') for x in soup.find_all(name="source")]
videoSources = [x.get('src') for x in soup.find_all(name="source")]
pprint(videoSources)
pprint(videoResolutions)
videoUrls = zip(videoSources, videoResolutions) video_resolutions = [x.get('res') for x in soup.find_all(name="source")]
topRes = 0 video_sources = [x.get('src') for x in soup.find_all(name="source")]
curTop = videoUrls[0][0]
for video in videoUrls:
if (self.calc_res(video[1]) > topRes):
topRes = self.calc_res(video[1])
curTop = video[0]
url = curTop
# frameUrl = player.get('data-src').strip() video_urls = zip(video_sources, video_resolutions)
return Scraper.get_top_resolution(video_urls)
def download_file(self, url, destination):
r = requests.get(url, stream=True, headers=self.headers)
written_chunks = 0
last_time = time.time()
with open(destination, 'wb') as f:
total_length = int(r.headers['Content-Length'])
speed_buffer = []
speed_buffer_size = 1000
print "downloading with requests"
local_filename = "s01e01.mp4"
# NOTE the stream=True parameter
r = requests.get(url, stream=True)
with open(local_filename, 'wb') as f:
for chunk in r.iter_content(chunk_size=1024): for chunk in r.iter_content(chunk_size=1024):
if chunk: # filter out keep-alive new chunks if chunk: # filter out keep-alive new chunks
print('writing chunk...\n') time_diff = time.time() - last_time
last_time = time.time()
speed_buffer.append(float(1024 / time_diff / 10))
if len(speed_buffer) > speed_buffer_size:
speed_buffer.pop(0)
f.write(chunk) f.write(chunk)
# f.flush() commented by recommendation from J.F.Sebastian written_chunks += 1
# r = requests.get(url)
# with open("s01e01.mp4", "wb") as code:
# code.write(r.content)
# pprint(url)
# pprint(frameUrl)
# iframe#player
completion = Scraper.get_completion(written_chunks * 1000, total_length / 1024 * 1000)
speed = Scraper.get_speed_string(float(sum(speed_buffer)) / speed_buffer_size)
def parse_stations(self, stations): if written_chunks % 20 == 0:
_ret = [] self.buffer_out(
for station in stations.split('\n'): 'Download in progress...\n%s - %s\n' % (
if len(station) > 0: completion, speed),
_ret.append(self.parse_station(station)) False)
return _ret self.buffer_out('Download complete')
def run(self, download=False):
def parse_station(self, station): self.buffer_out(self.info_header)
stat = station.split('|') self.buffer_out(self.url)
return { for season in range(1, 4):
'name': stat[0].strip(), for episode in range(1, 12):
'code': stat[1].strip() ep_url = self.get_episode_url(season, episode)
} if ep_url is None:
self.buffer_out('season %s episode %s - WARN could not retrieve url' % (season, episode))
continue
def find_stations(self, station_name, stations): self.buffer_out('season %s episode %s - url: %s' % (season, episode, ep_url))
_ret = [] if download:
for station in stations: destination = "s%se%s.mp4" % (season, episode)
if station_name.lower() in station['name'].lower(): self.download_file(ep_url, destination)
_ret.append(station) pass
return _ret
if __name__ == '__main__': if __name__ == '__main__':
scraper = Scraper() scraper = Scraper()
scraper.run()
stations = scraper.get_stations('elenco_stazioni.txt')
pprint(stations)