diff options
Diffstat (limited to 'test')
| -rw-r--r-- | test/helper.py | 16 | ||||
| -rw-r--r-- | test/swftests/.gitignore | 1 | ||||
| -rw-r--r-- | test/swftests/ArrayAccess.as | 19 | ||||
| -rw-r--r-- | test/swftests/ClassCall.as | 17 | ||||
| -rw-r--r-- | test/swftests/ClassConstruction.as | 15 | ||||
| -rw-r--r-- | test/swftests/LocalVars.as | 13 | ||||
| -rw-r--r-- | test/swftests/PrivateCall.as | 21 | ||||
| -rw-r--r-- | test/swftests/StaticAssignment.as | 13 | ||||
| -rw-r--r-- | test/swftests/StaticRetrieval.as | 16 | ||||
| -rw-r--r-- | test/test_YoutubeDL.py | 6 | ||||
| -rw-r--r-- | test/test_age_restriction.py | 2 | ||||
| -rw-r--r-- | test/test_all_urls.py | 5 | ||||
| -rw-r--r-- | test/test_download.py | 34 | ||||
| -rw-r--r-- | test/test_playlists.py | 111 | ||||
| -rw-r--r-- | test/test_subtitles.py | 2 | ||||
| -rw-r--r-- | test/test_swfinterp.py | 77 | ||||
| -rw-r--r-- | test/test_utils.py | 2 | ||||
| -rw-r--r-- | test/test_youtube_lists.py | 4 | ||||
| -rw-r--r-- | test/test_youtube_signature.py | 74 | 
19 files changed, 379 insertions, 69 deletions
| diff --git a/test/helper.py b/test/helper.py index 230d2bd67..22d763860 100644 --- a/test/helper.py +++ b/test/helper.py @@ -117,8 +117,9 @@ def expect_info_dict(self, expected_dict, got_dict):                  u'invalid value for field %s, expected %r, got %r' % (info_field, expected, got))      # Check for the presence of mandatory fields -    for key in ('id', 'url', 'title', 'ext'): -        self.assertTrue(got_dict.get(key), 'Missing mandatory field %s' % key) +    if got_dict.get('_type') != 'playlist': +        for key in ('id', 'url', 'title', 'ext'): +            self.assertTrue(got_dict.get(key), 'Missing mandatory field %s' % key)      # Check for mandatory fields that are automatically set by YoutubeDL      for key in ['webpage_url', 'extractor', 'extractor_key']:          self.assertTrue(got_dict.get(key), u'Missing field: %s' % key) @@ -137,8 +138,8 @@ def expect_info_dict(self, expected_dict, got_dict):  def assertRegexpMatches(self, text, regexp, msg=None): -    if hasattr(self, 'assertRegexpMatches'): -        return self.assertRegexpMatches(text, regexp, msg) +    if hasattr(self, 'assertRegexp'): +        return self.assertRegexp(text, regexp, msg)      else:          m = re.match(regexp, text)          if not m: @@ -148,3 +149,10 @@ def assertRegexpMatches(self, text, regexp, msg=None):              else:                  msg = note + ', ' + msg              self.assertTrue(m, msg) + + +def assertGreaterEqual(self, got, expected, msg=None): +    if not (got >= expected): +        if msg is None: +            msg = '%r not greater than or equal to %r' % (got, expected) +        self.assertTrue(got >= expected, msg) diff --git a/test/swftests/.gitignore b/test/swftests/.gitignore new file mode 100644 index 000000000..da97ff7ca --- /dev/null +++ b/test/swftests/.gitignore @@ -0,0 +1 @@ +*.swf diff --git a/test/swftests/ArrayAccess.as b/test/swftests/ArrayAccess.as new file mode 100644 index 000000000..e22caa386 --- /dev/null +++ b/test/swftests/ArrayAccess.as @@ -0,0 +1,19 @@ +// input: [["a", "b", "c", "d"]] +// output: ["c", "b", "a", "d"] + +package { +public class ArrayAccess { +    public static function main(ar:Array):Array { +    	var aa:ArrayAccess = new ArrayAccess(); +    	return aa.f(ar, 2); +    } + +    private function f(ar:Array, num:Number):Array{ +        var x:String = ar[0]; +        var y:String = ar[num % ar.length]; +        ar[0] = y; +        ar[num] = x; +        return ar; +    } +} +} diff --git a/test/swftests/ClassCall.as b/test/swftests/ClassCall.as new file mode 100644 index 000000000..aef58daf3 --- /dev/null +++ b/test/swftests/ClassCall.as @@ -0,0 +1,17 @@ +// input: [] +// output: 121 + +package { +public class ClassCall { +    public static function main():int{ +    	var f:OtherClass = new OtherClass(); +        return f.func(100,20); +    } +} +} + +class OtherClass { +	public function func(x: int, y: int):int { +		return x+y+1; +	} +} diff --git a/test/swftests/ClassConstruction.as b/test/swftests/ClassConstruction.as new file mode 100644 index 000000000..436479f8f --- /dev/null +++ b/test/swftests/ClassConstruction.as @@ -0,0 +1,15 @@ +// input: [] +// output: 0 + +package { +public class ClassConstruction { +    public static function main():int{ +    	var f:Foo = new Foo(); +        return 0; +    } +} +} + +class Foo { + +} diff --git a/test/swftests/LocalVars.as b/test/swftests/LocalVars.as new file mode 100644 index 000000000..b2911a9f3 --- /dev/null +++ b/test/swftests/LocalVars.as @@ -0,0 +1,13 @@ +// input: [1, 2] +// output: 3 + +package { +public class LocalVars { +    public static function main(a:int, b:int):int{ +        var c:int = a + b + b; +        var d:int = c - b; +        var e:int = d; +        return e; +    } +} +} diff --git a/test/swftests/PrivateCall.as b/test/swftests/PrivateCall.as new file mode 100644 index 000000000..f1c110a37 --- /dev/null +++ b/test/swftests/PrivateCall.as @@ -0,0 +1,21 @@ +// input: [] +// output: 9 + +package { +public class PrivateCall { +    public static function main():int{ +    	var f:OtherClass = new OtherClass(); +        return f.func(); +    } +} +} + +class OtherClass { +	private function pf():int { +		return 9; +	} + +	public function func():int { +		return this.pf(); +	} +} diff --git a/test/swftests/StaticAssignment.as b/test/swftests/StaticAssignment.as new file mode 100644 index 000000000..b061c219d --- /dev/null +++ b/test/swftests/StaticAssignment.as @@ -0,0 +1,13 @@ +// input: [1] +// output: 1 + +package { +public class StaticAssignment { +	public static var v:int; + +    public static function main(a:int):int{ +        v = a; +        return v; +    } +} +} diff --git a/test/swftests/StaticRetrieval.as b/test/swftests/StaticRetrieval.as new file mode 100644 index 000000000..c8352d819 --- /dev/null +++ b/test/swftests/StaticRetrieval.as @@ -0,0 +1,16 @@ +// input: [] +// output: 1 + +package { +public class StaticRetrieval { +	public static var v:int; + +    public static function main():int{ +        if (v) { +        	return 0; +        } else { +        	return 1; +        } +    } +} +} diff --git a/test/test_YoutubeDL.py b/test/test_YoutubeDL.py index 8735013f7..ab61e1976 100644 --- a/test/test_YoutubeDL.py +++ b/test/test_YoutubeDL.py @@ -67,7 +67,7 @@ class TestFormatSelection(unittest.TestCase):          downloaded = ydl.downloaded_info_dicts[0]          self.assertEqual(downloaded['ext'], 'mp4') -        # No prefer_free_formats => prefer mp4 and flv for greater compatibilty +        # No prefer_free_formats => prefer mp4 and flv for greater compatibility          ydl = YDL()          ydl.params['prefer_free_formats'] = False          formats = [ @@ -221,7 +221,7 @@ class TestFormatSelection(unittest.TestCase):              '138', '137', '248', '136', '247', '135', '246',              '245', '244', '134', '243', '133', '242', '160',              # Dash audio -            '141', '172', '140', '139', '171', +            '141', '172', '140', '171', '139',          ]          for f1id, f2id in zip(order, order[1:]): @@ -279,7 +279,7 @@ class TestFormatSelection(unittest.TestCase):          self.assertEqual(ydl._format_note({}), '')          assertRegexpMatches(self, ydl._format_note({              'vbr': 10, -        }), '^x\s*10k$') +        }), '^\s*10k$')  if __name__ == '__main__':      unittest.main() diff --git a/test/test_age_restriction.py b/test/test_age_restriction.py index c9cdb96cb..71e80b037 100644 --- a/test/test_age_restriction.py +++ b/test/test_age_restriction.py @@ -13,7 +13,7 @@ from youtube_dl import YoutubeDL  def _download_restricted(url, filename, age): -    """ Returns true iff the file has been downloaded """ +    """ Returns true if the file has been downloaded """      params = {          'age_limit': age, diff --git a/test/test_all_urls.py b/test/test_all_urls.py index 4b56137ce..b1ad30bf1 100644 --- a/test/test_all_urls.py +++ b/test/test_all_urls.py @@ -15,7 +15,6 @@ from youtube_dl.extractor import (      FacebookIE,      gen_extractors,      JustinTVIE, -    PBSIE,      YoutubeIE,  ) @@ -69,9 +68,6 @@ class TestAllURLsMatching(unittest.TestCase):      def test_youtube_show_matching(self):          self.assertMatch('http://www.youtube.com/show/airdisasters', ['youtube:show']) -    def test_youtube_truncated(self): -        self.assertMatch('http://www.youtube.com/watch?', ['youtube:truncated_url']) -      def test_youtube_search_matching(self):          self.assertMatch('http://www.youtube.com/results?search_query=making+mustard', ['youtube:search_url'])          self.assertMatch('https://www.youtube.com/results?baz=bar&search_query=youtube-dl+test+video&filters=video&lclk=video', ['youtube:search_url']) @@ -103,6 +99,7 @@ class TestAllURLsMatching(unittest.TestCase):      def test_facebook_matching(self):          self.assertTrue(FacebookIE.suitable('https://www.facebook.com/Shiniknoh#!/photo.php?v=10153317450565268')) +        self.assertTrue(FacebookIE.suitable('https://www.facebook.com/cindyweather?fref=ts#!/photo.php?v=10152183998945793'))      def test_no_duplicates(self):          ies = gen_extractors() diff --git a/test/test_download.py b/test/test_download.py index f171c10ba..c8d4ec2c8 100644 --- a/test/test_download.py +++ b/test/test_download.py @@ -10,7 +10,6 @@ from test.helper import (      get_params,      gettestcases,      expect_info_dict, -    md5,      try_rm,      report_warning,  ) @@ -24,7 +23,6 @@ import socket  import youtube_dl.YoutubeDL  from youtube_dl.utils import (      compat_http_client, -    compat_str,      compat_urllib_error,      compat_HTTPError,      DownloadError, @@ -65,15 +63,21 @@ def generator(test_case):      def test_template(self):          ie = youtube_dl.extractor.get_info_extractor(test_case['name'])          other_ies = [get_info_extractor(ie_key) for ie_key in test_case.get('add_ie', [])] +        is_playlist = any(k.startswith('playlist') for k in test_case) +        test_cases = test_case.get( +            'playlist', [] if is_playlist else [test_case]) +          def print_skipping(reason):              print('Skipping %s: %s' % (test_case['name'], reason))          if not ie.working():              print_skipping('IE marked as not _WORKING')              return -        if 'playlist' not in test_case: -            info_dict = test_case.get('info_dict', {}) -            if not test_case.get('file') and not (info_dict.get('id') and info_dict.get('ext')): + +        for tc in test_cases: +            info_dict = tc.get('info_dict', {}) +            if not tc.get('file') and not (info_dict.get('id') and info_dict.get('ext')):                  raise Exception('Test definition incorrect. The output file cannot be known. Are both \'id\' and \'ext\' keys present?') +          if 'skip' in test_case:              print_skipping(test_case['skip'])              return @@ -83,6 +87,9 @@ def generator(test_case):                  return          params = get_params(test_case.get('params', {})) +        if is_playlist and 'playlist' not in test_case: +            params.setdefault('extract_flat', True) +            params.setdefault('skip_download', True)          ydl = YoutubeDL(params)          ydl.add_default_info_extractors() @@ -95,7 +102,6 @@ def generator(test_case):          def get_tc_filename(tc):              return tc.get('file') or ydl.prepare_filename(tc.get('info_dict', {})) -        test_cases = test_case.get('playlist', [test_case])          def try_rm_tcs_files():              for tc in test_cases:                  tc_filename = get_tc_filename(tc) @@ -107,7 +113,10 @@ def generator(test_case):              try_num = 1              while True:                  try: -                    ydl.download([test_case['url']]) +                    # We're not using .download here sine that is just a shim +                    # for outside error handling, and returns the exit code +                    # instead of the result dict. +                    res_dict = ydl.extract_info(test_case['url'])                  except (DownloadError, ExtractorError) as err:                      # Check if the exception is not a network related one                      if not err.exc_info[0] in (compat_urllib_error.URLError, socket.timeout, UnavailableVideoError, compat_http_client.BadStatusLine) or (err.exc_info[0] == compat_HTTPError and err.exc_info[1].code == 503): @@ -123,6 +132,17 @@ def generator(test_case):                  else:                      break +            if is_playlist: +                self.assertEqual(res_dict['_type'], 'playlist') +                expect_info_dict(self, test_case.get('info_dict', {}), res_dict) +            if 'playlist_mincount' in test_case: +                self.assertGreaterEqual( +                    len(res_dict['entries']), +                    test_case['playlist_mincount'], +                    'Expected at least %d in playlist %s, but got only %d' % ( +                        test_case['playlist_mincount'], test_case['url'], +                        len(res_dict['entries']))) +              for tc in test_cases:                  tc_filename = get_tc_filename(tc)                  if not test_case.get('params', {}).get('skip_download', False): diff --git a/test/test_playlists.py b/test/test_playlists.py index cc871698a..6448fea38 100644 --- a/test/test_playlists.py +++ b/test/test_playlists.py @@ -1,6 +1,17 @@  #!/usr/bin/env python  # encoding: utf-8 +## DEPRECATED FILE! +# Add new tests to the extractors themselves, like this: +# _TEST = { +#    'url': 'http://example.com/playlist/42', +#    'playlist_mincount': 99, +#    'info_dict': { +#        'id': '42', +#        'title': 'Playlist number forty-two', +#    } +# } +  from __future__ import unicode_literals  # Allow direct execution @@ -11,6 +22,7 @@ sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))  from test.helper import (      assertRegexpMatches, +    assertGreaterEqual,      expect_info_dict,      FakeYDL,  ) @@ -28,7 +40,9 @@ from youtube_dl.extractor import (      SoundcloudSetIE,      SoundcloudUserIE,      SoundcloudPlaylistIE, +    TeacherTubeUserIE,      LivestreamIE, +    LivestreamOriginalIE,      NHLVideocenterIE,      BambuserChannelIE,      BandcampAlbumIE, @@ -39,6 +53,7 @@ from youtube_dl.extractor import (      KhanAcademyIE,      EveryonesMixtapeIE,      RutubeChannelIE, +    RutubePersonIE,      GoogleSearchIE,      GenericIE,      TEDIE, @@ -68,8 +83,8 @@ class TestPlaylists(unittest.TestCase):          ie = DailymotionUserIE(dl)          result = ie.extract('https://www.dailymotion.com/user/nqtv')          self.assertIsPlaylist(result) +        assertGreaterEqual(self, len(result['entries']), 100)          self.assertEqual(result['title'], 'Rémi Gaillard') -        self.assertTrue(len(result['entries']) >= 100)      def test_vimeo_channel(self):          dl = FakeYDL() @@ -108,15 +123,15 @@ class TestPlaylists(unittest.TestCase):          ie = VineUserIE(dl)          result = ie.extract('https://vine.co/Visa')          self.assertIsPlaylist(result) -        self.assertTrue(len(result['entries']) >= 50) +        assertGreaterEqual(self, len(result['entries']), 47)      def test_ustream_channel(self):          dl = FakeYDL()          ie = UstreamChannelIE(dl) -        result = ie.extract('http://www.ustream.tv/channel/young-americans-for-liberty') +        result = ie.extract('http://www.ustream.tv/channel/channeljapan')          self.assertIsPlaylist(result) -        self.assertEqual(result['id'], '5124905') -        self.assertTrue(len(result['entries']) >= 6) +        self.assertEqual(result['id'], '10874166') +        assertGreaterEqual(self, len(result['entries']), 54)      def test_soundcloud_set(self):          dl = FakeYDL() @@ -124,7 +139,7 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('https://soundcloud.com/the-concept-band/sets/the-royal-concept-ep')          self.assertIsPlaylist(result)          self.assertEqual(result['title'], 'The Royal Concept EP') -        self.assertTrue(len(result['entries']) >= 6) +        assertGreaterEqual(self, len(result['entries']), 6)      def test_soundcloud_user(self):          dl = FakeYDL() @@ -132,7 +147,15 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('https://soundcloud.com/the-concept-band')          self.assertIsPlaylist(result)          self.assertEqual(result['id'], '9615865') -        self.assertTrue(len(result['entries']) >= 12) +        assertGreaterEqual(self, len(result['entries']), 12) + +    def test_soundcloud_likes(self): +        dl = FakeYDL() +        ie = SoundcloudUserIE(dl) +        result = ie.extract('https://soundcloud.com/the-concept-band/likes') +        self.assertIsPlaylist(result) +        self.assertEqual(result['id'], '9615865') +        assertGreaterEqual(self, len(result['entries']), 1)      def test_soundcloud_playlist(self):          dl = FakeYDL() @@ -142,7 +165,7 @@ class TestPlaylists(unittest.TestCase):          self.assertEqual(result['id'], '4110309')          self.assertEqual(result['title'], 'TILT Brass - Bowery Poetry Club, August \'03 [Non-Site SCR 02]')          assertRegexpMatches( -            self, result['description'], r'TILT Brass - Bowery Poetry Club') +            self, result['description'], r'.*?TILT Brass - Bowery Poetry Club')          self.assertEqual(len(result['entries']), 6)      def test_livestream_event(self): @@ -151,7 +174,15 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('http://new.livestream.com/tedx/cityenglish')          self.assertIsPlaylist(result)          self.assertEqual(result['title'], 'TEDCity2.0 (English)') -        self.assertTrue(len(result['entries']) >= 4) +        assertGreaterEqual(self, len(result['entries']), 4) + +    def test_livestreamoriginal_folder(self): +        dl = FakeYDL() +        ie = LivestreamOriginalIE(dl) +        result = ie.extract('https://www.livestream.com/newplay/folder?dirId=a07bf706-d0e4-4e75-a747-b021d84f2fd3') +        self.assertIsPlaylist(result) +        self.assertEqual(result['id'], 'a07bf706-d0e4-4e75-a747-b021d84f2fd3') +        assertGreaterEqual(self, len(result['entries']), 28)      def test_nhl_videocenter(self):          dl = FakeYDL() @@ -168,15 +199,15 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('http://bambuser.com/channel/pixelversity')          self.assertIsPlaylist(result)          self.assertEqual(result['title'], 'pixelversity') -        self.assertTrue(len(result['entries']) >= 60) +        assertGreaterEqual(self, len(result['entries']), 60)      def test_bandcamp_album(self):          dl = FakeYDL()          ie = BandcampAlbumIE(dl) -        result = ie.extract('http://mpallante.bandcamp.com/album/nightmare-night-ep') +        result = ie.extract('http://nightbringer.bandcamp.com/album/hierophany-of-the-open-grave')          self.assertIsPlaylist(result) -        self.assertEqual(result['title'], 'Nightmare Night EP') -        self.assertTrue(len(result['entries']) >= 4) +        self.assertEqual(result['title'], 'Hierophany of the Open Grave') +        assertGreaterEqual(self, len(result['entries']), 9)      def test_smotri_community(self):          dl = FakeYDL() @@ -185,7 +216,7 @@ class TestPlaylists(unittest.TestCase):          self.assertIsPlaylist(result)          self.assertEqual(result['id'], 'kommuna')          self.assertEqual(result['title'], 'КПРФ') -        self.assertTrue(len(result['entries']) >= 4) +        assertGreaterEqual(self, len(result['entries']), 4)      def test_smotri_user(self):          dl = FakeYDL() @@ -194,7 +225,7 @@ class TestPlaylists(unittest.TestCase):          self.assertIsPlaylist(result)          self.assertEqual(result['id'], 'inspector')          self.assertEqual(result['title'], 'Inspector') -        self.assertTrue(len(result['entries']) >= 9) +        assertGreaterEqual(self, len(result['entries']), 9)      def test_AcademicEarthCourse(self):          dl = FakeYDL() @@ -209,20 +240,20 @@ class TestPlaylists(unittest.TestCase):      def test_ivi_compilation(self):          dl = FakeYDL()          ie = IviCompilationIE(dl) -        result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel') +        result = ie.extract('http://www.ivi.ru/watch/dvoe_iz_lartsa')          self.assertIsPlaylist(result) -        self.assertEqual(result['id'], 'dezhurnyi_angel') -        self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012)') -        self.assertTrue(len(result['entries']) >= 23) +        self.assertEqual(result['id'], 'dvoe_iz_lartsa') +        self.assertEqual(result['title'], 'Двое из ларца (2006 - 2008)') +        assertGreaterEqual(self, len(result['entries']), 24)      def test_ivi_compilation_season(self):          dl = FakeYDL()          ie = IviCompilationIE(dl) -        result = ie.extract('http://www.ivi.ru/watch/dezhurnyi_angel/season2') +        result = ie.extract('http://www.ivi.ru/watch/dvoe_iz_lartsa/season1')          self.assertIsPlaylist(result) -        self.assertEqual(result['id'], 'dezhurnyi_angel/season2') -        self.assertEqual(result['title'], 'Дежурный ангел (2010 - 2012) 2 сезон') -        self.assertTrue(len(result['entries']) >= 7) +        self.assertEqual(result['id'], 'dvoe_iz_lartsa/season1') +        self.assertEqual(result['title'], 'Двое из ларца (2006 - 2008) 1 сезон') +        assertGreaterEqual(self, len(result['entries']), 12)      def test_imdb_list(self):          dl = FakeYDL() @@ -241,7 +272,7 @@ class TestPlaylists(unittest.TestCase):          self.assertEqual(result['id'], 'cryptography')          self.assertEqual(result['title'], 'Journey into cryptography')          self.assertEqual(result['description'], 'How have humans protected their secret messages through history? What has changed today?') -        self.assertTrue(len(result['entries']) >= 3) +        assertGreaterEqual(self, len(result['entries']), 3)      def test_EveryonesMixtape(self):          dl = FakeYDL() @@ -255,10 +286,18 @@ class TestPlaylists(unittest.TestCase):      def test_rutube_channel(self):          dl = FakeYDL()          ie = RutubeChannelIE(dl) -        result = ie.extract('http://rutube.ru/tags/video/1409') +        result = ie.extract('http://rutube.ru/tags/video/1800/') +        self.assertIsPlaylist(result) +        self.assertEqual(result['id'], '1800') +        assertGreaterEqual(self, len(result['entries']), 68) + +    def test_rutube_person(self): +        dl = FakeYDL() +        ie = RutubePersonIE(dl) +        result = ie.extract('http://rutube.ru/video/person/313878/')          self.assertIsPlaylist(result) -        self.assertEqual(result['id'], '1409') -        self.assertTrue(len(result['entries']) >= 34) +        self.assertEqual(result['id'], '313878') +        assertGreaterEqual(self, len(result['entries']), 37)      def test_multiple_brightcove_videos(self):          # https://github.com/rg3/youtube-dl/issues/2283 @@ -295,7 +334,7 @@ class TestPlaylists(unittest.TestCase):          self.assertIsPlaylist(result)          self.assertEqual(result['id'], '10')          self.assertEqual(result['title'], 'Who are the hackers?') -        self.assertTrue(len(result['entries']) >= 6) +        assertGreaterEqual(self, len(result['entries']), 6)      def test_toypics_user(self):          dl = FakeYDL() @@ -303,7 +342,7 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('http://videos.toypics.net/Mikey')          self.assertIsPlaylist(result)          self.assertEqual(result['id'], 'Mikey') -        self.assertTrue(len(result['entries']) >= 17) +        assertGreaterEqual(self, len(result['entries']), 17)      def test_xtube_user(self):          dl = FakeYDL() @@ -311,7 +350,7 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('http://www.xtube.com/community/profile.php?user=greenshowers')          self.assertIsPlaylist(result)          self.assertEqual(result['id'], 'greenshowers') -        self.assertTrue(len(result['entries']) >= 155) +        assertGreaterEqual(self, len(result['entries']), 155)      def test_InstagramUser(self):          dl = FakeYDL() @@ -319,7 +358,7 @@ class TestPlaylists(unittest.TestCase):          result = ie.extract('http://instagram.com/porsche')          self.assertIsPlaylist(result)          self.assertEqual(result['id'], 'porsche') -        self.assertTrue(len(result['entries']) >= 2) +        assertGreaterEqual(self, len(result['entries']), 2)          test_video = next(              e for e in result['entries']              if e['id'] == '614605558512799803_462752227') @@ -358,7 +397,15 @@ class TestPlaylists(unittest.TestCase):          self.assertEqual(result['id'], '152147')          self.assertEqual(              result['title'], 'Brace Yourself - Today\'s Weirdest News') -        self.assertTrue(len(result['entries']) >= 10) +        assertGreaterEqual(self, len(result['entries']), 10) + +    def test_TeacherTubeUser(self): +        dl = FakeYDL() +        ie = TeacherTubeUserIE(dl) +        result = ie.extract('http://www.teachertube.com/user/profile/rbhagwati2') +        self.assertIsPlaylist(result) +        self.assertEqual(result['id'], 'rbhagwati2') +        assertGreaterEqual(self, len(result['entries']), 179)  if __name__ == '__main__':      unittest.main() diff --git a/test/test_subtitles.py b/test/test_subtitles.py index 5736fe581..48c302198 100644 --- a/test/test_subtitles.py +++ b/test/test_subtitles.py @@ -87,7 +87,7 @@ class TestYoutubeSubtitles(BaseTestSubtitles):      def test_youtube_nosubtitles(self):          self.DL.expect_warning(u'video doesn\'t have subtitles') -        self.url = 'sAjKT8FhjI8' +        self.url = 'n5BB19UTcdA'          self.DL.params['writesubtitles'] = True          self.DL.params['allsubtitles'] = True          subtitles = self.getSubtitles() diff --git a/test/test_swfinterp.py b/test/test_swfinterp.py new file mode 100644 index 000000000..b42cd74c7 --- /dev/null +++ b/test/test_swfinterp.py @@ -0,0 +1,77 @@ +#!/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__)))) + + +import errno +import io +import json +import re +import subprocess + +from youtube_dl.swfinterp import SWFInterpreter + + +TEST_DIR = os.path.join( +    os.path.dirname(os.path.abspath(__file__)), 'swftests') + + +class TestSWFInterpreter(unittest.TestCase): +    pass + + +def _make_testfunc(testfile): +    m = re.match(r'^(.*)\.(as)$', testfile) +    if not m: +        return +    test_id = m.group(1) + +    def test_func(self): +        as_file = os.path.join(TEST_DIR, testfile) +        swf_file = os.path.join(TEST_DIR, test_id + '.swf') +        if ((not os.path.exists(swf_file)) +                or os.path.getmtime(swf_file) < os.path.getmtime(as_file)): +            # Recompile +            try: +                subprocess.check_call(['mxmlc', '-output', swf_file, as_file]) +            except OSError as ose: +                if ose.errno == errno.ENOENT: +                    print('mxmlc not found! Skipping test.') +                    return +                raise + +        with open(swf_file, 'rb') as swf_f: +            swf_content = swf_f.read() +        swfi = SWFInterpreter(swf_content) + +        with io.open(as_file, 'r', encoding='utf-8') as as_f: +            as_content = as_f.read() + +        def _find_spec(key): +            m = re.search( +                r'(?m)^//\s*%s:\s*(.*?)\n' % re.escape(key), as_content) +            if not m: +                raise ValueError('Cannot find %s in %s' % (key, testfile)) +            return json.loads(m.group(1)) + +        input_args = _find_spec('input') +        output = _find_spec('output') + +        swf_class = swfi.extract_class(test_id) +        func = swfi.extract_function(swf_class, 'main') +        res = func(input_args) +        self.assertEqual(res, output) + +    test_func.__name__ = str('test_swf_' + test_id) +    setattr(TestSWFInterpreter, test_func.__name__, test_func) + + +for testfile in os.listdir(TEST_DIR): +    _make_testfunc(testfile) + +if __name__ == '__main__': +    unittest.main() diff --git a/test/test_utils.py b/test/test_utils.py index 51eb0b6b9..e26cc5b0c 100644 --- a/test/test_utils.py +++ b/test/test_utils.py @@ -280,7 +280,7 @@ class TestUtil(unittest.TestCase):          d = json.loads(stripped)          self.assertEqual(d, [{"id": "532cb", "x": 3}]) -    def test_uppercase_escpae(self): +    def test_uppercase_escape(self):          self.assertEqual(uppercase_escape(u'aä'), u'aä')          self.assertEqual(uppercase_escape(u'\\U0001d550'), u'𝕐') diff --git a/test/test_youtube_lists.py b/test/test_youtube_lists.py index 7d3b9c705..3aadedd64 100644 --- a/test/test_youtube_lists.py +++ b/test/test_youtube_lists.py @@ -112,11 +112,11 @@ class TestYoutubeLists(unittest.TestCase):      def test_youtube_mix(self):          dl = FakeYDL()          ie = YoutubePlaylistIE(dl) -        result = ie.extract('http://www.youtube.com/watch?v=lLJf9qJHR3E&list=RDrjFaenf1T-Y') +        result = ie.extract('https://www.youtube.com/watch?v=W01L70IGBgE&index=2&list=RDOQpdSVF_k_w')          entries = result['entries']          self.assertTrue(len(entries) >= 20)          original_video = entries[0] -        self.assertEqual(original_video['id'], 'rjFaenf1T-Y') +        self.assertEqual(original_video['id'], 'OQpdSVF_k_w')      def test_youtube_toptracks(self):          print('Skipping: The playlist page gives error 500') diff --git a/test/test_youtube_signature.py b/test/test_youtube_signature.py index 8417c55a6..604e76ab6 100644 --- a/test/test_youtube_signature.py +++ b/test/test_youtube_signature.py @@ -1,5 +1,7 @@  #!/usr/bin/env python +from __future__ import unicode_literals +  # Allow direct execution  import os  import sys @@ -16,23 +18,65 @@ from youtube_dl.utils import compat_str, compat_urlretrieve  _TESTS = [      ( -        u'https://s.ytimg.com/yts/jsbin/html5player-vflHOr_nV.js', -        u'js', +        'https://s.ytimg.com/yts/jsbin/html5player-vflHOr_nV.js', +        'js',          86, -        u'>=<;:/.-[+*)(\'&%$#"!ZYX0VUTSRQPONMLKJIHGFEDCBA\\yxwvutsrqponmlkjihgfedcba987654321', +        '>=<;:/.-[+*)(\'&%$#"!ZYX0VUTSRQPONMLKJIHGFEDCBA\\yxwvutsrqponmlkjihgfedcba987654321',      ),      ( -        u'https://s.ytimg.com/yts/jsbin/html5player-vfldJ8xgI.js', -        u'js', +        'https://s.ytimg.com/yts/jsbin/html5player-vfldJ8xgI.js', +        'js',          85, -        u'3456789a0cdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRS[UVWXYZ!"#$%&\'()*+,-./:;<=>?@', +        '3456789a0cdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRS[UVWXYZ!"#$%&\'()*+,-./:;<=>?@',      ),      ( -        u'https://s.ytimg.com/yts/jsbin/html5player-vfle-mVwz.js', -        u'js', +        'https://s.ytimg.com/yts/jsbin/html5player-vfle-mVwz.js', +        'js',          90, -        u']\\[@?>=<;:/.-,+*)(\'&%$#"hZYXWVUTSRQPONMLKJIHGFEDCBAzyxwvutsrqponmlkjiagfedcb39876', +        ']\\[@?>=<;:/.-,+*)(\'&%$#"hZYXWVUTSRQPONMLKJIHGFEDCBAzyxwvutsrqponmlkjiagfedcb39876', +    ), +    ( +        'https://s.ytimg.com/yts/jsbin/html5player-en_US-vfl0Cbn9e.js', +        'js', +        84, +        'O1I3456789abcde0ghijklmnopqrstuvwxyzABCDEFGHfJKLMN2PQRSTUVW@YZ!"#$%&\'()*+,-./:;<=', +    ), +    ( +        'https://s.ytimg.com/yts/jsbin/html5player-en_US-vflXGBaUN.js', +        'js', +        '2ACFC7A61CA478CD21425E5A57EBD73DDC78E22A.2094302436B2D377D14A3BBA23022D023B8BC25AA', +        'A52CB8B320D22032ABB3A41D773D2B6342034902.A22E87CDD37DBE75A5E52412DC874AC16A7CFCA2', +    ), +    ( +        'http://s.ytimg.com/yts/swfbin/player-vfl5vIhK2/watch_as3.swf', +        'swf', +        86, +        'O1I3456789abcde0ghijklmnopqrstuvwxyzABCDEFGHfJKLMN2PQRSTUVWXY\\!"#$%&\'()*+,-./:;<=>?'      ), +    ( +        'http://s.ytimg.com/yts/swfbin/player-vflmDyk47/watch_as3.swf', +        'swf', +        'F375F75BF2AFDAAF2666E43868D46816F83F13E81C46.3725A8218E446A0DECD33F79DC282994D6AA92C92C9', +        '9C29AA6D499282CD97F33DCED0A644E8128A5273.64C18E31F38361864D86834E6662FAADFA2FB57F' +    ), +    ( +        'https://s.ytimg.com/yts/jsbin/html5player-en_US-vflBb0OQx.js', +        'js', +        84, +        '123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQ0STUVWXYZ!"#$%&\'()*+,@./:;<=>' +    ), +    ( +        'https://s.ytimg.com/yts/jsbin/html5player-en_US-vfl9FYC6l.js', +        'js', +        83, +        '123456789abcdefghijklmnopqr0tuvwxyzABCDETGHIJKLMNOPQRS>UVWXYZ!"#$%&\'()*+,-./:;<=F' +    ), +    ( +        'https://s.ytimg.com/yts/jsbin/html5player-en_US-vflCGk6yw/html5player.js', +        'js', +        '4646B5181C6C3020DF1D9C7FCFEA.AD80ABF70C39BD369CCCAE780AFBB98FA6B6CB42766249D9488C288', +        '82C8849D94266724DC6B6AF89BBFA087EACCD963.B93C07FBA084ACAEFCF7C9D1FD0203C6C1815B6B' +    )  ] @@ -44,13 +88,13 @@ class TestSignature(unittest.TestCase):              os.mkdir(self.TESTDATA_DIR) -def make_tfunc(url, stype, sig_length, expected_sig): -    basename = url.rpartition('/')[2] -    m = re.match(r'.*-([a-zA-Z0-9_-]+)\.[a-z]+$', basename) -    assert m, '%r should follow URL format' % basename +def make_tfunc(url, stype, sig_input, expected_sig): +    m = re.match(r'.*-([a-zA-Z0-9_-]+)(?:/watch_as3|/html5player)?\.[a-z]+$', url) +    assert m, '%r should follow URL format' % url      test_id = m.group(1)      def test_func(self): +        basename = 'player-%s.%s' % (test_id, stype)          fn = os.path.join(self.TESTDATA_DIR, basename)          if not os.path.exists(fn): @@ -66,7 +110,9 @@ def make_tfunc(url, stype, sig_length, expected_sig):              with open(fn, 'rb') as testf:                  swfcode = testf.read()              func = ie._parse_sig_swf(swfcode) -        src_sig = compat_str(string.printable[:sig_length]) +        src_sig = ( +            compat_str(string.printable[:sig_input]) +            if isinstance(sig_input, int) else sig_input)          got_sig = func(src_sig)          self.assertEqual(got_sig, expected_sig) | 
