mirror of
				https://github.com/ytdl-org/youtube-dl
				synced 2025-10-26 15:22:52 +00:00 
			
		
		
		
	[feat]: Add support to external downloader aria2p (#31500)
* feat: add class Aria2pFD * feat: create call_downloader function * feat: a colorful download interface to aria2pFD * feat: change value name * Apply suggestions from code review Co-authored-by: dirkf <fieldhouse@gmx.net> * Typo in suggestion * fix: remove unused value * fix: add not function to return value(0 is normal); add total_seconds to download.eta(timedelta object); add waiting status when hook progress * fix: remove unuse method ..utils.format_bytes * fix: be up to flake8 * fix: be up to flake8 * Apply suggestions from code review * [feat] test external downloader aria2p * [feat] test external downloader aria2p * [fix] test_external_downloader.py * Apply suggestions from code review Co-authored-by: dirkf <fieldhouse@gmx.net> * Apply suggestions from code review Co-authored-by: dirkf <fieldhouse@gmx.net> * Update test/test_external_downloader.py Co-authored-by: dirkf <fieldhouse@gmx.net> * Update test/test_external_downloader.py Co-authored-by: dirkf <fieldhouse@gmx.net> * Update youtube_dl/downloader/external.py Co-authored-by: dirkf <fieldhouse@gmx.net> * refactoring code and fix bugs * Apply suggestions from code review * Rename test_external_downloader.py to test_downloader_external.py --------- Co-authored-by: dirkf <fieldhouse@gmx.net>
This commit is contained in:
		
							parent
							
								
									f33923cba7
								
							
						
					
					
						commit
						33db85c571
					
				| @ -89,6 +89,17 @@ class FakeYDL(YoutubeDL): | |||||||
|         self.report_warning = types.MethodType(report_warning, self) |         self.report_warning = types.MethodType(report_warning, self) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | class FakeLogger(object): | ||||||
|  |     def debug(self, msg): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  |     def warning(self, msg): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  |     def error(self, msg): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def gettestcases(include_onlymatching=False): | def gettestcases(include_onlymatching=False): | ||||||
|     for ie in youtube_dl.extractor.gen_extractors(): |     for ie in youtube_dl.extractor.gen_extractors(): | ||||||
|         for tc in ie.get_testcases(include_onlymatching): |         for tc in ie.get_testcases(include_onlymatching): | ||||||
|  | |||||||
							
								
								
									
										115
									
								
								test/test_downloader_external.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								test/test_downloader_external.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,115 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  | # coding: utf-8 | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | 
 | ||||||
|  | # Allow direct execution | ||||||
|  | import os | ||||||
|  | import re | ||||||
|  | import sys | ||||||
|  | import subprocess | ||||||
|  | import unittest | ||||||
|  | sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||||
|  | 
 | ||||||
|  | from test.helper import ( | ||||||
|  |     FakeLogger, | ||||||
|  |     http_server_port, | ||||||
|  |     try_rm, | ||||||
|  | ) | ||||||
|  | from youtube_dl import YoutubeDL | ||||||
|  | from youtube_dl.compat import compat_http_server | ||||||
|  | from youtube_dl.utils import encodeFilename | ||||||
|  | from youtube_dl.downloader.external import Aria2pFD | ||||||
|  | import threading | ||||||
|  | 
 | ||||||
|  | TEST_DIR = os.path.dirname(os.path.abspath(__file__)) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | TEST_SIZE = 10 * 1024 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class HTTPTestRequestHandler(compat_http_server.BaseHTTPRequestHandler): | ||||||
|  |     def log_message(self, format, *args): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  |     def send_content_range(self, total=None): | ||||||
|  |         range_header = self.headers.get('Range') | ||||||
|  |         start = end = None | ||||||
|  |         if range_header: | ||||||
|  |             mobj = re.match(r'bytes=(\d+)-(\d+)', range_header) | ||||||
|  |             if mobj: | ||||||
|  |                 start, end = (int(mobj.group(i)) for i in (1, 2)) | ||||||
|  |         valid_range = start is not None and end is not None | ||||||
|  |         if valid_range: | ||||||
|  |             content_range = 'bytes %d-%d' % (start, end) | ||||||
|  |             if total: | ||||||
|  |                 content_range += '/%d' % total | ||||||
|  |             self.send_header('Content-Range', content_range) | ||||||
|  |         return (end - start + 1) if valid_range else total | ||||||
|  | 
 | ||||||
|  |     def serve(self, range=True, content_length=True): | ||||||
|  |         self.send_response(200) | ||||||
|  |         self.send_header('Content-Type', 'video/mp4') | ||||||
|  |         size = TEST_SIZE | ||||||
|  |         if range: | ||||||
|  |             size = self.send_content_range(TEST_SIZE) | ||||||
|  |         if content_length: | ||||||
|  |             self.send_header('Content-Length', size) | ||||||
|  |         self.end_headers() | ||||||
|  |         self.wfile.write(b'#' * size) | ||||||
|  | 
 | ||||||
|  |     def do_GET(self): | ||||||
|  |         if self.path == '/regular': | ||||||
|  |             self.serve() | ||||||
|  |         elif self.path == '/no-content-length': | ||||||
|  |             self.serve(content_length=False) | ||||||
|  |         elif self.path == '/no-range': | ||||||
|  |             self.serve(range=False) | ||||||
|  |         elif self.path == '/no-range-no-content-length': | ||||||
|  |             self.serve(range=False, content_length=False) | ||||||
|  |         else: | ||||||
|  |             assert False, 'unrecognised server path' | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @unittest.skipUnless(Aria2pFD.available(), 'aria2p module not found') | ||||||
|  | class TestAria2pFD(unittest.TestCase): | ||||||
|  |     def setUp(self): | ||||||
|  |         self.httpd = compat_http_server.HTTPServer( | ||||||
|  |             ('127.0.0.1', 0), HTTPTestRequestHandler) | ||||||
|  |         self.port = http_server_port(self.httpd) | ||||||
|  |         self.server_thread = threading.Thread(target=self.httpd.serve_forever) | ||||||
|  |         self.server_thread.daemon = True | ||||||
|  |         self.server_thread.start() | ||||||
|  | 
 | ||||||
|  |     def download(self, params, ep): | ||||||
|  |         with subprocess.Popen( | ||||||
|  |             ['aria2c', '--enable-rpc'], | ||||||
|  |             stdout=subprocess.DEVNULL, | ||||||
|  |             stderr=subprocess.DEVNULL | ||||||
|  |         ) as process: | ||||||
|  |             if not process.poll(): | ||||||
|  |                 filename = 'testfile.mp4' | ||||||
|  |                 params['logger'] = FakeLogger() | ||||||
|  |                 params['outtmpl'] = filename | ||||||
|  |                 ydl = YoutubeDL(params) | ||||||
|  |                 try_rm(encodeFilename(filename)) | ||||||
|  |                 self.assertEqual(ydl.download(['http://127.0.0.1:%d/%s' % (self.port, ep)]), 0) | ||||||
|  |                 self.assertEqual(os.path.getsize(encodeFilename(filename)), TEST_SIZE) | ||||||
|  |                 try_rm(encodeFilename(filename)) | ||||||
|  |             process.kill() | ||||||
|  | 
 | ||||||
|  |     def download_all(self, params): | ||||||
|  |         for ep in ('regular', 'no-content-length', 'no-range', 'no-range-no-content-length'): | ||||||
|  |             self.download(params, ep) | ||||||
|  | 
 | ||||||
|  |     def test_regular(self): | ||||||
|  |         self.download_all({'external_downloader': 'aria2p'}) | ||||||
|  | 
 | ||||||
|  |     def test_chunked(self): | ||||||
|  |         self.download_all({ | ||||||
|  |             'external_downloader': 'aria2p', | ||||||
|  |             'http_chunk_size': 1000, | ||||||
|  |         }) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     unittest.main() | ||||||
| @ -9,7 +9,11 @@ import sys | |||||||
| import unittest | import unittest | ||||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||||
| 
 | 
 | ||||||
| from test.helper import http_server_port, try_rm | from test.helper import ( | ||||||
|  |     FakeLogger, | ||||||
|  |     http_server_port, | ||||||
|  |     try_rm, | ||||||
|  | ) | ||||||
| from youtube_dl import YoutubeDL | from youtube_dl import YoutubeDL | ||||||
| from youtube_dl.compat import compat_http_server | from youtube_dl.compat import compat_http_server | ||||||
| from youtube_dl.downloader.http import HttpFD | from youtube_dl.downloader.http import HttpFD | ||||||
| @ -66,17 +70,6 @@ class HTTPTestRequestHandler(compat_http_server.BaseHTTPRequestHandler): | |||||||
|             assert False |             assert False | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class FakeLogger(object): |  | ||||||
|     def debug(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
|     def warning(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
|     def error(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class TestHttpFD(unittest.TestCase): | class TestHttpFD(unittest.TestCase): | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|         self.httpd = compat_http_server.HTTPServer( |         self.httpd = compat_http_server.HTTPServer( | ||||||
|  | |||||||
| @ -8,7 +8,10 @@ import sys | |||||||
| import unittest | import unittest | ||||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||||
| 
 | 
 | ||||||
| from test.helper import http_server_port | from test.helper import ( | ||||||
|  |     FakeLogger, | ||||||
|  |     http_server_port, | ||||||
|  | ) | ||||||
| from youtube_dl import YoutubeDL | from youtube_dl import YoutubeDL | ||||||
| from youtube_dl.compat import compat_http_server, compat_urllib_request | from youtube_dl.compat import compat_http_server, compat_urllib_request | ||||||
| import ssl | import ssl | ||||||
| @ -52,17 +55,6 @@ class HTTPTestRequestHandler(compat_http_server.BaseHTTPRequestHandler): | |||||||
|             assert False |             assert False | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class FakeLogger(object): |  | ||||||
|     def debug(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
|     def warning(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
|     def error(self, msg): |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class TestHTTP(unittest.TestCase): | class TestHTTP(unittest.TestCase): | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|         self.httpd = compat_http_server.HTTPServer( |         self.httpd = compat_http_server.HTTPServer( | ||||||
|  | |||||||
| @ -200,6 +200,64 @@ class Aria2cFD(ExternalFD): | |||||||
|         return cmd |         return cmd | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | class Aria2pFD(ExternalFD): | ||||||
|  |     ''' Aria2pFD class | ||||||
|  |     This class support to use aria2p as downloader. | ||||||
|  |     (Aria2p, a command-line tool and Python library to interact with an aria2c daemon process | ||||||
|  |     through JSON-RPC.) | ||||||
|  |     It can help you to get download progress more easily. | ||||||
|  |     To use aria2p as downloader, you need to install aria2c and aria2p, aria2p can download with pip. | ||||||
|  |     Then run aria2c in the background and enable with the --enable-rpc option. | ||||||
|  |     ''' | ||||||
|  |     try: | ||||||
|  |         import aria2p | ||||||
|  |         __avail = True | ||||||
|  |     except ImportError: | ||||||
|  |         __avail = False | ||||||
|  | 
 | ||||||
|  |     @classmethod | ||||||
|  |     def available(cls): | ||||||
|  |         return cls.__avail | ||||||
|  | 
 | ||||||
|  |     def _call_downloader(self, tmpfilename, info_dict): | ||||||
|  |         aria2 = self.aria2p.API( | ||||||
|  |             self.aria2p.Client( | ||||||
|  |                 host='http://localhost', | ||||||
|  |                 port=6800, | ||||||
|  |                 secret='' | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         options = { | ||||||
|  |             'min-split-size': '1M', | ||||||
|  |             'max-connection-per-server': 4, | ||||||
|  |             'auto-file-renaming': 'false', | ||||||
|  |         } | ||||||
|  |         options['dir'] = os.path.dirname(tmpfilename) or os.path.abspath('.') | ||||||
|  |         options['out'] = os.path.basename(tmpfilename) | ||||||
|  |         options['header'] = [] | ||||||
|  |         for key, val in info_dict['http_headers'].items(): | ||||||
|  |             options['header'].append('{0}: {1}'.format(key, val)) | ||||||
|  |         download = aria2.add_uris([info_dict['url']], options) | ||||||
|  |         status = { | ||||||
|  |             'status': 'downloading', | ||||||
|  |             'tmpfilename': tmpfilename, | ||||||
|  |         } | ||||||
|  |         started = time.time() | ||||||
|  |         while download.status in ['active', 'waiting']: | ||||||
|  |             download = aria2.get_download(download.gid) | ||||||
|  |             status.update({ | ||||||
|  |                 'downloaded_bytes': download.completed_length, | ||||||
|  |                 'total_bytes': download.total_length, | ||||||
|  |                 'elapsed': time.time() - started, | ||||||
|  |                 'eta': download.eta.total_seconds(), | ||||||
|  |                 'speed': download.download_speed, | ||||||
|  |             }) | ||||||
|  |             self._hook_progress(status) | ||||||
|  |             time.sleep(.5) | ||||||
|  |         return download.status != 'complete' | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| class HttpieFD(ExternalFD): | class HttpieFD(ExternalFD): | ||||||
|     @classmethod |     @classmethod | ||||||
|     def available(cls): |     def available(cls): | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 teddy171
						teddy171