]> gitweb @ CieloNegro.org - youtube-dl.git/blob - youtube_dl/downloader/hls.py
Merge pull request #8130 from dyn888/master
[youtube-dl.git] / youtube_dl / downloader / hls.py
1 from __future__ import unicode_literals
2
3 import os
4 import re
5 import subprocess
6
7 from .common import FileDownloader
8 from .fragment import FragmentFD
9
10 from ..compat import compat_urlparse
11 from ..postprocessor.ffmpeg import FFmpegPostProcessor
12 from ..utils import (
13     encodeArgument,
14     encodeFilename,
15     sanitize_open,
16     handle_youtubedl_headers,
17 )
18
19
20 class HlsFD(FileDownloader):
21     def real_download(self, filename, info_dict):
22         url = info_dict['url']
23         self.report_destination(filename)
24         tmpfilename = self.temp_name(filename)
25
26         ffpp = FFmpegPostProcessor(downloader=self)
27         if not ffpp.available:
28             self.report_error('m3u8 download detected but ffmpeg or avconv could not be found. Please install one.')
29             return False
30         ffpp.check_version()
31
32         args = [ffpp.executable, '-y']
33
34         if info_dict['http_headers'] and re.match(r'^https?://', url):
35             # Trailing \r\n after each HTTP header is important to prevent warning from ffmpeg/avconv:
36             # [http @ 00000000003d2fa0] No trailing CRLF found in HTTP header.
37             headers = handle_youtubedl_headers(info_dict['http_headers'])
38             args += [
39                 '-headers',
40                 ''.join('%s: %s\r\n' % (key, val) for key, val in headers.items())]
41
42         args += ['-i', url, '-f', 'mp4', '-c', 'copy', '-bsf:a', 'aac_adtstoasc']
43
44         args = [encodeArgument(opt) for opt in args]
45         args.append(encodeFilename(ffpp._ffmpeg_filename_argument(tmpfilename), True))
46
47         self._debug_cmd(args)
48
49         proc = subprocess.Popen(args, stdin=subprocess.PIPE)
50         try:
51             retval = proc.wait()
52         except KeyboardInterrupt:
53             # subprocces.run would send the SIGKILL signal to ffmpeg and the
54             # mp4 file couldn't be played, but if we ask ffmpeg to quit it
55             # produces a file that is playable (this is mostly useful for live
56             # streams)
57             proc.communicate(b'q')
58             raise
59         if retval == 0:
60             fsize = os.path.getsize(encodeFilename(tmpfilename))
61             self.to_screen('\r[%s] %s bytes' % (args[0], fsize))
62             self.try_rename(tmpfilename, filename)
63             self._hook_progress({
64                 'downloaded_bytes': fsize,
65                 'total_bytes': fsize,
66                 'filename': filename,
67                 'status': 'finished',
68             })
69             return True
70         else:
71             self.to_stderr('\n')
72             self.report_error('%s exited with code %d' % (ffpp.basename, retval))
73             return False
74
75
76 class NativeHlsFD(FragmentFD):
77     """ A more limited implementation that does not require ffmpeg """
78
79     FD_NAME = 'hlsnative'
80
81     def real_download(self, filename, info_dict):
82         man_url = info_dict['url']
83         self.to_screen('[%s] Downloading m3u8 manifest' % self.FD_NAME)
84         manifest = self.ydl.urlopen(man_url).read()
85
86         s = manifest.decode('utf-8', 'ignore')
87         fragment_urls = []
88         for line in s.splitlines():
89             line = line.strip()
90             if line and not line.startswith('#'):
91                 segment_url = (
92                     line
93                     if re.match(r'^https?://', line)
94                     else compat_urlparse.urljoin(man_url, line))
95                 fragment_urls.append(segment_url)
96                 # We only download the first fragment during the test
97                 if self.params.get('test', False):
98                     break
99
100         ctx = {
101             'filename': filename,
102             'total_frags': len(fragment_urls),
103         }
104
105         self._prepare_and_start_frag_download(ctx)
106
107         frags_filenames = []
108         for i, frag_url in enumerate(fragment_urls):
109             frag_filename = '%s-Frag%d' % (ctx['tmpfilename'], i)
110             success = ctx['dl'].download(frag_filename, {'url': frag_url})
111             if not success:
112                 return False
113             down, frag_sanitized = sanitize_open(frag_filename, 'rb')
114             ctx['dest_stream'].write(down.read())
115             down.close()
116             frags_filenames.append(frag_sanitized)
117
118         self._finish_frag_download(ctx)
119
120         for frag_file in frags_filenames:
121             os.remove(encodeFilename(frag_file))
122
123         return True