import re import unicodedata def _format_size(value, digits, unit): if digits > 0: return "{{:.{}f}}{}".format(digits, unit).format(value) else: return "{{:d}}{}".format(unit).format(value) def format_size(bytes_, digits=1): if bytes_ < 1024: return _format_size(bytes_, digits, "B") kilo = bytes_ / 1024 if kilo < 1024: return _format_size(kilo, digits, "kB") mega = kilo / 1024 if mega < 1024: return _format_size(mega, digits, "MB") return _format_size(mega / 1024, digits, "GB") def format_duration(total_seconds): total_seconds = int(total_seconds) hours = total_seconds // 3600 remainder = total_seconds % 3600 minutes = remainder // 60 seconds = total_seconds % 60 if hours: return "{} h {} min".format(hours, minutes) if minutes: return "{} min {} sec".format(minutes, seconds) return "{} sec".format(seconds) def format_time(total_seconds, force_hours=False): total_seconds = int(total_seconds) hours = total_seconds // 3600 remainder = total_seconds % 3600 minutes = remainder // 60 seconds = total_seconds % 60 if hours or force_hours: return f"{hours:02}:{minutes:02}:{seconds:02}" return f"{minutes:02}:{seconds:02}" def read_int(msg, min, max, default=None): if default: msg = msg + f" [default {default}]" msg += ": " while True: try: val = input(msg) if default and not val: return default if min <= int(val) <= max: return int(val) except ValueError: pass def slugify(value): value = unicodedata.normalize('NFKC', str(value)) value = re.sub(r'[^\w\s_-]', '', value) value = re.sub(r'[\s_-]+', '_', value) return value.strip("_").lower() def titlify(value): value = unicodedata.normalize('NFKC', str(value)) value = re.sub(r'[^\w\s\[\]().-]', '', value) value = re.sub(r'\s+', ' ', value) return value.strip() VIDEO_PATTERNS = [ r"^(?P\d+)?$", r"^https://(www.)?twitch.tv/videos/(?P\d+)(\?.+)?$", ] CLIP_PATTERNS = [ r"^(?P[A-Za-z0-9]+(?:-[A-Za-z0-9_-]{16})?)$", r"^https://(www.)?twitch.tv/\w+/clip/(?P[A-Za-z0-9]+(?:-[A-Za-z0-9_-]{16})?)(\?.+)?$", r"^https://clips.twitch.tv/(?P[A-Za-z0-9]+(?:-[A-Za-z0-9_-]{16})?)(\?.+)?$", ] def parse_video_identifier(identifier): """Given a video ID or URL returns the video ID, or null if not matched""" for pattern in VIDEO_PATTERNS: match = re.match(pattern, identifier) if match: return match.group("id") def parse_clip_identifier(identifier): """Given a clip slug or URL returns the clip slug, or null if not matched""" for pattern in CLIP_PATTERNS: match = re.match(pattern, identifier) if match: return match.group("slug")