Compare commits
	
		
			57 Commits
		
	
	
		
			2013.11.06
			...
			2013.11.15
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					aa13b2dffd | ||
| 
						 | 
					fc2ef392be | ||
| 
						 | 
					463a908705 | ||
| 
						 | 
					d24ffe1cfa | ||
| 
						 | 
					78fb87b283 | ||
| 
						 | 
					ab2d524780 | ||
| 
						 | 
					85d61685f1 | ||
| 
						 | 
					b9643eed7c | ||
| 
						 | 
					feee2ecfa9 | ||
| 
						 | 
					a25a5cfeec | ||
| 
						 | 
					0e145dd541 | ||
| 
						 | 
					9f9be844fc | ||
| 
						 | 
					e3b9ab5e18 | ||
| 
						 | 
					c66d2baa9c | ||
| 
						 | 
					ca715127a2 | ||
| 
						 | 
					ea7a7af1d4 | ||
| 
						 | 
					80b9bbce86 | ||
| 
						 | 
					d37936386f | ||
| 
						 | 
					c3a3028f9f | ||
| 
						 | 
					6c5ad80cdc | ||
| 
						 | 
					b5bdc2699a | ||
| 
						 | 
					384b98cd8f | ||
| 
						 | 
					eb9b5bffef | ||
| 
						 | 
					8b8cbd8f6d | ||
| 
						 | 
					72b18c5d34 | ||
| 
						 | 
					eb0a839866 | ||
| 
						 | 
					1777d5a952 | ||
| 
						 | 
					d4b7da84c3 | ||
| 
						 | 
					801dbbdffd | ||
| 
						 | 
					0ed05a1d2d | ||
| 
						 | 
					1008bebade | ||
| 
						 | 
					ae84f879d7 | ||
| 
						 | 
					be6dfd1b49 | ||
| 
						 | 
					231516b6c9 | ||
| 
						 | 
					fb53d58dcf | ||
| 
						 | 
					2a9e9b210b | ||
| 
						 | 
					897d6cc43a | ||
| 
						 | 
					f470c6c812 | ||
| 
						 | 
					566d4e0425 | ||
| 
						 | 
					81be02d2f9 | ||
| 
						 | 
					c2b6a482d5 | ||
| 
						 | 
					12c167c881 | ||
| 
						 | 
					20aafee7fa | ||
| 
						 | 
					be07375b66 | ||
| 
						 | 
					dd5bcdc4c9 | ||
| 
						 | 
					6161d17579 | ||
| 
						 | 
					4ac5306ae7 | ||
| 
						 | 
					b1a80ec1a9 | ||
| 
						 | 
					5137ebac0b | ||
| 
						 | 
					a8eeb0597b | ||
| 
						 | 
					4ed3e51080 | ||
| 
						 | 
					38fcd4597a | ||
| 
						 | 
					38db46794f | ||
| 
						 | 
					a9a3876d55 | ||
| 
						 | 
					1f343eaabb | ||
| 
						 | 
					0a43ddf320 | ||
| 
						 | 
					31366066bd | 
@@ -92,12 +92,14 @@ which means you can modify it, redistribute it or use it however you like.
 | 
			
		||||
                               ownloads/%(uploader)s/%(title)s-%(id)s.%(ext)s' .
 | 
			
		||||
    --autonumber-size NUMBER   Specifies the number of digits in %(autonumber)s
 | 
			
		||||
                               when it is present in output filename template or
 | 
			
		||||
                               --autonumber option is given
 | 
			
		||||
                               --auto-number option is given
 | 
			
		||||
    --restrict-filenames       Restrict filenames to only ASCII characters, and
 | 
			
		||||
                               avoid "&" and spaces in filenames
 | 
			
		||||
    -a, --batch-file FILE      file containing URLs to download ('-' for stdin)
 | 
			
		||||
    -w, --no-overwrites        do not overwrite files
 | 
			
		||||
    -c, --continue             resume partially downloaded files
 | 
			
		||||
    -c, --continue             force resume of partially downloaded files. By
 | 
			
		||||
                               default, youtube-dl will resume downloads if
 | 
			
		||||
                               possible.
 | 
			
		||||
    --no-continue              do not resume partially downloaded files (restart
 | 
			
		||||
                               from beginning)
 | 
			
		||||
    --cookies FILE             file to read cookies from and dump cookie jar in
 | 
			
		||||
 
 | 
			
		||||
@@ -1,70 +0,0 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
 | 
			
		||||
# Allow direct execution
 | 
			
		||||
import os
 | 
			
		||||
import sys
 | 
			
		||||
import unittest
 | 
			
		||||
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
 | 
			
		||||
 | 
			
		||||
from test.helper import FakeYDL, global_setup, md5
 | 
			
		||||
global_setup()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
from youtube_dl.extractor import DailymotionIE
 | 
			
		||||
 | 
			
		||||
class TestDailymotionSubtitles(unittest.TestCase):
 | 
			
		||||
    def setUp(self):
 | 
			
		||||
        self.DL = FakeYDL()
 | 
			
		||||
        self.url = 'http://www.dailymotion.com/video/xczg00'
 | 
			
		||||
    def getInfoDict(self):
 | 
			
		||||
        IE = DailymotionIE(self.DL)
 | 
			
		||||
        info_dict = IE.extract(self.url)
 | 
			
		||||
        return info_dict
 | 
			
		||||
    def getSubtitles(self):
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        return info_dict['subtitles']
 | 
			
		||||
    def test_no_writesubtitles(self):
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(subtitles, None)
 | 
			
		||||
    def test_subtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f')
 | 
			
		||||
    def test_subtitles_lang(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['fr']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792')
 | 
			
		||||
    def test_allsubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles.keys()), 5)
 | 
			
		||||
    def test_list_subtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['listsubtitles'] = True
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        self.assertEqual(info_dict, None)
 | 
			
		||||
    def test_automatic_captions(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['writeautomaticsub'] = True
 | 
			
		||||
        self.DL.params['subtitleslang'] = ['en']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertTrue(len(subtitles.keys()) == 0)
 | 
			
		||||
    def test_nosubtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'video doesn\'t have subtitles')
 | 
			
		||||
        self.url = 'http://www.dailymotion.com/video/x12u166_le-zapping-tele-star-du-08-aout-2013_tv'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles), 0)
 | 
			
		||||
    def test_multiple_langs(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        langs = ['es', 'fr', 'de']
 | 
			
		||||
        self.DL.params['subtitleslangs'] = langs
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        for lang in langs:
 | 
			
		||||
            self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang)
 | 
			
		||||
 | 
			
		||||
if __name__ == '__main__':
 | 
			
		||||
    unittest.main()
 | 
			
		||||
@@ -17,6 +17,7 @@ from youtube_dl.extractor import (
 | 
			
		||||
    DailymotionUserIE,
 | 
			
		||||
    VimeoChannelIE,
 | 
			
		||||
    UstreamChannelIE,
 | 
			
		||||
    SoundcloudSetIE,
 | 
			
		||||
    SoundcloudUserIE,
 | 
			
		||||
    LivestreamIE,
 | 
			
		||||
    NHLVideocenterIE,
 | 
			
		||||
@@ -61,6 +62,14 @@ class TestPlaylists(unittest.TestCase):
 | 
			
		||||
        self.assertEqual(result['id'], u'5124905')
 | 
			
		||||
        self.assertTrue(len(result['entries']) >= 11)
 | 
			
		||||
 | 
			
		||||
    def test_soundcloud_set(self):
 | 
			
		||||
        dl = FakeYDL()
 | 
			
		||||
        ie = SoundcloudSetIE(dl)
 | 
			
		||||
        result = ie.extract('https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep')
 | 
			
		||||
        self.assertIsPlaylist(result)
 | 
			
		||||
        self.assertEqual(result['title'], u'The Royal Concept EP')
 | 
			
		||||
        self.assertTrue(len(result['entries']) >= 6)
 | 
			
		||||
 | 
			
		||||
    def test_soundcloud_user(self):
 | 
			
		||||
        dl = FakeYDL()
 | 
			
		||||
        ie = SoundcloudUserIE(dl)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										211
									
								
								test/test_subtitles.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								test/test_subtitles.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,211 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
 | 
			
		||||
# Allow direct execution
 | 
			
		||||
import os
 | 
			
		||||
import sys
 | 
			
		||||
import unittest
 | 
			
		||||
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
 | 
			
		||||
 | 
			
		||||
from test.helper import FakeYDL, global_setup, md5
 | 
			
		||||
global_setup()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
from youtube_dl.extractor import (
 | 
			
		||||
    YoutubeIE,
 | 
			
		||||
    DailymotionIE,
 | 
			
		||||
    TEDIE,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class BaseTestSubtitles(unittest.TestCase):
 | 
			
		||||
    url = None
 | 
			
		||||
    IE = None
 | 
			
		||||
    def setUp(self):
 | 
			
		||||
        self.DL = FakeYDL()
 | 
			
		||||
        self.ie = self.IE(self.DL)
 | 
			
		||||
 | 
			
		||||
    def getInfoDict(self):
 | 
			
		||||
        info_dict = self.ie.extract(self.url)
 | 
			
		||||
        return info_dict
 | 
			
		||||
 | 
			
		||||
    def getSubtitles(self):
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        return info_dict['subtitles']
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestYoutubeSubtitles(BaseTestSubtitles):
 | 
			
		||||
    url = 'QRS8MkLhQmM'
 | 
			
		||||
    IE = YoutubeIE
 | 
			
		||||
 | 
			
		||||
    def getSubtitles(self):
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        return info_dict[0]['subtitles']
 | 
			
		||||
 | 
			
		||||
    def test_youtube_no_writesubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = False
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(subtitles, None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_lang(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['it']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_allsubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles.keys()), 13)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_sbv_format(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitlesformat'] = 'sbv'
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '13aeaa0c245a8bed9a451cb643e3ad8b')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_vtt_format(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitlesformat'] = 'vtt'
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '356cdc577fde0c6783b9b822e7206ff7')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_list_subtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'Video doesn\'t have automatic captions')
 | 
			
		||||
        self.DL.params['listsubtitles'] = True
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        self.assertEqual(info_dict, None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_automatic_captions(self):
 | 
			
		||||
        self.url = '8YoUxe5ncPo'
 | 
			
		||||
        self.DL.params['writeautomaticsub'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['it']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertTrue(subtitles['it'] is not None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_nosubtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'video doesn\'t have subtitles')
 | 
			
		||||
        self.url = 'sAjKT8FhjI8'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles), 0)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_multiple_langs(self):
 | 
			
		||||
        self.url = 'QRS8MkLhQmM'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        langs = ['it', 'fr', 'de']
 | 
			
		||||
        self.DL.params['subtitleslangs'] = langs
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        for lang in langs:
 | 
			
		||||
            self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestDailymotionSubtitles(BaseTestSubtitles):
 | 
			
		||||
    url = 'http://www.dailymotion.com/video/xczg00'
 | 
			
		||||
    IE = DailymotionIE
 | 
			
		||||
 | 
			
		||||
    def test_no_writesubtitles(self):
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(subtitles, None)
 | 
			
		||||
 | 
			
		||||
    def test_subtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f')
 | 
			
		||||
 | 
			
		||||
    def test_subtitles_lang(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['fr']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792')
 | 
			
		||||
 | 
			
		||||
    def test_allsubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles.keys()), 5)
 | 
			
		||||
 | 
			
		||||
    def test_list_subtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['listsubtitles'] = True
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        self.assertEqual(info_dict, None)
 | 
			
		||||
 | 
			
		||||
    def test_automatic_captions(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['writeautomaticsub'] = True
 | 
			
		||||
        self.DL.params['subtitleslang'] = ['en']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertTrue(len(subtitles.keys()) == 0)
 | 
			
		||||
 | 
			
		||||
    def test_nosubtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'video doesn\'t have subtitles')
 | 
			
		||||
        self.url = 'http://www.dailymotion.com/video/x12u166_le-zapping-tele-star-du-08-aout-2013_tv'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles), 0)
 | 
			
		||||
 | 
			
		||||
    def test_multiple_langs(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        langs = ['es', 'fr', 'de']
 | 
			
		||||
        self.DL.params['subtitleslangs'] = langs
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        for lang in langs:
 | 
			
		||||
            self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestTedSubtitles(BaseTestSubtitles):
 | 
			
		||||
    url = 'http://www.ted.com/talks/dan_dennett_on_our_consciousness.html'
 | 
			
		||||
    IE = TEDIE
 | 
			
		||||
 | 
			
		||||
    def test_no_writesubtitles(self):
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(subtitles, None)
 | 
			
		||||
 | 
			
		||||
    def test_subtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '2154f31ff9b9f89a0aa671537559c21d')
 | 
			
		||||
 | 
			
		||||
    def test_subtitles_lang(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['fr']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['fr']), '7616cbc6df20ec2c1204083c83871cf6')
 | 
			
		||||
 | 
			
		||||
    def test_allsubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles.keys()), 28)
 | 
			
		||||
 | 
			
		||||
    def test_list_subtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['listsubtitles'] = True
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        self.assertEqual(info_dict, None)
 | 
			
		||||
 | 
			
		||||
    def test_automatic_captions(self):
 | 
			
		||||
        self.DL.expect_warning(u'Automatic Captions not supported by this server')
 | 
			
		||||
        self.DL.params['writeautomaticsub'] = True
 | 
			
		||||
        self.DL.params['subtitleslang'] = ['en']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertTrue(len(subtitles.keys()) == 0)
 | 
			
		||||
 | 
			
		||||
    def test_multiple_langs(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        langs = ['es', 'fr', 'de']
 | 
			
		||||
        self.DL.params['subtitleslangs'] = langs
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        for lang in langs:
 | 
			
		||||
            self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang)
 | 
			
		||||
 | 
			
		||||
if __name__ == '__main__':
 | 
			
		||||
    unittest.main()
 | 
			
		||||
@@ -1,95 +0,0 @@
 | 
			
		||||
#!/usr/bin/env python
 | 
			
		||||
 | 
			
		||||
# Allow direct execution
 | 
			
		||||
import os
 | 
			
		||||
import sys
 | 
			
		||||
import unittest
 | 
			
		||||
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
 | 
			
		||||
 | 
			
		||||
from test.helper import FakeYDL, global_setup, md5
 | 
			
		||||
global_setup()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
from youtube_dl.extractor import YoutubeIE
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TestYoutubeSubtitles(unittest.TestCase):
 | 
			
		||||
    def setUp(self):
 | 
			
		||||
        self.DL = FakeYDL()
 | 
			
		||||
        self.url = 'QRS8MkLhQmM'
 | 
			
		||||
 | 
			
		||||
    def getInfoDict(self):
 | 
			
		||||
        IE = YoutubeIE(self.DL)
 | 
			
		||||
        info_dict = IE.extract(self.url)
 | 
			
		||||
        return info_dict
 | 
			
		||||
 | 
			
		||||
    def getSubtitles(self):
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        return info_dict[0]['subtitles']
 | 
			
		||||
 | 
			
		||||
    def test_youtube_no_writesubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = False
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(subtitles, None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_lang(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['it']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_allsubtitles(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles.keys()), 13)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_sbv_format(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitlesformat'] = 'sbv'
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '13aeaa0c245a8bed9a451cb643e3ad8b')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_subtitles_vtt_format(self):
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['subtitlesformat'] = 'vtt'
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(md5(subtitles['en']), '356cdc577fde0c6783b9b822e7206ff7')
 | 
			
		||||
 | 
			
		||||
    def test_youtube_list_subtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'Video doesn\'t have automatic captions')
 | 
			
		||||
        self.DL.params['listsubtitles'] = True
 | 
			
		||||
        info_dict = self.getInfoDict()
 | 
			
		||||
        self.assertEqual(info_dict, None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_automatic_captions(self):
 | 
			
		||||
        self.url = '8YoUxe5ncPo'
 | 
			
		||||
        self.DL.params['writeautomaticsub'] = True
 | 
			
		||||
        self.DL.params['subtitleslangs'] = ['it']
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertTrue(subtitles['it'] is not None)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_nosubtitles(self):
 | 
			
		||||
        self.DL.expect_warning(u'video doesn\'t have subtitles')
 | 
			
		||||
        self.url = 'sAjKT8FhjI8'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        self.DL.params['allsubtitles'] = True
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        self.assertEqual(len(subtitles), 0)
 | 
			
		||||
 | 
			
		||||
    def test_youtube_multiple_langs(self):
 | 
			
		||||
        self.url = 'QRS8MkLhQmM'
 | 
			
		||||
        self.DL.params['writesubtitles'] = True
 | 
			
		||||
        langs = ['it', 'fr', 'de']
 | 
			
		||||
        self.DL.params['subtitleslangs'] = langs
 | 
			
		||||
        subtitles = self.getSubtitles()
 | 
			
		||||
        for lang in langs:
 | 
			
		||||
            self.assertTrue(subtitles.get(lang) is not None, u'Subtitles for \'%s\' not extracted' % lang)
 | 
			
		||||
 | 
			
		||||
if __name__ == '__main__':
 | 
			
		||||
    unittest.main()
 | 
			
		||||
@@ -234,8 +234,14 @@ class FileDownloader(object):
 | 
			
		||||
        if self.params.get('noprogress', False):
 | 
			
		||||
            return
 | 
			
		||||
        clear_line = (u'\x1b[K' if sys.stderr.isatty() and os.name != 'nt' else u'')
 | 
			
		||||
        eta_str = self.format_eta(eta)
 | 
			
		||||
        percent_str = self.format_percent(percent)
 | 
			
		||||
        if eta is not None:
 | 
			
		||||
            eta_str = self.format_eta(eta)
 | 
			
		||||
        else:
 | 
			
		||||
            eta_str = 'Unknown ETA'
 | 
			
		||||
        if percent is not None:
 | 
			
		||||
            percent_str = self.format_percent(percent)
 | 
			
		||||
        else:
 | 
			
		||||
            percent_str = 'Unknown %'
 | 
			
		||||
        speed_str = self.format_speed(speed)
 | 
			
		||||
        if self.params.get('progress_with_newline', False):
 | 
			
		||||
            self.to_screen(u'[download] %s of %s at %s ETA %s' %
 | 
			
		||||
@@ -274,7 +280,7 @@ class FileDownloader(object):
 | 
			
		||||
            self.to_screen(u'\r%s[download] 100%% of %s in %s' %
 | 
			
		||||
                (clear_line, data_len_str, self.format_seconds(tot_time)))
 | 
			
		||||
 | 
			
		||||
    def _download_with_rtmpdump(self, filename, url, player_url, page_url, play_path, tc_url):
 | 
			
		||||
    def _download_with_rtmpdump(self, filename, url, player_url, page_url, play_path, tc_url, live):
 | 
			
		||||
        self.report_destination(filename)
 | 
			
		||||
        tmpfilename = self.temp_name(filename)
 | 
			
		||||
        test = self.params.get('test', False)
 | 
			
		||||
@@ -301,6 +307,8 @@ class FileDownloader(object):
 | 
			
		||||
            basic_args += ['--tcUrl', url]
 | 
			
		||||
        if test:
 | 
			
		||||
            basic_args += ['--stop', '1']
 | 
			
		||||
        if live:
 | 
			
		||||
            basic_args += ['--live']
 | 
			
		||||
        args = basic_args + [[], ['--resume', '--skip', '1']][self.params.get('continuedl', False)]
 | 
			
		||||
        if self.params.get('verbose', False):
 | 
			
		||||
            try:
 | 
			
		||||
@@ -373,15 +381,20 @@ class FileDownloader(object):
 | 
			
		||||
        self.report_destination(filename)
 | 
			
		||||
        tmpfilename = self.temp_name(filename)
 | 
			
		||||
 | 
			
		||||
        args = ['ffmpeg', '-y', '-i', url, '-f', 'mp4', tmpfilename]
 | 
			
		||||
        # Check for ffmpeg first
 | 
			
		||||
        try:
 | 
			
		||||
            subprocess.call(['ffmpeg', '-h'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT)
 | 
			
		||||
        except (OSError, IOError):
 | 
			
		||||
            self.report_error(u'm3u8 download detected but "%s" could not be run' % args[0] )
 | 
			
		||||
            return False
 | 
			
		||||
        args = ['-y', '-i', url, '-f', 'mp4', '-c', 'copy',
 | 
			
		||||
            '-bsf:a', 'aac_adtstoasc', tmpfilename]
 | 
			
		||||
 | 
			
		||||
        retval = subprocess.call(args)
 | 
			
		||||
        for program in ['avconv', 'ffmpeg']:
 | 
			
		||||
            try:
 | 
			
		||||
                subprocess.call([program, '-version'], stdout=(open(os.path.devnull, 'w')), stderr=subprocess.STDOUT)
 | 
			
		||||
                break
 | 
			
		||||
            except (OSError, IOError):
 | 
			
		||||
                pass
 | 
			
		||||
        else:
 | 
			
		||||
            self.report_error(u'm3u8 download detected but ffmpeg or avconv could not be found')
 | 
			
		||||
        cmd = [program] + args
 | 
			
		||||
 | 
			
		||||
        retval = subprocess.call(cmd)
 | 
			
		||||
        if retval == 0:
 | 
			
		||||
            fsize = os.path.getsize(encodeFilename(tmpfilename))
 | 
			
		||||
            self.to_screen(u'\r[%s] %s bytes' % (args[0], fsize))
 | 
			
		||||
@@ -418,7 +431,8 @@ class FileDownloader(object):
 | 
			
		||||
                                                info_dict.get('player_url', None),
 | 
			
		||||
                                                info_dict.get('page_url', None),
 | 
			
		||||
                                                info_dict.get('play_path', None),
 | 
			
		||||
                                                info_dict.get('tc_url', None))
 | 
			
		||||
                                                info_dict.get('tc_url', None),
 | 
			
		||||
                                                info_dict.get('rtmp_live', False))
 | 
			
		||||
 | 
			
		||||
        # Attempt to download using mplayer
 | 
			
		||||
        if url.startswith('mms') or url.startswith('rtsp'):
 | 
			
		||||
@@ -557,12 +571,11 @@ class FileDownloader(object):
 | 
			
		||||
            # Progress message
 | 
			
		||||
            speed = self.calc_speed(start, time.time(), byte_counter - resume_len)
 | 
			
		||||
            if data_len is None:
 | 
			
		||||
                self.report_progress('Unknown %', data_len_str, speed, 'Unknown ETA')
 | 
			
		||||
                eta = None
 | 
			
		||||
                eta = percent = None
 | 
			
		||||
            else:
 | 
			
		||||
                percent = self.calc_percent(byte_counter, data_len)
 | 
			
		||||
                eta = self.calc_eta(start, time.time(), data_len - resume_len, byte_counter - resume_len)
 | 
			
		||||
                self.report_progress(percent, data_len_str, speed, eta)
 | 
			
		||||
            self.report_progress(percent, data_len_str, speed, eta)
 | 
			
		||||
 | 
			
		||||
            self._hook_progress({
 | 
			
		||||
                'downloaded_bytes': byte_counter,
 | 
			
		||||
 
 | 
			
		||||
@@ -501,7 +501,7 @@ class FFmpegMetadataPP(FFmpegPostProcessor):
 | 
			
		||||
 | 
			
		||||
        options = ['-c', 'copy']
 | 
			
		||||
        for (name, value) in metadata.items():
 | 
			
		||||
            options.extend(['-metadata', '%s="%s"' % (name, value)])
 | 
			
		||||
            options.extend(['-metadata', '%s=%s' % (name, value)])
 | 
			
		||||
        options.extend(['-f', ext])
 | 
			
		||||
 | 
			
		||||
        self._downloader.to_screen(u'[ffmpeg] Adding metadata to \'%s\'' % filename)
 | 
			
		||||
 
 | 
			
		||||
@@ -385,7 +385,7 @@ class YoutubeDL(object):
 | 
			
		||||
        result_type = ie_result.get('_type', 'video') # If not given we suppose it's a video, support the default old system
 | 
			
		||||
        if result_type == 'video':
 | 
			
		||||
            self.add_extra_info(ie_result, extra_info)
 | 
			
		||||
            return self.process_video_result(ie_result)
 | 
			
		||||
            return self.process_video_result(ie_result, download=download)
 | 
			
		||||
        elif result_type == 'url':
 | 
			
		||||
            # We have to add extra_info to the results because it may be
 | 
			
		||||
            # contained in a playlist
 | 
			
		||||
@@ -640,7 +640,7 @@ class YoutubeDL(object):
 | 
			
		||||
            # subtitles download errors are already managed as troubles in relevant IE
 | 
			
		||||
            # that way it will silently go on when used with unsupporting IE
 | 
			
		||||
            subtitles = info_dict['subtitles']
 | 
			
		||||
            sub_format = self.params.get('subtitlesformat')
 | 
			
		||||
            sub_format = self.params.get('subtitlesformat', 'srt')
 | 
			
		||||
            for sub_lang in subtitles.keys():
 | 
			
		||||
                sub = subtitles[sub_lang]
 | 
			
		||||
                if sub is None:
 | 
			
		||||
@@ -782,7 +782,7 @@ class YoutubeDL(object):
 | 
			
		||||
 | 
			
		||||
    def list_formats(self, info_dict):
 | 
			
		||||
        def line(format):
 | 
			
		||||
            return (u'%-15s%-10s%-12s%s' % (
 | 
			
		||||
            return (u'%-20s%-10s%-12s%s' % (
 | 
			
		||||
                format['format_id'],
 | 
			
		||||
                format['ext'],
 | 
			
		||||
                self.format_resolution(format),
 | 
			
		||||
 
 | 
			
		||||
@@ -32,6 +32,8 @@ __authors__  = (
 | 
			
		||||
    'Ismael Mejía',
 | 
			
		||||
    'Steffan \'Ruirize\' James',
 | 
			
		||||
    'Andras Elso',
 | 
			
		||||
    'Jelle van der Waa',
 | 
			
		||||
    'Marcin Cieślak',
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
__license__ = 'Public Domain'
 | 
			
		||||
@@ -349,7 +351,7 @@ def parseOpts(overrideArguments=None):
 | 
			
		||||
                  'for example with -o \'/my/downloads/%(uploader)s/%(title)s-%(id)s.%(ext)s\' .'))
 | 
			
		||||
    filesystem.add_option('--autonumber-size',
 | 
			
		||||
            dest='autonumber_size', metavar='NUMBER',
 | 
			
		||||
            help='Specifies the number of digits in %(autonumber)s when it is present in output filename template or --autonumber option is given')
 | 
			
		||||
            help='Specifies the number of digits in %(autonumber)s when it is present in output filename template or --auto-number option is given')
 | 
			
		||||
    filesystem.add_option('--restrict-filenames',
 | 
			
		||||
            action='store_true', dest='restrictfilenames',
 | 
			
		||||
            help='Restrict filenames to only ASCII characters, and avoid "&" and spaces in filenames', default=False)
 | 
			
		||||
@@ -358,7 +360,7 @@ def parseOpts(overrideArguments=None):
 | 
			
		||||
    filesystem.add_option('-w', '--no-overwrites',
 | 
			
		||||
            action='store_true', dest='nooverwrites', help='do not overwrite files', default=False)
 | 
			
		||||
    filesystem.add_option('-c', '--continue',
 | 
			
		||||
            action='store_true', dest='continue_dl', help='resume partially downloaded files', default=True)
 | 
			
		||||
            action='store_true', dest='continue_dl', help='force resume of partially downloaded files. By default, youtube-dl will resume downloads if possible.', default=True)
 | 
			
		||||
    filesystem.add_option('--no-continue',
 | 
			
		||||
            action='store_false', dest='continue_dl',
 | 
			
		||||
            help='do not resume partially downloaded files (restart from beginning)')
 | 
			
		||||
 
 | 
			
		||||
@@ -57,6 +57,7 @@ from .francetv import (
 | 
			
		||||
)
 | 
			
		||||
from .freesound import FreesoundIE
 | 
			
		||||
from .funnyordie import FunnyOrDieIE
 | 
			
		||||
from .gamekings import GamekingsIE
 | 
			
		||||
from .gamespot import GameSpotIE
 | 
			
		||||
from .gametrailers import GametrailersIE
 | 
			
		||||
from .generic import GenericIE
 | 
			
		||||
@@ -79,7 +80,7 @@ from .keezmovies import KeezMoviesIE
 | 
			
		||||
from .kickstarter import KickStarterIE
 | 
			
		||||
from .keek import KeekIE
 | 
			
		||||
from .liveleak import LiveLeakIE
 | 
			
		||||
from .livestream import LivestreamIE
 | 
			
		||||
from .livestream import LivestreamIE, LivestreamOriginalIE
 | 
			
		||||
from .metacafe import MetacafeIE
 | 
			
		||||
from .metacritic import MetacriticIE
 | 
			
		||||
from .mit import TechTVMITIE, MITIE
 | 
			
		||||
@@ -134,6 +135,7 @@ from .tube8 import Tube8IE
 | 
			
		||||
from .tudou import TudouIE
 | 
			
		||||
from .tumblr import TumblrIE
 | 
			
		||||
from .tutv import TutvIE
 | 
			
		||||
from .tvp import TvpIE
 | 
			
		||||
from .unistra import UnistraIE
 | 
			
		||||
from .ustream import UstreamIE, UstreamChannelIE
 | 
			
		||||
from .vbox7 import Vbox7IE
 | 
			
		||||
 
 | 
			
		||||
@@ -10,6 +10,7 @@ from ..utils import (
 | 
			
		||||
    unified_strdate,
 | 
			
		||||
    determine_ext,
 | 
			
		||||
    get_element_by_id,
 | 
			
		||||
    compat_str,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
# There are different sources of video in arte.tv, the extraction process 
 | 
			
		||||
@@ -181,20 +182,30 @@ class ArteTVPlus7IE(InfoExtractor):
 | 
			
		||||
                formats = all_formats
 | 
			
		||||
            else:
 | 
			
		||||
                raise ExtractorError(u'The formats list is empty')
 | 
			
		||||
        # We order the formats by quality
 | 
			
		||||
 | 
			
		||||
        if re.match(r'[A-Z]Q', formats[0]['quality']) is not None:
 | 
			
		||||
            sort_key = lambda f: ['HQ', 'MQ', 'EQ', 'SQ'].index(f['quality'])
 | 
			
		||||
            def sort_key(f):
 | 
			
		||||
                return ['HQ', 'MQ', 'EQ', 'SQ'].index(f['quality'])
 | 
			
		||||
        else:
 | 
			
		||||
            sort_key = lambda f: int(f.get('height',-1))
 | 
			
		||||
            def sort_key(f):
 | 
			
		||||
                return (
 | 
			
		||||
                    # Sort first by quality
 | 
			
		||||
                    int(f.get('height',-1)),
 | 
			
		||||
                    int(f.get('bitrate',-1)),
 | 
			
		||||
                    # The original version with subtitles has lower relevance
 | 
			
		||||
                    re.match(r'VO-ST(F|A)', f.get('versionCode', '')) is None,
 | 
			
		||||
                    # The version with sourds/mal subtitles has also lower relevance
 | 
			
		||||
                    re.match(r'VO?(F|A)-STM\1', f.get('versionCode', '')) is None,
 | 
			
		||||
                )
 | 
			
		||||
        formats = sorted(formats, key=sort_key)
 | 
			
		||||
        # Prefer videos without subtitles in the same language
 | 
			
		||||
        formats = sorted(formats, key=lambda f: re.match(r'VO(F|A)-STM\1', f.get('versionCode', '')) is None)
 | 
			
		||||
        # Pick the best quality
 | 
			
		||||
        def _format(format_info):
 | 
			
		||||
            quality = format_info['quality']
 | 
			
		||||
            m_quality = re.match(r'\w*? - (\d*)p', quality)
 | 
			
		||||
            if m_quality is not None:
 | 
			
		||||
                quality = m_quality.group(1)
 | 
			
		||||
            quality = ''
 | 
			
		||||
            height = format_info.get('height')
 | 
			
		||||
            if height is not None:
 | 
			
		||||
                quality = compat_str(height)
 | 
			
		||||
            bitrate = format_info.get('bitrate')
 | 
			
		||||
            if bitrate is not None:
 | 
			
		||||
                quality += '-%d' % bitrate
 | 
			
		||||
            if format_info.get('versionCode') is not None:
 | 
			
		||||
                format_id = u'%s-%s' % (quality, format_info['versionCode'])
 | 
			
		||||
            else:
 | 
			
		||||
@@ -203,7 +214,7 @@ class ArteTVPlus7IE(InfoExtractor):
 | 
			
		||||
                'format_id': format_id,
 | 
			
		||||
                'format_note': format_info.get('versionLibelle'),
 | 
			
		||||
                'width': format_info.get('width'),
 | 
			
		||||
                'height': format_info.get('height'),
 | 
			
		||||
                'height': height,
 | 
			
		||||
            }
 | 
			
		||||
            if format_info['mediaType'] == u'rtmp':
 | 
			
		||||
                info['url'] = format_info['streamer']
 | 
			
		||||
 
 | 
			
		||||
@@ -10,10 +10,12 @@ from ..utils import (
 | 
			
		||||
    find_xpath_attr,
 | 
			
		||||
    compat_urlparse,
 | 
			
		||||
    compat_str,
 | 
			
		||||
    compat_urllib_request,
 | 
			
		||||
 | 
			
		||||
    ExtractorError,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class BrightcoveIE(InfoExtractor):
 | 
			
		||||
    _VALID_URL = r'https?://.*brightcove\.com/(services|viewer).*\?(?P<query>.*)'
 | 
			
		||||
    _FEDERATED_URL_TEMPLATE = 'http://c.brightcove.com/services/viewer/htmlFederated?%s'
 | 
			
		||||
@@ -80,6 +82,9 @@ class BrightcoveIE(InfoExtractor):
 | 
			
		||||
        videoPlayer = find_xpath_attr(object_doc, './param', 'name', '@videoPlayer')
 | 
			
		||||
        if videoPlayer is not None:
 | 
			
		||||
            params['@videoPlayer'] = videoPlayer.attrib['value']
 | 
			
		||||
        linkBase = find_xpath_attr(object_doc, './param', 'name', 'linkBaseURL')
 | 
			
		||||
        if linkBase is not None:
 | 
			
		||||
            params['linkBaseURL'] = linkBase.attrib['value']
 | 
			
		||||
        data = compat_urllib_parse.urlencode(params)
 | 
			
		||||
        return cls._FEDERATED_URL_TEMPLATE % data
 | 
			
		||||
 | 
			
		||||
@@ -107,14 +112,18 @@ class BrightcoveIE(InfoExtractor):
 | 
			
		||||
 | 
			
		||||
        videoPlayer = query.get('@videoPlayer')
 | 
			
		||||
        if videoPlayer:
 | 
			
		||||
            return self._get_video_info(videoPlayer[0], query_str)
 | 
			
		||||
            return self._get_video_info(videoPlayer[0], query_str, query)
 | 
			
		||||
        else:
 | 
			
		||||
            player_key = query['playerKey']
 | 
			
		||||
            return self._get_playlist_info(player_key[0])
 | 
			
		||||
 | 
			
		||||
    def _get_video_info(self, video_id, query):
 | 
			
		||||
        request_url = self._FEDERATED_URL_TEMPLATE % query
 | 
			
		||||
        webpage = self._download_webpage(request_url, video_id)
 | 
			
		||||
    def _get_video_info(self, video_id, query_str, query):
 | 
			
		||||
        request_url = self._FEDERATED_URL_TEMPLATE % query_str
 | 
			
		||||
        req = compat_urllib_request.Request(request_url)
 | 
			
		||||
        linkBase = query.get('linkBaseURL')
 | 
			
		||||
        if linkBase is not None:
 | 
			
		||||
            req.add_header('Referer', linkBase[0])
 | 
			
		||||
        webpage = self._download_webpage(req, video_id)
 | 
			
		||||
 | 
			
		||||
        self.report_extraction(video_id)
 | 
			
		||||
        info = self._search_regex(r'var experienceJSON = ({.*?});', webpage, 'json')
 | 
			
		||||
 
 | 
			
		||||
@@ -65,6 +65,7 @@ class CinemassacreIE(InfoExtractor):
 | 
			
		||||
            {
 | 
			
		||||
                'url': url,
 | 
			
		||||
                'play_path': 'mp4:' + sd_file,
 | 
			
		||||
                'rtmp_live': True, # workaround
 | 
			
		||||
                'ext': 'flv',
 | 
			
		||||
                'format': 'sd',
 | 
			
		||||
                'format_id': 'sd',
 | 
			
		||||
@@ -72,6 +73,7 @@ class CinemassacreIE(InfoExtractor):
 | 
			
		||||
            {
 | 
			
		||||
                'url': url,
 | 
			
		||||
                'play_path': 'mp4:' + hd_file,
 | 
			
		||||
                'rtmp_live': True, # workaround
 | 
			
		||||
                'ext': 'flv',
 | 
			
		||||
                'format': 'hd',
 | 
			
		||||
                'format_id': 'hd',
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@ from ..utils import determine_ext
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class CNNIE(InfoExtractor):
 | 
			
		||||
    _VALID_URL = r'''(?x)https?://(edition\.)?cnn\.com/video/(data/.+?|\?)/
 | 
			
		||||
    _VALID_URL = r'''(?x)https?://((edition|www)\.)?cnn\.com/video/(data/.+?|\?)/
 | 
			
		||||
        (?P<path>.+?/(?P<title>[^/]+?)(?:\.cnn|(?=&)))'''
 | 
			
		||||
 | 
			
		||||
    _TESTS = [{
 | 
			
		||||
 
 | 
			
		||||
@@ -315,13 +315,21 @@ class InfoExtractor(object):
 | 
			
		||||
 | 
			
		||||
    # Helper functions for extracting OpenGraph info
 | 
			
		||||
    @staticmethod
 | 
			
		||||
    def _og_regex(prop):
 | 
			
		||||
        return r'<meta.+?property=[\'"]og:%s[\'"].+?content=(?:"(.+?)"|\'(.+?)\')' % re.escape(prop)
 | 
			
		||||
    def _og_regexes(prop):
 | 
			
		||||
        content_re = r'content=(?:"([^>]+?)"|\'(.+?)\')'
 | 
			
		||||
        property_re = r'property=[\'"]og:%s[\'"]' % re.escape(prop)
 | 
			
		||||
        template = r'<meta[^>]+?%s[^>]+?%s'
 | 
			
		||||
        return [
 | 
			
		||||
            template % (property_re, content_re),
 | 
			
		||||
            template % (content_re, property_re),
 | 
			
		||||
        ]
 | 
			
		||||
 | 
			
		||||
    def _og_search_property(self, prop, html, name=None, **kargs):
 | 
			
		||||
        if name is None:
 | 
			
		||||
            name = 'OpenGraph %s' % prop
 | 
			
		||||
        escaped = self._search_regex(self._og_regex(prop), html, name, flags=re.DOTALL, **kargs)
 | 
			
		||||
        escaped = self._search_regex(self._og_regexes(prop), html, name, flags=re.DOTALL, **kargs)
 | 
			
		||||
        if escaped is None:
 | 
			
		||||
            return None
 | 
			
		||||
        return unescapeHTML(escaped)
 | 
			
		||||
 | 
			
		||||
    def _og_search_thumbnail(self, html, **kargs):
 | 
			
		||||
@@ -334,8 +342,8 @@ class InfoExtractor(object):
 | 
			
		||||
        return self._og_search_property('title', html, **kargs)
 | 
			
		||||
 | 
			
		||||
    def _og_search_video_url(self, html, name='video url', secure=True, **kargs):
 | 
			
		||||
        regexes = [self._og_regex('video')]
 | 
			
		||||
        if secure: regexes.insert(0, self._og_regex('video:secure_url'))
 | 
			
		||||
        regexes = self._og_regexes('video')
 | 
			
		||||
        if secure: regexes = self._og_regexes('video:secure_url') + regexes
 | 
			
		||||
        return self._html_search_regex(regexes, html, name, **kargs)
 | 
			
		||||
 | 
			
		||||
    def _rta_search(self, html):
 | 
			
		||||
 
 | 
			
		||||
@@ -141,9 +141,9 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
            raise ExtractorError(u'Unable to extract video URL')
 | 
			
		||||
 | 
			
		||||
        # subtitles
 | 
			
		||||
        video_subtitles = self.extract_subtitles(video_id)
 | 
			
		||||
        video_subtitles = self.extract_subtitles(video_id, webpage)
 | 
			
		||||
        if self._downloader.params.get('listsubtitles', False):
 | 
			
		||||
            self._list_available_subtitles(video_id)
 | 
			
		||||
            self._list_available_subtitles(video_id, webpage)
 | 
			
		||||
            return
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
@@ -157,7 +157,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
            'age_limit': age_limit,
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    def _get_available_subtitles(self, video_id):
 | 
			
		||||
    def _get_available_subtitles(self, video_id, webpage):
 | 
			
		||||
        try:
 | 
			
		||||
            sub_list = self._download_webpage(
 | 
			
		||||
                'https://api.dailymotion.com/video/%s/subtitles?fields=id,language,url' % video_id,
 | 
			
		||||
 
 | 
			
		||||
@@ -33,5 +33,5 @@ class EitbIE(InfoExtractor):
 | 
			
		||||
            raise ExtractorError(u'Could not extract the Brightcove url')
 | 
			
		||||
        # The BrightcoveExperience object doesn't contain the video id, we set
 | 
			
		||||
        # it manually
 | 
			
		||||
        bc_url += '&%40videoPlayer={}'.format(chapter_id)
 | 
			
		||||
        bc_url += '&%40videoPlayer={0}'.format(chapter_id)
 | 
			
		||||
        return self.url_result(bc_url, BrightcoveIE.ie_key())
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										40
									
								
								youtube_dl/extractor/gamekings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								youtube_dl/extractor/gamekings.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,40 @@
 | 
			
		||||
import re
 | 
			
		||||
 | 
			
		||||
from .common import InfoExtractor
 | 
			
		||||
from ..utils import (
 | 
			
		||||
    determine_ext,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class GamekingsIE(InfoExtractor):
 | 
			
		||||
    _VALID_URL = r'http?://www\.gamekings\.tv/videos/(?P<name>[0-9a-z\-]+)'
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u"url": u"http://www.gamekings.tv/videos/phoenix-wright-ace-attorney-dual-destinies-review/",
 | 
			
		||||
        u'file': u'20130811.mp4',
 | 
			
		||||
        u'md5': u'2f32b1f7b80fdc5cb616efb4f387f8a3',
 | 
			
		||||
        u'info_dict': {
 | 
			
		||||
            u"title": u"Phoenix Wright: Ace Attorney \u2013 Dual Destinies Review",
 | 
			
		||||
            u"description": u"Melle en Steven hebben voor de review een week in de rechtbank doorbracht met Phoenix Wright: Ace Attorney - Dual Destinies.",
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url)
 | 
			
		||||
        name = mobj.group('name')
 | 
			
		||||
        webpage = self._download_webpage(url, name)
 | 
			
		||||
        video_url = self._og_search_video_url(webpage)
 | 
			
		||||
 | 
			
		||||
        video = re.search(r'[0-9]+', video_url)
 | 
			
		||||
        video_id = video.group(0)
 | 
			
		||||
 | 
			
		||||
        # Todo: add medium format
 | 
			
		||||
        video_url = video_url.replace(video_id, 'large/' + video_id)
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            'id': video_id,
 | 
			
		||||
            'ext': 'mp4',
 | 
			
		||||
            'url': video_url,
 | 
			
		||||
            'title': self._og_search_title(webpage),
 | 
			
		||||
            'description': self._og_search_description(webpage),
 | 
			
		||||
        }
 | 
			
		||||
@@ -55,15 +55,17 @@ class GenericIE(InfoExtractor):
 | 
			
		||||
            u'skip': u'There is a limit of 200 free downloads / month for the test song',
 | 
			
		||||
        },
 | 
			
		||||
        # embedded brightcove video
 | 
			
		||||
        # it also tests brightcove videos that need to set the 'Referer' in the
 | 
			
		||||
        # http requests
 | 
			
		||||
        {
 | 
			
		||||
            u'add_ie': ['Brightcove'],
 | 
			
		||||
            u'url': u'http://www.scientificamerican.com/article.cfm?id=soap-bubble-physics',
 | 
			
		||||
            u'url': u'http://www.bfmtv.com/video/bfmbusiness/cours-bourse/cours-bourse-l-analyse-technique-154522/',
 | 
			
		||||
            u'info_dict': {
 | 
			
		||||
                u'id': u'2365799484001',
 | 
			
		||||
                u'id': u'2765128793001',
 | 
			
		||||
                u'ext': u'mp4',
 | 
			
		||||
                u'title': u'Bubble Simulation',
 | 
			
		||||
                u'description': u'A visualization from a new computer model of foam behavior.',
 | 
			
		||||
                u'uploader': u'Scientific American',
 | 
			
		||||
                u'title': u'Le cours de bourse : l’analyse technique',
 | 
			
		||||
                u'description': u'md5:7e9ad046e968cb2d1114004aba466fd9',
 | 
			
		||||
                u'uploader': u'BFM BUSINESS',
 | 
			
		||||
            },
 | 
			
		||||
            u'params': {
 | 
			
		||||
                u'skip_download': True,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,10 @@
 | 
			
		||||
import re
 | 
			
		||||
import hashlib
 | 
			
		||||
 | 
			
		||||
from .common import InfoExtractor
 | 
			
		||||
from ..utils import determine_ext
 | 
			
		||||
 | 
			
		||||
_md5 = lambda s: hashlib.md5(s.encode('utf-8')).hexdigest()
 | 
			
		||||
 | 
			
		||||
class KankanIE(InfoExtractor):
 | 
			
		||||
    _VALID_URL = r'https?://(?:.*?\.)?kankan\.com/.+?/(?P<id>\d+)\.shtml'
 | 
			
		||||
@@ -30,7 +32,10 @@ class KankanIE(InfoExtractor):
 | 
			
		||||
                                                 video_id, u'Downloading video url info')
 | 
			
		||||
        ip = self._search_regex(r'ip:"(.+?)"', video_info_page, u'video url ip')
 | 
			
		||||
        path = self._search_regex(r'path:"(.+?)"', video_info_page, u'video url path')
 | 
			
		||||
        video_url = 'http://%s%s' % (ip, path)
 | 
			
		||||
        param1 = self._search_regex(r'param1:(\d+)', video_info_page, u'param1')
 | 
			
		||||
        param2 = self._search_regex(r'param2:(\d+)', video_info_page, u'param2')
 | 
			
		||||
        key = _md5('xl_mp43651' + param1 + param2)
 | 
			
		||||
        video_url = 'http://%s%s?key=%s&key1=%s' % (ip, path, key, param2)
 | 
			
		||||
 | 
			
		||||
        return {'id': video_id,
 | 
			
		||||
                'title': title,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,16 +1,19 @@
 | 
			
		||||
import re
 | 
			
		||||
import json
 | 
			
		||||
import xml.etree.ElementTree
 | 
			
		||||
 | 
			
		||||
from .common import InfoExtractor
 | 
			
		||||
from ..utils import (
 | 
			
		||||
    compat_urllib_parse_urlparse,
 | 
			
		||||
    compat_urlparse,
 | 
			
		||||
    get_meta_content,
 | 
			
		||||
    xpath_with_ns,
 | 
			
		||||
    ExtractorError,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class LivestreamIE(InfoExtractor):
 | 
			
		||||
    IE_NAME = u'livestream'
 | 
			
		||||
    _VALID_URL = r'http://new.livestream.com/.*?/(?P<event_name>.*?)(/videos/(?P<id>\d+))?/?$'
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u'url': u'http://new.livestream.com/CoheedandCambria/WebsterHall/videos/4719370',
 | 
			
		||||
@@ -54,3 +57,44 @@ class LivestreamIE(InfoExtractor):
 | 
			
		||||
            info = json.loads(self._download_webpage(api_url, video_id,
 | 
			
		||||
                                                     u'Downloading video info'))
 | 
			
		||||
            return self._extract_video_info(info)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
# The original version of Livestream uses a different system
 | 
			
		||||
class LivestreamOriginalIE(InfoExtractor):
 | 
			
		||||
    IE_NAME = u'livestream:original'
 | 
			
		||||
    _VALID_URL = r'https?://www\.livestream\.com/(?P<user>[^/]+)/video\?.*?clipId=(?P<id>.*?)(&|$)'
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u'url': u'http://www.livestream.com/dealbook/video?clipId=pla_8aa4a3f1-ba15-46a4-893b-902210e138fb',
 | 
			
		||||
        u'info_dict': {
 | 
			
		||||
            u'id': u'pla_8aa4a3f1-ba15-46a4-893b-902210e138fb',
 | 
			
		||||
            u'ext': u'flv',
 | 
			
		||||
            u'title': u'Spark 1 (BitCoin) with Cameron Winklevoss & Tyler Winklevoss of Winklevoss Capital',
 | 
			
		||||
        },
 | 
			
		||||
        u'params': {
 | 
			
		||||
            # rtmp
 | 
			
		||||
            u'skip_download': True,
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url)
 | 
			
		||||
        video_id = mobj.group('id')
 | 
			
		||||
        user = mobj.group('user')
 | 
			
		||||
        api_url = 'http://x{0}x.api.channel.livestream.com/2.0/clipdetails?extendedInfo=true&id={1}'.format(user, video_id)
 | 
			
		||||
 | 
			
		||||
        api_response = self._download_webpage(api_url, video_id)
 | 
			
		||||
        info = xml.etree.ElementTree.fromstring(api_response.encode('utf-8'))
 | 
			
		||||
        item = info.find('channel').find('item')
 | 
			
		||||
        ns = {'media': 'http://search.yahoo.com/mrss'}
 | 
			
		||||
        thumbnail_url = item.find(xpath_with_ns('media:thumbnail', ns)).attrib['url']
 | 
			
		||||
        # Remove the extension and number from the path (like 1.jpg)
 | 
			
		||||
        path = self._search_regex(r'(user-files/.+)_.*?\.jpg$', thumbnail_url, u'path')
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            'id': video_id,
 | 
			
		||||
            'title': item.find('title').text,
 | 
			
		||||
            'url': 'rtmp://extondemand.livestream.com/ondemand',
 | 
			
		||||
            'play_path': 'mp4:trans/dv15/mogulus-{0}.mp4'.format(path),
 | 
			
		||||
            'ext': 'flv',
 | 
			
		||||
            'thumbnail': thumbnail_url,
 | 
			
		||||
        }
 | 
			
		||||
 
 | 
			
		||||
@@ -62,18 +62,6 @@ class RTLnowIE(InfoExtractor):
 | 
			
		||||
            u'skip_download': True,
 | 
			
		||||
        },
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
        u'url': u'http://www.rtlnitronow.de/recht-ordnung/stadtpolizei-frankfurt-gerichtsvollzieher-leipzig.php?film_id=129679&player=1&season=1',
 | 
			
		||||
        u'file': u'129679.flv',
 | 
			
		||||
        u'info_dict': {
 | 
			
		||||
            u'upload_date': u'20131016', 
 | 
			
		||||
            u'title': u'Recht & Ordnung - Stadtpolizei Frankfurt/ Gerichtsvollzieher...',
 | 
			
		||||
            u'description': u'Stadtpolizei Frankfurt/ Gerichtsvollzieher Leipzig',
 | 
			
		||||
        },
 | 
			
		||||
        u'params': {
 | 
			
		||||
            u'skip_download': True,
 | 
			
		||||
        },
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
        u'url': u'http://www.n-tvnow.de/top-gear/episode-1-2013-01-01-00-00-00.php?film_id=124903&player=1&season=10',
 | 
			
		||||
        u'file': u'124903.flv',
 | 
			
		||||
 
 | 
			
		||||
@@ -29,17 +29,34 @@ class SoundcloudIE(InfoExtractor):
 | 
			
		||||
                    )
 | 
			
		||||
                    '''
 | 
			
		||||
    IE_NAME = u'soundcloud'
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u'url': u'http://soundcloud.com/ethmusic/lostin-powers-she-so-heavy',
 | 
			
		||||
        u'file': u'62986583.mp3',
 | 
			
		||||
        u'md5': u'ebef0a451b909710ed1d7787dddbf0d7',
 | 
			
		||||
        u'info_dict': {
 | 
			
		||||
            u"upload_date": u"20121011", 
 | 
			
		||||
            u"description": u"No Downloads untill we record the finished version this weekend, i was too pumped n i had to post it , earl is prolly gonna b hella p.o'd", 
 | 
			
		||||
            u"uploader": u"E.T. ExTerrestrial Music", 
 | 
			
		||||
            u"title": u"Lostin Powers - She so Heavy (SneakPreview) Adrian Ackers Blueprint 1"
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    _TESTS = [
 | 
			
		||||
        {
 | 
			
		||||
            u'url': u'http://soundcloud.com/ethmusic/lostin-powers-she-so-heavy',
 | 
			
		||||
            u'file': u'62986583.mp3',
 | 
			
		||||
            u'md5': u'ebef0a451b909710ed1d7787dddbf0d7',
 | 
			
		||||
            u'info_dict': {
 | 
			
		||||
                u"upload_date": u"20121011", 
 | 
			
		||||
                u"description": u"No Downloads untill we record the finished version this weekend, i was too pumped n i had to post it , earl is prolly gonna b hella p.o'd", 
 | 
			
		||||
                u"uploader": u"E.T. ExTerrestrial Music", 
 | 
			
		||||
                u"title": u"Lostin Powers - She so Heavy (SneakPreview) Adrian Ackers Blueprint 1"
 | 
			
		||||
            }
 | 
			
		||||
        },
 | 
			
		||||
        # not streamable song
 | 
			
		||||
        {
 | 
			
		||||
            u'url': u'https://soundcloud.com/the-concept-band/goldrushed-mastered?in=the-concept-band/sets/the-royal-concept-ep',
 | 
			
		||||
            u'info_dict': {
 | 
			
		||||
                u'id': u'47127627',
 | 
			
		||||
                u'ext': u'mp3',
 | 
			
		||||
                u'title': u'Goldrushed',
 | 
			
		||||
                u'uploader': u'The Royal Concept',
 | 
			
		||||
                u'upload_date': u'20120521',
 | 
			
		||||
            },
 | 
			
		||||
            u'params': {
 | 
			
		||||
                # rtmp
 | 
			
		||||
                u'skip_download': True,
 | 
			
		||||
            },
 | 
			
		||||
        },
 | 
			
		||||
    ]
 | 
			
		||||
 | 
			
		||||
    _CLIENT_ID = 'b45b1aa10f1ac2941910a7f0d10f8e28'
 | 
			
		||||
 | 
			
		||||
@@ -56,24 +73,39 @@ class SoundcloudIE(InfoExtractor):
 | 
			
		||||
        return 'http://api.soundcloud.com/resolve.json?url=' + url + '&client_id=' + cls._CLIENT_ID
 | 
			
		||||
 | 
			
		||||
    def _extract_info_dict(self, info, full_title=None, quiet=False):
 | 
			
		||||
        video_id = info['id']
 | 
			
		||||
        name = full_title or video_id
 | 
			
		||||
        track_id = compat_str(info['id'])
 | 
			
		||||
        name = full_title or track_id
 | 
			
		||||
        if quiet == False:
 | 
			
		||||
            self.report_extraction(name)
 | 
			
		||||
 | 
			
		||||
        thumbnail = info['artwork_url']
 | 
			
		||||
        if thumbnail is not None:
 | 
			
		||||
            thumbnail = thumbnail.replace('-large', '-t500x500')
 | 
			
		||||
        return {
 | 
			
		||||
            'id':       info['id'],
 | 
			
		||||
        result = {
 | 
			
		||||
            'id':       track_id,
 | 
			
		||||
            'url':      info['stream_url'] + '?client_id=' + self._CLIENT_ID,
 | 
			
		||||
            'uploader': info['user']['username'],
 | 
			
		||||
            'upload_date': unified_strdate(info['created_at']),
 | 
			
		||||
            'title':    info['title'],
 | 
			
		||||
            'ext':      u'mp3',
 | 
			
		||||
            'ext':      info.get('original_format', u'mp3'),
 | 
			
		||||
            'description': info['description'],
 | 
			
		||||
            'thumbnail': thumbnail,
 | 
			
		||||
        }
 | 
			
		||||
        if info.get('downloadable', False):
 | 
			
		||||
            result['url'] = 'https://api.soundcloud.com/tracks/{0}/download?client_id={1}'.format(track_id, self._CLIENT_ID)
 | 
			
		||||
        if not info.get('streamable', False):
 | 
			
		||||
            # We have to get the rtmp url
 | 
			
		||||
            stream_json = self._download_webpage(
 | 
			
		||||
                'http://api.soundcloud.com/i1/tracks/{0}/streams?client_id={1}'.format(track_id, self._CLIENT_ID),
 | 
			
		||||
                track_id, u'Downloading track url')
 | 
			
		||||
            rtmp_url = json.loads(stream_json)['rtmp_mp3_128_url']
 | 
			
		||||
            # The url doesn't have an rtmp app, we have to extract the playpath
 | 
			
		||||
            url, path = rtmp_url.split('mp3:', 1)
 | 
			
		||||
            result.update({
 | 
			
		||||
                'url': url,
 | 
			
		||||
                'play_path': 'mp3:' + path,
 | 
			
		||||
            })
 | 
			
		||||
        return result
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url, flags=re.VERBOSE)
 | 
			
		||||
@@ -106,70 +138,8 @@ class SoundcloudIE(InfoExtractor):
 | 
			
		||||
class SoundcloudSetIE(SoundcloudIE):
 | 
			
		||||
    _VALID_URL = r'^(?:https?://)?(?:www\.)?soundcloud\.com/([\w\d-]+)/sets/([\w\d-]+)(?:[?].*)?$'
 | 
			
		||||
    IE_NAME = u'soundcloud:set'
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u"url":"https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep",
 | 
			
		||||
        u"playlist": [
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"30510138.mp3",
 | 
			
		||||
                u"md5":"f9136bf103901728f29e419d2c70f55d",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20111213",
 | 
			
		||||
                    u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"D-D-Dance"
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"47127625.mp3",
 | 
			
		||||
                u"md5":"09b6758a018470570f8fd423c9453dd8",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20120521",
 | 
			
		||||
                    u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"The Royal Concept - Gimme Twice"
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"47127627.mp3",
 | 
			
		||||
                u"md5":"154abd4e418cea19c3b901f1e1306d9c",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20120521",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"Goldrushed"
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"47127629.mp3",
 | 
			
		||||
                u"md5":"2f5471edc79ad3f33a683153e96a79c1",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20120521",
 | 
			
		||||
                    u"description": u"The Royal Concept from Stockholm\r\nFilip / Povel / David / Magnus\r\nwww.royalconceptband.com",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"In the End"
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"47127631.mp3",
 | 
			
		||||
                u"md5":"f9ba87aa940af7213f98949254f1c6e2",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20120521",
 | 
			
		||||
                    u"description": u"The Royal Concept from Stockholm\r\nFilip / David / Povel / Magnus\r\nwww.theroyalconceptband.com",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"Knocked Up"
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
                u"file":"75206121.mp3",
 | 
			
		||||
                u"md5":"f9d1fe9406717e302980c30de4af9353",
 | 
			
		||||
                u"info_dict": {
 | 
			
		||||
                    u"upload_date": u"20130116",
 | 
			
		||||
                    u"description": u"The unreleased track World on Fire premiered on the CW's hit show Arrow (8pm/7pm central).  \r\nAs a gift to our fans we would like to offer you a free download of the track!  ",
 | 
			
		||||
                    u"uploader": u"The Royal Concept",
 | 
			
		||||
                    u"title": u"World On Fire"
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        ]
 | 
			
		||||
    }
 | 
			
		||||
    # it's in tests/test_playlists.py
 | 
			
		||||
    _TESTS = []
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url)
 | 
			
		||||
@@ -208,7 +178,7 @@ class SoundcloudUserIE(SoundcloudIE):
 | 
			
		||||
    IE_NAME = u'soundcloud:user'
 | 
			
		||||
 | 
			
		||||
    # it's in tests/test_playlists.py
 | 
			
		||||
    _TEST = None
 | 
			
		||||
    _TESTS = []
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url)
 | 
			
		||||
 
 | 
			
		||||
@@ -12,9 +12,9 @@ class SubtitlesInfoExtractor(InfoExtractor):
 | 
			
		||||
        return any([self._downloader.params.get('writesubtitles', False),
 | 
			
		||||
                    self._downloader.params.get('writeautomaticsub')])
 | 
			
		||||
 | 
			
		||||
    def _list_available_subtitles(self, video_id, webpage=None):
 | 
			
		||||
    def _list_available_subtitles(self, video_id, webpage):
 | 
			
		||||
        """ outputs the available subtitles for the video """
 | 
			
		||||
        sub_lang_list = self._get_available_subtitles(video_id)
 | 
			
		||||
        sub_lang_list = self._get_available_subtitles(video_id, webpage)
 | 
			
		||||
        auto_captions_list = self._get_available_automatic_caption(video_id, webpage)
 | 
			
		||||
        sub_lang = ",".join(list(sub_lang_list.keys()))
 | 
			
		||||
        self.to_screen(u'%s: Available subtitles for video: %s' %
 | 
			
		||||
@@ -23,7 +23,7 @@ class SubtitlesInfoExtractor(InfoExtractor):
 | 
			
		||||
        self.to_screen(u'%s: Available automatic captions for video: %s' %
 | 
			
		||||
                       (video_id, auto_lang))
 | 
			
		||||
 | 
			
		||||
    def extract_subtitles(self, video_id, video_webpage=None):
 | 
			
		||||
    def extract_subtitles(self, video_id, webpage):
 | 
			
		||||
        """
 | 
			
		||||
        returns {sub_lang: sub} ,{} if subtitles not found or None if the
 | 
			
		||||
        subtitles aren't requested.
 | 
			
		||||
@@ -32,9 +32,9 @@ class SubtitlesInfoExtractor(InfoExtractor):
 | 
			
		||||
            return None
 | 
			
		||||
        available_subs_list = {}
 | 
			
		||||
        if self._downloader.params.get('writeautomaticsub', False):
 | 
			
		||||
            available_subs_list.update(self._get_available_automatic_caption(video_id, video_webpage))
 | 
			
		||||
            available_subs_list.update(self._get_available_automatic_caption(video_id, webpage))
 | 
			
		||||
        if self._downloader.params.get('writesubtitles', False):
 | 
			
		||||
            available_subs_list.update(self._get_available_subtitles(video_id))
 | 
			
		||||
            available_subs_list.update(self._get_available_subtitles(video_id, webpage))
 | 
			
		||||
 | 
			
		||||
        if not available_subs_list:  # error, it didn't get the available subtitles
 | 
			
		||||
            return {}
 | 
			
		||||
@@ -74,7 +74,7 @@ class SubtitlesInfoExtractor(InfoExtractor):
 | 
			
		||||
            return
 | 
			
		||||
        return sub
 | 
			
		||||
 | 
			
		||||
    def _get_available_subtitles(self, video_id):
 | 
			
		||||
    def _get_available_subtitles(self, video_id, webpage):
 | 
			
		||||
        """
 | 
			
		||||
        returns {sub_lang: url} or {} if not available
 | 
			
		||||
        Must be redefined by the subclasses
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,14 @@
 | 
			
		||||
import json
 | 
			
		||||
import re
 | 
			
		||||
 | 
			
		||||
from .common import InfoExtractor
 | 
			
		||||
from .subtitles import SubtitlesInfoExtractor
 | 
			
		||||
 | 
			
		||||
from ..utils import (
 | 
			
		||||
    compat_str,
 | 
			
		||||
    RegexNotFoundError,
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
class TEDIE(InfoExtractor):
 | 
			
		||||
class TEDIE(SubtitlesInfoExtractor):
 | 
			
		||||
    _VALID_URL=r'''http://www\.ted\.com/
 | 
			
		||||
                   (
 | 
			
		||||
                        ((?P<type_playlist>playlists)/(?P<playlist_id>\d+)) # We have a playlist
 | 
			
		||||
@@ -32,33 +36,32 @@ class TEDIE(InfoExtractor):
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        m=re.match(self._VALID_URL, url, re.VERBOSE)
 | 
			
		||||
        if m.group('type_talk'):
 | 
			
		||||
            return [self._talk_info(url)]
 | 
			
		||||
            return self._talk_info(url)
 | 
			
		||||
        else :
 | 
			
		||||
            playlist_id=m.group('playlist_id')
 | 
			
		||||
            name=m.group('name')
 | 
			
		||||
            self.to_screen(u'Getting info of playlist %s: "%s"' % (playlist_id,name))
 | 
			
		||||
            return [self._playlist_videos_info(url,name,playlist_id)]
 | 
			
		||||
 | 
			
		||||
    def _playlist_videos_info(self,url,name,playlist_id=0):
 | 
			
		||||
 | 
			
		||||
    def _playlist_videos_info(self, url, name, playlist_id):
 | 
			
		||||
        '''Returns the videos of the playlist'''
 | 
			
		||||
        video_RE=r'''
 | 
			
		||||
                     <li\ id="talk_(\d+)"([.\s]*?)data-id="(?P<video_id>\d+)"
 | 
			
		||||
                     ([.\s]*?)data-playlist_item_id="(\d+)"
 | 
			
		||||
                     ([.\s]*?)data-mediaslug="(?P<mediaSlug>.+?)"
 | 
			
		||||
                     '''
 | 
			
		||||
        video_name_RE=r'<p\ class="talk-title"><a href="(?P<talk_url>/talks/(.+).html)">(?P<fullname>.+?)</a></p>'
 | 
			
		||||
        webpage=self._download_webpage(url, playlist_id, 'Downloading playlist webpage')
 | 
			
		||||
        m_videos=re.finditer(video_RE,webpage,re.VERBOSE)
 | 
			
		||||
        m_names=re.finditer(video_name_RE,webpage)
 | 
			
		||||
 | 
			
		||||
        webpage = self._download_webpage(
 | 
			
		||||
            url, playlist_id, u'Downloading playlist webpage')
 | 
			
		||||
        matches = re.finditer(
 | 
			
		||||
            r'<p\s+class="talk-title[^"]*"><a\s+href="(?P<talk_url>/talks/[^"]+\.html)">[^<]*</a></p>',
 | 
			
		||||
            webpage)
 | 
			
		||||
 | 
			
		||||
        playlist_title = self._html_search_regex(r'div class="headline">\s*?<h1>\s*?<span>(.*?)</span>',
 | 
			
		||||
                                                 webpage, 'playlist title')
 | 
			
		||||
 | 
			
		||||
        playlist_entries = []
 | 
			
		||||
        for m_video, m_name in zip(m_videos,m_names):
 | 
			
		||||
            talk_url='http://www.ted.com%s' % m_name.group('talk_url')
 | 
			
		||||
            playlist_entries.append(self.url_result(talk_url, 'TED'))
 | 
			
		||||
        return self.playlist_result(playlist_entries, playlist_id = playlist_id, playlist_title = playlist_title)
 | 
			
		||||
        playlist_entries = [
 | 
			
		||||
            self.url_result(u'http://www.ted.com' + m.group('talk_url'), 'TED')
 | 
			
		||||
            for m in matches
 | 
			
		||||
        ]
 | 
			
		||||
        return self.playlist_result(
 | 
			
		||||
            playlist_entries, playlist_id=playlist_id, playlist_title=playlist_title)
 | 
			
		||||
 | 
			
		||||
    def _talk_info(self, url, video_id=0):
 | 
			
		||||
        """Return the video for the talk in the url"""
 | 
			
		||||
@@ -81,16 +84,35 @@ class TEDIE(InfoExtractor):
 | 
			
		||||
            'ext': 'mp4',
 | 
			
		||||
            'url': stream['file'],
 | 
			
		||||
            'format': stream['id']
 | 
			
		||||
            } for stream in info['htmlStreams']]
 | 
			
		||||
        info = {
 | 
			
		||||
            'id': info['id'],
 | 
			
		||||
        } for stream in info['htmlStreams']]
 | 
			
		||||
 | 
			
		||||
        video_id = info['id']
 | 
			
		||||
 | 
			
		||||
        # subtitles
 | 
			
		||||
        video_subtitles = self.extract_subtitles(video_id, webpage)
 | 
			
		||||
        if self._downloader.params.get('listsubtitles', False):
 | 
			
		||||
            self._list_available_subtitles(video_id, webpage)
 | 
			
		||||
            return
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            'id': video_id,
 | 
			
		||||
            'title': title,
 | 
			
		||||
            'thumbnail': thumbnail,
 | 
			
		||||
            'description': desc,
 | 
			
		||||
            'subtitles': video_subtitles,
 | 
			
		||||
            'formats': formats,
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        # TODO: Remove when #980 has been merged
 | 
			
		||||
        info.update(info['formats'][-1])
 | 
			
		||||
 | 
			
		||||
        return info
 | 
			
		||||
    def _get_available_subtitles(self, video_id, webpage):
 | 
			
		||||
        try:
 | 
			
		||||
            options = self._search_regex(r'(?:<select name="subtitles_language_select" id="subtitles_language_select">)(.*?)(?:</select>)', webpage, 'subtitles_language_select', flags=re.DOTALL)
 | 
			
		||||
            languages = re.findall(r'(?:<option value=")(\S+)"', options)
 | 
			
		||||
            if languages:
 | 
			
		||||
                sub_lang_list = {}
 | 
			
		||||
                for l in languages:
 | 
			
		||||
                    url = 'http://www.ted.com/talks/subtitles/id/%s/lang/%s/format/srt' % (video_id, l)
 | 
			
		||||
                    sub_lang_list[l] = url
 | 
			
		||||
                return sub_lang_list
 | 
			
		||||
        except RegexNotFoundError as err:
 | 
			
		||||
            self._downloader.report_warning(u'video doesn\'t have subtitles')
 | 
			
		||||
        return {}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										41
									
								
								youtube_dl/extractor/tvp.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								youtube_dl/extractor/tvp.py
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,41 @@
 | 
			
		||||
import json
 | 
			
		||||
import re
 | 
			
		||||
 | 
			
		||||
from .common import InfoExtractor
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class TvpIE(InfoExtractor):
 | 
			
		||||
    IE_NAME = u'tvp.pl'
 | 
			
		||||
    _VALID_URL = r'https?://www\.tvp\.pl/.*?wideo/(?P<date>\d+)/(?P<id>\d+)'
 | 
			
		||||
 | 
			
		||||
    _TEST = {
 | 
			
		||||
        u'url': u'http://www.tvp.pl/warszawa/magazyny/campusnews/wideo/31102013/12878238',
 | 
			
		||||
        u'md5': u'148408967a6a468953c0a75cbdaf0d7a',
 | 
			
		||||
        u'file': u'12878238.wmv',
 | 
			
		||||
        u'info_dict': {
 | 
			
		||||
            u'title': u'31.10.2013 - Odcinek 2',
 | 
			
		||||
            u'description': u'31.10.2013 - Odcinek 2',
 | 
			
		||||
        },
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    def _real_extract(self, url):
 | 
			
		||||
        mobj = re.match(self._VALID_URL, url)
 | 
			
		||||
        video_id = mobj.group('id')
 | 
			
		||||
        webpage = self._download_webpage(url, video_id)
 | 
			
		||||
        json_url = 'http://www.tvp.pl/pub/stat/videofileinfo?video_id=%s' % video_id
 | 
			
		||||
        json_params = self._download_webpage(
 | 
			
		||||
            json_url, video_id, u"Downloading video metadata")
 | 
			
		||||
 | 
			
		||||
        params = json.loads(json_params)
 | 
			
		||||
        self.report_extraction(video_id)
 | 
			
		||||
        video_url = params['video_url']
 | 
			
		||||
 | 
			
		||||
        title = self._og_search_title(webpage, fatal=True)
 | 
			
		||||
        return {
 | 
			
		||||
            'id': video_id,
 | 
			
		||||
            'title': title,
 | 
			
		||||
            'ext': 'wmv',
 | 
			
		||||
            'url': video_url,
 | 
			
		||||
            'description': self._og_search_description(webpage),
 | 
			
		||||
            'thumbnail': self._og_search_thumbnail(webpage),
 | 
			
		||||
        }
 | 
			
		||||
@@ -27,7 +27,7 @@ class VineIE(InfoExtractor):
 | 
			
		||||
        video_url = self._html_search_regex(r'<meta property="twitter:player:stream" content="(.+?)"',
 | 
			
		||||
            webpage, u'video URL')
 | 
			
		||||
 | 
			
		||||
        uploader = self._html_search_regex(r'<div class="user">.*?<h2>(.+?)</h2>',
 | 
			
		||||
        uploader = self._html_search_regex(r'<p class="username">(.*?)</p>',
 | 
			
		||||
            webpage, u'uploader', fatal=False, flags=re.DOTALL)
 | 
			
		||||
 | 
			
		||||
        return [{
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,7 @@ from ..utils import (
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
class XNXXIE(InfoExtractor):
 | 
			
		||||
    _VALID_URL = r'^(?:https?://)?video\.xnxx\.com/video([0-9]+)/(.*)'
 | 
			
		||||
    _VALID_URL = r'^(?:https?://)?(?:video|www)\.xnxx\.com/video([0-9]+)/(.*)'
 | 
			
		||||
    VIDEO_URL_RE = r'flv_url=(.*?)&'
 | 
			
		||||
    VIDEO_TITLE_RE = r'<title>(.*?)\s+-\s+XNXX.COM'
 | 
			
		||||
    VIDEO_THUMB_RE = r'url_bigthumb=(.*?)&'
 | 
			
		||||
 
 | 
			
		||||
@@ -1019,6 +1019,8 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
        """Turn the encrypted s field into a working signature"""
 | 
			
		||||
 | 
			
		||||
        if player_url is not None:
 | 
			
		||||
            if player_url.startswith(u'//'):
 | 
			
		||||
                player_url = u'https:' + player_url
 | 
			
		||||
            try:
 | 
			
		||||
                player_id = (player_url, len(s))
 | 
			
		||||
                if player_id not in self._player_cache:
 | 
			
		||||
@@ -1082,7 +1084,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
        else:
 | 
			
		||||
            raise ExtractorError(u'Unable to decrypt signature, key length %d not supported; retrying might work' % (len(s)))
 | 
			
		||||
 | 
			
		||||
    def _get_available_subtitles(self, video_id):
 | 
			
		||||
    def _get_available_subtitles(self, video_id, webpage):
 | 
			
		||||
        try:
 | 
			
		||||
            sub_list = self._download_webpage(
 | 
			
		||||
                'http://video.google.com/timedtext?hl=en&type=list&v=%s' % video_id,
 | 
			
		||||
@@ -1098,7 +1100,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
            params = compat_urllib_parse.urlencode({
 | 
			
		||||
                'lang': lang,
 | 
			
		||||
                'v': video_id,
 | 
			
		||||
                'fmt': self._downloader.params.get('subtitlesformat'),
 | 
			
		||||
                'fmt': self._downloader.params.get('subtitlesformat', 'srt'),
 | 
			
		||||
                'name': l[0].encode('utf-8'),
 | 
			
		||||
            })
 | 
			
		||||
            url = u'http://www.youtube.com/api/timedtext?' + params
 | 
			
		||||
@@ -1111,7 +1113,7 @@ class YoutubeIE(YoutubeBaseInfoExtractor, SubtitlesInfoExtractor):
 | 
			
		||||
    def _get_available_automatic_caption(self, video_id, webpage):
 | 
			
		||||
        """We need the webpage for getting the captions url, pass it as an
 | 
			
		||||
           argument to speed up the process."""
 | 
			
		||||
        sub_format = self._downloader.params.get('subtitlesformat')
 | 
			
		||||
        sub_format = self._downloader.params.get('subtitlesformat', 'srt')
 | 
			
		||||
        self.to_screen(u'%s: Looking for automatic captions' % video_id)
 | 
			
		||||
        mobj = re.search(r';ytplayer.config = ({.*?});', webpage)
 | 
			
		||||
        err_msg = u'Couldn\'t find automatic captions for %s' % video_id
 | 
			
		||||
@@ -1592,20 +1594,31 @@ class YoutubeChannelIE(InfoExtractor):
 | 
			
		||||
        # Download channel page
 | 
			
		||||
        channel_id = mobj.group(1)
 | 
			
		||||
        video_ids = []
 | 
			
		||||
        url = 'https://www.youtube.com/channel/%s/videos' % channel_id
 | 
			
		||||
        channel_page = self._download_webpage(url, channel_id)
 | 
			
		||||
        if re.search(r'channel-header-autogenerated-label', channel_page) is not None:
 | 
			
		||||
            autogenerated = True
 | 
			
		||||
        else:
 | 
			
		||||
            autogenerated = False
 | 
			
		||||
 | 
			
		||||
        # Download all channel pages using the json-based channel_ajax query
 | 
			
		||||
        for pagenum in itertools.count(1):
 | 
			
		||||
            url = self._MORE_PAGES_URL % (pagenum, channel_id)
 | 
			
		||||
            page = self._download_webpage(url, channel_id,
 | 
			
		||||
                                          u'Downloading page #%s' % pagenum)
 | 
			
		||||
 | 
			
		||||
            page = json.loads(page)
 | 
			
		||||
 | 
			
		||||
            ids_in_page = self.extract_videos_from_page(page['content_html'])
 | 
			
		||||
            video_ids.extend(ids_in_page)
 | 
			
		||||
 | 
			
		||||
            if self._MORE_PAGES_INDICATOR not in page['load_more_widget_html']:
 | 
			
		||||
                break
 | 
			
		||||
        if autogenerated:
 | 
			
		||||
            # The videos are contained in a single page
 | 
			
		||||
            # the ajax pages can't be used, they are empty
 | 
			
		||||
            video_ids = self.extract_videos_from_page(channel_page)
 | 
			
		||||
        else:
 | 
			
		||||
            # Download all channel pages using the json-based channel_ajax query
 | 
			
		||||
            for pagenum in itertools.count(1):
 | 
			
		||||
                url = self._MORE_PAGES_URL % (pagenum, channel_id)
 | 
			
		||||
                page = self._download_webpage(url, channel_id,
 | 
			
		||||
                                              u'Downloading page #%s' % pagenum)
 | 
			
		||||
    
 | 
			
		||||
                page = json.loads(page)
 | 
			
		||||
    
 | 
			
		||||
                ids_in_page = self.extract_videos_from_page(page['content_html'])
 | 
			
		||||
                video_ids.extend(ids_in_page)
 | 
			
		||||
    
 | 
			
		||||
                if self._MORE_PAGES_INDICATOR not in page['load_more_widget_html']:
 | 
			
		||||
                    break
 | 
			
		||||
 | 
			
		||||
        self._downloader.to_screen(u'[youtube] Channel %s: Found %i videos' % (channel_id, len(video_ids)))
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,2 +1,2 @@
 | 
			
		||||
 | 
			
		||||
__version__ = '2013.11.06.1'
 | 
			
		||||
__version__ = '2013.11.15.1'
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user