| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | # encoding: utf-8 | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | import re | 
					
						
							| 
									
										
										
										
											2014-10-29 11:04:48 +01:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | from .common import InfoExtractor | 
					
						
							|  |  |  | from ..utils import ( | 
					
						
							|  |  |  |     compat_urllib_parse, | 
					
						
							|  |  |  |     compat_urllib_request, | 
					
						
							|  |  |  |     compat_urlparse, | 
					
						
							|  |  |  |     unified_strdate, | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |     parse_duration, | 
					
						
							|  |  |  |     int_or_none, | 
					
						
							| 
									
										
										
										
											2014-11-05 19:52:34 +01:00
										 |  |  |     ExtractorError, | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | class NiconicoIE(InfoExtractor): | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |     IE_NAME = 'niconico' | 
					
						
							|  |  |  |     IE_DESC = 'ニコニコ動画' | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _TEST = { | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |         'url': 'http://www.nicovideo.jp/watch/sm22312215', | 
					
						
							|  |  |  |         'md5': 'd1a75c0823e2f629128c43e1212760f9', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': 'sm22312215', | 
					
						
							|  |  |  |             'ext': 'mp4', | 
					
						
							|  |  |  |             'title': 'Big Buck Bunny', | 
					
						
							|  |  |  |             'uploader': 'takuya0301', | 
					
						
							|  |  |  |             'uploader_id': '2698420', | 
					
						
							|  |  |  |             'upload_date': '20131123', | 
					
						
							|  |  |  |             'description': '(c) copyright 2008, Blender Foundation / www.bigbuckbunny.org', | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |             'duration': 33, | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |         'params': { | 
					
						
							|  |  |  |             'username': 'ydl.niconico@gmail.com', | 
					
						
							|  |  |  |             'password': 'youtube-dl', | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |         }, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |     _VALID_URL = r'https?://(?:www\.|secure\.)?nicovideo\.jp/watch/((?:[a-z]{2})?[0-9]+)' | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |     _NETRC_MACHINE = 'niconico' | 
					
						
							| 
									
										
										
										
											2014-10-12 23:18:42 +03:00
										 |  |  |     # Determine whether the downloader used authentication to download video | 
					
						
							|  |  |  |     _AUTHENTICATED = False | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _real_initialize(self): | 
					
						
							| 
									
										
										
										
											2014-10-12 23:18:42 +03:00
										 |  |  |         self._login() | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _login(self): | 
					
						
							|  |  |  |         (username, password) = self._get_login_info() | 
					
						
							| 
									
										
										
										
											2014-10-12 23:18:42 +03:00
										 |  |  |         # No authentication to be performed | 
					
						
							|  |  |  |         if not username: | 
					
						
							|  |  |  |             return True | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Log in | 
					
						
							|  |  |  |         login_form_strs = { | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             'mail': username, | 
					
						
							|  |  |  |             'password': password, | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |         # Convert to UTF-8 *before* urlencode because Python 2.x's urlencode | 
					
						
							|  |  |  |         # chokes on unicode | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |         login_form = dict((k.encode('utf-8'), v.encode('utf-8')) for k, v in login_form_strs.items()) | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |         login_data = compat_urllib_parse.urlencode(login_form).encode('utf-8') | 
					
						
							|  |  |  |         request = compat_urllib_request.Request( | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             'https://secure.nicovideo.jp/secure/login', login_data) | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |         login_results = self._download_webpage( | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             request, None, note='Logging in', errnote='Unable to log in') | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |         if re.search(r'(?i)<h1 class="mb8p4">Log in error</h1>', login_results) is not None: | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             self._downloader.report_warning('unable to log in: bad username or password') | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2014-10-12 23:18:42 +03:00
										 |  |  |         # Successful login | 
					
						
							|  |  |  |         self._AUTHENTICATED = True | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |         mobj = re.match(self._VALID_URL, url) | 
					
						
							|  |  |  |         video_id = mobj.group(1) | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-24 06:53:50 +01:00
										 |  |  |         # Get video webpage. We are not actually interested in it, but need | 
					
						
							|  |  |  |         # the cookies in order to be able to download the info webpage | 
					
						
							|  |  |  |         self._download_webpage('http://www.nicovideo.jp/watch/' + video_id, video_id) | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-26 18:48:52 +01:00
										 |  |  |         video_info = self._download_xml( | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |             'http://ext.nicovideo.jp/api/getthumbinfo/' + video_id, video_id, | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             note='Downloading video info page') | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-12 23:18:42 +03:00
										 |  |  |         if self._AUTHENTICATED: | 
					
						
							| 
									
										
										
										
											2014-07-02 02:32:54 +09:00
										 |  |  |             # Get flv info | 
					
						
							|  |  |  |             flv_info_webpage = self._download_webpage( | 
					
						
							|  |  |  |                 'http://flapi.nicovideo.jp/api/getflv?v=' + video_id, | 
					
						
							|  |  |  |                 video_id, 'Downloading flv info') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # Get external player info | 
					
						
							|  |  |  |             ext_player_info = self._download_webpage( | 
					
						
							|  |  |  |                 'http://ext.nicovideo.jp/thumb_watch/' + video_id, video_id) | 
					
						
							|  |  |  |             thumb_play_key = self._search_regex( | 
					
						
							|  |  |  |                 r'\'thumbPlayKey\'\s*:\s*\'(.*?)\'', ext_player_info, 'thumbPlayKey') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Get flv info | 
					
						
							|  |  |  |             flv_info_data = compat_urllib_parse.urlencode({ | 
					
						
							|  |  |  |                 'k': thumb_play_key, | 
					
						
							|  |  |  |                 'v': video_id | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |             flv_info_request = compat_urllib_request.Request( | 
					
						
							|  |  |  |                 'http://ext.nicovideo.jp/thumb_watch', flv_info_data, | 
					
						
							|  |  |  |                 {'Content-Type': 'application/x-www-form-urlencoded'}) | 
					
						
							|  |  |  |             flv_info_webpage = self._download_webpage( | 
					
						
							|  |  |  |                 flv_info_request, video_id, | 
					
						
							|  |  |  |                 note='Downloading flv info', errnote='Unable to download flv info') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 19:52:34 +01:00
										 |  |  |         if 'deleted=' in flv_info_webpage: | 
					
						
							|  |  |  |             raise ExtractorError('The video has been deleted.', | 
					
						
							|  |  |  |                 expected=True) | 
					
						
							| 
									
										
										
										
											2013-11-24 06:37:14 +01:00
										 |  |  |         video_real_url = compat_urlparse.parse_qs(flv_info_webpage)['url'][0] | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Start extracting information | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |         title = video_info.find('.//title').text | 
					
						
							|  |  |  |         extension = video_info.find('.//movie_type').text | 
					
						
							|  |  |  |         video_format = extension.upper() | 
					
						
							|  |  |  |         thumbnail = video_info.find('.//thumbnail_url').text | 
					
						
							|  |  |  |         description = video_info.find('.//description').text | 
					
						
							|  |  |  |         upload_date = unified_strdate(video_info.find('.//first_retrieve').text.split('+')[0]) | 
					
						
							|  |  |  |         view_count = int_or_none(video_info.find('.//view_counter').text) | 
					
						
							|  |  |  |         comment_count = int_or_none(video_info.find('.//comment_num').text) | 
					
						
							|  |  |  |         duration = parse_duration(video_info.find('.//length').text) | 
					
						
							|  |  |  |         webpage_url = video_info.find('.//watch_url').text | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-02 03:13:12 +09:00
										 |  |  |         if video_info.find('.//ch_id') is not None: | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |             uploader_id = video_info.find('.//ch_id').text | 
					
						
							|  |  |  |             uploader = video_info.find('.//ch_name').text | 
					
						
							| 
									
										
										
										
											2014-07-02 03:13:12 +09:00
										 |  |  |         elif video_info.find('.//user_id') is not None: | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |             uploader_id = video_info.find('.//user_id').text | 
					
						
							|  |  |  |             uploader = video_info.find('.//user_nickname').text | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             uploader_id = uploader = None | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             'id': video_id, | 
					
						
							|  |  |  |             'url': video_real_url, | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |             'title': title, | 
					
						
							|  |  |  |             'ext': extension, | 
					
						
							| 
									
										
										
										
											2014-03-27 21:01:09 +01:00
										 |  |  |             'format': video_format, | 
					
						
							| 
									
										
										
										
											2014-07-04 22:05:46 +07:00
										 |  |  |             'thumbnail': thumbnail, | 
					
						
							|  |  |  |             'description': description, | 
					
						
							|  |  |  |             'uploader': uploader, | 
					
						
							|  |  |  |             'upload_date': upload_date, | 
					
						
							|  |  |  |             'uploader_id': uploader_id, | 
					
						
							|  |  |  |             'view_count': view_count, | 
					
						
							|  |  |  |             'comment_count': comment_count, | 
					
						
							|  |  |  |             'duration': duration, | 
					
						
							|  |  |  |             'webpage_url': webpage_url, | 
					
						
							| 
									
										
										
										
											2013-11-23 18:19:44 +09:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-10-29 11:04:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NiconicoPlaylistIE(InfoExtractor): | 
					
						
							|  |  |  |     _VALID_URL = r'https?://www\.nicovideo\.jp/mylist/(?P<id>\d+)' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _TEST = { | 
					
						
							|  |  |  |         'url': 'http://www.nicovideo.jp/mylist/27411728', | 
					
						
							|  |  |  |         'info_dict': { | 
					
						
							|  |  |  |             'id': '27411728', | 
					
						
							|  |  |  |             'title': 'AKB48のオールナイトニッポン', | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'playlist_mincount': 225, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _real_extract(self, url): | 
					
						
							|  |  |  |         list_id = self._match_id(url) | 
					
						
							|  |  |  |         webpage = self._download_webpage(url, list_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         entries_json = self._search_regex(r'Mylist\.preload\(\d+, (\[.*\])\);', | 
					
						
							|  |  |  |             webpage, 'entries') | 
					
						
							|  |  |  |         entries = json.loads(entries_json) | 
					
						
							|  |  |  |         entries = [{ | 
					
						
							|  |  |  |             '_type': 'url', | 
					
						
							|  |  |  |             'ie_key': NiconicoIE.ie_key(), | 
					
						
							| 
									
										
										
										
											2014-11-08 14:53:23 +01:00
										 |  |  |             'url': ('http://www.nicovideo.jp/watch/%s' % | 
					
						
							|  |  |  |                 entry['item_data']['video_id']), | 
					
						
							| 
									
										
										
										
											2014-10-29 11:04:48 +01:00
										 |  |  |         } for entry in entries] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             '_type': 'playlist', | 
					
						
							|  |  |  |             'title': self._search_regex(r'\s+name: "(.*?)"', webpage, 'title'), | 
					
						
							|  |  |  |             'id': list_id, | 
					
						
							|  |  |  |             'entries': entries, | 
					
						
							|  |  |  |         } |