2018-01-25 10:09:20 +00:00
|
|
|
import re
|
|
|
|
import subprocess
|
|
|
|
import tempfile
|
|
|
|
|
2018-01-26 20:50:55 +00:00
|
|
|
from datetime import datetime
|
2018-01-25 10:09:20 +00:00
|
|
|
from concurrent.futures import ThreadPoolExecutor, as_completed
|
|
|
|
from functools import partial
|
|
|
|
from urllib.request import urlretrieve
|
|
|
|
|
|
|
|
from twitchdl import twitch
|
|
|
|
from twitchdl.output import print_out
|
|
|
|
|
|
|
|
|
|
|
|
def read_int(msg, min, max, default):
|
|
|
|
msg = msg + " [default {}]: ".format(default)
|
|
|
|
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
val = input(msg)
|
|
|
|
if not val:
|
|
|
|
return default
|
|
|
|
if min <= int(val) <= max:
|
|
|
|
return int(val)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2018-01-26 20:50:55 +00:00
|
|
|
def format_size(bytes_):
|
|
|
|
if bytes_ < 1024:
|
|
|
|
return str(bytes_)
|
|
|
|
|
|
|
|
kilo = bytes_ / 1024
|
|
|
|
if kilo < 1024:
|
|
|
|
return "{:.1f}K".format(kilo)
|
|
|
|
|
|
|
|
mega = kilo / 1024
|
|
|
|
if mega < 1024:
|
|
|
|
return "{:.1f}M".format(mega)
|
|
|
|
|
|
|
|
return "{:.1f}G".format(mega / 1024)
|
|
|
|
|
|
|
|
|
|
|
|
def format_duration(total_seconds):
|
|
|
|
total_seconds = int(total_seconds)
|
|
|
|
hours = total_seconds // 3600
|
|
|
|
remainder = total_seconds % 3600
|
2018-01-25 10:09:20 +00:00
|
|
|
minutes = remainder // 60
|
2018-01-26 20:50:55 +00:00
|
|
|
seconds = total_seconds % 60
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
if hours:
|
2018-01-26 20:50:55 +00:00
|
|
|
return "{} h {} min".format(hours, minutes)
|
|
|
|
|
|
|
|
if minutes:
|
|
|
|
return "{} min {} sec".format(minutes, seconds)
|
2018-01-25 10:09:20 +00:00
|
|
|
|
2018-01-26 20:50:55 +00:00
|
|
|
return "{} sec".format(seconds)
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _print_video(video):
|
|
|
|
published_at = video['published_at'].replace('T', ' @ ').replace('Z', '')
|
2018-01-26 20:50:55 +00:00
|
|
|
length = format_duration(video['length'])
|
2018-01-25 10:09:20 +00:00
|
|
|
name = video['channel']['display_name']
|
|
|
|
|
|
|
|
print_out("\n<bold>{}</bold>".format(video['_id'][1:]))
|
|
|
|
print_out("<green>{}</green>".format(video["title"]))
|
|
|
|
print_out("<cyan>{}</cyan> playing <cyan>{}</cyan>".format(name, video['game']))
|
|
|
|
print_out("Published <cyan>{}</cyan> Length: <cyan>{}</cyan> ".format(published_at, length))
|
|
|
|
|
|
|
|
|
|
|
|
def videos(channel_name, **kwargs):
|
|
|
|
videos = twitch.get_channel_videos(channel_name)
|
|
|
|
|
|
|
|
print("Found {} videos".format(videos["_total"]))
|
|
|
|
|
|
|
|
for video in videos['videos']:
|
|
|
|
_print_video(video)
|
|
|
|
|
|
|
|
|
|
|
|
def _select_playlist_by_quality(playlists):
|
|
|
|
print("\nAvailable qualities:")
|
|
|
|
for no, v in playlists.items():
|
|
|
|
print("{}) {}".format(no, v[0]))
|
|
|
|
|
|
|
|
keys = list(playlists.keys())
|
|
|
|
no = read_int("Choose quality", min=min(keys), max=max(keys), default=keys[0])
|
|
|
|
|
|
|
|
return playlists[no][1]
|
|
|
|
|
|
|
|
|
|
|
|
def _print_progress(futures):
|
|
|
|
counter = 1
|
|
|
|
total = len(futures)
|
2018-01-26 20:50:55 +00:00
|
|
|
total_size = 0
|
|
|
|
start_time = datetime.now()
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
for future in as_completed(futures):
|
2018-01-26 20:50:55 +00:00
|
|
|
file, headers = future.result()
|
|
|
|
percentage = 100 * counter // total
|
|
|
|
total_size += int(headers.get("Content-Length"))
|
|
|
|
duration = (datetime.now() - start_time).seconds
|
|
|
|
speed = total_size // duration if duration else 0
|
|
|
|
remaining = (total - counter) * duration / counter
|
|
|
|
|
|
|
|
msg = "Downloaded VOD {}/{} ({}%) total <cyan>{}B</cyan> at <cyan>{}B/s</cyan> remaining <cyan>{}</cyan>".format(
|
|
|
|
counter, total, percentage, format_size(total_size), format_size(speed), format_duration(remaining))
|
|
|
|
|
|
|
|
print_out("\r" + msg.ljust(80), end='')
|
2018-01-25 10:09:20 +00:00
|
|
|
counter += 1
|
|
|
|
|
|
|
|
|
|
|
|
def _download_files(base_url, directory, filenames, max_workers):
|
|
|
|
args = [(base_url.format(f), "/".join([directory, f])) for f in filenames]
|
|
|
|
|
|
|
|
fns = [partial(urlretrieve, url, path) for url, path in args]
|
|
|
|
|
|
|
|
with ThreadPoolExecutor(max_workers=max_workers) as executor:
|
|
|
|
futures = [executor.submit(fn) for fn in fns]
|
|
|
|
_print_progress(futures)
|
|
|
|
|
|
|
|
return [f.result()[0] for f in futures]
|
|
|
|
|
|
|
|
|
|
|
|
def _join_vods(directory, filenames, target):
|
|
|
|
input_path = "{}/files.txt".format(directory)
|
|
|
|
|
|
|
|
with open(input_path, 'w') as f:
|
|
|
|
for filename in filenames:
|
|
|
|
f.write('file {}\n'.format(filename))
|
|
|
|
|
|
|
|
result = subprocess.run([
|
|
|
|
"ffmpeg",
|
|
|
|
"-f", "concat",
|
|
|
|
"-i", input_path,
|
|
|
|
"-c", "copy",
|
|
|
|
target,
|
|
|
|
"-stats",
|
|
|
|
"-loglevel", "warning",
|
|
|
|
])
|
|
|
|
|
|
|
|
result.check_returncode()
|
|
|
|
|
|
|
|
|
|
|
|
def _video_target_filename(video, format):
|
|
|
|
dttm = re.sub(r'\D+', '-', video['published_at'][:16])
|
|
|
|
name = " - ".join([dttm, video['channel']['display_name'], video['game']])
|
|
|
|
return "{}.{}".format(name, format)
|
|
|
|
|
|
|
|
|
|
|
|
def download(video_id, max_workers, format='mkv', **kwargs):
|
|
|
|
print("Looking up video...")
|
|
|
|
video = twitch.get_video(video_id)
|
|
|
|
|
|
|
|
print("Fetching access token...")
|
|
|
|
access_token = twitch.get_access_token(video_id)
|
|
|
|
|
|
|
|
print("Fetching playlists...")
|
|
|
|
playlists = twitch.get_playlists(video_id, access_token)
|
|
|
|
playlist_url = _select_playlist_by_quality(playlists)
|
|
|
|
|
|
|
|
print("\nFetching playlist...")
|
|
|
|
base_url, filenames = twitch.get_playlist_urls(playlist_url)
|
|
|
|
|
|
|
|
target = _video_target_filename(video, format)
|
|
|
|
|
|
|
|
with tempfile.TemporaryDirectory() as directory:
|
2018-01-26 20:50:55 +00:00
|
|
|
print("Downloading with {} workers...".format(max_workers))
|
2018-01-25 10:09:20 +00:00
|
|
|
_download_files(base_url, directory, filenames, max_workers)
|
|
|
|
|
|
|
|
print("\n\nJoining files...")
|
|
|
|
_join_vods(directory, filenames, target)
|
|
|
|
|
|
|
|
print("\nDownloaded: {}".format(target))
|