1 from __future__ import unicode_literals
6 from .common import FileDownloader
13 class ExternalFD(FileDownloader):
14 def real_download(self, filename, info_dict):
15 self.report_destination(filename)
16 tmpfilename = self.temp_name(filename)
18 retval = self._call_downloader(tmpfilename, info_dict)
20 fsize = os.path.getsize(encodeFilename(tmpfilename))
21 self.to_screen('\r[%s] Downloaded %s bytes' % (self.get_basename(), fsize))
22 self.try_rename(tmpfilename, filename)
24 'downloaded_bytes': fsize,
32 self.report_error('%s exited with code %d' % (
33 self.get_basename(), retval))
37 def get_basename(cls):
38 return cls.__name__[:-2].lower()
42 return self.params.get('external_downloader')
45 def supports(cls, info_dict):
46 return info_dict['protocol'] in ('http', 'https', 'ftp', 'ftps')
48 def _source_address(self, command_option):
49 source_address = self.params.get('source_address')
50 if source_address is None:
52 return [command_option, source_address]
54 def _option(self, command_option, param):
55 param = self.params.get(param)
58 if isinstance(param, bool):
59 return [command_option]
60 return [command_option, param]
62 def _no_check_certificate(self, command_option):
63 return [command_option] if self.params.get('nocheckcertificate', False) else []
65 def _configuration_args(self, default=[]):
66 ex_args = self.params.get('external_downloader_args')
69 assert isinstance(ex_args, list)
72 def _call_downloader(self, tmpfilename, info_dict):
73 """ Either overwrite this or implement _make_cmd """
74 cmd = [encodeArgument(a) for a in self._make_cmd(tmpfilename, info_dict)]
79 cmd, stderr=subprocess.PIPE)
80 _, stderr = p.communicate()
82 self.to_stderr(stderr)
86 class CurlFD(ExternalFD):
87 def _make_cmd(self, tmpfilename, info_dict):
88 cmd = [self.exe, '--location', '-o', tmpfilename]
89 for key, val in info_dict['http_headers'].items():
90 cmd += ['--header', '%s: %s' % (key, val)]
91 cmd += self._source_address('--interface')
92 cmd += self._configuration_args()
93 cmd += ['--', info_dict['url']]
97 class AxelFD(ExternalFD):
98 def _make_cmd(self, tmpfilename, info_dict):
99 cmd = [self.exe, '-o', tmpfilename]
100 for key, val in info_dict['http_headers'].items():
101 cmd += ['-H', '%s: %s' % (key, val)]
102 cmd += self._configuration_args()
103 cmd += ['--', info_dict['url']]
107 class WgetFD(ExternalFD):
108 def _make_cmd(self, tmpfilename, info_dict):
109 cmd = [self.exe, '-O', tmpfilename, '-nv', '--no-cookies']
110 for key, val in info_dict['http_headers'].items():
111 cmd += ['--header', '%s: %s' % (key, val)]
112 cmd += self._source_address('--bind-address')
113 cmd += self._option('--proxy', 'proxy')
114 cmd += self._no_check_certificate('--no-check-certificate')
115 cmd += self._configuration_args()
116 cmd += ['--', info_dict['url']]
120 class Aria2cFD(ExternalFD):
121 def _make_cmd(self, tmpfilename, info_dict):
122 cmd = [self.exe, '-c']
123 cmd += self._configuration_args([
124 '--min-split-size', '1M', '--max-connection-per-server', '4'])
125 dn = os.path.dirname(tmpfilename)
128 cmd += ['--out', os.path.basename(tmpfilename)]
129 for key, val in info_dict['http_headers'].items():
130 cmd += ['--header', '%s: %s' % (key, val)]
131 cmd += self._source_address('--interface')
132 cmd += self._option('--all-proxy', 'proxy')
133 cmd += ['--', info_dict['url']]
137 class HttpieFD(ExternalFD):
138 def _make_cmd(self, tmpfilename, info_dict):
139 cmd = ['http', '--download', '--output', tmpfilename, info_dict['url']]
140 for key, val in info_dict['http_headers'].items():
141 cmd += ['%s:%s' % (key, val)]
145 (klass.get_basename(), klass)
146 for name, klass in globals().items()
147 if name.endswith('FD') and name != 'ExternalFD'
151 def list_external_downloaders():
152 return sorted(_BY_NAME.keys())
155 def get_external_downloader(external_downloader):
156 """ Given the name of the executable, see whether we support the given
158 # Drop .exe extension on Windows
159 bn = os.path.splitext(os.path.basename(external_downloader))[0]