| 
									
										
										
										
											2016-10-02 13:39:18 +02:00
										 |  |  | # coding: utf-8 | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2014-11-02 11:23:40 +01:00
										 |  |  | from ..compat import ( | 
					
						
							| 
									
										
										
										
											2014-03-08 12:24:43 +01:00
										 |  |  |     compat_HTTPError, | 
					
						
							| 
									
										
										
										
											2016-06-19 03:56:01 +07:00
										 |  |  |     compat_str, | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |     compat_urlparse, | 
					
						
							| 
									
										
										
										
											2014-11-02 11:23:40 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							| 
									
										
										
										
											2016-01-09 03:06:09 +06:00
										 |  |  |     determine_ext, | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |     InAdvancePagedList, | 
					
						
							|  |  |  |     int_or_none, | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |     NO_DEFAULT, | 
					
						
							| 
									
										
										
										
											2013-10-23 14:38:03 +02:00
										 |  |  |     RegexNotFoundError, | 
					
						
							| 
									
										
										
										
											2015-11-20 20:33:49 +06:00
										 |  |  |     sanitized_Request, | 
					
						
							| 
									
										
										
										
											2015-02-18 00:27:57 +01:00
										 |  |  |     smuggle_url, | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |     std_headers, | 
					
						
							| 
									
										
										
										
											2015-03-06 18:16:56 +01:00
										 |  |  |     unified_strdate, | 
					
						
							| 
									
										
										
										
											2013-10-15 12:05:13 +02:00
										 |  |  |     unsmuggle_url, | 
					
						
							| 
									
										
										
										
											2014-04-24 14:44:27 +02:00
										 |  |  |     urlencode_postdata, | 
					
						
							| 
									
										
										
										
											2015-06-21 18:23:58 +08:00
										 |  |  |     unescapeHTML, | 
					
						
							| 
									
										
										
										
											2015-12-16 09:43:53 +01:00
										 |  |  |     parse_filesize, | 
					
						
							| 
									
										
										
										
											2016-06-19 03:56:01 +07:00
										 |  |  |     try_get, | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-22 03:17:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  | class VimeoBaseInfoExtractor(InfoExtractor): | 
					
						
							|  |  |  |     _NETRC_MACHINE = 'vimeo' | 
					
						
							|  |  |  |     _LOGIN_REQUIRED = False | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  |     _LOGIN_URL = 'https://vimeo.com/log_in' | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _login(self): | 
					
						
							|  |  |  |         (username, password) = self._get_login_info() | 
					
						
							|  |  |  |         if username is None: | 
					
						
							|  |  |  |             if self._LOGIN_REQUIRED: | 
					
						
							| 
									
										
										
										
											2014-05-04 22:27:56 +02:00
										 |  |  |                 raise ExtractorError('No login info available, needed for using %s.' % self.IE_NAME, expected=True) | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |             return | 
					
						
							|  |  |  |         self.report_login() | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  |         webpage = self._download_webpage(self._LOGIN_URL, None, False) | 
					
						
							| 
									
										
										
										
											2015-10-11 02:31:37 +06:00
										 |  |  |         token, vuid = self._extract_xsrft_and_vuid(webpage) | 
					
						
							| 
									
										
										
										
											2016-03-26 01:46:57 +06:00
										 |  |  |         data = urlencode_postdata({ | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  |             'action': 'login', | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |             'email': username, | 
					
						
							|  |  |  |             'password': password, | 
					
						
							|  |  |  |             'service': 'vimeo', | 
					
						
							|  |  |  |             'token': token, | 
					
						
							| 
									
										
										
										
											2016-03-26 01:46:57 +06:00
										 |  |  |         }) | 
					
						
							| 
									
										
										
										
											2015-11-20 20:33:49 +06:00
										 |  |  |         login_request = sanitized_Request(self._LOGIN_URL, data) | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |         login_request.add_header('Content-Type', 'application/x-www-form-urlencoded') | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  |         login_request.add_header('Referer', self._LOGIN_URL) | 
					
						
							| 
									
										
										
										
											2015-11-11 22:32:13 +06:00
										 |  |  |         self._set_vimeo_cookie('vuid', vuid) | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |         self._download_webpage(login_request, None, False, 'Wrong login info') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |     def _verify_video_password(self, url, video_id, webpage): | 
					
						
							|  |  |  |         password = self._downloader.params.get('videopassword') | 
					
						
							|  |  |  |         if password is None: | 
					
						
							|  |  |  |             raise ExtractorError('This video is protected by a password, use the --video-password option', expected=True) | 
					
						
							|  |  |  |         token, vuid = self._extract_xsrft_and_vuid(webpage) | 
					
						
							|  |  |  |         data = urlencode_postdata({ | 
					
						
							|  |  |  |             'password': password, | 
					
						
							|  |  |  |             'token': token, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         if url.startswith('http://'): | 
					
						
							|  |  |  |             # vimeo only supports https now, but the user can give an http url | 
					
						
							|  |  |  |             url = url.replace('http://', 'https://') | 
					
						
							|  |  |  |         password_request = sanitized_Request(url + '/password', data) | 
					
						
							|  |  |  |         password_request.add_header('Content-Type', 'application/x-www-form-urlencoded') | 
					
						
							|  |  |  |         password_request.add_header('Referer', url) | 
					
						
							|  |  |  |         self._set_vimeo_cookie('vuid', vuid) | 
					
						
							|  |  |  |         return self._download_webpage( | 
					
						
							|  |  |  |             password_request, video_id, | 
					
						
							|  |  |  |             'Verifying the password', 'Wrong password') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-11 02:31:37 +06:00
										 |  |  |     def _extract_xsrft_and_vuid(self, webpage): | 
					
						
							|  |  |  |         xsrft = self._search_regex( | 
					
						
							| 
									
										
										
										
											2016-02-12 03:16:26 +06:00
										 |  |  |             r'(?:(?P<q1>["\'])xsrft(?P=q1)\s*:|xsrft\s*[=:])\s*(?P<q>["\'])(?P<xsrft>.+?)(?P=q)', | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  |             webpage, 'login token', group='xsrft') | 
					
						
							| 
									
										
										
										
											2015-10-11 02:31:37 +06:00
										 |  |  |         vuid = self._search_regex( | 
					
						
							|  |  |  |             r'["\']vuid["\']\s*:\s*(["\'])(?P<vuid>.+?)\1', | 
					
						
							|  |  |  |             webpage, 'vuid', group='vuid') | 
					
						
							|  |  |  |         return xsrft, vuid | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-11 22:32:13 +06:00
										 |  |  |     def _set_vimeo_cookie(self, name, value): | 
					
						
							|  |  |  |         self._set_cookie('vimeo.com', name, value) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |     def _vimeo_sort_formats(self, formats): | 
					
						
							|  |  |  |         # Bitrates are completely broken. Single m3u8 may contain entries in kbps and bps | 
					
						
							|  |  |  |         # at the same time without actual units specified. This lead to wrong sorting. | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         self._sort_formats(formats, field_preference=('preference', 'height', 'width', 'fps', 'tbr', 'format_id')) | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _parse_config(self, config, video_id): | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         video_data = config['video'] | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         # Extract title | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         video_title = video_data['title'] | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Extract uploader, uploader_url and uploader_id | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         video_uploader = video_data.get('owner', {}).get('name') | 
					
						
							|  |  |  |         video_uploader_url = video_data.get('owner', {}).get('url') | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         video_uploader_id = video_uploader_url.split('/')[-1] if video_uploader_url else None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Extract video thumbnail | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         video_thumbnail = video_data.get('thumbnail') | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         if video_thumbnail is None: | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |             video_thumbs = video_data.get('thumbs') | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |             if video_thumbs and isinstance(video_thumbs, dict): | 
					
						
							|  |  |  |                 _, video_thumbnail = sorted((int(width if width.isdigit() else 0), t_url) for (width, t_url) in video_thumbs.items())[-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Extract video duration | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         video_duration = int_or_none(video_data.get('duration')) | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         formats = [] | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  |         config_files = video_data.get('files') or config['request'].get('files', {}) | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         for f in config_files.get('progressive', []): | 
					
						
							|  |  |  |             video_url = f.get('url') | 
					
						
							|  |  |  |             if not video_url: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             formats.append({ | 
					
						
							|  |  |  |                 'url': video_url, | 
					
						
							|  |  |  |                 'format_id': 'http-%s' % f.get('quality'), | 
					
						
							|  |  |  |                 'width': int_or_none(f.get('width')), | 
					
						
							|  |  |  |                 'height': int_or_none(f.get('height')), | 
					
						
							|  |  |  |                 'fps': int_or_none(f.get('fps')), | 
					
						
							|  |  |  |                 'tbr': int_or_none(f.get('bitrate')), | 
					
						
							|  |  |  |             }) | 
					
						
							| 
									
										
										
										
											2016-12-20 12:25:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for files_type in ('hls', 'dash'): | 
					
						
							|  |  |  |             for cdn_name, cdn_data in config_files.get(files_type, {}).get('cdns', {}).items(): | 
					
						
							|  |  |  |                 manifest_url = cdn_data.get('url') | 
					
						
							|  |  |  |                 if not manifest_url: | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 format_id = '%s-%s' % (files_type, cdn_name) | 
					
						
							|  |  |  |                 if files_type == 'hls': | 
					
						
							|  |  |  |                     formats.extend(self._extract_m3u8_formats( | 
					
						
							|  |  |  |                         manifest_url, video_id, 'mp4', | 
					
						
							|  |  |  |                         'm3u8_native', m3u8_id=format_id, | 
					
						
							|  |  |  |                         note='Downloading %s m3u8 information' % cdn_name, | 
					
						
							|  |  |  |                         fatal=False)) | 
					
						
							|  |  |  |                 elif files_type == 'dash': | 
					
						
							|  |  |  |                     formats.extend(self._extract_mpd_formats( | 
					
						
							|  |  |  |                         manifest_url.replace('/master.json', '/master.mpd'), video_id, format_id, | 
					
						
							|  |  |  |                         'Downloading %s MPD information' % cdn_name, | 
					
						
							|  |  |  |                         fatal=False)) | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         subtitles = {} | 
					
						
							|  |  |  |         text_tracks = config['request'].get('text_tracks') | 
					
						
							|  |  |  |         if text_tracks: | 
					
						
							|  |  |  |             for tt in text_tracks: | 
					
						
							|  |  |  |                 subtitles[tt['lang']] = [{ | 
					
						
							|  |  |  |                     'ext': 'vtt', | 
					
						
							|  |  |  |                     'url': 'https://vimeo.com' + tt['url'], | 
					
						
							|  |  |  |                 }] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             'title': video_title, | 
					
						
							|  |  |  |             'uploader': video_uploader, | 
					
						
							|  |  |  |             'uploader_id': video_uploader_id, | 
					
						
							|  |  |  |             'uploader_url': video_uploader_url, | 
					
						
							|  |  |  |             'thumbnail': video_thumbnail, | 
					
						
							|  |  |  |             'duration': video_duration, | 
					
						
							|  |  |  |             'formats': formats, | 
					
						
							|  |  |  |             'subtitles': subtitles, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-15 18:52:07 +01:00
										 |  |  | class VimeoIE(VimeoBaseInfoExtractor): | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |     """Information extractor for vimeo.com.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # _VALID_URL matches Vimeo URLs | 
					
						
							| 
									
										
										
										
											2013-12-22 03:17:56 +01:00
										 |  |  |     _VALID_URL = r'''(?x)
 | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |                     https?:// | 
					
						
							|  |  |  |                         (?: | 
					
						
							|  |  |  |                             (?: | 
					
						
							|  |  |  |                                 www| | 
					
						
							|  |  |  |                                 (?P<player>player) | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                             \. | 
					
						
							|  |  |  |                         )? | 
					
						
							|  |  |  |                         vimeo(?P<pro>pro)?\.com/ | 
					
						
							| 
									
										
										
										
											2016-06-23 01:28:36 +07:00
										 |  |  |                         (?!(?:channels|album)/[^/?#]+/?(?:$|[?#])|[^/]+/review/|ondemand/) | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |                         (?:.*?/)? | 
					
						
							|  |  |  |                         (?: | 
					
						
							|  |  |  |                             (?: | 
					
						
							|  |  |  |                                 play_redirect_hls| | 
					
						
							|  |  |  |                                 moogaloop\.swf)\?clip_id= | 
					
						
							|  |  |  |                             )? | 
					
						
							|  |  |  |                         (?:videos?/)? | 
					
						
							|  |  |  |                         (?P<id>[0-9]+) | 
					
						
							| 
									
										
										
										
											2016-04-18 21:40:28 +06:00
										 |  |  |                         (?:/[\da-f]+)? | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |                         /?(?:[?&].*)?(?:[#].*)?$ | 
					
						
							|  |  |  |                     '''
 | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo' | 
					
						
							| 
									
										
										
										
											2013-08-21 13:48:19 +02:00
										 |  |  |     _TESTS = [ | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             'url': 'http://vimeo.com/56015672#at=0', | 
					
						
							|  |  |  |             'md5': '8879b6cc097e987f02484baf890129e5', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							| 
									
										
										
										
											2014-02-17 11:44:24 +01:00
										 |  |  |                 'id': '56015672', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2015-10-11 02:22:42 +06:00
										 |  |  |                 'title': "youtube-dl test video - \u2605 \" ' \u5e78 / \\ \u00e4 \u21ad \U0001d550", | 
					
						
							|  |  |  |                 'description': 'md5:2d3305bad981a06ff79f027f19865021', | 
					
						
							|  |  |  |                 'upload_date': '20121220', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/user7108434', | 
					
						
							| 
									
										
										
										
											2015-10-11 02:22:42 +06:00
										 |  |  |                 'uploader_id': 'user7108434', | 
					
						
							|  |  |  |                 'uploader': 'Filippo Valsorda', | 
					
						
							|  |  |  |                 'duration': 10, | 
					
						
							| 
									
										
										
										
											2013-08-21 13:48:19 +02:00
										 |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             'url': 'http://vimeopro.com/openstreetmapus/state-of-the-map-us-2013/video/68093876', | 
					
						
							|  |  |  |             'md5': '3b5ca6aa22b60dfeeadf50b72e44ed82', | 
					
						
							|  |  |  |             'note': 'Vimeo Pro video (#1197)', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							| 
									
										
										
										
											2014-05-04 22:27:56 +02:00
										 |  |  |                 'id': '68093876', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/openstreetmapus', | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'uploader_id': 'openstreetmapus', | 
					
						
							|  |  |  |                 'uploader': 'OpenStreetMap US', | 
					
						
							|  |  |  |                 'title': 'Andy Allan - Putting the Carto into OpenStreetMap Cartography', | 
					
						
							| 
									
										
										
										
											2015-10-11 02:22:42 +06:00
										 |  |  |                 'description': 'md5:fd69a7b8d8c34a4e1d2ec2e4afd6ec30', | 
					
						
							| 
									
										
										
										
											2014-05-10 01:46:40 +07:00
										 |  |  |                 'duration': 1595, | 
					
						
							| 
									
										
										
										
											2013-08-21 13:48:19 +02:00
										 |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2013-09-03 10:48:56 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             'url': 'http://player.vimeo.com/video/54469442', | 
					
						
							|  |  |  |             'md5': '619b811a4417aa4abe78dc653becf511', | 
					
						
							|  |  |  |             'note': 'Videos that embed the url in the player page', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							| 
									
										
										
										
											2014-05-04 22:27:56 +02:00
										 |  |  |                 'id': '54469442', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2014-07-21 13:11:24 +02:00
										 |  |  |                 'title': 'Kathy Sierra: Building the minimum Badass User, Business of Software 2012', | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'uploader': 'The BLN & Business of Software', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/theblnbusinessofsoftware', | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'uploader_id': 'theblnbusinessofsoftware', | 
					
						
							| 
									
										
										
										
											2014-05-10 01:46:40 +07:00
										 |  |  |                 'duration': 3610, | 
					
						
							| 
									
										
										
										
											2014-09-29 22:23:21 +02:00
										 |  |  |                 'description': None, | 
					
						
							| 
									
										
										
										
											2013-09-03 10:48:56 +02:00
										 |  |  |             }, | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |         }, | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             'url': 'http://vimeo.com/68375962', | 
					
						
							|  |  |  |             'md5': 'aaf896bdb7ddd6476df50007a0ac0ae7', | 
					
						
							|  |  |  |             'note': 'Video protected with password', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							| 
									
										
										
										
											2014-05-04 22:27:56 +02:00
										 |  |  |                 'id': '68375962', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'title': 'youtube-dl password protected test video', | 
					
						
							|  |  |  |                 'upload_date': '20130614', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/user18948128', | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'uploader_id': 'user18948128', | 
					
						
							|  |  |  |                 'uploader': 'Jaime Marquínez Ferrándiz', | 
					
						
							| 
									
										
										
										
											2014-05-10 01:46:40 +07:00
										 |  |  |                 'duration': 10, | 
					
						
							| 
									
										
										
										
											2017-01-11 22:57:08 +07:00
										 |  |  |                 'description': 'md5:dca3ea23adb29ee387127bc4ddfce63f', | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |             }, | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             'params': { | 
					
						
							|  |  |  |                 'videopassword': 'youtube-dl', | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2014-08-04 00:04:47 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'http://vimeo.com/channels/keypeele/75629013', | 
					
						
							|  |  |  |             'md5': '2f86a05afe9d7abc0b9126d229bbe15d', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '75629013', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							|  |  |  |                 'title': 'Key & Peele: Terrorist Interrogation', | 
					
						
							|  |  |  |                 'description': 'md5:8678b246399b070816b12313e8b4eb5c', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/atencio', | 
					
						
							| 
									
										
										
										
											2014-08-04 00:04:47 +07:00
										 |  |  |                 'uploader_id': 'atencio', | 
					
						
							|  |  |  |                 'uploader': 'Peter Atencio', | 
					
						
							| 
									
										
										
										
											2015-03-06 18:16:56 +01:00
										 |  |  |                 'upload_date': '20130927', | 
					
						
							| 
									
										
										
										
											2014-08-04 00:04:47 +07:00
										 |  |  |                 'duration': 187, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2014-02-03 20:02:58 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'http://vimeo.com/76979871', | 
					
						
							|  |  |  |             'note': 'Video with subtitles', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '76979871', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							|  |  |  |                 'title': 'The New Vimeo Player (You Know, For Videos)', | 
					
						
							|  |  |  |                 'description': 'md5:2ec900bf97c3f389378a96aee11260ea', | 
					
						
							|  |  |  |                 'upload_date': '20131015', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/staff', | 
					
						
							| 
									
										
										
										
											2014-02-03 20:02:58 +07:00
										 |  |  |                 'uploader_id': 'staff', | 
					
						
							|  |  |  |                 'uploader': 'Vimeo Staff', | 
					
						
							| 
									
										
										
										
											2014-05-10 01:46:40 +07:00
										 |  |  |                 'duration': 62, | 
					
						
							| 
									
										
										
										
											2014-02-03 20:02:58 +07:00
										 |  |  |             } | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2014-10-17 15:49:16 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |             # from https://www.ouya.tv/game/Pier-Solar-and-the-Great-Architects/ | 
					
						
							|  |  |  |             'url': 'https://player.vimeo.com/video/98044508', | 
					
						
							|  |  |  |             'note': 'The js code contains assignments to the same variable as the config', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '98044508', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							|  |  |  |                 'title': 'Pier Solar OUYA Official Trailer', | 
					
						
							|  |  |  |                 'uploader': 'Tulio Gonçalves', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/user28849593', | 
					
						
							| 
									
										
										
										
											2014-10-17 15:49:16 +02:00
										 |  |  |                 'uploader_id': 'user28849593', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-12-16 17:00:17 +01:00
										 |  |  |         { | 
					
						
							|  |  |  |             # contains original format | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/33951933', | 
					
						
							| 
									
										
										
										
											2017-01-11 22:57:08 +07:00
										 |  |  |             'md5': '53c688fa95a55bf4b7293d37a89c5c53', | 
					
						
							| 
									
										
										
										
											2015-12-16 17:00:17 +01:00
										 |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '33951933', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							|  |  |  |                 'title': 'FOX CLASSICS - Forever Classic ID - A Full Minute', | 
					
						
							|  |  |  |                 'uploader': 'The DMCI', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/dmci', | 
					
						
							| 
									
										
										
										
											2015-12-16 17:00:17 +01:00
										 |  |  |                 'uploader_id': 'dmci', | 
					
						
							|  |  |  |                 'upload_date': '20111220', | 
					
						
							|  |  |  |                 'description': 'md5:ae23671e82d05415868f7ad1aec21147', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-06-23 00:51:36 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             # only available via https://vimeo.com/channels/tributes/6213729 and | 
					
						
							|  |  |  |             # not via https://vimeo.com/6213729 | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/channels/tributes/6213729', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '6213729', | 
					
						
							| 
									
										
										
										
											2017-01-11 22:57:08 +07:00
										 |  |  |                 'ext': 'mov', | 
					
						
							| 
									
										
										
										
											2016-06-23 00:51:36 +07:00
										 |  |  |                 'title': 'Vimeo Tribute: The Shining', | 
					
						
							|  |  |  |                 'uploader': 'Casey Donahue', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/caseydonahue', | 
					
						
							| 
									
										
										
										
											2016-06-23 00:51:36 +07:00
										 |  |  |                 'uploader_id': 'caseydonahue', | 
					
						
							|  |  |  |                 'upload_date': '20090821', | 
					
						
							|  |  |  |                 'description': 'md5:bdbf314014e58713e6e5b66eb252f4a6', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             'params': { | 
					
						
							|  |  |  |                 'skip_download': True, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             'expected_warnings': ['Unable to download JSON metadata'], | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-10-23 18:48:50 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             # redirects to ondemand extractor and should be passed throught it | 
					
						
							|  |  |  |             # for successful extraction | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/73445910', | 
					
						
							|  |  |  |             'info_dict': { | 
					
						
							|  |  |  |                 'id': '73445910', | 
					
						
							|  |  |  |                 'ext': 'mp4', | 
					
						
							|  |  |  |                 'title': 'The Reluctant Revolutionary', | 
					
						
							|  |  |  |                 'uploader': '10Ft Films', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 'uploader_url': r're:https?://(?:www\.)?vimeo\.com/tenfootfilms', | 
					
						
							| 
									
										
										
										
											2016-10-23 18:48:50 +07:00
										 |  |  |                 'uploader_id': 'tenfootfilms', | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             'params': { | 
					
						
							|  |  |  |                 'skip_download': True, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-06-23 00:51:36 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'http://vimeo.com/moogaloop.swf?clip_id=2539741', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-10-20 16:38:44 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/109815029', | 
					
						
							|  |  |  |             'note': 'Video not completely processed, "failed" seed status', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2015-11-19 22:31:16 +06:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/groups/travelhd/videos/22439234', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-06-23 01:28:36 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/album/2632481/video/79010983', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-01-09 03:07:29 +06:00
										 |  |  |         { | 
					
						
							|  |  |  |             # source file returns 403: Forbidden | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/7809605', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-04-18 21:40:28 +06:00
										 |  |  |         { | 
					
						
							|  |  |  |             'url': 'https://vimeo.com/160743502/abd0e13fb4', | 
					
						
							|  |  |  |             'only_matching': True, | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-08-21 13:48:19 +02:00
										 |  |  |     ] | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-12 21:48:45 +07:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _smuggle_referrer(url, referrer_url): | 
					
						
							|  |  |  |         return smuggle_url(url, {'http_headers': {'Referer': referrer_url}}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 18:23:58 +08:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2016-10-06 23:22:52 +07:00
										 |  |  |     def _extract_urls(url, webpage): | 
					
						
							|  |  |  |         urls = [] | 
					
						
							| 
									
										
										
										
											2015-06-21 18:23:58 +08:00
										 |  |  |         # Look for embedded (iframe) Vimeo player | 
					
						
							| 
									
										
										
										
											2016-10-06 23:22:52 +07:00
										 |  |  |         for mobj in re.finditer( | 
					
						
							| 
									
										
										
										
											2016-10-07 22:15:26 +07:00
										 |  |  |                 r'<iframe[^>]+?src=(["\'])(?P<url>(?:https?:)?//player\.vimeo\.com/video/.+?)\1', | 
					
						
							|  |  |  |                 webpage): | 
					
						
							| 
									
										
										
										
											2016-10-06 23:22:52 +07:00
										 |  |  |             urls.append(VimeoIE._smuggle_referrer(unescapeHTML(mobj.group('url')), url)) | 
					
						
							|  |  |  |         PLAIN_EMBED_RE = ( | 
					
						
							|  |  |  |             # Look for embedded (swf embed) Vimeo player | 
					
						
							|  |  |  |             r'<embed[^>]+?src=(["\'])(?P<url>(?:https?:)?//(?:www\.)?vimeo\.com/moogaloop\.swf.+?)\1', | 
					
						
							|  |  |  |             # Look more for non-standard embedded Vimeo player | 
					
						
							|  |  |  |             r'<video[^>]+src=(["\'])(?P<url>(?:https?:)?//(?:www\.)?vimeo\.com/[0-9]+)\1', | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         for embed_re in PLAIN_EMBED_RE: | 
					
						
							|  |  |  |             for mobj in re.finditer(embed_re, webpage): | 
					
						
							|  |  |  |                 urls.append(mobj.group('url')) | 
					
						
							|  |  |  |         return urls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _extract_url(url, webpage): | 
					
						
							|  |  |  |         urls = VimeoIE._extract_urls(url, webpage) | 
					
						
							|  |  |  |         return urls[0] if urls else None | 
					
						
							| 
									
										
										
										
											2015-06-21 18:23:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |     def _verify_player_video_password(self, url, video_id): | 
					
						
							| 
									
										
										
										
											2016-02-14 14:25:04 +06:00
										 |  |  |         password = self._downloader.params.get('videopassword') | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |         if password is None: | 
					
						
							|  |  |  |             raise ExtractorError('This video is protected by a password, use the --video-password option') | 
					
						
							| 
									
										
										
										
											2016-03-26 01:46:57 +06:00
										 |  |  |         data = urlencode_postdata({'password': password}) | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |         pass_url = url + '/check-password' | 
					
						
							| 
									
										
										
										
											2015-11-20 20:33:49 +06:00
										 |  |  |         password_request = sanitized_Request(pass_url, data) | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |         password_request.add_header('Content-Type', 'application/x-www-form-urlencoded') | 
					
						
							| 
									
										
										
										
											2016-04-22 00:02:39 +06:00
										 |  |  |         password_request.add_header('Referer', url) | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |         return self._download_json( | 
					
						
							|  |  |  |             password_request, video_id, | 
					
						
							| 
									
										
										
										
											2016-04-22 00:02:39 +06:00
										 |  |  |             'Verifying the password', 'Wrong password') | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-07 23:24:34 +02:00
										 |  |  |     def _real_initialize(self): | 
					
						
							|  |  |  |         self._login() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 20:43:16 +01:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2016-01-08 10:41:24 +06:00
										 |  |  |         url, data = unsmuggle_url(url, {}) | 
					
						
							| 
									
										
										
										
											2016-03-06 17:01:05 +08:00
										 |  |  |         headers = std_headers.copy() | 
					
						
							| 
									
										
										
										
											2016-01-08 10:41:24 +06:00
										 |  |  |         if 'http_headers' in data: | 
					
						
							|  |  |  |             headers.update(data['http_headers']) | 
					
						
							| 
									
										
										
										
											2014-08-25 09:35:37 +02:00
										 |  |  |         if 'Referer' not in headers: | 
					
						
							|  |  |  |             headers['Referer'] = url | 
					
						
							| 
									
										
										
										
											2013-10-15 12:05:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |         # Extract ID from URL | 
					
						
							|  |  |  |         mobj = re.match(self._VALID_URL, url) | 
					
						
							|  |  |  |         video_id = mobj.group('id') | 
					
						
							| 
									
										
										
										
											2014-09-29 22:23:21 +02:00
										 |  |  |         orig_url = url | 
					
						
							| 
									
										
										
										
											2013-11-03 12:11:13 +01:00
										 |  |  |         if mobj.group('pro') or mobj.group('player'): | 
					
						
							| 
									
										
										
										
											2015-03-06 22:39:05 +01:00
										 |  |  |             url = 'https://player.vimeo.com/video/' + video_id | 
					
						
							| 
									
										
										
										
											2016-06-23 00:51:36 +07:00
										 |  |  |         elif any(p in url for p in ('play_redirect_hls', 'moogaloop.swf')): | 
					
						
							| 
									
										
										
										
											2015-04-28 14:56:48 +02:00
										 |  |  |             url = 'https://vimeo.com/' + video_id | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Retrieve video webpage to extract further information | 
					
						
							| 
									
										
										
										
											2016-03-06 17:01:05 +08:00
										 |  |  |         request = sanitized_Request(url, headers=headers) | 
					
						
							| 
									
										
										
										
											2014-03-08 12:24:43 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2016-10-23 18:48:50 +07:00
										 |  |  |             webpage, urlh = self._download_webpage_handle(request, video_id) | 
					
						
							|  |  |  |             # Some URLs redirect to ondemand can't be extracted with | 
					
						
							|  |  |  |             # this extractor right away thus should be passed through | 
					
						
							|  |  |  |             # ondemand extractor (e.g. https://vimeo.com/73445910) | 
					
						
							|  |  |  |             if VimeoOndemandIE.suitable(urlh.geturl()): | 
					
						
							|  |  |  |                 return self.url_result(urlh.geturl(), VimeoOndemandIE.ie_key()) | 
					
						
							| 
									
										
										
										
											2014-03-08 12:24:43 +01:00
										 |  |  |         except ExtractorError as ee: | 
					
						
							|  |  |  |             if isinstance(ee.cause, compat_HTTPError) and ee.cause.code == 403: | 
					
						
							|  |  |  |                 errmsg = ee.cause.read() | 
					
						
							|  |  |  |                 if b'Because of its privacy settings, this video cannot be played here' in errmsg: | 
					
						
							|  |  |  |                     raise ExtractorError( | 
					
						
							|  |  |  |                         'Cannot download embed-only video without embedding ' | 
					
						
							|  |  |  |                         'URL. Please call youtube-dl with the URL of the page ' | 
					
						
							|  |  |  |                         'that embeds this video.', | 
					
						
							|  |  |  |                         expected=True) | 
					
						
							|  |  |  |             raise | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # 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) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-27 03:05:08 +06:00
										 |  |  |         vimeo_config = self._search_regex( | 
					
						
							| 
									
										
										
										
											2015-10-20 10:30:31 +02:00
										 |  |  |             r'vimeo\.config\s*=\s*(?:({.+?})|_extend\([^,]+,\s+({.+?})\));', webpage, | 
					
						
							| 
									
										
										
										
											2015-03-27 03:05:08 +06:00
										 |  |  |             'vimeo config', default=None) | 
					
						
							|  |  |  |         if vimeo_config: | 
					
						
							|  |  |  |             seed_status = self._parse_json(vimeo_config, video_id).get('seed_status', {}) | 
					
						
							|  |  |  |             if seed_status.get('state') == 'failed': | 
					
						
							|  |  |  |                 raise ExtractorError( | 
					
						
							| 
									
										
										
										
											2015-10-20 10:30:31 +02:00
										 |  |  |                     '%s said: %s' % (self.IE_NAME, seed_status['title']), | 
					
						
							| 
									
										
										
										
											2015-03-27 03:05:08 +06:00
										 |  |  |                     expected=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |         # Extract the config JSON | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 config_url = self._html_search_regex( | 
					
						
							| 
									
										
										
										
											2015-10-17 18:18:40 +02:00
										 |  |  |                     r' data-config-url="(.+?)"', webpage, | 
					
						
							|  |  |  |                     'config URL', default=None) | 
					
						
							|  |  |  |                 if not config_url: | 
					
						
							| 
									
										
										
										
											2015-10-17 22:48:14 +06:00
										 |  |  |                     # Sometimes new react-based page is served instead of old one that require | 
					
						
							|  |  |  |                     # different config URL extraction approach (see | 
					
						
							|  |  |  |                     # https://github.com/rg3/youtube-dl/pull/7209) | 
					
						
							| 
									
										
										
										
											2015-10-17 18:18:40 +02:00
										 |  |  |                     vimeo_clip_page_config = self._search_regex( | 
					
						
							|  |  |  |                         r'vimeo\.clip_page_config\s*=\s*({.+?});', webpage, | 
					
						
							|  |  |  |                         'vimeo clip page config') | 
					
						
							| 
									
										
										
										
											2015-10-17 22:48:14 +06:00
										 |  |  |                     config_url = self._parse_json( | 
					
						
							|  |  |  |                         vimeo_clip_page_config, video_id)['player']['config_url'] | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |                 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-12-10 20:28:12 +01:00
										 |  |  |                 # We try to find out to which variable is assigned the config dic | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |                 m_variable_name = re.search(r'(\w)\.video\.id', webpage) | 
					
						
							| 
									
										
										
										
											2013-12-10 20:28:12 +01:00
										 |  |  |                 if m_variable_name is not None: | 
					
						
							| 
									
										
										
										
											2014-10-17 15:49:16 +02:00
										 |  |  |                     config_re = r'%s=({[^}].+?});' % re.escape(m_variable_name.group(1)) | 
					
						
							| 
									
										
										
										
											2013-12-10 20:28:12 +01:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     config_re = [r' = {config:({.+?}),assets:', r'(?:[abc])=({.+?});'] | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 config = self._search_regex(config_re, webpage, 'info section', | 
					
						
							| 
									
										
										
										
											2014-11-23 21:39:15 +01:00
										 |  |  |                                             flags=re.DOTALL) | 
					
						
							| 
									
										
										
										
											2013-10-23 16:31:53 +02:00
										 |  |  |                 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): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 raise ExtractorError('The author has restricted the access to this video, try with the "--referer" option') | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |             if re.search(r'<form[^>]+?id="pw_form"', webpage) is not None: | 
					
						
							| 
									
										
										
										
											2016-01-08 10:41:24 +06:00
										 |  |  |                 if '_video_password_verified' in data: | 
					
						
							| 
									
										
										
										
											2015-02-18 00:27:57 +01:00
										 |  |  |                     raise ExtractorError('video password verification failed!') | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |                 self._verify_video_password(url, video_id, webpage) | 
					
						
							| 
									
										
										
										
											2015-02-18 00:27:57 +01:00
										 |  |  |                 return self._real_extract( | 
					
						
							|  |  |  |                     smuggle_url(url, {'_video_password_verified': 'verified'})) | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 raise ExtractorError('Unable to extract info section', | 
					
						
							| 
									
										
										
										
											2013-10-23 11:38:51 +02:00
										 |  |  |                                      cause=e) | 
					
						
							| 
									
										
										
										
											2014-01-06 23:35:24 +01:00
										 |  |  |         else: | 
					
						
							|  |  |  |             if config.get('view') == 4: | 
					
						
							| 
									
										
										
										
											2014-01-07 09:51:57 +01:00
										 |  |  |                 config = self._verify_player_video_password(url, video_id) | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-19 03:56:01 +07:00
										 |  |  |         def is_rented(): | 
					
						
							|  |  |  |             if '>You rented this title.<' in webpage: | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |             if config.get('user', {}).get('purchased'): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |             label = try_get( | 
					
						
							|  |  |  |                 config, lambda x: x['video']['vod']['purchase_options'][0]['label_string'], compat_str) | 
					
						
							|  |  |  |             if label and label.startswith('You rented this'): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if is_rented(): | 
					
						
							| 
									
										
										
										
											2016-01-08 10:41:24 +06:00
										 |  |  |             feature_id = config.get('video', {}).get('vod', {}).get('feature_id') | 
					
						
							|  |  |  |             if feature_id and not data.get('force_feature_id', False): | 
					
						
							|  |  |  |                 return self.url_result(smuggle_url( | 
					
						
							|  |  |  |                     'https://player.vimeo.com/player/%s' % feature_id, | 
					
						
							|  |  |  |                     {'force_feature_id': True}), 'Vimeo') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |         # Extract video description | 
					
						
							| 
									
										
										
										
											2014-09-29 22:23:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-29 04:58:29 +02:00
										 |  |  |         video_description = self._html_search_regex( | 
					
						
							| 
									
										
										
										
											2014-09-29 22:23:21 +02:00
										 |  |  |             r'(?s)<div\s+class="[^"]*description[^"]*"[^>]*>(.*?)</div>', | 
					
						
							|  |  |  |             webpage, 'description', default=None) | 
					
						
							|  |  |  |         if not video_description: | 
					
						
							|  |  |  |             video_description = self._html_search_meta( | 
					
						
							|  |  |  |                 'description', webpage, default=None) | 
					
						
							|  |  |  |         if not video_description and mobj.group('pro'): | 
					
						
							|  |  |  |             orig_webpage = self._download_webpage( | 
					
						
							|  |  |  |                 orig_url, video_id, | 
					
						
							|  |  |  |                 note='Downloading webpage for description', | 
					
						
							|  |  |  |                 fatal=False) | 
					
						
							|  |  |  |             if orig_webpage: | 
					
						
							|  |  |  |                 video_description = self._html_search_meta( | 
					
						
							|  |  |  |                     'description', orig_webpage, default=None) | 
					
						
							|  |  |  |         if not video_description and not mobj.group('player'): | 
					
						
							|  |  |  |             self._downloader.report_warning('Cannot find video description') | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Extract upload date | 
					
						
							|  |  |  |         video_upload_date = None | 
					
						
							| 
									
										
										
										
											2015-03-06 18:16:56 +01:00
										 |  |  |         mobj = re.search(r'<time[^>]+datetime="([^"]+)"', webpage) | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  |         if mobj is not None: | 
					
						
							| 
									
										
										
										
											2015-03-06 18:16:56 +01:00
										 |  |  |             video_upload_date = unified_strdate(mobj.group(1)) | 
					
						
							| 
									
										
										
										
											2013-06-23 20:18:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-06 13:03:08 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |             view_count = int(self._search_regex(r'UserPlays:(\d+)', webpage, 'view count')) | 
					
						
							|  |  |  |             like_count = int(self._search_regex(r'UserLikes:(\d+)', webpage, 'like count')) | 
					
						
							|  |  |  |             comment_count = int(self._search_regex(r'UserComments:(\d+)', webpage, 'comment count')) | 
					
						
							| 
									
										
										
										
											2013-12-06 13:03:08 +01:00
										 |  |  |         except RegexNotFoundError: | 
					
						
							|  |  |  |             # This info is only available in vimeo.com/{id} urls | 
					
						
							|  |  |  |             view_count = None | 
					
						
							|  |  |  |             like_count = None | 
					
						
							|  |  |  |             comment_count = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-05 09:10:57 -07:00
										 |  |  |         formats = [] | 
					
						
							| 
									
										
										
										
											2015-12-16 09:43:53 +01:00
										 |  |  |         download_request = sanitized_Request('https://vimeo.com/%s?action=load_download_config' % video_id, headers={ | 
					
						
							|  |  |  |             'X-Requested-With': 'XMLHttpRequest'}) | 
					
						
							|  |  |  |         download_data = self._download_json(download_request, video_id, fatal=False) | 
					
						
							|  |  |  |         if download_data: | 
					
						
							|  |  |  |             source_file = download_data.get('source_file') | 
					
						
							| 
									
										
										
										
											2016-01-09 03:06:09 +06:00
										 |  |  |             if isinstance(source_file, dict): | 
					
						
							|  |  |  |                 download_url = source_file.get('download_url') | 
					
						
							|  |  |  |                 if download_url and not source_file.get('is_cold') and not source_file.get('is_defrosting'): | 
					
						
							|  |  |  |                     source_name = source_file.get('public_name', 'Original') | 
					
						
							|  |  |  |                     if self._is_valid_url(download_url, video_id, '%s video' % source_name): | 
					
						
							| 
									
										
										
										
											2016-01-21 12:43:45 +01:00
										 |  |  |                         ext = source_file.get('extension', determine_ext(download_url)).lower() | 
					
						
							| 
									
										
										
										
											2016-01-09 03:06:09 +06:00
										 |  |  |                         formats.append({ | 
					
						
							|  |  |  |                             'url': download_url, | 
					
						
							|  |  |  |                             'ext': ext, | 
					
						
							|  |  |  |                             'width': int_or_none(source_file.get('width')), | 
					
						
							|  |  |  |                             'height': int_or_none(source_file.get('height')), | 
					
						
							|  |  |  |                             'filesize': parse_filesize(source_file.get('size')), | 
					
						
							|  |  |  |                             'format_id': source_name, | 
					
						
							|  |  |  |                             'preference': 1, | 
					
						
							|  |  |  |                         }) | 
					
						
							| 
									
										
										
										
											2014-02-03 20:02:58 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         info_dict = self._parse_config(config, video_id) | 
					
						
							|  |  |  |         formats.extend(info_dict['formats']) | 
					
						
							|  |  |  |         self._vimeo_sort_formats(formats) | 
					
						
							|  |  |  |         info_dict.update({ | 
					
						
							| 
									
										
										
										
											2014-02-03 20:24:11 +07:00
										 |  |  |             'id': video_id, | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |             'formats': formats, | 
					
						
							| 
									
										
										
										
											2014-02-03 20:24:11 +07:00
										 |  |  |             'upload_date': video_upload_date, | 
					
						
							|  |  |  |             'description': video_description, | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return info_dict | 
					
						
							| 
									
										
										
										
											2013-07-29 13:12:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  | class VimeoOndemandIE(VimeoBaseInfoExtractor): | 
					
						
							| 
									
										
										
										
											2016-03-10 00:52:52 +06:00
										 |  |  |     IE_NAME = 'vimeo:ondemand' | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |     _VALID_URL = r'https?://(?:www\.)?vimeo\.com/ondemand/(?P<id>[^/?#&]+)' | 
					
						
							|  |  |  |     _TESTS = [{ | 
					
						
							|  |  |  |         # ondemand video not available via https://vimeo.com/id | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/ondemand/20704', | 
					
						
							|  |  |  |         'md5': 'c424deda8c7f73c1dfb3edd7630e2f35', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '105442900', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							|  |  |  |             'title': 'המעבדה - במאי יותם פלדמן', | 
					
						
							|  |  |  |             'uploader': 'גם סרטים', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |             'uploader_url': r're:https?://(?:www\.)?vimeo\.com/gumfilms', | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |             'uploader_id': 'gumfilms', | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2017-01-11 22:51:03 +07:00
										 |  |  |         'params': { | 
					
						
							|  |  |  |             'format': 'best[protocol=https]', | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-09-12 21:48:45 +07:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         # requires Referer to be passed along with og:video:url | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/ondemand/36938/126682985', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '126682985', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							|  |  |  |             'title': 'Rävlock, rätt läte på rätt plats', | 
					
						
							|  |  |  |             'uploader': 'Lindroth & Norin', | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |             'uploader_url': r're:https?://(?:www\.)?vimeo\.com/user14430847', | 
					
						
							| 
									
										
										
										
											2016-09-12 21:48:45 +07:00
										 |  |  |             'uploader_id': 'user14430847', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'params': { | 
					
						
							|  |  |  |             'skip_download': True, | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/ondemand/nazmaalik', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							|  |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/ondemand/141692381', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							|  |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/ondemand/thelastcolony/150274832', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							|  |  |  |     }] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         video_id = self._match_id(url) | 
					
						
							|  |  |  |         webpage = self._download_webpage(url, video_id) | 
					
						
							| 
									
										
										
										
											2016-09-12 21:48:45 +07:00
										 |  |  |         return self.url_result( | 
					
						
							|  |  |  |             # Some videos require Referer to be passed along with og:video:url | 
					
						
							|  |  |  |             # similarly to generic vimeo embeds (e.g. | 
					
						
							|  |  |  |             # https://vimeo.com/ondemand/36938/126682985). | 
					
						
							|  |  |  |             VimeoIE._smuggle_referrer(self._og_search_video_url(webpage), url), | 
					
						
							|  |  |  |             VimeoIE.ie_key()) | 
					
						
							| 
									
										
										
										
											2016-03-10 00:51:07 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  | class VimeoChannelIE(VimeoBaseInfoExtractor): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo:channel' | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/channels/(?P<id>[^/?#]+)/?(?:$|[?#])' | 
					
						
							| 
									
										
										
										
											2013-07-29 13:12:09 +02:00
										 |  |  |     _MORE_PAGES_INDICATOR = r'<a.+?rel="next"' | 
					
						
							| 
									
										
										
										
											2015-08-10 23:58:01 +06:00
										 |  |  |     _TITLE = None | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  |     _TITLE_RE = r'<link rel="alternate"[^>]+?title="(.*?)"' | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         'url': 'https://vimeo.com/channels/tributes', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-02-18 00:33:31 +01:00
										 |  |  |             'id': 'tributes', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |             'title': 'Vimeo Tributes', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_mincount': 25, | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											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): | 
					
						
							| 
									
										
										
										
											2015-08-10 23:58:01 +06:00
										 |  |  |         return self._TITLE or self._html_search_regex(self._TITLE_RE, webpage, 'list title') | 
					
						
							| 
									
										
										
										
											2013-12-06 22:01:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |     def _login_list_password(self, page_url, list_id, webpage): | 
					
						
							|  |  |  |         login_form = self._search_regex( | 
					
						
							|  |  |  |             r'(?s)<form[^>]+?id="pw_form"(.*?)</form>', | 
					
						
							|  |  |  |             webpage, 'login form', default=None) | 
					
						
							|  |  |  |         if not login_form: | 
					
						
							|  |  |  |             return webpage | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-14 14:25:04 +06:00
										 |  |  |         password = self._downloader.params.get('videopassword') | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |         if password is None: | 
					
						
							|  |  |  |             raise ExtractorError('This album is protected by a password, use the --video-password option', expected=True) | 
					
						
							| 
									
										
										
										
											2015-07-14 22:36:30 +06:00
										 |  |  |         fields = self._hidden_inputs(login_form) | 
					
						
							| 
									
										
										
										
											2015-10-11 02:31:37 +06:00
										 |  |  |         token, vuid = self._extract_xsrft_and_vuid(webpage) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |         fields['token'] = token | 
					
						
							|  |  |  |         fields['password'] = password | 
					
						
							| 
									
										
										
										
											2016-03-26 01:46:57 +06:00
										 |  |  |         post = urlencode_postdata(fields) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |         password_path = self._search_regex( | 
					
						
							|  |  |  |             r'action="([^"]+)"', login_form, 'password URL') | 
					
						
							|  |  |  |         password_url = compat_urlparse.urljoin(page_url, password_path) | 
					
						
							| 
									
										
										
										
											2015-11-20 20:33:49 +06:00
										 |  |  |         password_request = sanitized_Request(password_url, post) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |         password_request.add_header('Content-type', 'application/x-www-form-urlencoded') | 
					
						
							| 
									
										
										
										
											2015-11-11 22:32:13 +06:00
										 |  |  |         self._set_vimeo_cookie('vuid', vuid) | 
					
						
							|  |  |  |         self._set_vimeo_cookie('xsrft', token) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return self._download_webpage( | 
					
						
							|  |  |  |             password_request, list_id, | 
					
						
							|  |  |  |             'Verifying the password', 'Wrong password') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 21:29:32 +06:00
										 |  |  |     def _title_and_entries(self, list_id, base_url): | 
					
						
							| 
									
										
										
										
											2013-07-29 13:12:09 +02:00
										 |  |  |         for pagenum in itertools.count(1): | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |             page_url = self._page_url(base_url, pagenum) | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  |             webpage = self._download_webpage( | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |                 page_url, list_id, | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |                 'Downloading page %s' % pagenum) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if pagenum == 1: | 
					
						
							|  |  |  |                 webpage = self._login_list_password(page_url, list_id, webpage) | 
					
						
							| 
									
										
										
										
											2015-11-19 21:29:32 +06:00
										 |  |  |                 yield self._extract_list_title(webpage) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-23 01:28:36 +07:00
										 |  |  |             # Try extracting href first since not all videos are available via | 
					
						
							|  |  |  |             # short https://vimeo.com/id URL (e.g. https://vimeo.com/channels/tributes/6213729) | 
					
						
							|  |  |  |             clips = re.findall( | 
					
						
							|  |  |  |                 r'id="clip_(\d+)"[^>]*>\s*<a[^>]+href="(/(?:[^/]+/)*\1)', webpage) | 
					
						
							|  |  |  |             if clips: | 
					
						
							|  |  |  |                 for video_id, video_url in clips: | 
					
						
							|  |  |  |                     yield self.url_result( | 
					
						
							|  |  |  |                         compat_urlparse.urljoin(base_url, video_url), | 
					
						
							|  |  |  |                         VimeoIE.ie_key(), video_id=video_id) | 
					
						
							|  |  |  |             # More relaxed fallback | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 for video_id in re.findall(r'id=["\']clip_(\d+)', webpage): | 
					
						
							|  |  |  |                     yield self.url_result( | 
					
						
							|  |  |  |                         'https://vimeo.com/%s' % video_id, | 
					
						
							|  |  |  |                         VimeoIE.ie_key(), video_id=video_id) | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-29 13:12:09 +02:00
										 |  |  |             if re.search(self._MORE_PAGES_INDICATOR, webpage, re.DOTALL) is None: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 21:29:32 +06:00
										 |  |  |     def _extract_videos(self, list_id, base_url): | 
					
						
							|  |  |  |         title_and_entries = self._title_and_entries(list_id, base_url) | 
					
						
							|  |  |  |         list_title = next(title_and_entries) | 
					
						
							|  |  |  |         return self.playlist_result(title_and_entries, list_id, list_title) | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         mobj = re.match(self._VALID_URL, url) | 
					
						
							| 
									
										
										
										
											2014-05-04 22:27:56 +02:00
										 |  |  |         channel_id = mobj.group('id') | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         return self._extract_videos(channel_id, 'https://vimeo.com/channels/%s' % channel_id) | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class VimeoUserIE(VimeoChannelIE): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo:user' | 
					
						
							| 
									
										
										
										
											2015-08-11 01:17:41 +06:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/(?!(?:[0-9]+|watchlater)(?:$|[?#/]))(?P<name>[^/]+)(?:/videos|[#?]|$)' | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  |     _TITLE_RE = r'<a[^>]+?class="user">([^<>]+?)</a>' | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         'url': 'https://vimeo.com/nkistudio/videos', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'title': 'Nki', | 
					
						
							| 
									
										
										
										
											2015-02-18 00:33:31 +01:00
										 |  |  |             'id': 'nkistudio', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'playlist_mincount': 66, | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-12-01 22:36:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         mobj = re.match(self._VALID_URL, url) | 
					
						
							|  |  |  |         name = mobj.group('name') | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         return self._extract_videos(name, 'https://vimeo.com/%s' % name) | 
					
						
							| 
									
										
										
										
											2013-12-06 21:47:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class VimeoAlbumIE(VimeoChannelIE): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo:album' | 
					
						
							| 
									
										
										
										
											2016-06-23 01:56:58 +07:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/album/(?P<id>\d+)(?:$|[?#]|/(?!video))' | 
					
						
							| 
									
										
										
										
											2013-12-06 21:47:32 +01:00
										 |  |  |     _TITLE_RE = r'<header id="page_header">\n\s*<h1>(.*?)</h1>' | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-03-06 22:16:26 +01:00
										 |  |  |         'url': 'https://vimeo.com/album/2632481', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-02-18 00:33:31 +01:00
										 |  |  |             'id': '2632481', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |             'title': 'Staff Favorites: November 2013', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_mincount': 13, | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'note': 'Password-protected album', | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/album/3253534', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'title': 'test', | 
					
						
							|  |  |  |             'id': '3253534', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_count': 1, | 
					
						
							|  |  |  |         'params': { | 
					
						
							|  |  |  |             'videopassword': 'youtube-dl', | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-06-23 01:56:58 +07:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/album/2632481/sort:plays/format:thumbnail', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							| 
									
										
										
										
											2016-06-23 02:00:03 +07:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         # TODO: respect page number | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/album/2632481/page:2/sort:plays/format:thumbnail', | 
					
						
							|  |  |  |         'only_matching': True, | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-12-06 21:47:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _page_url(self, base_url, pagenum): | 
					
						
							|  |  |  |         return '%s/page:%d/' % (base_url, pagenum) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2015-02-10 04:53:21 +01:00
										 |  |  |         album_id = self._match_id(url) | 
					
						
							| 
									
										
										
										
											2015-03-06 22:16:26 +01:00
										 |  |  |         return self._extract_videos(album_id, 'https://vimeo.com/album/%s' % album_id) | 
					
						
							| 
									
										
										
										
											2013-12-06 22:01:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class VimeoGroupsIE(VimeoAlbumIE): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo:group' | 
					
						
							| 
									
										
										
										
											2015-11-19 22:30:58 +06:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/groups/(?P<name>[^/]+)(?:/(?!videos?/\d+)|$)' | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         'url': 'https://vimeo.com/groups/rolexawards', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-02-18 00:33:31 +01:00
										 |  |  |             'id': 'rolexawards', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |             'title': 'Rolex Awards for Enterprise', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_mincount': 73, | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											2013-12-06 22:01:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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') | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         return self._extract_videos(name, 'https://vimeo.com/groups/%s' % name) | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  | class VimeoReviewIE(VimeoBaseInfoExtractor): | 
					
						
							| 
									
										
										
										
											2014-01-06 23:38:16 +01:00
										 |  |  |     IE_NAME = 'vimeo:review' | 
					
						
							|  |  |  |     IE_DESC = 'Review pages on vimeo' | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/[^/]+/review/(?P<id>[^/]+)' | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  |         'url': 'https://vimeo.com/user21297594/review/75524534/3c257a1b5d', | 
					
						
							|  |  |  |         'md5': 'c507a72f780cacc12b2248bb4006d253', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-02-01 12:12:27 +01:00
										 |  |  |             'id': '75524534', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  |             'title': "DICK HARDWICK 'Comedian'", | 
					
						
							|  |  |  |             'uploader': 'Richard Hardwick', | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |             'uploader_id': 'user21297594', | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'note': 'video player needs Referer', | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |         'url': 'https://vimeo.com/user22258446/review/91613211/13f927e053', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |         'md5': '6295fdab8f4bf6a002d058b2c6dce276', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '91613211', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							| 
									
										
										
										
											2014-10-27 00:13:40 +01:00
										 |  |  |             'title': 're:(?i)^Death by dogma versus assembling agile . Sander Hoogendoorn', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |             'uploader': 'DevWeek Events', | 
					
						
							|  |  |  |             'duration': 2773, | 
					
						
							| 
									
										
										
										
											2017-01-02 20:08:07 +08:00
										 |  |  |             'thumbnail': r're:^https?://.*\.jpg$', | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |             'uploader_id': 'user22258446', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |     }, { | 
					
						
							|  |  |  |         'note': 'Password protected', | 
					
						
							|  |  |  |         'url': 'https://vimeo.com/user37284429/review/138823582/c4d865efde', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '138823582', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							|  |  |  |             'title': 'EFFICIENT PICKUP MASTERCLASS MODULE 1', | 
					
						
							|  |  |  |             'uploader': 'TMB', | 
					
						
							|  |  |  |             'uploader_id': 'user37284429', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'params': { | 
					
						
							|  |  |  |             'videopassword': 'holygrail', | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-10-12 01:48:06 +08:00
										 |  |  |         'skip': 'video gone', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:13:42 +02:00
										 |  |  |     }] | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |     def _real_initialize(self): | 
					
						
							|  |  |  |         self._login() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_config_url(self, webpage_url, video_id, video_password_verified=False): | 
					
						
							|  |  |  |         webpage = self._download_webpage(webpage_url, video_id) | 
					
						
							| 
									
										
										
										
											2016-10-12 01:48:06 +08:00
										 |  |  |         data = self._parse_json(self._search_regex( | 
					
						
							|  |  |  |             r'window\s*=\s*_extend\(window,\s*({.+?})\);', webpage, 'data', | 
					
						
							|  |  |  |             default=NO_DEFAULT if video_password_verified else '{}'), video_id) | 
					
						
							|  |  |  |         config_url = data.get('vimeo_esi', {}).get('config', {}).get('configUrl') | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |         if config_url is None: | 
					
						
							|  |  |  |             self._verify_video_password(webpage_url, video_id, webpage) | 
					
						
							|  |  |  |             config_url = self._get_config_url( | 
					
						
							|  |  |  |                 webpage_url, video_id, video_password_verified=True) | 
					
						
							|  |  |  |         return config_url | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 17:31:47 +01:00
										 |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         video_id = self._match_id(url) | 
					
						
							| 
									
										
										
										
											2016-06-24 16:48:37 +08:00
										 |  |  |         config_url = self._get_config_url(url, video_id) | 
					
						
							|  |  |  |         config = self._download_json(config_url, video_id) | 
					
						
							| 
									
										
										
										
											2016-06-11 21:35:08 +08:00
										 |  |  |         info_dict = self._parse_config(config, video_id) | 
					
						
							|  |  |  |         self._vimeo_sort_formats(info_dict['formats']) | 
					
						
							|  |  |  |         info_dict['id'] = video_id | 
					
						
							|  |  |  |         return info_dict | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 23:35:08 +06:00
										 |  |  | class VimeoWatchLaterIE(VimeoChannelIE): | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |     IE_NAME = 'vimeo:watchlater' | 
					
						
							|  |  |  |     IE_DESC = 'Vimeo watch later list, "vimeowatchlater" keyword (requires authentication)' | 
					
						
							| 
									
										
										
										
											2015-08-10 23:58:01 +06:00
										 |  |  |     _VALID_URL = r'https://vimeo\.com/(?:home/)?watchlater|:vimeowatchlater' | 
					
						
							|  |  |  |     _TITLE = 'Watch Later' | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |     _LOGIN_REQUIRED = True | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |     _TESTS = [{ | 
					
						
							| 
									
										
										
										
											2015-08-10 23:58:01 +06:00
										 |  |  |         'url': 'https://vimeo.com/watchlater', | 
					
						
							| 
									
										
										
										
											2014-08-27 11:36:01 +02:00
										 |  |  |         'only_matching': True, | 
					
						
							|  |  |  |     }] | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _real_initialize(self): | 
					
						
							|  |  |  |         self._login() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _page_url(self, base_url, pagenum): | 
					
						
							|  |  |  |         url = '%s/page:%d/' % (base_url, pagenum) | 
					
						
							| 
									
										
										
										
											2015-11-20 20:33:49 +06:00
										 |  |  |         request = sanitized_Request(url) | 
					
						
							| 
									
										
										
										
											2014-04-24 21:51:20 +02:00
										 |  |  |         # Set the header to get a partial html page with the ids, | 
					
						
							|  |  |  |         # the normal page doesn't contain them. | 
					
						
							|  |  |  |         request.add_header('X-Requested-With', 'XMLHttpRequest') | 
					
						
							|  |  |  |         return request | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2015-08-10 23:58:01 +06:00
										 |  |  |         return self._extract_videos('watchlater', 'https://vimeo.com/watchlater') | 
					
						
							| 
									
										
										
										
											2014-09-28 12:14:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class VimeoLikesIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |     _VALID_URL = r'https://(?:www\.)?vimeo\.com/user(?P<id>[0-9]+)/likes/?(?:$|[?#]|sort:)' | 
					
						
							| 
									
										
										
										
											2014-09-28 12:14:16 +02:00
										 |  |  |     IE_NAME = 'vimeo:likes' | 
					
						
							|  |  |  |     IE_DESC = 'Vimeo user likes' | 
					
						
							|  |  |  |     _TEST = { | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |         'url': 'https://vimeo.com/user755559/likes/', | 
					
						
							|  |  |  |         'playlist_mincount': 293, | 
					
						
							| 
									
										
										
										
											2016-02-14 15:37:17 +06:00
										 |  |  |         'info_dict': { | 
					
						
							| 
									
										
										
										
											2015-02-18 00:33:31 +01:00
										 |  |  |             'id': 'user755559_likes', | 
					
						
							| 
									
										
										
										
											2016-02-14 15:37:17 +06:00
										 |  |  |             'description': 'See all the videos urza likes', | 
					
						
							|  |  |  |             'title': 'Videos urza likes', | 
					
						
							| 
									
										
										
										
											2014-09-28 12:14:16 +02:00
										 |  |  |         }, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         user_id = self._match_id(url) | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |         webpage = self._download_webpage(url, user_id) | 
					
						
							|  |  |  |         page_count = self._int( | 
					
						
							|  |  |  |             self._search_regex( | 
					
						
							|  |  |  |                 r'''(?x)<li><a\s+href="[^"]+"\s+data-page="([0-9]+)">
 | 
					
						
							|  |  |  |                     .*?</a></li>\s*<li\s+class="pagination_next"> | 
					
						
							|  |  |  |                 ''', webpage, 'page count'),
 | 
					
						
							|  |  |  |             'page count', fatal=True) | 
					
						
							|  |  |  |         PAGE_SIZE = 12 | 
					
						
							|  |  |  |         title = self._html_search_regex( | 
					
						
							|  |  |  |             r'(?s)<h1>(.+?)</h1>', webpage, 'title', fatal=False) | 
					
						
							|  |  |  |         description = self._html_search_meta('description', webpage) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def _get_page(idx): | 
					
						
							| 
									
										
										
										
											2015-03-12 19:08:16 +01:00
										 |  |  |             page_url = 'https://vimeo.com/user%s/likes/page:%d/sort:date' % ( | 
					
						
							|  |  |  |                 user_id, idx + 1) | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |             webpage = self._download_webpage( | 
					
						
							|  |  |  |                 page_url, user_id, | 
					
						
							|  |  |  |                 note='Downloading page %d/%d' % (idx + 1, page_count)) | 
					
						
							|  |  |  |             video_list = self._search_regex( | 
					
						
							|  |  |  |                 r'(?s)<ol class="js-browse_list[^"]+"[^>]*>(.*?)</ol>', | 
					
						
							|  |  |  |                 webpage, 'video content') | 
					
						
							|  |  |  |             paths = re.findall( | 
					
						
							|  |  |  |                 r'<li[^>]*>\s*<a\s+href="([^"]+)"', video_list) | 
					
						
							|  |  |  |             for path in paths: | 
					
						
							|  |  |  |                 yield { | 
					
						
							|  |  |  |                     '_type': 'url', | 
					
						
							|  |  |  |                     'url': compat_urlparse.urljoin(page_url, path), | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pl = InAdvancePagedList(_get_page, page_count, PAGE_SIZE) | 
					
						
							| 
									
										
										
										
											2014-09-28 12:14:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							| 
									
										
										
										
											2014-09-29 00:36:06 +02:00
										 |  |  |             '_type': 'playlist', | 
					
						
							|  |  |  |             'id': 'user%s_likes' % user_id, | 
					
						
							|  |  |  |             'title': title, | 
					
						
							|  |  |  |             'description': description, | 
					
						
							|  |  |  |             'entries': pl, | 
					
						
							| 
									
										
										
										
											2014-09-28 12:14:16 +02:00
										 |  |  |         } |