2013-10-23 16:31:53 +02:00
# encoding: utf-8
2013-06-23 20:18:21 +02:00
import json
import re
2013-07-29 13:12:09 +02:00
import itertools
2013-06-23 20:18:21 +02:00
from . common import InfoExtractor
from . . utils import (
compat_urllib_parse ,
compat_urllib_request ,
clean_html ,
get_element_by_attribute ,
ExtractorError ,
2013-10-23 14:38:03 +02:00
RegexNotFoundError ,
2013-06-23 20:18:21 +02:00
std_headers ,
2013-10-15 12:05:13 +02:00
unsmuggle_url ,
2013-06-23 20:18:21 +02:00
)
class VimeoIE ( InfoExtractor ) :
""" Information extractor for vimeo.com. """
# _VALID_URL matches Vimeo URLs
2013-12-06 22:01:41 +01:00
_VALID_URL = r ' (?P<proto>https?://)?(?:(?:www|(?P<player>player)) \ .)?vimeo(?P<pro>pro)? \ .com/(?:.*?/)?(?P<direct_link>play_redirect_hls \ ?clip_id=)?(?:videos?/)?(?P<id>[0-9]+)/?(?:[?].*)?(?:#.*)?$ '
2013-07-07 23:24:34 +02:00
_NETRC_MACHINE = ' vimeo '
2013-06-23 20:18:21 +02:00
IE_NAME = u ' vimeo '
2013-08-21 13:48:19 +02:00
_TESTS = [
{
2013-10-28 11:32:22 +01:00
u ' url ' : u ' http://vimeo.com/56015672#at=0 ' ,
2013-08-21 13:48:19 +02:00
u ' file ' : u ' 56015672.mp4 ' ,
2013-10-29 21:44:35 +01:00
u ' md5 ' : u ' 8879b6cc097e987f02484baf890129e5 ' ,
2013-08-21 13:48:19 +02:00
u ' info_dict ' : {
u " upload_date " : u " 20121220 " ,
u " description " : u " This is a test case for youtube-dl. \n For more information, see github.com/rg3/youtube-dl \n Test chars: \u2605 \" ' \u5e78 / \\ \u00e4 \u21ad \U0001d550 " ,
u " uploader_id " : u " user7108434 " ,
u " uploader " : u " Filippo Valsorda " ,
u " title " : u " youtube-dl test video - \u2605 \" ' \u5e78 / \\ \u00e4 \u21ad \U0001d550 " ,
} ,
} ,
{
u ' url ' : u ' http://vimeopro.com/openstreetmapus/state-of-the-map-us-2013/video/68093876 ' ,
u ' file ' : u ' 68093876.mp4 ' ,
u ' md5 ' : u ' 3b5ca6aa22b60dfeeadf50b72e44ed82 ' ,
u ' note ' : u ' Vimeo Pro video (#1197) ' ,
u ' info_dict ' : {
u ' uploader_id ' : u ' openstreetmapus ' ,
u ' uploader ' : u ' OpenStreetMap US ' ,
u ' title ' : u ' Andy Allan - Putting the Carto into OpenStreetMap Cartography ' ,
} ,
} ,
2013-09-03 10:48:56 +02:00
{
u ' url ' : u ' http://player.vimeo.com/video/54469442 ' ,
u ' file ' : u ' 54469442.mp4 ' ,
u ' md5 ' : u ' 619b811a4417aa4abe78dc653becf511 ' ,
u ' note ' : u ' Videos that embed the url in the player page ' ,
u ' info_dict ' : {
u ' title ' : u ' Kathy Sierra: Building the minimum Badass User, Business of Software ' ,
u ' uploader ' : u ' The BLN & Business of Software ' ,
} ,
2013-10-23 16:31:53 +02:00
} ,
{
u ' url ' : u ' http://vimeo.com/68375962 ' ,
u ' file ' : u ' 68375962.mp4 ' ,
u ' md5 ' : u ' aaf896bdb7ddd6476df50007a0ac0ae7 ' ,
u ' note ' : u ' Video protected with password ' ,
u ' info_dict ' : {
u ' title ' : u ' youtube-dl password protected test video ' ,
u ' upload_date ' : u ' 20130614 ' ,
u ' uploader_id ' : u ' user18948128 ' ,
u ' uploader ' : u ' Jaime Marquínez Ferrándiz ' ,
} ,
u ' params ' : {
u ' videopassword ' : u ' youtube-dl ' ,
} ,
} ,
2013-08-21 13:48:19 +02:00
]
2013-06-23 20:18:21 +02:00
2013-07-07 23:24:34 +02:00
def _login ( self ) :
( username , password ) = self . _get_login_info ( )
if username is None :
return
self . report_login ( )
login_url = ' https://vimeo.com/log_in '
webpage = self . _download_webpage ( login_url , None , False )
token = re . search ( r ' xsrft: \' (.*?) \' ' , webpage ) . group ( 1 )
data = compat_urllib_parse . urlencode ( { ' email ' : username ,
' password ' : password ,
' action ' : ' login ' ,
' service ' : ' vimeo ' ,
' token ' : token ,
} )
login_request = compat_urllib_request . Request ( login_url , data )
login_request . add_header ( ' Content-Type ' , ' application/x-www-form-urlencoded ' )
login_request . add_header ( ' Cookie ' , ' xsrft= %s ' % token )
self . _download_webpage ( login_request , None , False , u ' Wrong login info ' )
2013-06-23 20:18:21 +02:00
def _verify_video_password ( self , url , video_id , webpage ) :
2013-06-25 22:22:32 +02:00
password = self . _downloader . params . get ( ' videopassword ' , None )
2013-06-23 20:18:21 +02:00
if password is None :
2013-06-25 22:22:32 +02:00
raise ExtractorError ( u ' This video is protected by a password, use the --video-password option ' )
2013-06-23 20:18:21 +02:00
token = re . search ( r ' xsrft: \' (.*?) \' ' , webpage ) . group ( 1 )
data = compat_urllib_parse . urlencode ( { ' password ' : password ,
' token ' : token } )
# I didn't manage to use the password with https
if url . startswith ( ' https ' ) :
pass_url = url . replace ( ' https ' , ' http ' )
else :
pass_url = url
password_request = compat_urllib_request . Request ( pass_url + ' /password ' , data )
password_request . add_header ( ' Content-Type ' , ' application/x-www-form-urlencoded ' )
password_request . add_header ( ' Cookie ' , ' xsrft= %s ' % token )
self . _download_webpage ( password_request , video_id ,
u ' Verifying the password ' ,
u ' Wrong password ' )
2013-07-07 23:24:34 +02:00
def _real_initialize ( self ) :
self . _login ( )
2013-06-23 20:18:21 +02:00
def _real_extract ( self , url , new_video = True ) :
2013-10-15 12:05:13 +02:00
url , data = unsmuggle_url ( url )
headers = std_headers
if data is not None :
headers = headers . copy ( )
headers . update ( data )
2013-06-23 20:18:21 +02:00
# Extract ID from URL
mobj = re . match ( self . _VALID_URL , url )
if mobj is None :
raise ExtractorError ( u ' Invalid URL: %s ' % url )
video_id = mobj . group ( ' id ' )
2013-11-03 12:11:13 +01:00
if mobj . group ( ' pro ' ) or mobj . group ( ' player ' ) :
2013-08-21 13:48:19 +02:00
url = ' http://player.vimeo.com/video/ ' + video_id
2013-11-03 12:11:13 +01:00
else :
2013-06-23 20:18:21 +02:00
url = ' https://vimeo.com/ ' + video_id
# Retrieve video webpage to extract further information
2013-10-15 12:05:13 +02:00
request = compat_urllib_request . Request ( url , None , headers )
2013-06-23 20:18:21 +02:00
webpage = self . _download_webpage ( request , video_id )
# Now we begin extracting as much information as we can from what we
# retrieved. First we extract the information common to all extractors,
# and latter we extract those that are Vimeo specific.
self . report_extraction ( video_id )
# Extract the config JSON
try :
2013-10-23 16:31:53 +02:00
try :
config_url = self . _html_search_regex (
r ' data-config-url= " (.+?) " ' , webpage , u ' config URL ' )
config_json = self . _download_webpage ( config_url , video_id )
config = json . loads ( config_json )
except RegexNotFoundError :
# For pro videos or player.vimeo.com urls
2013-11-22 18:20:31 +01:00
config = self . _search_regex ( [ r ' = { config:( { .+?}),assets: ' , r ' (?:c|b)=( { .+?}); ' ] ,
2013-10-23 16:31:53 +02:00
webpage , u ' info section ' , flags = re . DOTALL )
config = json . loads ( config )
2013-10-23 11:38:51 +02:00
except Exception as e :
2013-06-23 20:18:21 +02:00
if re . search ( ' The creator of this video has not given you permission to embed it on this domain. ' , webpage ) :
raise ExtractorError ( u ' The author has restricted the access to this video, try with the " --referer " option ' )
2013-10-23 16:31:53 +02:00
if re . search ( ' <form[^>]+?id= " pw_form " ' , webpage ) is not None :
2013-06-23 20:18:21 +02:00
self . _verify_video_password ( url , video_id , webpage )
return self . _real_extract ( url )
else :
2013-10-23 11:38:51 +02:00
raise ExtractorError ( u ' Unable to extract info section ' ,
cause = e )
2013-06-23 20:18:21 +02:00
# Extract title
video_title = config [ " video " ] [ " title " ]
# Extract uploader and uploader_id
video_uploader = config [ " video " ] [ " owner " ] [ " name " ]
video_uploader_id = config [ " video " ] [ " owner " ] [ " url " ] . split ( ' / ' ) [ - 1 ] if config [ " video " ] [ " owner " ] [ " url " ] else None
# Extract video thumbnail
2013-09-03 10:48:56 +02:00
video_thumbnail = config [ " video " ] . get ( " thumbnail " )
if video_thumbnail is None :
_ , video_thumbnail = sorted ( ( int ( width ) , t_url ) for ( width , t_url ) in config [ " video " ] [ " thumbs " ] . items ( ) ) [ - 1 ]
2013-06-23 20:18:21 +02:00
# Extract video description
2013-09-03 11:11:36 +02:00
video_description = None
try :
video_description = get_element_by_attribute ( " itemprop " , " description " , webpage )
if video_description : video_description = clean_html ( video_description )
except AssertionError as err :
# On some pages like (http://player.vimeo.com/video/54469442) the
# html tags are not closed, python 2.6 cannot handle it
if err . args [ 0 ] == ' we should not get here! ' :
pass
else :
raise
2013-06-23 20:18:21 +02:00
# Extract upload date
video_upload_date = None
mobj = re . search ( r ' <meta itemprop= " dateCreated " content= " ( \ d {4} )-( \ d {2} )-( \ d {2} )T ' , webpage )
if mobj is not None :
video_upload_date = mobj . group ( 1 ) + mobj . group ( 2 ) + mobj . group ( 3 )
2013-12-06 13:03:08 +01:00
try :
view_count = int ( self . _search_regex ( r ' UserPlays:( \ d+) ' , webpage , u ' view count ' ) )
like_count = int ( self . _search_regex ( r ' UserLikes:( \ d+) ' , webpage , u ' like count ' ) )
comment_count = int ( self . _search_regex ( r ' UserComments:( \ d+) ' , webpage , u ' comment count ' ) )
except RegexNotFoundError :
# This info is only available in vimeo.com/{id} urls
view_count = None
like_count = None
comment_count = None
2013-06-23 20:18:21 +02:00
# Vimeo specific: extract request signature and timestamp
sig = config [ ' request ' ] [ ' signature ' ]
timestamp = config [ ' request ' ] [ ' timestamp ' ]
# Vimeo specific: extract video codec and quality information
# First consider quality, then codecs, then take everything
2013-07-05 18:10:57 +02:00
codecs = [ ( ' vp6 ' , ' flv ' ) , ( ' vp8 ' , ' flv ' ) , ( ' h264 ' , ' mp4 ' ) ]
2013-11-03 14:03:17 +01:00
files = { ' hd ' : [ ] , ' sd ' : [ ] , ' other ' : [ ] }
2013-09-03 10:48:56 +02:00
config_files = config [ " video " ] . get ( " files " ) or config [ " request " ] . get ( " files " )
2013-06-23 20:18:21 +02:00
for codec_name , codec_extension in codecs :
2013-07-05 18:10:57 +02:00
for quality in config_files . get ( codec_name , [ ] ) :
format_id = ' - ' . join ( ( codec_name , quality ) ) . lower ( )
key = quality if quality in files else ' other '
video_url = None
if isinstance ( config_files [ codec_name ] , dict ) :
file_info = config_files [ codec_name ] [ quality ]
video_url = file_info . get ( ' url ' )
2013-06-23 20:18:21 +02:00
else :
2013-07-05 18:10:57 +02:00
file_info = { }
if video_url is None :
video_url = " http://player.vimeo.com/play_redirect?clip_id= %s &sig= %s &time= %s &quality= %s &codecs= %s &type=moogaloop_local&embed_location= " \
% ( video_id , sig , timestamp , quality , codec_name . upper ( ) )
files [ key ] . append ( {
' ext ' : codec_extension ,
' url ' : video_url ,
' format_id ' : format_id ,
' width ' : file_info . get ( ' width ' ) ,
' height ' : file_info . get ( ' height ' ) ,
} )
formats = [ ]
for key in ( ' other ' , ' sd ' , ' hd ' ) :
formats + = files [ key ]
if len ( formats ) == 0 :
2013-06-23 20:18:21 +02:00
raise ExtractorError ( u ' No known codec found ' )
2013-11-03 12:11:13 +01:00
return {
2013-06-23 20:18:21 +02:00
' id ' : video_id ,
' uploader ' : video_uploader ,
' uploader_id ' : video_uploader_id ,
' upload_date ' : video_upload_date ,
' title ' : video_title ,
' thumbnail ' : video_thumbnail ,
' description ' : video_description ,
2013-07-05 18:10:57 +02:00
' formats ' : formats ,
2013-11-03 12:11:13 +01:00
' webpage_url ' : url ,
2013-12-06 13:03:08 +01:00
' view_count ' : view_count ,
' like_count ' : like_count ,
' comment_count ' : comment_count ,
2013-11-03 12:11:13 +01:00
}
2013-07-29 13:12:09 +02:00
class VimeoChannelIE ( InfoExtractor ) :
IE_NAME = u ' vimeo:channel '
_VALID_URL = r ' (?:https?://)?vimeo. \ com/channels/(?P<id>[^/]+) '
_MORE_PAGES_INDICATOR = r ' <a.+?rel= " next " '
2013-12-01 22:36:18 +01:00
_TITLE_RE = r ' <link rel= " alternate " [^>]+?title= " (.*?) " '
2013-07-29 13:12:09 +02:00
2013-12-06 21:47:32 +01:00
def _page_url ( self , base_url , pagenum ) :
return ' %s /videos/page: %d / ' % ( base_url , pagenum )
2013-12-06 22:01:41 +01:00
def _extract_list_title ( self , webpage ) :
return self . _html_search_regex ( self . _TITLE_RE , webpage , u ' list title ' )
2013-12-01 22:36:18 +01:00
def _extract_videos ( self , list_id , base_url ) :
2013-07-29 13:12:09 +02:00
video_ids = [ ]
for pagenum in itertools . count ( 1 ) :
2013-12-01 22:36:18 +01:00
webpage = self . _download_webpage (
2013-12-06 21:47:32 +01:00
self . _page_url ( base_url , pagenum ) , list_id ,
2013-12-01 22:36:18 +01:00
u ' Downloading page %s ' % pagenum )
2013-07-29 13:12:09 +02:00
video_ids . extend ( re . findall ( r ' id= " clip_( \ d+?) " ' , webpage ) )
if re . search ( self . _MORE_PAGES_INDICATOR , webpage , re . DOTALL ) is None :
break
entries = [ self . url_result ( ' http://vimeo.com/ %s ' % video_id , ' Vimeo ' )
for video_id in video_ids ]
return { ' _type ' : ' playlist ' ,
2013-12-01 22:36:18 +01:00
' id ' : list_id ,
2013-12-06 22:01:41 +01:00
' title ' : self . _extract_list_title ( webpage ) ,
2013-07-29 13:12:09 +02:00
' entries ' : entries ,
}
2013-12-01 22:36:18 +01:00
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
channel_id = mobj . group ( ' id ' )
return self . _extract_videos ( channel_id , ' http://vimeo.com/channels/ %s ' % channel_id )
class VimeoUserIE ( VimeoChannelIE ) :
IE_NAME = u ' vimeo:user '
_VALID_URL = r ' (?:https?://)?vimeo. \ com/(?P<name>[^/]+) '
_TITLE_RE = r ' <a[^>]+?class= " user " >([^<>]+?)</a> '
@classmethod
def suitable ( cls , url ) :
2013-12-06 22:01:41 +01:00
if VimeoChannelIE . suitable ( url ) or VimeoIE . suitable ( url ) or VimeoAlbumIE . suitable ( url ) or VimeoGroupsIE . suitable ( url ) :
2013-12-01 22:36:18 +01:00
return False
return super ( VimeoUserIE , cls ) . suitable ( url )
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
name = mobj . group ( ' name ' )
return self . _extract_videos ( name , ' http://vimeo.com/ %s ' % name )
2013-12-06 21:47:32 +01:00
class VimeoAlbumIE ( VimeoChannelIE ) :
IE_NAME = u ' vimeo:album '
_VALID_URL = r ' (?:https?://)?vimeo. \ com/album/(?P<id> \ d+) '
_TITLE_RE = r ' <header id= " page_header " > \ n \ s*<h1>(.*?)</h1> '
def _page_url ( self , base_url , pagenum ) :
return ' %s /page: %d / ' % ( base_url , pagenum )
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
album_id = mobj . group ( ' id ' )
return self . _extract_videos ( album_id , ' http://vimeo.com/album/ %s ' % album_id )
2013-12-06 22:01:41 +01:00
class VimeoGroupsIE ( VimeoAlbumIE ) :
IE_NAME = u ' vimeo:group '
_VALID_URL = r ' (?:https?://)?vimeo. \ com/groups/(?P<name>[^/]+) '
def _extract_list_title ( self , webpage ) :
return self . _og_search_title ( webpage )
def _real_extract ( self , url ) :
mobj = re . match ( self . _VALID_URL , url )
name = mobj . group ( ' name ' )
return self . _extract_videos ( name , ' http://vimeo.com/groups/ %s ' % name )