From a525f1d913d283bb285dc13d28e702379512d264 Mon Sep 17 00:00:00 2001 From: FildCommander Date: Sat, 6 Jul 2024 11:12:41 +0200 Subject: [PATCH 1/4] Updated Tube8 Video Download --- youtube_dl/extractor/tube8.py | 140 ++++++++++++++++++---------------- 1 file changed, 75 insertions(+), 65 deletions(-) diff --git a/youtube_dl/extractor/tube8.py b/youtube_dl/extractor/tube8.py index db93b0182..fcb929285 100644 --- a/youtube_dl/extractor/tube8.py +++ b/youtube_dl/extractor/tube8.py @@ -2,85 +2,95 @@ from __future__ import unicode_literals import re +from .common import InfoExtractor from ..utils import ( - int_or_none, - str_to_int, + traverse_obj, + T, + url_or_none, + parse_iso8601, ) -from .keezmovies import KeezMoviesIE -class Tube8IE(KeezMoviesIE): - _VALID_URL = r'https?://(?:www\.)?tube8\.com/(?:[^/]+/)+(?P[^/]+)/(?P\d+)' +class Tube8IE(InfoExtractor): + _VALID_URL = r'https?:\/\/(?:www\.)?tube8\.com\/+[^\/]+\/(?P\d+)' _TESTS = [{ - 'url': 'http://www.tube8.com/teen/kasia-music-video/229795/', - 'md5': '65e20c48e6abff62ed0c3965fff13a39', + 'url': 'https://www.tube8.com/porn-video/189530841/', + 'md5': '532408f59e89a32027d873af6289c85a', 'info_dict': { - 'id': '229795', - 'display_id': 'kasia-music-video', + 'id': '189530841', 'ext': 'mp4', - 'description': 'hot teen Kasia grinding', - 'uploader': 'unknown', - 'title': 'Kasia music video', - 'age_limit': 18, - 'duration': 230, - 'categories': ['Teen'], - 'tags': ['dancing'], - }, - }, { - 'url': 'http://www.tube8.com/shemale/teen/blonde-cd-gets-kidnapped-by-two-blacks-and-punished-for-being-a-slutty-girl/19569151/', - 'only_matching': True, + 'title': 'Found dildo. She let it cum in her tight ass to keep the secret', + 'thumbnail': r're:^https?://.*\.jpg$', + 'uploader': 'MaryKrylova', + 'timestamp': 1718961736, + 'upload_date': '20240621', + } }] - @staticmethod - def _extract_urls(webpage): - return re.findall( - r']+\bsrc=["\']((?:https?:)?//(?:www\.)?tube8\.com/embed/(?:[^/]+/)+\d+)', - webpage) - def _real_extract(self, url): - webpage, info = self._extract_info(url) + video_id = self._match_valid_url(url).group('id') + webpage = self._download_webpage(url, video_id) - if not info['title']: - info['title'] = self._html_search_regex( - r'videoTitle\s*=\s*"([^"]+)', webpage, 'title') + playervars = self._search_json( + r'\bplayervars\s*:', webpage, 'playervars', video_id) - description = self._html_search_regex( - r'(?s)Description:\s*
(.+?)
', webpage, 'description', fatal=False) - uploader = self._html_search_regex( - r'\s*(.+?)\s*<', - webpage, 'uploader', fatal=False) + extra_info = self._search_json( + r'application/ld\+json[\"\']?>\s?', webpage, 'extra_info', video_id) - like_count = int_or_none(self._search_regex( - r'rupVar\s*=\s*"(\d+)"', webpage, 'like count', fatal=False)) - dislike_count = int_or_none(self._search_regex( - r'rdownVar\s*=\s*"(\d+)"', webpage, 'dislike count', fatal=False)) - view_count = str_to_int(self._search_regex( - r'Views:\s*\s*
([\d,\.]+)', - webpage, 'view count', fatal=False)) - comment_count = str_to_int(self._search_regex( - r'(\d+)', - webpage, 'comment count', fatal=False)) + uploader = traverse_obj(extra_info, ( + '@graph', lambda _, v: v.get('author'), 'author'))[0] - category = self._search_regex( - r'Category:\s*\s*
\s*]+href=[^>]+>([^<]+)', - webpage, 'category', fatal=False) - categories = [category] if category else None + thumbnail = traverse_obj(extra_info, ( + '@graph', lambda _, v: v.get('thumbnail'), 'thumbnail'))[0] - tags_str = self._search_regex( - r'(?s)Tags:\s*\s*
(.+?)]+href=[^>]+>([^<]+)', tags_str)] if tags_str else None + timestamp = parse_iso8601(traverse_obj(extra_info, ( + '@graph', lambda _, v: v.get('datePublished'), 'datePublished'))[0]) - info.update({ - 'description': description, + # Borrowed from youporn extractor + def get_fmt(x): + v_url = url_or_none(x.get('videoUrl')) + if v_url: + x['videoUrl'] = v_url + return (x['format'], x) + + defs_by_format = dict(traverse_obj(playervars, ( + 'mediaDefinitions', lambda _, v: v.get('format'), T(get_fmt)))) + + title = traverse_obj(playervars, 'video_title') + if not thumbnail: + thumbnail = traverse_obj(playervars, 'image_url') + + # Borrowed from youporn extractor + def get_format_data(f): + if f not in defs_by_format: + return [] + return self._download_json( + defs_by_format[f]['videoUrl'], video_id, '{0}-formats'.format(f)) + + formats = [] + for mp4_url in traverse_obj( + get_format_data('mp4'), + (lambda _, v: not isinstance(v['defaultQuality'], bool), 'videoUrl'), + (Ellipsis, 'videoUrl')): + mobj = re.search(r'(?P\d{3,4})[pP]_(?P\d+)[kK]_\d+', mp4_url) + if mobj: + height = int(mobj.group('height')) + tbr = int(mobj.group('bitrate')) + formats.append({ + 'format_id': '%dp-%dk' % (height, tbr), + 'url': mp4_url, + 'ext': 'mp4', + 'tbr': tbr, + 'height': height, + }) + + self._sort_formats(formats) + + return { + 'id': video_id, + 'title': title, + 'thumbnail': thumbnail, + 'formats': formats, 'uploader': uploader, - 'view_count': view_count, - 'like_count': like_count, - 'dislike_count': dislike_count, - 'comment_count': comment_count, - 'categories': categories, - 'tags': tags, - }) - - return info + 'timestamp': timestamp, + } From 5f38e78a96d7eaa410cf2b31742686223816b23e Mon Sep 17 00:00:00 2001 From: FildCommander Date: Sat, 6 Jul 2024 12:10:32 +0200 Subject: [PATCH 2/4] Added Support for Tube8 Channel downloads --- youtube_dl/extractor/extractors.py | 5 +- youtube_dl/extractor/tube8.py | 166 ++++++++++++++++++++++++++++- 2 files changed, 169 insertions(+), 2 deletions(-) diff --git a/youtube_dl/extractor/extractors.py b/youtube_dl/extractor/extractors.py index 3da5f8020..26cff0db2 100644 --- a/youtube_dl/extractor/extractors.py +++ b/youtube_dl/extractor/extractors.py @@ -1327,7 +1327,10 @@ from .trovo import ( ) from .trunews import TruNewsIE from .trutv import TruTVIE -from .tube8 import Tube8IE +from .tube8 import ( + Tube8IE, + Tube8ListIE, +) from .tubitv import TubiTvIE from .tumblr import TumblrIE from .tunein import ( diff --git a/youtube_dl/extractor/tube8.py b/youtube_dl/extractor/tube8.py index fcb929285..e1d332bd6 100644 --- a/youtube_dl/extractor/tube8.py +++ b/youtube_dl/extractor/tube8.py @@ -1,6 +1,8 @@ from __future__ import unicode_literals +import itertools import re +from time import sleep from .common import InfoExtractor from ..utils import ( @@ -8,11 +10,16 @@ from ..utils import ( T, url_or_none, parse_iso8601, + get_element_by_class, + get_element_by_id, + int_or_none, + parse_qs, + urljoin, ) class Tube8IE(InfoExtractor): - _VALID_URL = r'https?:\/\/(?:www\.)?tube8\.com\/+[^\/]+\/(?P\d+)' + _VALID_URL = r'https?:\/\/(?:www\.)?tube8\.com\/+porn-video+\/(?P\d+)' _TESTS = [{ 'url': 'https://www.tube8.com/porn-video/189530841/', 'md5': '532408f59e89a32027d873af6289c85a', @@ -94,3 +101,160 @@ class Tube8IE(InfoExtractor): 'uploader': uploader, 'timestamp': timestamp, } + + +# Currently only user channels +class Tube8ListIE(InfoExtractor): + _PLAYLIST_TITLEBAR_RE = r'\s+[Vv]ideos\s*$' + _PAGE_RETRY_COUNT = 0 # ie, no retry + _PAGE_RETRY_DELAY = 2 # seconds + + _VALID_URL = r'https?:\/\/(?:www\.)?tube8\.com\.?\/+user-videos\/+(?P\d+)\/+(?P[^\/]+)\/?.*' + _TESTS = [{ + 'url': 'https://www.tube8.com/user-videos/195075441/MaryKrylova/', + 'info_dict': { + 'id': '195075441', + }, + 'playlist_mincount': 29, + }, { + 'url': 'https://www.tube8.com/user-videos/195048331/FoxyElf/', + 'info_dict': { + 'id': '195048331', + }, + 'playlist_mincount': 86, + }] + + # Borrowed from youporn extractor + @classmethod + def _get_title_from_slug(cls, title_slug): + return re.sub(r'[_-]', ' ', title_slug) + + # Borrowed from youporn extractor + def _get_next_url(self, url, pl_id, html): + return urljoin(url, self._search_regex( + r''']*?\bhref\s*=\s*("|')(?P(?:(?!\1)[^>])+)\1''', + get_element_by_id('next', html) or '', 'next page', + group='url', default=None)) + + # Borrowed from youporn extractor + def _entries(self, url, pl_id, html=None, page_num=None): + + # separates page sections + PLAYLIST_SECTION_RE = ( + r''']*\bclass\s*=\s*('|")(?:[\w$-]+\s+|\s)*?title-bar(?:\s+[\w$-]+|\s)*\1[^>]*>''' + ) + # contains video link + VIDEO_URL_RE = r'''(?x) + ]*\bdata-video-id\s*=\s*('|")\d+\1[^>]*>\s* + (?:\s*)* + ]*\bhref\s*=\s*('|")(?P(?:(?!\2)[^>])+)\2 + ''' + + def yield_pages(url, html=html, page_num=page_num): + fatal = not html + for pnum in itertools.count(start=page_num or 1): + if not html: + html = self._download_webpage( + url, pl_id, note='Downloading page %d' % pnum, + fatal=fatal) + if not html: + break + fatal = False + yield (url, html, pnum) + # explicit page: extract just that page + if page_num is not None: + break + next_url = self._get_next_url(url, pl_id, html) + if not next_url or next_url == url: + break + url, html = next_url, None + + def retry_page(msg, tries_left, page_data): + if tries_left <= 0: + return + self.report_warning(msg, pl_id) + sleep(self._PAGE_RETRY_DELAY) + return next( + yield_pages(page_data[0], page_num=page_data[2]), None) + + def yield_entries(html): + for frag in re.split(PLAYLIST_SECTION_RE, html): + if not frag: + continue + t_text = get_element_by_class('title-text', frag or '') + if not (t_text and re.search(self._PLAYLIST_TITLEBAR_RE, t_text)): + continue + for m in re.finditer(VIDEO_URL_RE, frag): + video_url = urljoin(url, m.group('url')) + if video_url: + yield self.url_result(video_url) + + last_first_url = None + for page_data in yield_pages(url, html=html, page_num=page_num): + # page_data: url, html, page_num + first_url = None + tries_left = self._PAGE_RETRY_COUNT + 1 + while tries_left > 0: + tries_left -= 1 + for from_ in yield_entries(page_data[1]): + # may get the same page twice instead of empty page + # or (site bug) intead of actual next page + if not first_url: + first_url = from_['url'] + if first_url == last_first_url: + # sometimes (/porntags/) the site serves the previous page + # instead but may provide the correct page after a delay + page_data = retry_page( + 'Retrying duplicate page...', tries_left, page_data) + if page_data: + first_url = None + break + continue + yield from_ + else: + if not first_url and 'no-result-paragarph1' in page_data[1]: + page_data = retry_page( + 'Retrying empty page...', tries_left, page_data) + if page_data: + continue + else: + # success/failure + break + # may get an infinite (?) sequence of empty pages + if not first_url: + break + last_first_url = first_url + + # Borrowed from youporn extractor + def _real_extract(self, url, html=None): + # exceptionally, id may be None + m_dict = self._match_valid_url(url).groupdict() + pl_id, page_type, sort = (m_dict.get(k) for k in ('id', 'type', 'sort')) + + qs = parse_qs(url) + for q, v in qs.items(): + if v: + qs[q] = v[-1] + else: + del qs[q] + + base_id = pl_id or 'Tube8' + title = self._get_title_from_slug(base_id) + if page_type: + title = '%s %s' % (page_type.capitalize(), title) + base_id = [base_id.lower()] + if sort is None: + title += ' videos' + else: + title = '%s videos by %s' % (title, re.sub(r'[_-]', ' ', sort)) + base_id.append(sort) + if qs: + ps = ['%s=%s' % item for item in sorted(qs.items())] + title += ' (%s)' % ','.join(ps) + base_id.extend(ps) + pl_id = '/'.join(base_id) + + return self.playlist_result( + self._entries(url, pl_id, html=html, + page_num=int_or_none(qs.get('page'))), + playlist_id=pl_id) From d8414d01bd364d54b7803d48a6048454c4089fb2 Mon Sep 17 00:00:00 2001 From: "P. S" <50270838+FildCommander@users.noreply.github.com> Date: Mon, 15 Jul 2024 18:38:13 +0200 Subject: [PATCH 3/4] Apply suggestions from code review Co-authored-by: dirkf --- youtube_dl/extractor/tube8.py | 70 ++++++++++++++++++----------------- 1 file changed, 37 insertions(+), 33 deletions(-) diff --git a/youtube_dl/extractor/tube8.py b/youtube_dl/extractor/tube8.py index e1d332bd6..dd3baf577 100644 --- a/youtube_dl/extractor/tube8.py +++ b/youtube_dl/extractor/tube8.py @@ -1,3 +1,4 @@ +# coding: utf-8 from __future__ import unicode_literals import itertools @@ -34,24 +35,30 @@ class Tube8IE(InfoExtractor): } }] + _EMBED_REGEX = r'