[extractor/txxx] Add extractors (#5240)

Authored by: chio0hai
Closes #5021
This commit is contained in:
chio0hai 2023-02-03 13:47:00 -05:00 committed by GitHub
parent b032ff0f03
commit 389896df85
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 424 additions and 14 deletions

View File

@ -2048,6 +2048,10 @@ from .twitter import (
TwitterSpacesIE, TwitterSpacesIE,
TwitterShortenerIE, TwitterShortenerIE,
) )
from .txxx import (
TxxxIE,
PornTopIE,
)
from .udemy import ( from .udemy import (
UdemyIE, UdemyIE,
UdemyCourseIE UdemyCourseIE

View File

@ -864,20 +864,6 @@ class GenericIE(InfoExtractor):
'thumbnail': r're:^https?://.*\.jpg$', 'thumbnail': r're:^https?://.*\.jpg$',
}, },
}, },
{
# JWPlayer config passed as variable
'url': 'http://www.txxx.com/videos/3326530/ariele/',
'info_dict': {
'id': '3326530_hq',
'ext': 'mp4',
'title': 'ARIELE | Tube Cup',
'uploader': 'www.txxx.com',
'age_limit': 18,
},
'params': {
'skip_download': True,
}
},
{ {
# Video.js embed, multiple formats # Video.js embed, multiple formats
'url': 'http://ortcam.com/solidworks-урок-6-настройка-чертежа_33f9b7351.html', 'url': 'http://ortcam.com/solidworks-урок-6-настройка-чертежа_33f9b7351.html',

418
yt_dlp/extractor/txxx.py Normal file
View File

@ -0,0 +1,418 @@
import base64
import re
from .common import InfoExtractor
from ..utils import (
ExtractorError,
int_or_none,
js_to_json,
merge_dicts,
parse_duration,
traverse_obj,
try_call,
urljoin,
variadic,
)
def decode_base64(text):
return base64.b64decode(text.translate(text.maketrans({
'\u0405': 'S',
'\u0406': 'I',
'\u0408': 'J',
'\u0410': 'A',
'\u0412': 'B',
'\u0415': 'E',
'\u041a': 'K',
'\u041c': 'M',
'\u041d': 'H',
'\u041e': 'O',
'\u0420': 'P',
'\u0421': 'C',
'\u0425': 'X',
',': '/',
'.': '+',
'~': '=',
}))).decode()
def get_formats(host, video_file):
return [{
'url': urljoin(f'https://{host}', decode_base64(video['video_url'])),
'format_id': try_call(lambda: variadic(video['format'])[0].lstrip('_')),
'quality': index,
} for index, video in enumerate(video_file) if video.get('video_url')]
class TxxxIE(InfoExtractor):
_DOMAINS = (
'hclips.com',
'hdzog.com',
'hdzog.tube',
'hotmovs.com',
'hotmovs.tube',
'inporn.com',
'privatehomeclips.com',
'tubepornclassic.com',
'txxx.com',
'txxx.tube',
'upornia.com',
'upornia.tube',
'vjav.com',
'vjav.tube',
'vxxx.com',
'voyeurhit.com',
'voyeurhit.tube',
)
_VALID_URL = rf'''(?x)
https?://(?:www\.)?(?P<host>{"|".join(map(re.escape, _DOMAINS))})/
(?:videos?[/-]|embed/)(?P<id>\d+)(?:/(?P<display_id>[^/?#]+))?
'''
_EMBED_REGEX = [rf'<iframe[^>]+?src=(["\'])(?P<url>(?:https?:)?//(?:www\.)?(?:{"|".join(map(re.escape, _DOMAINS))})/embed/[^"\']*)\1']
_TESTS = [{
'url': 'https://txxx.com/videos/16574965/digital-desire-malena-morgan/',
'md5': 'c54e4ace54320aaf8e2a72df87859391',
'info_dict': {
'id': '16574965',
'display_id': 'digital-desire-malena-morgan',
'ext': 'mp4',
'title': 'Digital Desire - Malena Morgan',
'uploader': 'Lois Argentum',
'duration': 694,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://txxx.tube/videos/16574965/digital-desire-malena-morgan/',
'md5': 'c54e4ace54320aaf8e2a72df87859391',
'info_dict': {
'id': '16574965',
'display_id': 'digital-desire-malena-morgan',
'ext': 'mp4',
'title': 'Digital Desire - Malena Morgan',
'uploader': 'Lois Argentum',
'duration': 694,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://vxxx.com/video-68925/',
'md5': '1fcff3748b0c5b41fe41d0afa22409e1',
'info_dict': {
'id': '68925',
'display_id': '68925',
'ext': 'mp4',
'title': 'Malena Morgan',
'uploader': 'Huge Hughes',
'duration': 694,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://hclips.com/videos/6291073/malena-morgan-masturbates-her-sweet/',
'md5': 'a5dd4f83363972ee043313cff85e7e26',
'info_dict': {
'id': '6291073',
'display_id': 'malena-morgan-masturbates-her-sweet',
'ext': 'mp4',
'title': 'Malena Morgan masturbates her sweet',
'uploader': 'John Salt',
'duration': 426,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://hdzog.com/videos/67063/gorgeous-malena-morgan-will-seduce-you-at-the-first-glance/',
'md5': 'f8bdedafd45d1ec2875c43fe33a846d3',
'info_dict': {
'id': '67063',
'display_id': 'gorgeous-malena-morgan-will-seduce-you-at-the-first-glance',
'ext': 'mp4',
'title': 'Gorgeous Malena Morgan will seduce you at the first glance',
'uploader': 'momlesson',
'duration': 601,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://hdzog.tube/videos/67063/gorgeous-malena-morgan-will-seduce-you-at-the-first-glance/',
'md5': 'f8bdedafd45d1ec2875c43fe33a846d3',
'info_dict': {
'id': '67063',
'display_id': 'gorgeous-malena-morgan-will-seduce-you-at-the-first-glance',
'ext': 'mp4',
'title': 'Gorgeous Malena Morgan will seduce you at the first glance',
'uploader': 'momlesson',
'duration': 601,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://hotmovs.com/videos/8789287/unbelievable-malena-morgan-performing-in-incredible-masturantion/',
'md5': '71d32c51584876472db87e561171a386',
'info_dict': {
'id': '8789287',
'display_id': 'unbelievable-malena-morgan-performing-in-incredible-masturantion',
'ext': 'mp4',
'title': 'Unbelievable Malena Morgan performing in incredible masturantion',
'uploader': 'Davit Sanchez',
'duration': 940,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://hotmovs.tube/videos/8789287/unbelievable-malena-morgan-performing-in-incredible-masturantion/',
'md5': '71d32c51584876472db87e561171a386',
'info_dict': {
'id': '8789287',
'display_id': 'unbelievable-malena-morgan-performing-in-incredible-masturantion',
'ext': 'mp4',
'title': 'Unbelievable Malena Morgan performing in incredible masturantion',
'uploader': 'Davit Sanchez',
'duration': 940,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://inporn.com/video/517897/malena-morgan-solo/',
'md5': '344db467481edf78f193cdf5820a7cfb',
'info_dict': {
'id': '517897',
'display_id': 'malena-morgan-solo',
'ext': 'mp4',
'title': 'Malena Morgan - Solo',
'uploader': 'Ashley Oxy',
'duration': 480,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://privatehomeclips.com/videos/3630599/malena-morgan-cam-show/',
'md5': 'ea657273e352493c5fb6357fbfa4f126',
'info_dict': {
'id': '3630599',
'display_id': 'malena-morgan-cam-show',
'ext': 'mp4',
'title': 'malena morgan cam show',
'uploader': 'Member9915',
'duration': 290,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://tubepornclassic.com/videos/1015455/mimi-rogers-full-body-massage-nude-compilation/',
'md5': '2e9a6cf610c9862e86e0ce24f08f4427',
'info_dict': {
'id': '1015455',
'display_id': 'mimi-rogers-full-body-massage-nude-compilation',
'ext': 'mp4',
'title': 'Mimi Rogers - Full Body Massage (Nude) compilation',
'uploader': '88bhuto',
'duration': 286,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://upornia.com/videos/1498858/twistys-malena-morgan-starring-at-dr-morgan-baller/',
'md5': '7ff7033340bc88a173198b7c22600e4f',
'info_dict': {
'id': '1498858',
'display_id': 'twistys-malena-morgan-starring-at-dr-morgan-baller',
'ext': 'mp4',
'title': 'Twistys - Malena Morgan starring at Dr. Morgan-Baller',
'uploader': 'mindgeek',
'duration': 480,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://upornia.tube/videos/1498858/twistys-malena-morgan-starring-at-dr-morgan-baller/',
'md5': '7ff7033340bc88a173198b7c22600e4f',
'info_dict': {
'id': '1498858',
'display_id': 'twistys-malena-morgan-starring-at-dr-morgan-baller',
'ext': 'mp4',
'title': 'Twistys - Malena Morgan starring at Dr. Morgan-Baller',
'uploader': 'mindgeek',
'duration': 480,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://vjav.com/videos/11761/yui-hatano-in-if-yui-was-my-girlfriend2/',
'md5': '6de5bc1f13bdfc3491a77f23edb1676f',
'info_dict': {
'id': '11761',
'display_id': 'yui-hatano-in-if-yui-was-my-girlfriend2',
'ext': 'mp4',
'title': 'Yui Hatano in If Yui Was My Girlfriend',
'uploader': 'Matheus69',
'duration': 3310,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://vjav.tube/videos/11761/yui-hatano-in-if-yui-was-my-girlfriend2/',
'md5': '6de5bc1f13bdfc3491a77f23edb1676f',
'info_dict': {
'id': '11761',
'display_id': 'yui-hatano-in-if-yui-was-my-girlfriend2',
'ext': 'mp4',
'title': 'Yui Hatano in If Yui Was My Girlfriend',
'uploader': 'Matheus69',
'duration': 3310,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://voyeurhit.com/videos/332875/charlotte-stokely-elle-alexandra-malena-morgan-lingerie/',
'md5': '12b4666e9c3e60dafe9182e5d12aae33',
'info_dict': {
'id': '332875',
'display_id': 'charlotte-stokely-elle-alexandra-malena-morgan-lingerie',
'ext': 'mp4',
'title': 'Charlotte Stokely, Elle Alexandra, Malena Morgan-Lingerie',
'uploader': 'Kyle Roberts',
'duration': 655,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}, {
'url': 'https://voyeurhit.tube/videos/332875/charlotte-stokely-elle-alexandra-malena-morgan-lingerie/',
'md5': '12b4666e9c3e60dafe9182e5d12aae33',
'info_dict': {
'id': '332875',
'display_id': 'charlotte-stokely-elle-alexandra-malena-morgan-lingerie',
'ext': 'mp4',
'title': 'Charlotte Stokely, Elle Alexandra, Malena Morgan-Lingerie',
'uploader': 'Kyle Roberts',
'duration': 655,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}]
_WEBPAGE_TESTS = [{
'url': 'https://pornzog.com/video/9125519/michelle-malone-dreamgirls-wild-wet-3/',
'info_dict': {
'id': '5119660',
'display_id': '5119660',
'ext': 'mp4',
'title': 'Michelle Malone - Dreamgirls - Wild Wet 3',
'uploader': 'FallenAngel12',
'duration': 402,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
}
}]
def _call_api(self, url, video_id, fatal=False, **kwargs):
content = self._download_json(url, video_id, fatal=fatal, **kwargs)
if traverse_obj(content, 'error'):
raise self._error_or_warning(ExtractorError(
f'Txxx said: {content["error"]}', expected=True), fatal=fatal)
return content or {}
def _real_extract(self, url):
video_id, host, display_id = self._match_valid_url(url).group('id', 'host', 'display_id')
headers = {'Referer': url, 'X-Requested-With': 'XMLHttpRequest'}
video_file = self._call_api(
f'https://{host}/api/videofile.php?video_id={video_id}&lifetime=8640000',
video_id, fatal=True, note='Downloading video file info', headers=headers)
slug = f'{int(1E6 * (int(video_id) // 1E6))}/{1000 * (int(video_id) // 1000)}'
video_info = self._call_api(
f'https://{host}/api/json/video/86400/{slug}/{video_id}.json',
video_id, note='Downloading video info', headers=headers)
return {
'id': video_id,
'display_id': display_id,
'title': traverse_obj(video_info, ('video', 'title')),
'uploader': traverse_obj(video_info, ('video', 'user', 'username')),
'duration': parse_duration(traverse_obj(video_info, ('video', 'duration'))),
'view_count': int_or_none(traverse_obj(video_info, ('video', 'statistics', 'viewed'))),
'like_count': int_or_none(traverse_obj(video_info, ('video', 'statistics', 'likes'))),
'dislike_count': int_or_none(traverse_obj(video_info, ('video', 'statistics', 'dislikes'))),
'age_limit': 18,
'formats': get_formats(host, video_file),
}
class PornTopIE(InfoExtractor):
_VALID_URL = r'https?://(?P<host>(?:www\.)?porntop\.com)/video/(?P<id>\d+)(?:/(?P<display_id>[^/?]+))?'
_TESTS = [{
'url': 'https://porntop.com/video/101569/triple-threat-with-lia-lor-malena-morgan-and-dani-daniels/',
'md5': '612ba7b3cb99455b382972948e200b08',
'info_dict': {
'id': '101569',
'display_id': 'triple-threat-with-lia-lor-malena-morgan-and-dani-daniels',
'ext': 'mp4',
'title': 'Triple Threat With Lia Lor, Malena Morgan And Dani Daniels',
'description': 'md5:285357d9d3a00ce5acb29f39f826dbf6',
'uploader': 'PatrickBush',
'duration': 480,
'view_count': int,
'like_count': int,
'dislike_count': int,
'age_limit': 18,
'timestamp': 1609455029,
'upload_date': '20201231',
'thumbnail': 'https://tn.porntop.com/media/tn/sources/101569_1.jpg',
}
}]
def _real_extract(self, url):
video_id, host, display_id = self._match_valid_url(url).group('id', 'host', 'display_id')
webpage = self._download_webpage(url, video_id)
json_ld = self._json_ld(self._search_json(
r'\bschemaJson\s*=', webpage, 'JSON-LD', video_id, transform_source=js_to_json,
contains_pattern='{[^<]+?VideoObject[^<]+};'), video_id, fatal=True)
video_file = self._parse_json(decode_base64(self._search_regex(
r"window\.initPlayer\(.*}}},\s*'(?P<json_b64c>[^']+)'",
webpage, 'json_urls', group='json_b64c')), video_id)
return merge_dicts({
'id': video_id,
'display_id': display_id,
'age_limit': 18,
'formats': get_formats(host, video_file),
}, json_ld)

View File

@ -3385,6 +3385,8 @@ def js_to_json(code, vars={}, *, strict=False):
if not strict: if not strict:
code = re.sub(r'new Date\((".+")\)', r'\g<1>', code) code = re.sub(r'new Date\((".+")\)', r'\g<1>', code)
code = re.sub(r'new \w+\((.*?)\)', lambda m: json.dumps(m.group(0)), code) code = re.sub(r'new \w+\((.*?)\)', lambda m: json.dumps(m.group(0)), code)
code = re.sub(r'parseInt\([^\d]+(\d+)[^\d]+\)', r'\1', code)
code = re.sub(r'\(function\([^)]*\)\s*\{[^}]*\}\s*\)\s*\(\s*(["\'][^)]*["\'])\s*\)', r'\1', code)
return re.sub(rf'''(?sx) return re.sub(rf'''(?sx)
{STRING_RE}| {STRING_RE}|