diff options
| author | Jesús <heckyel@hyperbola.info> | 2022-04-06 03:37:17 +0800 | 
|---|---|---|
| committer | Jesús <heckyel@hyperbola.info> | 2022-04-06 03:37:17 +0800 | 
| commit | 1e5a50b71d8f0eae6007bedc329eecb24bb5aba3 (patch) | |
| tree | a8611cda6596391cb6fb645e1469dcd356b63924 /test | |
| parent | f52fb3bceeb9d22b5106c1796fecec474a0cc138 (diff) | |
| download | hypervideo-1e5a50b71d8f0eae6007bedc329eecb24bb5aba3.tar.lz hypervideo-1e5a50b71d8f0eae6007bedc329eecb24bb5aba3.tar.xz hypervideo-1e5a50b71d8f0eae6007bedc329eecb24bb5aba3.zip | |
update from upstream
Diffstat (limited to 'test')
| -rw-r--r-- | test/helper.py | 47 | ||||
| -rw-r--r-- | test/parameters.json | 2 | ||||
| -rw-r--r-- | test/test_InfoExtractor.py | 184 | ||||
| -rw-r--r-- | test/test_YoutubeDL.py | 56 | ||||
| -rw-r--r-- | test/test_aes.py | 18 | ||||
| -rw-r--r-- | test/test_all_urls.py | 1 | ||||
| -rw-r--r-- | test/test_cookies.py | 36 | ||||
| -rwxr-xr-x[-rw-r--r--] | test/test_download.py | 2 | ||||
| -rw-r--r-- | test/test_netrc.py | 13 | ||||
| -rw-r--r-- | test/test_postprocessors.py | 4 | ||||
| -rw-r--r-- | test/test_subtitles.py | 4 | ||||
| -rw-r--r-- | test/test_utils.py | 214 | ||||
| -rw-r--r-- | test/test_verbose_output.py | 16 | ||||
| -rw-r--r-- | test/test_youtube_lists.py | 42 | 
14 files changed, 510 insertions, 129 deletions
| diff --git a/test/helper.py b/test/helper.py index 0d8822e..1f1ccfa 100644 --- a/test/helper.py +++ b/test/helper.py @@ -194,6 +194,45 @@ def expect_dict(self, got_dict, expected_dict):          expect_value(self, got, expected, info_field) +def sanitize_got_info_dict(got_dict): +    IGNORED_FIELDS = ( +        *YoutubeDL._format_fields, + +        # Lists +        'formats', 'thumbnails', 'subtitles', 'automatic_captions', 'comments', 'entries', + +        # Auto-generated +        'autonumber', 'playlist', 'format_index', 'video_ext', 'audio_ext', 'duration_string', 'epoch', +        'fulltitle', 'extractor', 'extractor_key', 'filepath', 'infojson_filename', 'original_url', 'n_entries', + +        # Only live_status needs to be checked +        'is_live', 'was_live', +    ) + +    IGNORED_PREFIXES = ('', 'playlist', 'requested', 'webpage') + +    def sanitize(key, value): +        if isinstance(value, str) and len(value) > 100 and key != 'thumbnail': +            return f'md5:{md5(value)}' +        elif isinstance(value, list) and len(value) > 10: +            return f'count:{len(value)}' +        elif key.endswith('_count') and isinstance(value, int): +            return int +        return value + +    test_info_dict = { +        key: sanitize(key, value) for key, value in got_dict.items() +        if value is not None and key not in IGNORED_FIELDS and not any( +            key.startswith(f'{prefix}_') for prefix in IGNORED_PREFIXES) +    } + +    # display_id may be generated from id +    if test_info_dict.get('display_id') == test_info_dict.get('id'): +        test_info_dict.pop('display_id') + +    return test_info_dict + +  def expect_info_dict(self, got_dict, expected_dict):      expect_dict(self, got_dict, expected_dict)      # Check for the presence of mandatory fields @@ -207,15 +246,15 @@ def expect_info_dict(self, got_dict, expected_dict):      for key in ['webpage_url', 'extractor', 'extractor_key']:          self.assertTrue(got_dict.get(key), 'Missing field: %s' % key) -    # Are checkable fields missing from the test case definition? -    test_info_dict = dict((key, value if not isinstance(value, compat_str) or len(value) < 250 else 'md5:' + md5(value)) -                          for key, value in got_dict.items() -                          if value and key in ('id', 'title', 'description', 'uploader', 'upload_date', 'timestamp', 'uploader_id', 'location', 'age_limit')) +    test_info_dict = sanitize_got_info_dict(got_dict) +      missing_keys = set(test_info_dict.keys()) - set(expected_dict.keys())      if missing_keys:          def _repr(v):              if isinstance(v, compat_str):                  return "'%s'" % v.replace('\\', '\\\\').replace("'", "\\'").replace('\n', '\\n') +            elif isinstance(v, type): +                return v.__name__              else:                  return repr(v)          info_dict_str = '' diff --git a/test/parameters.json b/test/parameters.json index 9ca7d2c..06fe3e3 100644 --- a/test/parameters.json +++ b/test/parameters.json @@ -9,7 +9,7 @@      "forcetitle": false,      "forceurl": false,      "force_write_download_archive": false, -    "format": "best", +    "format": "b/bv",      "ignoreerrors": false,      "listformats": null,      "logtostderr": false, diff --git a/test/test_InfoExtractor.py b/test/test_InfoExtractor.py index e892095..8494105 100644 --- a/test/test_InfoExtractor.py +++ b/test/test_InfoExtractor.py @@ -99,10 +99,10 @@ class TestInfoExtractor(unittest.TestCase):          self.assertRaises(RegexNotFoundError, ie._html_search_meta, ('z', 'x'), html, None, fatal=True)      def test_search_json_ld_realworld(self): -        # https://github.com/ytdl-org/youtube-dl/issues/23306 -        expect_dict( -            self, -            self.ie._search_json_ld(r'''<script type="application/ld+json"> +        _TESTS = [ +            # https://github.com/ytdl-org/youtube-dl/issues/23306 +            ( +                r'''<script type="application/ld+json">  {  "@context": "http://schema.org/",  "@type": "VideoObject", @@ -135,17 +135,171 @@ class TestInfoExtractor(unittest.TestCase):  "name": "Kleio Valentien",  "url": "https://www.eporner.com/pornstar/kleio-valentien/"  }]} -</script>''', None), -            { -                'title': '1 On 1 With Kleio', -                'description': 'Kleio Valentien', -                'url': 'https://gvideo.eporner.com/xN49A1cT3eB/xN49A1cT3eB.mp4', -                'timestamp': 1449347075, -                'duration': 743.0, -                'view_count': 1120958, -                'width': 1920, -                'height': 1080, -            }) +                </script>''', +                { +                    'title': '1 On 1 With Kleio', +                    'description': 'Kleio Valentien', +                    'url': 'https://gvideo.eporner.com/xN49A1cT3eB/xN49A1cT3eB.mp4', +                    'timestamp': 1449347075, +                    'duration': 743.0, +                    'view_count': 1120958, +                    'width': 1920, +                    'height': 1080, +                }, +                {}, +            ), +            ( +                r'''<script type="application/ld+json"> +      { +      "@context": "https://schema.org", +      "@graph": [ +      { +      "@type": "NewsArticle", +      "mainEntityOfPage": { +      "@type": "WebPage", +      "@id": "https://www.ant1news.gr/Society/article/620286/symmoria-anilikon-dikigoros-thymaton-ithelan-na-toys-apoteleiosoyn" +      }, +      "headline": "Συμμορία ανηλίκων – δικηγόρος θυμάτων: ήθελαν να τους αποτελειώσουν", +      "name": "Συμμορία ανηλίκων – δικηγόρος θυμάτων: ήθελαν να τους αποτελειώσουν", +      "description": "Τα παιδιά δέχθηκαν την επίθεση επειδή αρνήθηκαν να γίνουν μέλη της συμμορίας, ανέφερε ο Γ. Ζαχαρόπουλος.", +      "image": { +      "@type": "ImageObject", +      "url": "https://ant1media.azureedge.net/imgHandler/1100/a635c968-be71-447c-bf9c-80d843ece21e.jpg", +      "width": 1100, +      "height": 756            }, +      "datePublished": "2021-11-10T08:50:00+03:00", +      "dateModified": "2021-11-10T08:52:53+03:00", +      "author": { +      "@type": "Person", +      "@id": "https://www.ant1news.gr/", +      "name": "Ant1news", +      "image": "https://www.ant1news.gr/images/logo-e5d7e4b3e714c88e8d2eca96130142f6.png", +      "url": "https://www.ant1news.gr/" +      }, +      "publisher": { +      "@type": "Organization", +      "@id": "https://www.ant1news.gr#publisher", +      "name": "Ant1news", +      "url": "https://www.ant1news.gr", +      "logo": { +      "@type": "ImageObject", +      "url": "https://www.ant1news.gr/images/logo-e5d7e4b3e714c88e8d2eca96130142f6.png", +      "width": 400, +      "height": 400                }, +      "sameAs": [ +      "https://www.facebook.com/Ant1news.gr", +      "https://twitter.com/antennanews", +      "https://www.youtube.com/channel/UC0smvAbfczoN75dP0Hw4Pzw", +      "https://www.instagram.com/ant1news/" +      ] +      }, + +      "keywords": "μαχαίρωμα,συμμορία ανηλίκων,ΕΙΔΗΣΕΙΣ,ΕΙΔΗΣΕΙΣ ΣΗΜΕΡΑ,ΝΕΑ,Κοινωνία - Ant1news", + + +      "articleSection": "Κοινωνία" +      } +      ] +      } +                </script>''', +                { +                    'timestamp': 1636523400, +                    'title': 'md5:91fe569e952e4d146485740ae927662b', +                }, +                {'expected_type': 'NewsArticle'}, +            ), +            ( +                r'''<script type="application/ld+json"> +                {"url":"/vrtnu/a-z/het-journaal/2021/het-journaal-het-journaal-19u-20211231/", +                "name":"Het journaal 19u", +                "description":"Het journaal 19u van vrijdag 31 december 2021.", +                "potentialAction":{"url":"https://vrtnu.page.link/pfVy6ihgCAJKgHqe8","@type":"ShareAction"}, +                "mainEntityOfPage":{"@id":"1640092242445","@type":"WebPage"}, +                "publication":[{ +                    "startDate":"2021-12-31T19:00:00.000+01:00", +                    "endDate":"2022-01-30T23:55:00.000+01:00", +                    "publishedBy":{"name":"een","@type":"Organization"}, +                    "publishedOn":{"url":"https://www.vrt.be/vrtnu/","name":"VRT NU","@type":"BroadcastService"}, +                    "@id":"pbs-pub-3a7ec233-da95-4c1e-9b2b-cf5fdfebcbe8", +                    "@type":"BroadcastEvent" +                    }], +                "video":{ +                    "name":"Het journaal - Aflevering 365 (Seizoen 2021)", +                    "description":"Het journaal 19u van vrijdag 31 december 2021. Bekijk aflevering 365 van seizoen 2021 met VRT NU via de site of app.", +                    "thumbnailUrl":"//images.vrt.be/width1280/2021/12/31/80d5ed00-6a64-11ec-b07d-02b7b76bf47f.jpg", +                    "expires":"2022-01-30T23:55:00.000+01:00", +                    "hasPart":[ +                        {"name":"Explosie Turnhout","startOffset":70,"@type":"Clip"}, +                        {"name":"Jaarwisseling","startOffset":440,"@type":"Clip"}, +                        {"name":"Natuurbranden Colorado","startOffset":1179,"@type":"Clip"}, +                        {"name":"Klimaatverandering","startOffset":1263,"@type":"Clip"}, +                        {"name":"Zacht weer","startOffset":1367,"@type":"Clip"}, +                        {"name":"Financiële balans","startOffset":1383,"@type":"Clip"}, +                        {"name":"Club Brugge","startOffset":1484,"@type":"Clip"}, +                        {"name":"Mentale gezondheid bij topsporters","startOffset":1575,"@type":"Clip"}, +                        {"name":"Olympische Winterspelen","startOffset":1728,"@type":"Clip"}, +                        {"name":"Sober oudjaar in Nederland","startOffset":1873,"@type":"Clip"} +                        ], +                    "duration":"PT34M39.23S", +                    "uploadDate":"2021-12-31T19:00:00.000+01:00", +                    "@id":"vid-9457d0c6-b8ac-4aba-b5e1-15aa3a3295b5", +                    "@type":"VideoObject" +                }, +                "genre":["Nieuws en actua"], +                "episodeNumber":365, +                "partOfSeries":{"name":"Het journaal","@id":"222831405527","@type":"TVSeries"}, +                "partOfSeason":{"name":"Seizoen 2021","@id":"961809365527","@type":"TVSeason"}, +                "@context":"https://schema.org","@id":"961685295527","@type":"TVEpisode"}</script> +                ''', +                { +                    'chapters': [ +                        {"title": "Explosie Turnhout", "start_time": 70, "end_time": 440}, +                        {"title": "Jaarwisseling", "start_time": 440, "end_time": 1179}, +                        {"title": "Natuurbranden Colorado", "start_time": 1179, "end_time": 1263}, +                        {"title": "Klimaatverandering", "start_time": 1263, "end_time": 1367}, +                        {"title": "Zacht weer", "start_time": 1367, "end_time": 1383}, +                        {"title": "Financiële balans", "start_time": 1383, "end_time": 1484}, +                        {"title": "Club Brugge", "start_time": 1484, "end_time": 1575}, +                        {"title": "Mentale gezondheid bij topsporters", "start_time": 1575, "end_time": 1728}, +                        {"title": "Olympische Winterspelen", "start_time": 1728, "end_time": 1873}, +                        {"title": "Sober oudjaar in Nederland", "start_time": 1873, "end_time": 2079.23} +                    ], +                    'title': 'Het journaal - Aflevering 365 (Seizoen 2021)' +                }, {} +            ), +            ( +                # test multiple thumbnails in a list +                r''' +<script type="application/ld+json"> +{"@context":"https://schema.org", +"@type":"VideoObject", +"thumbnailUrl":["https://www.rainews.it/cropgd/640x360/dl/img/2021/12/30/1640886376927_GettyImages.jpg"]} +</script>''', +                { +                    'thumbnails': [{'url': 'https://www.rainews.it/cropgd/640x360/dl/img/2021/12/30/1640886376927_GettyImages.jpg'}], +                }, +                {}, +            ), +            ( +                # test single thumbnail +                r''' +<script type="application/ld+json"> +{"@context":"https://schema.org", +"@type":"VideoObject", +"thumbnailUrl":"https://www.rainews.it/cropgd/640x360/dl/img/2021/12/30/1640886376927_GettyImages.jpg"} +</script>''', +                { +                    'thumbnails': [{'url': 'https://www.rainews.it/cropgd/640x360/dl/img/2021/12/30/1640886376927_GettyImages.jpg'}], +                }, +                {}, +            ) +        ] +        for html, expected_dict, search_json_ld_kwargs in _TESTS: +            expect_dict( +                self, +                self.ie._search_json_ld(html, None, **search_json_ld_kwargs), +                expected_dict +            )      def test_download_json(self):          uri = encode_data_uri(b'{"foo": "blah"}', 'application/json') diff --git a/test/test_YoutubeDL.py b/test/test_YoutubeDL.py index c9dd498..fe0fd35 100644 --- a/test/test_YoutubeDL.py +++ b/test/test_YoutubeDL.py @@ -30,8 +30,7 @@ class YDL(FakeYDL):          self.msgs = []      def process_info(self, info_dict): -        info_dict.pop('__original_infodict', None) -        self.downloaded_info_dicts.append(info_dict) +        self.downloaded_info_dicts.append(info_dict.copy())      def to_screen(self, msg):          self.msgs.append(msg) @@ -137,7 +136,7 @@ class TestFormatSelection(unittest.TestCase):          test('webm/mp4', '47')          test('3gp/40/mp4', '35')          test('example-with-dashes', 'example-with-dashes') -        test('all', '35', 'example-with-dashes', '45', '47', '2')  # Order doesn't actually matter for this +        test('all', '2', '47', '45', 'example-with-dashes', '35')          test('mergeall', '2+47+45+example-with-dashes+35', multi=True)      def test_format_selection_audio(self): @@ -520,7 +519,7 @@ class TestFormatSelection(unittest.TestCase):          ydl = YDL({'format': 'all[width>=400][width<=600]'})          ydl.process_ie_result(info_dict)          downloaded_ids = [info['format_id'] for info in ydl.downloaded_info_dicts] -        self.assertEqual(downloaded_ids, ['B', 'C', 'D']) +        self.assertEqual(downloaded_ids, ['D', 'C', 'B'])          ydl = YDL({'format': 'best[height<40]'})          try: @@ -645,6 +644,7 @@ class TestYoutubeDL(unittest.TestCase):          'ext': 'mp4',          'width': None,          'height': 1080, +        'filesize': 1024,          'title1': '$PATH',          'title2': '%PATH%',          'title3': 'foo/bar\\test', @@ -656,7 +656,7 @@ class TestYoutubeDL(unittest.TestCase):          'playlist_autonumber': 2,          '_last_playlist_index': 100,          'n_entries': 10, -        'formats': [{'id': 'id1'}, {'id': 'id2'}, {'id': 'id3'}] +        'formats': [{'id': 'id 1'}, {'id': 'id 2'}, {'id': 'id 3'}]      }      def test_prepare_outtmpl_and_filename(self): @@ -717,6 +717,7 @@ class TestYoutubeDL(unittest.TestCase):          test('%(id)s', '.abcd', info={'id': '.abcd'})          test('%(id)s', 'ab__cd', info={'id': 'ab__cd'})          test('%(id)s', ('ab:cd', 'ab -cd'), info={'id': 'ab:cd'}) +        test('%(id.0)s', '-', info={'id': '--'})          # Invalid templates          self.assertTrue(isinstance(YoutubeDL.validate_outtmpl('%(title)'), ValueError)) @@ -737,6 +738,7 @@ class TestYoutubeDL(unittest.TestCase):          test(NA_TEST_OUTTMPL, 'NA-NA-def-1234.mp4')          test(NA_TEST_OUTTMPL, 'none-none-def-1234.mp4', outtmpl_na_placeholder='none')          test(NA_TEST_OUTTMPL, '--def-1234.mp4', outtmpl_na_placeholder='') +        test('%(non_existent.0)s', 'NA')          # String formatting          FMT_TEST_OUTTMPL = '%%(height)%s.%%(ext)s' @@ -762,23 +764,33 @@ class TestYoutubeDL(unittest.TestCase):          test('a%(width|)d', 'a', outtmpl_na_placeholder='none')          FORMATS = self.outtmpl_info['formats'] -        sanitize = lambda x: x.replace(':', ' -').replace('"', "'") +        sanitize = lambda x: x.replace(':', ' -').replace('"', "'").replace('\n', ' ')          # Custom type casting -        test('%(formats.:.id)l', 'id1, id2, id3') -        test('%(formats.:.id)#l', ('id1\nid2\nid3', 'id1 id2 id3')) +        test('%(formats.:.id)l', 'id 1, id 2, id 3') +        test('%(formats.:.id)#l', ('id 1\nid 2\nid 3', 'id 1 id 2 id 3'))          test('%(ext)l', 'mp4') -        test('%(formats.:.id) 15l', '  id1, id2, id3') +        test('%(formats.:.id) 18l', '  id 1, id 2, id 3')          test('%(formats)j', (json.dumps(FORMATS), sanitize(json.dumps(FORMATS)))) +        test('%(formats)#j', (json.dumps(FORMATS, indent=4), sanitize(json.dumps(FORMATS, indent=4))))          test('%(title5).3B', 'á')          test('%(title5)U', 'áéí 𝐀')          test('%(title5)#U', 'a\u0301e\u0301i\u0301 𝐀')          test('%(title5)+U', 'áéí A')          test('%(title5)+#U', 'a\u0301e\u0301i\u0301 A') +        test('%(height)D', '1k') +        test('%(filesize)#D', '1Ki') +        test('%(height)5.2D', ' 1.08k') +        test('%(title4)#S', 'foo_bar_test') +        test('%(title4).10S', ('foo \'bar\' ', 'foo \'bar\'' + ('#' if compat_os_name == 'nt' else ' ')))          if compat_os_name == 'nt':              test('%(title4)q', ('"foo \\"bar\\" test"', "'foo _'bar_' test'")) +            test('%(formats.:.id)#q', ('"id 1" "id 2" "id 3"', "'id 1' 'id 2' 'id 3'")) +            test('%(formats.0.id)#q', ('"id 1"', "'id 1'"))          else:              test('%(title4)q', ('\'foo "bar" test\'', "'foo 'bar' test'")) +            test('%(formats.:.id)#q', "'id 1' 'id 2' 'id 3'") +            test('%(formats.0.id)#q', "'id 1'")          # Internal formatting          test('%(timestamp-1000>%H-%M-%S)s', '11-43-20') @@ -802,6 +814,13 @@ class TestYoutubeDL(unittest.TestCase):          test('%(width-100,height+width|def)s', 'def')          test('%(timestamp-x>%H\\,%M\\,%S,timestamp>%H\\,%M\\,%S)s', '12,00,00') +        # Replacement +        test('%(id&foo)s.bar', 'foo.bar') +        test('%(title&foo)s.bar', 'NA.bar') +        test('%(title&foo|baz)s.bar', 'baz.bar') +        test('%(x,id&foo|baz)s.bar', 'foo.bar') +        test('%(x,title&foo|baz)s.bar', 'baz.bar') +          # Laziness          def gen():              yield from range(5) @@ -879,20 +898,6 @@ class TestYoutubeDL(unittest.TestCase):          os.unlink(filename)      def test_match_filter(self): -        class FilterYDL(YDL): -            def __init__(self, *args, **kwargs): -                super(FilterYDL, self).__init__(*args, **kwargs) -                self.params['simulate'] = True - -            def process_info(self, info_dict): -                super(YDL, self).process_info(info_dict) - -            def _match_entry(self, info_dict, incomplete=False): -                res = super(FilterYDL, self)._match_entry(info_dict, incomplete) -                if res is None: -                    self.downloaded_info_dicts.append(info_dict) -                return res -          first = {              'id': '1',              'url': TEST_URL, @@ -920,7 +925,7 @@ class TestYoutubeDL(unittest.TestCase):          videos = [first, second]          def get_videos(filter_=None): -            ydl = FilterYDL({'match_filter': filter_}) +            ydl = YDL({'match_filter': filter_, 'simulate': True})              for v in videos:                  ydl.process_ie_result(v, download=True)              return [v['id'] for v in ydl.downloaded_info_dicts] @@ -928,7 +933,7 @@ class TestYoutubeDL(unittest.TestCase):          res = get_videos()          self.assertEqual(res, ['1', '2']) -        def f(v): +        def f(v, incomplete):              if v['id'] == '1':                  return None              else: @@ -1135,6 +1140,7 @@ class TestYoutubeDL(unittest.TestCase):          self.assertTrue(entries[1] is None)          self.assertEqual(len(ydl.downloaded_info_dicts), 1)          downloaded = ydl.downloaded_info_dicts[0] +        entries[2].pop('requested_downloads', None)          self.assertEqual(entries[2], downloaded)          self.assertEqual(downloaded['url'], TEST_URL)          self.assertEqual(downloaded['title'], 'Video Transparent 2') diff --git a/test/test_aes.py b/test/test_aes.py index 746e447..9d260b5 100644 --- a/test/test_aes.py +++ b/test/test_aes.py @@ -10,6 +10,8 @@ sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))  from hypervideo_dl.aes import (      aes_decrypt,      aes_encrypt, +    aes_ecb_encrypt, +    aes_ecb_decrypt,      aes_cbc_decrypt,      aes_cbc_decrypt_bytes,      aes_cbc_encrypt, @@ -17,7 +19,8 @@ from hypervideo_dl.aes import (      aes_ctr_encrypt,      aes_gcm_decrypt_and_verify,      aes_gcm_decrypt_and_verify_bytes, -    aes_decrypt_text +    aes_decrypt_text, +    BLOCK_SIZE_BYTES,  )  from hypervideo_dl.compat import compat_pycrypto_AES  from hypervideo_dl.utils import bytes_to_intlist, intlist_to_bytes @@ -94,6 +97,19 @@ class TestAES(unittest.TestCase):          decrypted = (aes_decrypt_text(encrypted, password, 32))          self.assertEqual(decrypted, self.secret_msg) +    def test_ecb_encrypt(self): +        data = bytes_to_intlist(self.secret_msg) +        data += [0x08] * (BLOCK_SIZE_BYTES - len(data) % BLOCK_SIZE_BYTES) +        encrypted = intlist_to_bytes(aes_ecb_encrypt(data, self.key, self.iv)) +        self.assertEqual( +            encrypted, +            b'\xaa\x86]\x81\x97>\x02\x92\x9d\x1bR[[L/u\xd3&\xd1(h\xde{\x81\x94\xba\x02\xae\xbd\xa6\xd0:') + +    def test_ecb_decrypt(self): +        data = bytes_to_intlist(b'\xaa\x86]\x81\x97>\x02\x92\x9d\x1bR[[L/u\xd3&\xd1(h\xde{\x81\x94\xba\x02\xae\xbd\xa6\xd0:') +        decrypted = intlist_to_bytes(aes_ecb_decrypt(data, self.key, self.iv)) +        self.assertEqual(decrypted.rstrip(b'\x08'), self.secret_msg) +  if __name__ == '__main__':      unittest.main() diff --git a/test/test_all_urls.py b/test/test_all_urls.py index d9e4bad..74634cb 100644 --- a/test/test_all_urls.py +++ b/test/test_all_urls.py @@ -38,7 +38,6 @@ class TestAllURLsMatching(unittest.TestCase):          assertTab('https://www.youtube.com/AsapSCIENCE')          assertTab('https://www.youtube.com/embedded')          assertTab('https://www.youtube.com/playlist?list=UUBABnxM4Ar9ten8Mdjj1j0Q') -        assertTab('https://www.youtube.com/course?list=ECUl4u3cNGP61MdtwGTqZA0MreSaDybji8')          assertTab('https://www.youtube.com/playlist?list=PLwP_SiAcdui0KVebT0mU9Apz359a4ubsC')          assertTab('https://www.youtube.com/watch?v=AV6J6_AeFEQ&playnext=1&list=PL4023E734DA416012')  # 668          self.assertFalse('youtube:playlist' in self.matching_ies('PLtS2H6bU1M')) diff --git a/test/test_cookies.py b/test/test_cookies.py index fb034fc..053e45b 100644 --- a/test/test_cookies.py +++ b/test/test_cookies.py @@ -8,6 +8,8 @@ from hypervideo_dl.cookies import (      WindowsChromeCookieDecryptor,      parse_safari_cookies,      pbkdf2_sha1, +    _get_linux_desktop_environment, +    _LinuxDesktopEnvironment,  ) @@ -42,6 +44,37 @@ class MonkeyPatch:  class TestCookies(unittest.TestCase): +    def test_get_desktop_environment(self): +        """ based on https://chromium.googlesource.com/chromium/src/+/refs/heads/main/base/nix/xdg_util_unittest.cc """ +        test_cases = [ +            ({}, _LinuxDesktopEnvironment.OTHER), + +            ({'DESKTOP_SESSION': 'gnome'}, _LinuxDesktopEnvironment.GNOME), +            ({'DESKTOP_SESSION': 'mate'}, _LinuxDesktopEnvironment.GNOME), +            ({'DESKTOP_SESSION': 'kde4'}, _LinuxDesktopEnvironment.KDE), +            ({'DESKTOP_SESSION': 'kde'}, _LinuxDesktopEnvironment.KDE), +            ({'DESKTOP_SESSION': 'xfce'}, _LinuxDesktopEnvironment.XFCE), + +            ({'GNOME_DESKTOP_SESSION_ID': 1}, _LinuxDesktopEnvironment.GNOME), +            ({'KDE_FULL_SESSION': 1}, _LinuxDesktopEnvironment.KDE), + +            ({'XDG_CURRENT_DESKTOP': 'X-Cinnamon'}, _LinuxDesktopEnvironment.CINNAMON), +            ({'XDG_CURRENT_DESKTOP': 'GNOME'}, _LinuxDesktopEnvironment.GNOME), +            ({'XDG_CURRENT_DESKTOP': 'GNOME:GNOME-Classic'}, _LinuxDesktopEnvironment.GNOME), +            ({'XDG_CURRENT_DESKTOP': 'GNOME : GNOME-Classic'}, _LinuxDesktopEnvironment.GNOME), + +            ({'XDG_CURRENT_DESKTOP': 'Unity', 'DESKTOP_SESSION': 'gnome-fallback'}, _LinuxDesktopEnvironment.GNOME), +            ({'XDG_CURRENT_DESKTOP': 'KDE', 'KDE_SESSION_VERSION': '5'}, _LinuxDesktopEnvironment.KDE), +            ({'XDG_CURRENT_DESKTOP': 'KDE'}, _LinuxDesktopEnvironment.KDE), +            ({'XDG_CURRENT_DESKTOP': 'Pantheon'}, _LinuxDesktopEnvironment.PANTHEON), +            ({'XDG_CURRENT_DESKTOP': 'Unity'}, _LinuxDesktopEnvironment.UNITY), +            ({'XDG_CURRENT_DESKTOP': 'Unity:Unity7'}, _LinuxDesktopEnvironment.UNITY), +            ({'XDG_CURRENT_DESKTOP': 'Unity:Unity8'}, _LinuxDesktopEnvironment.UNITY), +        ] + +        for env, expected_desktop_environment in test_cases: +            self.assertEqual(_get_linux_desktop_environment(env), expected_desktop_environment) +      def test_chrome_cookie_decryptor_linux_derive_key(self):          key = LinuxChromeCookieDecryptor.derive_key(b'abc')          self.assertEqual(key, b'7\xa1\xec\xd4m\xfcA\xc7\xb19Z\xd0\x19\xdcM\x17') @@ -58,8 +91,7 @@ class TestCookies(unittest.TestCase):              self.assertEqual(decryptor.decrypt(encrypted_value), value)      def test_chrome_cookie_decryptor_linux_v11(self): -        with MonkeyPatch(cookies, {'_get_linux_keyring_password': lambda *args, **kwargs: b'', -                                   'KEYRING_AVAILABLE': True}): +        with MonkeyPatch(cookies, {'_get_linux_keyring_password': lambda *args, **kwargs: b''}):              encrypted_value = b'v11#\x81\x10>`w\x8f)\xc0\xb2\xc1\r\xf4\x1al\xdd\x93\xfd\xf8\xf8N\xf2\xa9\x83\xf1\xe9o\x0elVQd'              value = 'tz=Europe.London'              decryptor = LinuxChromeCookieDecryptor('Chrome', Logger()) diff --git a/test/test_download.py b/test/test_download.py index 8b5eea5..3cca13b 100644..100755 --- a/test/test_download.py +++ b/test/test_download.py @@ -53,7 +53,7 @@ class YoutubeDL(hypervideo_dl.YoutubeDL):          raise ExtractorError(message)      def process_info(self, info_dict): -        self.processed_info_dicts.append(info_dict) +        self.processed_info_dicts.append(info_dict.copy())          return super(YoutubeDL, self).process_info(info_dict) diff --git a/test/test_netrc.py b/test/test_netrc.py index 50b9e5b..c7f5272 100644 --- a/test/test_netrc.py +++ b/test/test_netrc.py @@ -7,18 +7,19 @@ import unittest  sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) -from hypervideo_dl.extractor import ( -    gen_extractors, -) +from hypervideo_dl.extractor import gen_extractor_classes +from hypervideo_dl.extractor.common import InfoExtractor + +NO_LOGIN = InfoExtractor._perform_login  class TestNetRc(unittest.TestCase):      def test_netrc_present(self): -        for ie in gen_extractors(): -            if not hasattr(ie, '_login'): +        for ie in gen_extractor_classes(): +            if ie._perform_login is NO_LOGIN:                  continue              self.assertTrue( -                hasattr(ie, '_NETRC_MACHINE'), +                ie._NETRC_MACHINE,                  'Extractor %s supports login, but is missing a _NETRC_MACHINE property' % ie.IE_NAME) diff --git a/test/test_postprocessors.py b/test/test_postprocessors.py index 42f37b8..e0b8347 100644 --- a/test/test_postprocessors.py +++ b/test/test_postprocessors.py @@ -124,11 +124,11 @@ class TestModifyChaptersPP(unittest.TestCase):          chapters = self._chapters([70], ['c']) + [              self._sponsor_chapter(10, 20, 'sponsor'),              self._sponsor_chapter(30, 40, 'preview'), -            self._sponsor_chapter(50, 60, 'sponsor')] +            self._sponsor_chapter(50, 60, 'filler')]          expected = self._chapters(              [10, 20, 30, 40, 50, 60, 70],              ['c', '[SponsorBlock]: Sponsor', 'c', '[SponsorBlock]: Preview/Recap', -             'c', '[SponsorBlock]: Sponsor', 'c']) +             'c', '[SponsorBlock]: Filler Tangent', 'c'])          self._remove_marked_arrange_sponsors_test_impl(chapters, expected, [])      def test_remove_marked_arrange_sponsors_UniqueNamesForOverlappingSponsors(self): diff --git a/test/test_subtitles.py b/test/test_subtitles.py index e94df35..10fa0ca 100644 --- a/test/test_subtitles.py +++ b/test/test_subtitles.py @@ -13,7 +13,7 @@ from test.helper import FakeYDL, md5, is_download_test  from hypervideo_dl.extractor import (      YoutubeIE,      DailymotionIE, -    TEDIE, +    TedTalkIE,      VimeoIE,      WallaIE,      CeskaTelevizeIE, @@ -141,7 +141,7 @@ class TestDailymotionSubtitles(BaseTestSubtitles):  @is_download_test  class TestTedSubtitles(BaseTestSubtitles):      url = 'http://www.ted.com/talks/dan_dennett_on_our_consciousness.html' -    IE = TEDIE +    IE = TedTalkIE      def test_allsubtitles(self):          self.DL.params['writesubtitles'] = True diff --git a/test/test_utils.py b/test/test_utils.py index 1cd2b2f..039900c 100644 --- a/test/test_utils.py +++ b/test/test_utils.py @@ -23,6 +23,7 @@ from hypervideo_dl.utils import (      caesar,      clean_html,      clean_podcast_url, +    Config,      date_from_str,      datetime_from_str,      DateRange, @@ -37,11 +38,18 @@ from hypervideo_dl.utils import (      ExtractorError,      find_xpath_attr,      fix_xml_ampersands, +    format_bytes,      float_or_none,      get_element_by_class,      get_element_by_attribute,      get_elements_by_class,      get_elements_by_attribute, +    get_element_html_by_class, +    get_element_html_by_attribute, +    get_elements_html_by_class, +    get_elements_html_by_attribute, +    get_elements_text_and_html_by_attribute, +    get_element_text_and_html_by_tag,      InAdvancePagedList,      int_or_none,      intlist_to_bytes, @@ -116,6 +124,7 @@ from hypervideo_dl.compat import (      compat_chr,      compat_etree_fromstring,      compat_getenv, +    compat_HTMLParseError,      compat_os_name,      compat_setenv,  ) @@ -151,10 +160,12 @@ class TestUtil(unittest.TestCase):              sanitize_filename('New World record at 0:12:34'),              'New World record at 0_12_34') -        self.assertEqual(sanitize_filename('--gasdgf'), '_-gasdgf') +        self.assertEqual(sanitize_filename('--gasdgf'), '--gasdgf')          self.assertEqual(sanitize_filename('--gasdgf', is_id=True), '--gasdgf') -        self.assertEqual(sanitize_filename('.gasdgf'), 'gasdgf') +        self.assertEqual(sanitize_filename('--gasdgf', is_id=False), '_-gasdgf') +        self.assertEqual(sanitize_filename('.gasdgf'), '.gasdgf')          self.assertEqual(sanitize_filename('.gasdgf', is_id=True), '.gasdgf') +        self.assertEqual(sanitize_filename('.gasdgf', is_id=False), 'gasdgf')          forbidden = '"\0\\/'          for fc in forbidden: @@ -616,6 +627,8 @@ class TestUtil(unittest.TestCase):          self.assertEqual(parse_duration('3h 11m 53s'), 11513)          self.assertEqual(parse_duration('3 hours 11 minutes 53 seconds'), 11513)          self.assertEqual(parse_duration('3 hours 11 mins 53 secs'), 11513) +        self.assertEqual(parse_duration('3 hours, 11 minutes, 53 seconds'), 11513) +        self.assertEqual(parse_duration('3 hours, 11 mins, 53 secs'), 11513)          self.assertEqual(parse_duration('62m45s'), 3765)          self.assertEqual(parse_duration('6m59s'), 419)          self.assertEqual(parse_duration('49s'), 49) @@ -634,6 +647,8 @@ class TestUtil(unittest.TestCase):          self.assertEqual(parse_duration('PT1H0.040S'), 3600.04)          self.assertEqual(parse_duration('PT00H03M30SZ'), 210)          self.assertEqual(parse_duration('P0Y0M0DT0H4M20.880S'), 260.88) +        self.assertEqual(parse_duration('01:02:03:050'), 3723.05) +        self.assertEqual(parse_duration('103:050'), 103.05)      def test_fix_xml_ampersands(self):          self.assertEqual( @@ -1122,7 +1137,7 @@ class TestUtil(unittest.TestCase):      def test_clean_html(self):          self.assertEqual(clean_html('a:\nb'), 'a: b') -        self.assertEqual(clean_html('a:\n   "b"'), 'a:    "b"') +        self.assertEqual(clean_html('a:\n   "b"'), 'a: "b"')          self.assertEqual(clean_html('a<br>\xa0b'), 'a\nb')      def test_intlist_to_bytes(self): @@ -1156,19 +1171,29 @@ class TestUtil(unittest.TestCase):          self.assertEqual(parse_count('1000'), 1000)          self.assertEqual(parse_count('1.000'), 1000)          self.assertEqual(parse_count('1.1k'), 1100) +        self.assertEqual(parse_count('1.1 k'), 1100) +        self.assertEqual(parse_count('1,1 k'), 1100)          self.assertEqual(parse_count('1.1kk'), 1100000)          self.assertEqual(parse_count('1.1kk '), 1100000) +        self.assertEqual(parse_count('1,1kk'), 1100000) +        self.assertEqual(parse_count('100 views'), 100) +        self.assertEqual(parse_count('1,100 views'), 1100)          self.assertEqual(parse_count('1.1kk views'), 1100000) +        self.assertEqual(parse_count('10M views'), 10000000) +        self.assertEqual(parse_count('has 10M views'), 10000000)      def test_parse_resolution(self):          self.assertEqual(parse_resolution(None), {})          self.assertEqual(parse_resolution(''), {}) -        self.assertEqual(parse_resolution('1920x1080'), {'width': 1920, 'height': 1080}) -        self.assertEqual(parse_resolution('1920×1080'), {'width': 1920, 'height': 1080}) +        self.assertEqual(parse_resolution(' 1920x1080'), {'width': 1920, 'height': 1080}) +        self.assertEqual(parse_resolution('1920×1080 '), {'width': 1920, 'height': 1080})          self.assertEqual(parse_resolution('1920 x 1080'), {'width': 1920, 'height': 1080})          self.assertEqual(parse_resolution('720p'), {'height': 720})          self.assertEqual(parse_resolution('4k'), {'height': 2160})          self.assertEqual(parse_resolution('8K'), {'height': 4320}) +        self.assertEqual(parse_resolution('pre_1920x1080_post'), {'width': 1920, 'height': 1080}) +        self.assertEqual(parse_resolution('ep1x2'), {}) +        self.assertEqual(parse_resolution('1920, 1080'), {'width': 1920, 'height': 1080})      def test_parse_bitrate(self):          self.assertEqual(parse_bitrate(None), None) @@ -1219,12 +1244,49 @@ ffmpeg version 2.4.4 Copyright (c) 2000-2014 the FFmpeg ...'''), '2.4.4')      def test_render_table(self):          self.assertEqual(              render_table( +                ['a', 'empty', 'bcd'], +                [[123, '', 4], [9999, '', 51]]), +            'a    empty bcd\n' +            '123        4\n' +            '9999       51') + +        self.assertEqual( +            render_table( +                ['a', 'empty', 'bcd'], +                [[123, '', 4], [9999, '', 51]], +                hide_empty=True), +            'a    bcd\n' +            '123  4\n' +            '9999 51') + +        self.assertEqual( +            render_table( +                ['\ta', 'bcd'], +                [['1\t23', 4], ['\t9999', 51]]), +            '   a bcd\n' +            '1 23 4\n' +            '9999 51') + +        self.assertEqual( +            render_table(                  ['a', 'bcd'], -                [[123, 4], [9999, 51]]), +                [[123, 4], [9999, 51]], +                delim='-'),              'a    bcd\n' +            '--------\n'              '123  4\n'              '9999 51') +        self.assertEqual( +            render_table( +                ['a', 'bcd'], +                [[123, 4], [9999, 51]], +                delim='-', extra_gap=2), +            'a      bcd\n' +            '----------\n' +            '123    4\n' +            '9999   51') +      def test_match_str(self):          # Unary          self.assertFalse(match_str('xy', {'x': 1200})) @@ -1390,21 +1452,21 @@ The first line    </body>  </tt>'''.encode('utf-8')          srt_data = '''1 -00:00:02,080 --> 00:00:05,839 +00:00:02,080 --> 00:00:05,840  <font color="white" face="sansSerif" size="16">default style<font color="red">custom style</font></font>  2 -00:00:02,080 --> 00:00:05,839 +00:00:02,080 --> 00:00:05,840  <b><font color="cyan" face="sansSerif" size="16"><font color="lime">part 1  </font>part 2</font></b>  3 -00:00:05,839 --> 00:00:09,560 +00:00:05,840 --> 00:00:09,560  <u><font color="lime">line 3  part 3</font></u>  4 -00:00:09,560 --> 00:00:12,359 +00:00:09,560 --> 00:00:12,360  <i><u><font color="yellow"><font color="lime">inner   </font>style</font></u></i> @@ -1526,46 +1588,116 @@ Line 1          self.assertEqual(urshift(3, 1), 1)          self.assertEqual(urshift(-3, 1), 2147483646) +    GET_ELEMENT_BY_CLASS_TEST_STRING = ''' +        <span class="foo bar">nice</span> +    ''' +      def test_get_element_by_class(self): -        html = ''' -            <span class="foo bar">nice</span> -        ''' +        html = self.GET_ELEMENT_BY_CLASS_TEST_STRING          self.assertEqual(get_element_by_class('foo', html), 'nice')          self.assertEqual(get_element_by_class('no-such-class', html), None) +    def test_get_element_html_by_class(self): +        html = self.GET_ELEMENT_BY_CLASS_TEST_STRING + +        self.assertEqual(get_element_html_by_class('foo', html), html.strip()) +        self.assertEqual(get_element_by_class('no-such-class', html), None) + +    GET_ELEMENT_BY_ATTRIBUTE_TEST_STRING = ''' +        <div itemprop="author" itemscope>foo</div> +    ''' +      def test_get_element_by_attribute(self): -        html = ''' -            <span class="foo bar">nice</span> -        ''' +        html = self.GET_ELEMENT_BY_CLASS_TEST_STRING          self.assertEqual(get_element_by_attribute('class', 'foo bar', html), 'nice')          self.assertEqual(get_element_by_attribute('class', 'foo', html), None)          self.assertEqual(get_element_by_attribute('class', 'no-such-foo', html), None) -        html = ''' -            <div itemprop="author" itemscope>foo</div> -        ''' +        html = self.GET_ELEMENT_BY_ATTRIBUTE_TEST_STRING          self.assertEqual(get_element_by_attribute('itemprop', 'author', html), 'foo') +    def test_get_element_html_by_attribute(self): +        html = self.GET_ELEMENT_BY_CLASS_TEST_STRING + +        self.assertEqual(get_element_html_by_attribute('class', 'foo bar', html), html.strip()) +        self.assertEqual(get_element_html_by_attribute('class', 'foo', html), None) +        self.assertEqual(get_element_html_by_attribute('class', 'no-such-foo', html), None) + +        html = self.GET_ELEMENT_BY_ATTRIBUTE_TEST_STRING + +        self.assertEqual(get_element_html_by_attribute('itemprop', 'author', html), html.strip()) + +    GET_ELEMENTS_BY_CLASS_TEST_STRING = ''' +        <span class="foo bar">nice</span><span class="foo bar">also nice</span> +    ''' +    GET_ELEMENTS_BY_CLASS_RES = ['<span class="foo bar">nice</span>', '<span class="foo bar">also nice</span>'] +      def test_get_elements_by_class(self): -        html = ''' -            <span class="foo bar">nice</span><span class="foo bar">also nice</span> -        ''' +        html = self.GET_ELEMENTS_BY_CLASS_TEST_STRING          self.assertEqual(get_elements_by_class('foo', html), ['nice', 'also nice'])          self.assertEqual(get_elements_by_class('no-such-class', html), []) +    def test_get_elements_html_by_class(self): +        html = self.GET_ELEMENTS_BY_CLASS_TEST_STRING + +        self.assertEqual(get_elements_html_by_class('foo', html), self.GET_ELEMENTS_BY_CLASS_RES) +        self.assertEqual(get_elements_html_by_class('no-such-class', html), []) +      def test_get_elements_by_attribute(self): -        html = ''' -            <span class="foo bar">nice</span><span class="foo bar">also nice</span> -        ''' +        html = self.GET_ELEMENTS_BY_CLASS_TEST_STRING          self.assertEqual(get_elements_by_attribute('class', 'foo bar', html), ['nice', 'also nice'])          self.assertEqual(get_elements_by_attribute('class', 'foo', html), [])          self.assertEqual(get_elements_by_attribute('class', 'no-such-foo', html), []) +    def test_get_elements_html_by_attribute(self): +        html = self.GET_ELEMENTS_BY_CLASS_TEST_STRING + +        self.assertEqual(get_elements_html_by_attribute('class', 'foo bar', html), self.GET_ELEMENTS_BY_CLASS_RES) +        self.assertEqual(get_elements_html_by_attribute('class', 'foo', html), []) +        self.assertEqual(get_elements_html_by_attribute('class', 'no-such-foo', html), []) + +    def test_get_elements_text_and_html_by_attribute(self): +        html = self.GET_ELEMENTS_BY_CLASS_TEST_STRING + +        self.assertEqual( +            list(get_elements_text_and_html_by_attribute('class', 'foo bar', html)), +            list(zip(['nice', 'also nice'], self.GET_ELEMENTS_BY_CLASS_RES))) +        self.assertEqual(list(get_elements_text_and_html_by_attribute('class', 'foo', html)), []) +        self.assertEqual(list(get_elements_text_and_html_by_attribute('class', 'no-such-foo', html)), []) + +    GET_ELEMENT_BY_TAG_TEST_STRING = ''' +    random text lorem ipsum</p> +    <div> +        this should be returned +        <span>this should also be returned</span> +        <div> +            this should also be returned +        </div> +        closing tag above should not trick, so this should also be returned +    </div> +    but this text should not be returned +    ''' +    GET_ELEMENT_BY_TAG_RES_OUTERDIV_HTML = GET_ELEMENT_BY_TAG_TEST_STRING.strip()[32:276] +    GET_ELEMENT_BY_TAG_RES_OUTERDIV_TEXT = GET_ELEMENT_BY_TAG_RES_OUTERDIV_HTML[5:-6] +    GET_ELEMENT_BY_TAG_RES_INNERSPAN_HTML = GET_ELEMENT_BY_TAG_TEST_STRING.strip()[78:119] +    GET_ELEMENT_BY_TAG_RES_INNERSPAN_TEXT = GET_ELEMENT_BY_TAG_RES_INNERSPAN_HTML[6:-7] + +    def test_get_element_text_and_html_by_tag(self): +        html = self.GET_ELEMENT_BY_TAG_TEST_STRING + +        self.assertEqual( +            get_element_text_and_html_by_tag('div', html), +            (self.GET_ELEMENT_BY_TAG_RES_OUTERDIV_TEXT, self.GET_ELEMENT_BY_TAG_RES_OUTERDIV_HTML)) +        self.assertEqual( +            get_element_text_and_html_by_tag('span', html), +            (self.GET_ELEMENT_BY_TAG_RES_INNERSPAN_TEXT, self.GET_ELEMENT_BY_TAG_RES_INNERSPAN_HTML)) +        self.assertRaises(compat_HTMLParseError, get_element_text_and_html_by_tag, 'article', html) +      def test_iri_to_uri(self):          self.assertEqual(              iri_to_uri('https://www.google.com/search?q=foo&ie=utf-8&oe=utf-8&client=firefox-b'), @@ -1617,9 +1749,9 @@ Line 1          self.assertEqual(repr(LazyList(it)), repr(it))          self.assertEqual(str(LazyList(it)), str(it)) -        self.assertEqual(list(LazyList(it).reverse()), it[::-1]) -        self.assertEqual(list(LazyList(it).reverse()[1:3:7]), it[::-1][1:3:7]) -        self.assertEqual(list(LazyList(it).reverse()[::-1]), it) +        self.assertEqual(list(LazyList(it, reverse=True)), it[::-1]) +        self.assertEqual(list(reversed(LazyList(it))[::-1]), it) +        self.assertEqual(list(reversed(LazyList(it))[1:3:7]), it[::-1][1:3:7])      def test_LazyList_laziness(self): @@ -1632,15 +1764,37 @@ Line 1          test(ll, 5, 5, range(6))          test(ll, -3, 7, range(10)) -        ll = LazyList(range(10)).reverse() +        ll = LazyList(range(10), reverse=True)          test(ll, -1, 0, range(1))          test(ll, 3, 6, range(10))          ll = LazyList(itertools.count())          test(ll, 10, 10, range(11)) -        ll.reverse() +        ll = reversed(ll)          test(ll, -15, 14, range(15)) +    def test_format_bytes(self): +        self.assertEqual(format_bytes(0), '0.00B') +        self.assertEqual(format_bytes(1000), '1000.00B') +        self.assertEqual(format_bytes(1024), '1.00KiB') +        self.assertEqual(format_bytes(1024**2), '1.00MiB') +        self.assertEqual(format_bytes(1024**3), '1.00GiB') +        self.assertEqual(format_bytes(1024**4), '1.00TiB') +        self.assertEqual(format_bytes(1024**5), '1.00PiB') +        self.assertEqual(format_bytes(1024**6), '1.00EiB') +        self.assertEqual(format_bytes(1024**7), '1.00ZiB') +        self.assertEqual(format_bytes(1024**8), '1.00YiB') +        self.assertEqual(format_bytes(1024**9), '1024.00YiB') + +    def test_hide_login_info(self): +        self.assertEqual(Config.hide_login_info(['-u', 'foo', '-p', 'bar']), +                         ['-u', 'PRIVATE', '-p', 'PRIVATE']) +        self.assertEqual(Config.hide_login_info(['-u']), ['-u']) +        self.assertEqual(Config.hide_login_info(['-u', 'foo', '-u', 'bar']), +                         ['-u', 'PRIVATE', '-u', 'PRIVATE']) +        self.assertEqual(Config.hide_login_info(['--username=foo']), +                         ['--username=PRIVATE']) +  if __name__ == '__main__':      unittest.main() diff --git a/test/test_verbose_output.py b/test/test_verbose_output.py index 050fd76..98c6d70 100644 --- a/test/test_verbose_output.py +++ b/test/test_verbose_output.py @@ -19,52 +19,52 @@ class TestVerboseOutput(unittest.TestCase):              [                  sys.executable, 'hypervideo_dl/__main__.py', '-v',                  '--username', 'johnsmith@gmail.com', -                '--password', 'secret', +                '--password', 'my_secret_password',              ], cwd=rootDir, stdout=subprocess.PIPE, stderr=subprocess.PIPE)          sout, serr = outp.communicate()          self.assertTrue(b'--username' in serr)          self.assertTrue(b'johnsmith' not in serr)          self.assertTrue(b'--password' in serr) -        self.assertTrue(b'secret' not in serr) +        self.assertTrue(b'my_secret_password' not in serr)      def test_private_info_shortarg(self):          outp = subprocess.Popen(              [                  sys.executable, 'hypervideo_dl/__main__.py', '-v',                  '-u', 'johnsmith@gmail.com', -                '-p', 'secret', +                '-p', 'my_secret_password',              ], cwd=rootDir, stdout=subprocess.PIPE, stderr=subprocess.PIPE)          sout, serr = outp.communicate()          self.assertTrue(b'-u' in serr)          self.assertTrue(b'johnsmith' not in serr)          self.assertTrue(b'-p' in serr) -        self.assertTrue(b'secret' not in serr) +        self.assertTrue(b'my_secret_password' not in serr)      def test_private_info_eq(self):          outp = subprocess.Popen(              [                  sys.executable, 'hypervideo_dl/__main__.py', '-v',                  '--username=johnsmith@gmail.com', -                '--password=secret', +                '--password=my_secret_password',              ], cwd=rootDir, stdout=subprocess.PIPE, stderr=subprocess.PIPE)          sout, serr = outp.communicate()          self.assertTrue(b'--username' in serr)          self.assertTrue(b'johnsmith' not in serr)          self.assertTrue(b'--password' in serr) -        self.assertTrue(b'secret' not in serr) +        self.assertTrue(b'my_secret_password' not in serr)      def test_private_info_shortarg_eq(self):          outp = subprocess.Popen(              [                  sys.executable, 'hypervideo_dl/__main__.py', '-v',                  '-u=johnsmith@gmail.com', -                '-p=secret', +                '-p=my_secret_password',              ], cwd=rootDir, stdout=subprocess.PIPE, stderr=subprocess.PIPE)          sout, serr = outp.communicate()          self.assertTrue(b'-u' in serr)          self.assertTrue(b'johnsmith' not in serr)          self.assertTrue(b'-p' in serr) -        self.assertTrue(b'secret' not in serr) +        self.assertTrue(b'my_secret_password' not in serr)  if __name__ == '__main__': diff --git a/test/test_youtube_lists.py b/test/test_youtube_lists.py index 2da1a50..b94b733 100644 --- a/test/test_youtube_lists.py +++ b/test/test_youtube_lists.py @@ -9,11 +9,9 @@ sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))  from test.helper import FakeYDL, is_download_test -  from hypervideo_dl.extractor import ( -    YoutubePlaylistIE, -    YoutubeTabIE,      YoutubeIE, +    YoutubeTabIE,  ) @@ -26,38 +24,20 @@ class TestYoutubeLists(unittest.TestCase):      def test_youtube_playlist_noplaylist(self):          dl = FakeYDL()          dl.params['noplaylist'] = True -        ie = YoutubePlaylistIE(dl) -        result = ie.extract('https://www.youtube.com/watch?v=FXxLjLQi3Fg&list=PLwiyx1dc3P2JR9N8gQaQN_BCvlSlap7re') +        ie = YoutubeTabIE(dl) +        result = ie.extract('https://www.youtube.com/watch?v=OmJ-4B-mS-Y&list=PLydZ2Hrp_gPRJViZjLFKaBMgCQOYEEkyp&index=2')          self.assertEqual(result['_type'], 'url') -        self.assertEqual(YoutubeIE().extract_id(result['url']), 'FXxLjLQi3Fg') - -    def test_youtube_course(self): -        dl = FakeYDL() -        ie = YoutubePlaylistIE(dl) -        # TODO find a > 100 (paginating?) videos course -        result = ie.extract('https://www.youtube.com/course?list=ECUl4u3cNGP61MdtwGTqZA0MreSaDybji8') -        entries = list(result['entries']) -        self.assertEqual(YoutubeIE().extract_id(entries[0]['url']), 'j9WZyLZCBzs') -        self.assertEqual(len(entries), 25) -        self.assertEqual(YoutubeIE().extract_id(entries[-1]['url']), 'rYefUsYuEp0') +        self.assertEqual(result['ie_key'], YoutubeIE.ie_key()) +        self.assertEqual(YoutubeIE.extract_id(result['url']), 'OmJ-4B-mS-Y')      def test_youtube_mix(self):          dl = FakeYDL() -        ie = YoutubePlaylistIE(dl) -        result = ie.extract('https://www.youtube.com/watch?v=W01L70IGBgE&index=2&list=RDOQpdSVF_k_w') -        entries = result['entries'] +        ie = YoutubeTabIE(dl) +        result = ie.extract('https://www.youtube.com/watch?v=tyITL_exICo&list=RDCLAK5uy_kLWIr9gv1XLlPbaDS965-Db4TrBoUTxQ8') +        entries = list(result['entries'])          self.assertTrue(len(entries) >= 50)          original_video = entries[0] -        self.assertEqual(original_video['id'], 'OQpdSVF_k_w') - -    def test_youtube_toptracks(self): -        print('Skipping: The playlist page gives error 500') -        return -        dl = FakeYDL() -        ie = YoutubePlaylistIE(dl) -        result = ie.extract('https://www.youtube.com/playlist?list=MCUS') -        entries = result['entries'] -        self.assertEqual(len(entries), 100) +        self.assertEqual(original_video['id'], 'tyITL_exICo')      def test_youtube_flat_playlist_extraction(self):          dl = FakeYDL() @@ -68,10 +48,10 @@ class TestYoutubeLists(unittest.TestCase):          entries = list(result['entries'])          self.assertTrue(len(entries) == 1)          video = entries[0] -        self.assertEqual(video['_type'], 'url_transparent') +        self.assertEqual(video['_type'], 'url')          self.assertEqual(video['ie_key'], 'Youtube')          self.assertEqual(video['id'], 'BaW_jenozKc') -        self.assertEqual(video['url'], 'BaW_jenozKc') +        self.assertEqual(video['url'], 'https://www.youtube.com/watch?v=BaW_jenozKc')          self.assertEqual(video['title'], 'youtube-dl test video "\'/\\ä↭𝕐')          self.assertEqual(video['duration'], 10)          self.assertEqual(video['uploader'], 'Philipp Hagemeister') | 
