2015-08-30 03:28:43 +02:00
|
|
|
import builtins
|
|
|
|
import re
|
|
|
|
import gzip
|
|
|
|
import urllib.request as request
|
2015-09-04 08:29:20 +02:00
|
|
|
import shutil
|
2020-05-21 01:57:46 +02:00
|
|
|
import json
|
2020-03-10 14:33:25 +01:00
|
|
|
|
|
|
|
import pirate.data
|
2020-05-21 01:57:46 +02:00
|
|
|
import pirate.torrent
|
2015-08-30 03:28:43 +02:00
|
|
|
|
2015-09-20 08:29:12 +02:00
|
|
|
import colorama
|
2020-05-21 14:21:55 +02:00
|
|
|
import veryprettytable as pretty
|
2015-09-20 08:29:12 +02:00
|
|
|
|
2020-03-10 14:33:25 +01:00
|
|
|
from io import BytesIO
|
2015-08-30 03:28:43 +02:00
|
|
|
|
|
|
|
|
2015-09-20 23:14:00 +02:00
|
|
|
class Printer:
|
|
|
|
def __init__(self, enable_color):
|
|
|
|
self.enable_color = enable_color
|
|
|
|
|
|
|
|
def print(self, *args, **kwargs):
|
|
|
|
if kwargs.get('color', False) and self.enable_color:
|
|
|
|
colorama.init()
|
|
|
|
color_dict = {
|
|
|
|
'default': '',
|
|
|
|
'header': colorama.Back.BLACK + colorama.Fore.WHITE,
|
|
|
|
'alt': colorama.Fore.YELLOW,
|
|
|
|
'zebra_0': '',
|
|
|
|
'zebra_1': colorama.Fore.BLUE,
|
|
|
|
'WARN': colorama.Fore.MAGENTA,
|
|
|
|
'ERROR': colorama.Fore.RED}
|
|
|
|
|
|
|
|
c = color_dict[kwargs.pop('color')]
|
|
|
|
args = (c + args[0],) + args[1:] + (colorama.Style.RESET_ALL,)
|
|
|
|
kwargs.pop('color', None)
|
|
|
|
return builtins.print(*args, **kwargs)
|
|
|
|
else:
|
|
|
|
kwargs.pop('color', None)
|
|
|
|
return builtins.print(*args, **kwargs)
|
|
|
|
|
2016-07-07 03:51:13 +02:00
|
|
|
# TODO: extract the name from the search results
|
|
|
|
# instead of from the magnet link when possible
|
2015-09-20 23:14:00 +02:00
|
|
|
def search_results(self, results, local=None):
|
|
|
|
columns = shutil.get_terminal_size((80, 20)).columns
|
|
|
|
even = True
|
2015-08-30 03:28:43 +02:00
|
|
|
|
|
|
|
if local:
|
2020-05-21 14:21:55 +02:00
|
|
|
table = pretty.VeryPrettyTable(['LINK', 'DATE', 'SIZE', 'NAME'])
|
2018-05-28 07:46:14 +02:00
|
|
|
|
|
|
|
table.align['SIZE'] = 'r'
|
|
|
|
table.align['NAME'] = 'l'
|
2015-08-30 03:28:43 +02:00
|
|
|
else:
|
2020-05-21 14:21:55 +02:00
|
|
|
table = pretty.VeryPrettyTable(['LINK', 'SEED', 'LEECH',
|
|
|
|
'RATIO', 'SIZE',
|
|
|
|
'UPLOAD', 'NAME'])
|
2015-09-20 23:14:00 +02:00
|
|
|
table.align['NAME'] = 'l'
|
|
|
|
table.align['SEED'] = 'r'
|
|
|
|
table.align['LEECH'] = 'r'
|
|
|
|
table.align['RATIO'] = 'r'
|
|
|
|
table.align['SIZE'] = 'r'
|
|
|
|
table.align['UPLOAD'] = 'l'
|
2015-08-30 03:28:43 +02:00
|
|
|
|
2015-09-20 23:14:00 +02:00
|
|
|
table.max_width = columns
|
|
|
|
table.border = False
|
|
|
|
table.padding_width = 1
|
2015-08-30 03:28:43 +02:00
|
|
|
|
2015-09-20 23:14:00 +02:00
|
|
|
for n, result in enumerate(results):
|
2020-05-21 01:57:46 +02:00
|
|
|
torrent_name = result['name']
|
2015-09-20 23:14:00 +02:00
|
|
|
|
|
|
|
if local:
|
2020-05-21 01:57:46 +02:00
|
|
|
content = [n, result['date'], result['size'],
|
|
|
|
torrent_name[:columns - 42]]
|
2015-09-20 23:14:00 +02:00
|
|
|
else:
|
2020-05-21 01:57:46 +02:00
|
|
|
no_seeders = int(result['seeders'])
|
2015-09-20 23:14:00 +02:00
|
|
|
no_leechers = int(result['leechers'])
|
2020-05-21 01:57:46 +02:00
|
|
|
size = result['size']
|
2015-09-20 23:14:00 +02:00
|
|
|
date = result['uploaded']
|
|
|
|
|
|
|
|
# compute the S/L ratio (Higher is better)
|
|
|
|
try:
|
|
|
|
ratio = no_seeders / no_leechers
|
|
|
|
except ZeroDivisionError:
|
|
|
|
ratio = float('inf')
|
|
|
|
|
2016-07-07 03:51:13 +02:00
|
|
|
content = [n, no_seeders, no_leechers,
|
|
|
|
'{:.1f}'.format(ratio),
|
2020-05-21 01:57:46 +02:00
|
|
|
size, date, torrent_name[:columns - 50]]
|
2015-09-20 23:14:00 +02:00
|
|
|
|
2015-09-20 23:44:52 +02:00
|
|
|
if even or not self.enable_color:
|
2015-09-20 23:14:00 +02:00
|
|
|
table.add_row(content)
|
|
|
|
else:
|
|
|
|
table.add_row(content, fore_color='blue')
|
|
|
|
|
|
|
|
# Alternate between colors
|
|
|
|
even = not even
|
|
|
|
self.print(table)
|
|
|
|
|
2020-05-21 02:28:03 +02:00
|
|
|
def descriptions(self, chosen_links, results, site, timeout):
|
2015-09-20 23:14:00 +02:00
|
|
|
for link in chosen_links:
|
2020-05-21 01:57:46 +02:00
|
|
|
result = results[link]
|
|
|
|
req = request.Request(
|
2020-05-21 14:21:55 +02:00
|
|
|
site + '/t.php?id=' + str(result['id']),
|
2020-05-21 01:57:46 +02:00
|
|
|
headers=pirate.data.default_headers)
|
2015-09-20 23:14:00 +02:00
|
|
|
req.add_header('Accept-encoding', 'gzip')
|
2020-05-21 02:28:03 +02:00
|
|
|
f = request.urlopen(req, timeout=timeout)
|
2015-09-20 23:14:00 +02:00
|
|
|
|
|
|
|
if f.info().get('Content-Encoding') == 'gzip':
|
|
|
|
f = gzip.GzipFile(fileobj=BytesIO(f.read()))
|
|
|
|
|
2020-05-21 01:57:46 +02:00
|
|
|
res = json.load(f)
|
2015-09-20 23:14:00 +02:00
|
|
|
|
|
|
|
# Replace HTML links with markdown style versions
|
|
|
|
desc = re.sub(r'<a href="\s*([^"]+?)\s*"[^>]*>(\s*)([^<]+?)(\s*'
|
2020-05-21 01:57:46 +02:00
|
|
|
r')</a>', r'\2[\3](\1)\4', res['descr'])
|
2015-09-20 23:14:00 +02:00
|
|
|
|
2020-05-21 01:57:46 +02:00
|
|
|
self.print('Description for "{}":'.format(result['name']),
|
|
|
|
color='zebra_1')
|
2015-09-20 23:14:00 +02:00
|
|
|
self.print(desc, color='zebra_0')
|
|
|
|
|
2020-05-21 02:28:03 +02:00
|
|
|
def file_lists(self, chosen_links, results, site, timeout):
|
2020-05-21 01:57:46 +02:00
|
|
|
# the API may returns object instead of list
|
|
|
|
def get(obj):
|
|
|
|
try:
|
|
|
|
return obj[0]
|
|
|
|
except KeyError:
|
|
|
|
return obj['0']
|
2020-03-10 14:33:25 +01:00
|
|
|
|
2015-09-20 23:14:00 +02:00
|
|
|
for link in chosen_links:
|
2020-05-21 01:57:46 +02:00
|
|
|
result = results[link]
|
|
|
|
req = request.Request(
|
2020-05-21 14:21:55 +02:00
|
|
|
site + '/f.php?id=' + str(result['id']),
|
2020-05-21 01:57:46 +02:00
|
|
|
headers=pirate.data.default_headers)
|
2015-09-20 23:14:00 +02:00
|
|
|
req.add_header('Accept-encoding', 'gzip')
|
2020-05-21 02:28:03 +02:00
|
|
|
f = request.urlopen(req, timeout=timeout)
|
2015-09-20 23:14:00 +02:00
|
|
|
|
|
|
|
if f.info().get('Content-Encoding') == 'gzip':
|
|
|
|
f = gzip.GzipFile(fileobj=BytesIO(f.read()))
|
|
|
|
|
2020-05-21 01:57:46 +02:00
|
|
|
res = json.load(f)
|
|
|
|
|
|
|
|
if len(res) == 1 and 'not found' in get(res[0]['name']):
|
2015-09-20 23:14:00 +02:00
|
|
|
self.print('File list not available.')
|
|
|
|
return
|
|
|
|
|
2020-05-21 01:57:46 +02:00
|
|
|
self.print('Files in {}:'.format(result['name']), color='zebra_1')
|
2015-09-20 23:14:00 +02:00
|
|
|
cur_color = 'zebra_0'
|
|
|
|
|
2020-05-21 01:57:46 +02:00
|
|
|
for f in res:
|
|
|
|
name = get(f['name'])
|
|
|
|
size = pirate.torrent.pretty_size(int(get(f['size'])))
|
|
|
|
self.print('{:>11} {}'.format(
|
|
|
|
size, name),
|
|
|
|
color=cur_color)
|
2016-07-07 03:51:13 +02:00
|
|
|
cur_color = 'zebra_0' if cur_color == 'zebra_1' else 'zebra_1'
|