2013-06-23 22:27:16 +02:00
from . common import InfoExtractor
2016-05-05 21:02:54 +06:00
from . . utils import (
ExtractorError ,
2024-05-26 21:27:21 +02:00
determine_ext ,
2016-05-05 21:02:54 +06:00
int_or_none ,
2019-10-05 22:04:49 +07:00
merge_dicts ,
2016-05-05 21:02:54 +06:00
str_to_int ,
unified_strdate ,
2018-07-21 19:08:28 +07:00
url_or_none ,
2024-01-28 09:15:29 +07:00
urljoin ,
2016-05-05 21:02:54 +06:00
)
2013-06-23 22:27:16 +02:00
class RedTubeIE ( InfoExtractor ) :
2024-02-01 00:56:29 +07:00
_VALID_URL = r ' https?://(?:(?: \ w+ \ .)?redtube \ .com(?: \ .br)?/|embed \ .redtube \ .com/ \ ?.*? \ bid=)(?P<id>[0-9]+) '
2022-08-01 06:53:25 +05:30
_EMBED_REGEX = [ r ' <iframe[^>]+?src=[ " \' ](?P<url>(?:https?:)?//embed \ .redtube \ .com/ \ ?.*? \ bid= \ d+) ' ]
2016-11-06 21:39:29 +07:00
_TESTS = [ {
2021-12-07 14:59:54 +01:00
' url ' : ' https://www.redtube.com/38864951 ' ,
' md5 ' : ' 4fba70cbca3aefd25767ab4b523c9878 ' ,
2014-01-21 14:16:44 +01:00
' info_dict ' : {
2021-12-07 14:59:54 +01:00
' id ' : ' 38864951 ' ,
2014-11-26 12:52:45 +01:00
' ext ' : ' mp4 ' ,
2021-12-07 14:59:54 +01:00
' title ' : ' Public Sex on the Balcony in Freezing Paris! Amateur Couple LeoLulu ' ,
' description ' : ' Watch video Public Sex on the Balcony in Freezing Paris! Amateur Couple LeoLulu on Redtube, home of free Blowjob porn videos and Blonde sex movies online. Video length: (10:46) - Uploaded by leolulu - Verified User - Starring Pornstar: Leolulu ' ,
' upload_date ' : ' 20210111 ' ,
' timestamp ' : 1610343109 ,
' duration ' : 646 ,
2016-05-05 21:02:54 +06:00
' view_count ' : int ,
2015-03-25 20:09:01 +06:00
' age_limit ' : 18 ,
2021-12-07 14:59:54 +01:00
' thumbnail ' : r ' re:https:// \ wi-ph \ .rdtcdn \ .com/videos/.+/.+ \ .jpg ' ,
} ,
2016-11-06 21:39:29 +07:00
} , {
' url ' : ' http://embed.redtube.com/?bgcolor=000000&id=1443286 ' ,
' only_matching ' : True ,
2020-09-20 06:39:42 +02:00
} , {
' url ' : ' http://it.redtube.com/66418 ' ,
' only_matching ' : True ,
2024-02-01 00:56:29 +07:00
} , {
' url ' : ' https://www.redtube.com.br/103224331 ' ,
' only_matching ' : True ,
2016-11-06 21:39:29 +07:00
} ]
2013-06-23 22:27:16 +02:00
2013-10-04 11:41:57 +02:00
def _real_extract ( self , url ) :
2014-11-26 12:52:45 +01:00
video_id = self . _match_id ( url )
2016-11-06 21:39:29 +07:00
webpage = self . _download_webpage (
2023-11-14 22:40:38 +01:00
f ' https://www.redtube.com/ { video_id } ' , video_id )
2013-06-23 22:27:16 +02:00
2020-01-02 22:45:42 +07:00
ERRORS = (
( ( ' video-deleted-info ' , ' >This video has been removed ' ) , ' has been removed ' ) ,
( ( ' private_video_text ' , ' >This video is private ' , ' >Send a friend request to its owner to be able to view it ' ) , ' is private ' ) ,
)
for patterns , message in ERRORS :
if any ( p in webpage for p in patterns ) :
raise ExtractorError (
2024-06-12 01:09:58 +02:00
f ' Video { video_id } { message } ' , expected = True )
2015-03-25 20:08:35 +06:00
2019-10-05 22:04:49 +07:00
info = self . _search_json_ld ( webpage , video_id , default = { } )
if not info . get ( ' title ' ) :
info [ ' title ' ] = self . _html_search_regex (
2020-05-19 14:11:05 -05:00
( r ' <h( \ d)[^>]+class= " (?:video_title_text|videoTitle|video_title)[^ " ]* " >(?P<title>(?:(?! \ 1).)+)</h \ 1> ' ,
2024-06-12 01:09:58 +02:00
r ' (?:videoTitle|title) \ s*: \ s*([ " \' ])(?P<title>(?:(?! \ 1).)+) \ 1 ' ) ,
2019-10-05 22:04:49 +07:00
webpage , ' title ' , group = ' title ' ,
default = None ) or self . _og_search_title ( webpage )
2016-05-05 21:02:54 +06:00
formats = [ ]
sources = self . _parse_json (
self . _search_regex (
r ' sources \ s*: \ s*( { .+?}) ' , webpage , ' source ' , default = ' {} ' ) ,
video_id , fatal = False )
if sources and isinstance ( sources , dict ) :
for format_id , format_url in sources . items ( ) :
if format_url :
formats . append ( {
' url ' : format_url ,
' format_id ' : format_id ,
' height ' : int_or_none ( format_id ) ,
} )
2017-09-05 22:45:07 +07:00
medias = self . _parse_json (
self . _search_regex (
2020-05-20 03:39:41 +07:00
r ' mediaDefinition[ " \' ]? \ s*: \ s*( \ [.+?} \ s* \ ]) ' , webpage ,
2017-09-05 22:45:07 +07:00
' media definitions ' , default = ' {} ' ) ,
video_id , fatal = False )
2021-12-07 14:59:54 +01:00
for media in medias if isinstance ( medias , list ) else [ ] :
2024-01-28 09:15:29 +07:00
format_url = urljoin ( ' https://www.redtube.com ' , media . get ( ' videoUrl ' ) )
2021-12-07 14:59:54 +01:00
if not format_url :
continue
format_id = media . get ( ' format ' )
quality = media . get ( ' quality ' )
if format_id == ' hls ' or ( format_id == ' mp4 ' and not quality ) :
more_media = self . _download_json ( format_url , video_id , fatal = False )
else :
more_media = [ media ]
for media in more_media if isinstance ( more_media , list ) else [ ] :
2018-07-21 19:08:28 +07:00
format_url = url_or_none ( media . get ( ' videoUrl ' ) )
if not format_url :
2017-09-05 22:45:07 +07:00
continue
2021-12-07 14:59:54 +01:00
format_id = media . get ( ' format ' )
if format_id == ' hls ' or determine_ext ( format_url ) == ' m3u8 ' :
2020-05-20 03:39:41 +07:00
formats . extend ( self . _extract_m3u8_formats (
format_url , video_id , ' mp4 ' ,
2021-12-07 14:59:54 +01:00
entry_protocol = ' m3u8_native ' , m3u8_id = format_id or ' hls ' ,
2020-05-20 03:39:41 +07:00
fatal = False ) )
continue
2017-09-05 22:45:07 +07:00
format_id = media . get ( ' quality ' )
formats . append ( {
' url ' : format_url ,
2021-09-05 05:22:45 +05:30
' ext ' : ' mp4 ' ,
2017-09-05 22:45:07 +07:00
' format_id ' : format_id ,
' height ' : int_or_none ( format_id ) ,
} )
if not formats :
2016-05-05 21:02:54 +06:00
video_url = self . _html_search_regex (
r ' <source src= " (.+?) " type= " video/mp4 " > ' , webpage , ' video URL ' )
2021-09-05 05:22:45 +05:30
formats . append ( { ' url ' : video_url , ' ext ' : ' mp4 ' } )
2016-05-05 21:02:54 +06:00
thumbnail = self . _og_search_thumbnail ( webpage )
upload_date = unified_strdate ( self . _search_regex (
2019-10-05 22:04:49 +07:00
r ' <span[^>]+>(?:ADDED|Published on) ([^<]+)< ' ,
webpage , ' upload date ' , default = None ) )
2018-02-21 15:55:28 +01:00
duration = int_or_none ( self . _og_search_property (
' video:duration ' , webpage , default = None ) or self . _search_regex (
r ' videoDuration \ s*: \ s*( \ d+) ' , webpage , ' duration ' , default = None ) )
2016-05-05 21:02:54 +06:00
view_count = str_to_int ( self . _search_regex (
2018-02-02 22:32:53 +07:00
( r ' <div[^>]*>Views</div> \ s*<div[^>]*> \ s*([ \ d,.]+) ' ,
2019-10-05 22:04:49 +07:00
r ' <span[^>]*>VIEWS</span> \ s*</td> \ s*<td> \ s*([ \ d,.]+) ' ,
r ' <span[^>]+ \ bclass=[ " \' ]video_view_count[^>]*> \ s*([ \ d,.]+) ' ) ,
webpage , ' view count ' , default = None ) )
2016-05-05 21:02:54 +06:00
2013-10-06 16:39:35 +02:00
# No self-labeling, but they describe themselves as
# "Home of Videos Porno"
age_limit = 18
2019-10-05 22:04:49 +07:00
return merge_dicts ( info , {
2014-01-21 14:16:44 +01:00
' id ' : video_id ,
2014-11-26 12:52:45 +01:00
' ext ' : ' mp4 ' ,
2016-05-05 21:02:54 +06:00
' thumbnail ' : thumbnail ,
' upload_date ' : upload_date ,
' duration ' : duration ,
' view_count ' : view_count ,
2013-10-06 16:39:35 +02:00
' age_limit ' : age_limit ,
2016-05-05 21:02:54 +06:00
' formats ' : formats ,
2019-10-05 22:04:49 +07:00
} )