twitch-dl/twitchdl/commands.py

173 lines
5.1 KiB
Python
Raw Normal View History

2019-08-23 10:36:05 +00:00
import m3u8
import os
import pathlib
2018-01-25 10:09:20 +00:00
import re
2019-08-23 10:36:05 +00:00
import requests
import shutil
2018-01-25 10:09:20 +00:00
import subprocess
import tempfile
2019-08-23 10:36:05 +00:00
from urllib.parse import urlparse
2018-01-25 10:09:20 +00:00
2020-04-11 11:08:42 +00:00
from twitchdl import twitch, utils
from twitchdl.download import download_files
from twitchdl.exceptions import ConsoleError
2020-04-11 11:08:42 +00:00
from twitchdl.output import print_out, print_video
2018-01-25 10:09:20 +00:00
def videos(channel_name, limit, offset, sort, **kwargs):
print_out("Looking up user...")
user = twitch.get_user(channel_name)
if not user:
raise ConsoleError("User {} not found.".format(channel_name))
print_out("Loading videos...")
videos = twitch.get_channel_videos(user["id"], limit, offset, sort)
count = len(videos['videos'])
if not count:
print_out("No videos found")
return
first = offset + 1
last = offset + len(videos['videos'])
total = videos["_total"]
print_out("<yellow>Showing videos {}-{} of {}</yellow>".format(first, last, total))
2018-01-25 10:09:20 +00:00
for video in videos['videos']:
2020-04-11 11:08:42 +00:00
print_video(video)
2018-01-25 10:09:20 +00:00
def _select_quality(playlists):
2019-04-30 11:34:54 +00:00
print_out("\nAvailable qualities:")
2019-08-23 10:36:05 +00:00
for n, p in enumerate(playlists):
name = p.media[0].name if p.media else ""
resolution = "x".join(str(r) for r in p.stream_info.resolution)
print_out("{}) {} [{}]".format(n + 1, name, resolution))
2018-01-25 10:09:20 +00:00
2020-04-11 11:08:42 +00:00
no = utils.read_int("Choose quality", min=1, max=len(playlists) + 1, default=1)
2018-01-25 10:09:20 +00:00
2019-08-23 10:36:05 +00:00
return playlists[no - 1]
2018-01-25 10:09:20 +00:00
def _join_vods(directory, file_paths, target):
2018-01-25 10:09:20 +00:00
input_path = "{}/files.txt".format(directory)
with open(input_path, 'w') as f:
for path in file_paths:
f.write('file {}\n'.format(os.path.basename(path)))
2018-01-25 10:09:20 +00:00
result = subprocess.run([
"ffmpeg",
"-f", "concat",
"-i", input_path,
"-c", "copy",
target,
"-stats",
"-loglevel", "warning",
])
result.check_returncode()
def _video_target_filename(video, format):
match = re.search(r"^(\d{4})-(\d{2})-(\d{2})T", video['published_at'])
date = "".join(match.groups())
name = "_".join([
date,
video['_id'][1:],
video['channel']['name'],
2020-04-11 11:08:42 +00:00
utils.slugify(video['title']),
])
return name + "." + format
2018-01-25 10:09:20 +00:00
2019-08-23 10:36:05 +00:00
def _parse_video_id(video_id):
"""This can be either a integer ID or an URL to the video on twitch."""
if re.search(r"^\d+$", video_id):
return int(video_id)
match = re.search(r"^https://www.twitch.tv/videos/(\d+)(\?.+)?$", video_id)
if match:
return int(match.group(1))
raise ConsoleError("Invalid video ID given, expected integer ID or Twitch URL")
2019-08-23 10:36:05 +00:00
def _get_files(playlist, start, end):
"""Extract files for download from playlist."""
vod_start = 0
for segment in playlist.segments:
vod_end = vod_start + segment.duration
# `vod_end > start` is used here becuase it's better to download a bit
# more than a bit less, similar for the end condition
start_condition = not start or vod_end > start
end_condition = not end or vod_start < end
if start_condition and end_condition:
yield segment.uri
vod_start = vod_end
def _crete_temp_dir(base_uri):
"""Create a temp dir to store downloads if it doesn't exist."""
path = urlparse(base_uri).path
directory = '{}/twitch-dl{}'.format(tempfile.gettempdir(), path)
pathlib.Path(directory).mkdir(parents=True, exist_ok=True)
return directory
def download(video_id, max_workers, format='mkv', start=None, end=None, keep=False, **kwargs):
2019-08-23 10:36:05 +00:00
video_id = _parse_video_id(video_id)
if start and end and end <= start:
raise ConsoleError("End time must be greater than start time")
2019-04-30 11:34:54 +00:00
print_out("Looking up video...")
2018-01-25 10:09:20 +00:00
video = twitch.get_video(video_id)
2019-04-30 11:34:54 +00:00
print_out("Found: <blue>{}</blue> by <yellow>{}</yellow>".format(
video['title'], video['channel']['display_name']))
print_out("Fetching access token...")
2018-01-25 10:09:20 +00:00
access_token = twitch.get_access_token(video_id)
2019-04-30 11:34:54 +00:00
print_out("Fetching playlists...")
2018-01-25 10:09:20 +00:00
playlists = twitch.get_playlists(video_id, access_token)
parsed = m3u8.loads(playlists)
selected = _select_quality(parsed.playlists)
2018-01-25 10:09:20 +00:00
2019-04-30 11:34:54 +00:00
print_out("\nFetching playlist...")
2019-08-23 10:36:05 +00:00
response = requests.get(selected.uri)
response.raise_for_status()
playlist = m3u8.loads(response.text)
2019-08-23 10:36:05 +00:00
base_uri = re.sub("/[^/]+$", "/", selected.uri)
target_dir = _crete_temp_dir(base_uri)
filenames = list(_get_files(playlist, start, end))
2018-01-25 10:09:20 +00:00
# Save playlists for debugging purposes
with open(target_dir + "playlists.m3u8", "w") as f:
f.write(playlists)
with open(target_dir + "playlist.m3u8", "w") as f:
f.write(response.text)
2019-08-23 10:36:05 +00:00
print_out("\nDownloading {} VODs using {} workers to {}".format(
len(filenames), max_workers, target_dir))
file_paths = download_files(base_uri, target_dir, filenames, max_workers)
2018-01-25 10:09:20 +00:00
2019-04-30 11:34:54 +00:00
print_out("\n\nJoining files...")
target = _video_target_filename(video, format)
_join_vods(target_dir, file_paths, target)
2018-01-25 10:09:20 +00:00
if keep:
2019-08-23 10:36:05 +00:00
print_out("\nTemporary files not deleted: {}".format(target_dir))
else:
2019-08-23 10:36:05 +00:00
print_out("\nDeleting temporary files...")
shutil.rmtree(target_dir)
2018-01-25 10:09:20 +00:00
print_out("Downloaded: {}".format(target))