| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  | import io | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | import os | 
					
						
							|  |  |  | import subprocess | 
					
						
							|  |  |  | import time | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  | import re | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from .common import AudioConversionError, PostProcessor | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-02 11:23:40 +01:00
										 |  |  | from ..utils import ( | 
					
						
							| 
									
										
										
										
											2014-05-16 15:47:54 +02:00
										 |  |  |     encodeArgument, | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |     encodeFilename, | 
					
						
							| 
									
										
										
										
											2014-11-02 10:50:30 +01:00
										 |  |  |     get_exe_version, | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  |     is_outdated_version, | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |     PostProcessingError, | 
					
						
							|  |  |  |     prepend_extension, | 
					
						
							|  |  |  |     shell_quote, | 
					
						
							|  |  |  |     subtitles_filename, | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  |     dfxp2srt, | 
					
						
							| 
									
										
										
										
											2015-06-21 18:53:17 +08:00
										 |  |  |     ISO639Utils, | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |     replace_extension, | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-13 12:15:29 +01:00
										 |  |  | EXT_TO_OUT_FORMATS = { | 
					
						
							| 
									
										
										
										
											2017-03-16 12:50:45 +01:00
										 |  |  |     'aac': 'adts', | 
					
						
							|  |  |  |     'flac': 'flac', | 
					
						
							|  |  |  |     'm4a': 'ipod', | 
					
						
							|  |  |  |     'mka': 'matroska', | 
					
						
							|  |  |  |     'mkv': 'matroska', | 
					
						
							|  |  |  |     'mpg': 'mpeg', | 
					
						
							|  |  |  |     'ogv': 'ogg', | 
					
						
							|  |  |  |     'ts': 'mpegts', | 
					
						
							|  |  |  |     'wma': 'asf', | 
					
						
							|  |  |  |     'wmv': 'asf', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | ACODECS = { | 
					
						
							|  |  |  |     'mp3': 'libmp3lame', | 
					
						
							|  |  |  |     'aac': 'aac', | 
					
						
							|  |  |  |     'flac': 'flac', | 
					
						
							|  |  |  |     'm4a': 'aac', | 
					
						
							| 
									
										
										
										
											2017-10-02 04:43:25 +07:00
										 |  |  |     'opus': 'libopus', | 
					
						
							| 
									
										
										
										
											2017-03-16 12:50:45 +01:00
										 |  |  |     'vorbis': 'libvorbis', | 
					
						
							|  |  |  |     'wav': None, | 
					
						
							| 
									
										
										
										
											2016-03-13 12:15:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | class FFmpegPostProcessorError(PostProcessingError): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-23 02:55:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | class FFmpegPostProcessor(PostProcessor): | 
					
						
							| 
									
										
										
										
											2015-04-18 11:52:36 +02:00
										 |  |  |     def __init__(self, downloader=None): | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         PostProcessor.__init__(self, downloader) | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |         self._determine_executables() | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  |     def check_version(self): | 
					
						
							| 
									
										
										
										
											2015-02-17 17:26:41 +01:00
										 |  |  |         if not self.available: | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             raise FFmpegPostProcessorError('ffmpeg or avconv not found. Please install one.') | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 17:27:29 +01:00
										 |  |  |         required_version = '10-0' if self.basename == 'avconv' else '1.0' | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  |         if is_outdated_version( | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |                 self._versions[self.basename], required_version): | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             warning = 'Your copy of %s is outdated, update %s to version %s or newer if you encounter any errors.' % ( | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |                 self.basename, self.basename, required_version) | 
					
						
							| 
									
										
										
										
											2014-10-26 21:00:42 +01:00
										 |  |  |             if self._downloader: | 
					
						
							|  |  |  |                 self._downloader.report_warning(warning) | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |     @staticmethod | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |     def get_versions(downloader=None): | 
					
						
							|  |  |  |         return FFmpegPostProcessor(downloader)._versions | 
					
						
							| 
									
										
										
										
											2015-01-10 05:45:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |     def _determine_executables(self): | 
					
						
							|  |  |  |         programs = ['avprobe', 'avconv', 'ffmpeg', 'ffprobe'] | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |         prefer_ffmpeg = True | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 15:09:44 +01:00
										 |  |  |         def get_ffmpeg_version(path): | 
					
						
							|  |  |  |             ver = get_exe_version(path, args=['-version']) | 
					
						
							|  |  |  |             if ver: | 
					
						
							|  |  |  |                 regexs = [ | 
					
						
							| 
									
										
										
										
											2019-01-12 00:30:06 +07:00
										 |  |  |                     r'(?:\d+:)?([0-9.]+)-[0-9]+ubuntu[0-9.]+$',  # Ubuntu, see [1] | 
					
						
							| 
									
										
										
										
											2019-01-11 23:47:23 +07:00
										 |  |  |                     r'n([0-9.]+)$',  # Arch Linux | 
					
						
							| 
									
										
										
										
											2019-01-12 00:30:06 +07:00
										 |  |  |                     # 1. http://www.ducea.com/2006/06/17/ubuntu-package-version-naming-explanation/ | 
					
						
							| 
									
										
										
										
											2019-01-11 15:09:44 +01:00
										 |  |  |                 ] | 
					
						
							|  |  |  |                 for regex in regexs: | 
					
						
							|  |  |  |                     mobj = re.match(regex, ver) | 
					
						
							|  |  |  |                     if mobj: | 
					
						
							|  |  |  |                         ver = mobj.group(1) | 
					
						
							|  |  |  |             return ver | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |         self.basename = None | 
					
						
							|  |  |  |         self.probe_basename = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._paths = None | 
					
						
							|  |  |  |         self._versions = None | 
					
						
							|  |  |  |         if self._downloader: | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |             prefer_ffmpeg = self._downloader.params.get('prefer_ffmpeg', True) | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |             location = self._downloader.params.get('ffmpeg_location') | 
					
						
							|  |  |  |             if location is not None: | 
					
						
							|  |  |  |                 if not os.path.exists(location): | 
					
						
							|  |  |  |                     self._downloader.report_warning( | 
					
						
							|  |  |  |                         'ffmpeg-location %s does not exist! ' | 
					
						
							|  |  |  |                         'Continuing without avconv/ffmpeg.' % (location)) | 
					
						
							|  |  |  |                     self._versions = {} | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 elif not os.path.isdir(location): | 
					
						
							|  |  |  |                     basename = os.path.splitext(os.path.basename(location))[0] | 
					
						
							|  |  |  |                     if basename not in programs: | 
					
						
							|  |  |  |                         self._downloader.report_warning( | 
					
						
							|  |  |  |                             'Cannot identify executable %s, its basename should be one of %s. ' | 
					
						
							|  |  |  |                             'Continuing without avconv/ffmpeg.' % | 
					
						
							|  |  |  |                             (location, ', '.join(programs))) | 
					
						
							|  |  |  |                         self._versions = {} | 
					
						
							|  |  |  |                         return None | 
					
						
							|  |  |  |                     location = os.path.dirname(os.path.abspath(location)) | 
					
						
							|  |  |  |                     if basename in ('ffmpeg', 'ffprobe'): | 
					
						
							|  |  |  |                         prefer_ffmpeg = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 self._paths = dict( | 
					
						
							|  |  |  |                     (p, os.path.join(location, p)) for p in programs) | 
					
						
							|  |  |  |                 self._versions = dict( | 
					
						
							| 
									
										
										
										
											2019-01-11 15:09:44 +01:00
										 |  |  |                     (p, get_ffmpeg_version(self._paths[p])) for p in programs) | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |         if self._versions is None: | 
					
						
							|  |  |  |             self._versions = dict( | 
					
						
							| 
									
										
										
										
											2019-01-11 15:09:44 +01:00
										 |  |  |                 (p, get_ffmpeg_version(p)) for p in programs) | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |             self._paths = dict((p, p) for p in programs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |         if prefer_ffmpeg is False: | 
					
						
							| 
									
										
										
										
											2014-10-26 16:31:52 +01:00
										 |  |  |             prefs = ('avconv', 'ffmpeg') | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             prefs = ('ffmpeg', 'avconv') | 
					
						
							| 
									
										
										
										
											2014-10-26 16:31:52 +01:00
										 |  |  |         for p in prefs: | 
					
						
							|  |  |  |             if self._versions[p]: | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |                 self.basename = p | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2014-01-08 17:53:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |         if prefer_ffmpeg is False: | 
					
						
							| 
									
										
										
										
											2014-10-26 21:03:16 +01:00
										 |  |  |             prefs = ('avprobe', 'ffprobe') | 
					
						
							| 
									
										
										
										
											2018-06-29 01:09:14 +07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             prefs = ('ffprobe', 'avprobe') | 
					
						
							| 
									
										
										
										
											2014-10-26 21:03:16 +01:00
										 |  |  |         for p in prefs: | 
					
						
							|  |  |  |             if self._versions[p]: | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |                 self.probe_basename = p | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-17 17:26:41 +01:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |     def available(self): | 
					
						
							|  |  |  |         return self.basename is not None | 
					
						
							| 
									
										
										
										
											2014-10-26 21:03:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def executable(self): | 
					
						
							|  |  |  |         return self._paths[self.basename] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-03 14:09:50 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def probe_available(self): | 
					
						
							|  |  |  |         return self.probe_basename is not None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def probe_executable(self): | 
					
						
							|  |  |  |         return self._paths[self.probe_basename] | 
					
						
							| 
									
										
										
										
											2014-01-08 17:53:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |     def get_audio_codec(self, path): | 
					
						
							| 
									
										
										
										
											2019-01-24 20:23:04 +01:00
										 |  |  |         if not self.probe_available and not self.available: | 
					
						
							|  |  |  |             raise PostProcessingError('ffprobe/avprobe and ffmpeg/avconv not found. Please install one.') | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2019-01-24 20:23:04 +01:00
										 |  |  |             if self.probe_available: | 
					
						
							|  |  |  |                 cmd = [ | 
					
						
							|  |  |  |                     encodeFilename(self.probe_executable, True), | 
					
						
							|  |  |  |                     encodeArgument('-show_streams')] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 cmd = [ | 
					
						
							|  |  |  |                     encodeFilename(self.executable, True), | 
					
						
							|  |  |  |                     encodeArgument('-i')] | 
					
						
							|  |  |  |             cmd.append(encodeFilename(self._ffmpeg_filename_argument(path), True)) | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |             if self._downloader.params.get('verbose', False): | 
					
						
							| 
									
										
										
										
											2019-01-24 20:23:04 +01:00
										 |  |  |                 self._downloader.to_screen( | 
					
						
							|  |  |  |                     '[debug] %s command line: %s' % (self.basename, shell_quote(cmd))) | 
					
						
							|  |  |  |             handle = subprocess.Popen( | 
					
						
							|  |  |  |                 cmd, stderr=subprocess.PIPE, | 
					
						
							|  |  |  |                 stdout=subprocess.PIPE, stdin=subprocess.PIPE) | 
					
						
							|  |  |  |             stdout_data, stderr_data = handle.communicate() | 
					
						
							|  |  |  |             expected_ret = 0 if self.probe_available else 1 | 
					
						
							|  |  |  |             if handle.wait() != expected_ret: | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |                 return None | 
					
						
							|  |  |  |         except (IOError, OSError): | 
					
						
							|  |  |  |             return None | 
					
						
							| 
									
										
										
										
											2019-01-24 20:23:04 +01:00
										 |  |  |         output = (stdout_data if self.probe_available else stderr_data).decode('ascii', 'ignore') | 
					
						
							|  |  |  |         if self.probe_available: | 
					
						
							|  |  |  |             audio_codec = None | 
					
						
							|  |  |  |             for line in output.split('\n'): | 
					
						
							|  |  |  |                 if line.startswith('codec_name='): | 
					
						
							|  |  |  |                     audio_codec = line.split('=')[1].strip() | 
					
						
							|  |  |  |                 elif line.strip() == 'codec_type=audio' and audio_codec is not None: | 
					
						
							|  |  |  |                     return audio_codec | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # Stream #FILE_INDEX:STREAM_INDEX[STREAM_ID](LANGUAGE): CODEC_TYPE: CODEC_NAME | 
					
						
							|  |  |  |             mobj = re.search( | 
					
						
							|  |  |  |                 r'Stream\s*#\d+:\d+(?:\[0x[0-9a-f]+\])?(?:\([a-z]{3}\))?:\s*Audio:\s*([0-9a-z]+)', | 
					
						
							|  |  |  |                 output) | 
					
						
							|  |  |  |             if mobj: | 
					
						
							|  |  |  |                 return mobj.group(1) | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |     def run_ffmpeg_multiple_files(self, input_paths, out_path, opts): | 
					
						
							| 
									
										
										
										
											2014-10-26 16:46:34 +01:00
										 |  |  |         self.check_version() | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-10 06:13:18 +01:00
										 |  |  |         oldest_mtime = min( | 
					
						
							|  |  |  |             os.stat(encodeFilename(path)).st_mtime for path in input_paths) | 
					
						
							| 
									
										
										
										
											2015-01-10 06:10:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 22:42:03 +06:00
										 |  |  |         opts += self._configuration_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         files_cmd = [] | 
					
						
							|  |  |  |         for path in input_paths: | 
					
						
							| 
									
										
										
										
											2015-09-17 15:22:19 +02:00
										 |  |  |             files_cmd.extend([ | 
					
						
							|  |  |  |                 encodeArgument('-i'), | 
					
						
							|  |  |  |                 encodeFilename(self._ffmpeg_filename_argument(path), True) | 
					
						
							|  |  |  |             ]) | 
					
						
							| 
									
										
										
										
											2015-02-13 11:14:01 +01:00
										 |  |  |         cmd = ([encodeFilename(self.executable, True), encodeArgument('-y')] + | 
					
						
							| 
									
										
										
										
											2019-01-29 01:59:56 +07:00
										 |  |  |                [encodeArgument('-loglevel'), encodeArgument('repeat+info')] + | 
					
						
							| 
									
										
										
										
											2015-01-10 06:10:18 +01:00
										 |  |  |                files_cmd + | 
					
						
							|  |  |  |                [encodeArgument(o) for o in opts] + | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |                [encodeFilename(self._ffmpeg_filename_argument(out_path), True)]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self._downloader.params.get('verbose', False): | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             self._downloader.to_screen('[debug] ffmpeg command line: %s' % shell_quote(cmd)) | 
					
						
							| 
									
										
										
										
											2015-02-13 22:25:34 +01:00
										 |  |  |         p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, stdin=subprocess.PIPE) | 
					
						
							| 
									
										
										
										
											2014-03-30 06:02:41 +02:00
										 |  |  |         stdout, stderr = p.communicate() | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         if p.returncode != 0: | 
					
						
							|  |  |  |             stderr = stderr.decode('utf-8', 'replace') | 
					
						
							|  |  |  |             msg = stderr.strip().split('\n')[-1] | 
					
						
							|  |  |  |             raise FFmpegPostProcessorError(msg) | 
					
						
							| 
									
										
										
										
											2015-04-08 21:40:31 +06:00
										 |  |  |         self.try_utime(out_path, oldest_mtime, oldest_mtime) | 
					
						
							| 
									
										
										
										
											2015-04-07 22:33:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |     def run_ffmpeg(self, path, out_path, opts): | 
					
						
							|  |  |  |         self.run_ffmpeg_multiple_files([path], out_path, opts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _ffmpeg_filename_argument(self, fn): | 
					
						
							| 
									
										
										
										
											2015-09-17 15:22:19 +02:00
										 |  |  |         # Always use 'file:' because the filename may contain ':' (ffmpeg | 
					
						
							|  |  |  |         # interprets that as a protocol) or can start with '-' (-- is broken in | 
					
						
							|  |  |  |         # ffmpeg, see https://ffmpeg.org/trac/ffmpeg/ticket/2127 for details) | 
					
						
							| 
									
										
										
										
											2016-04-16 21:49:13 +01:00
										 |  |  |         # Also leave '-' intact in order not to break streaming to stdout. | 
					
						
							| 
									
										
										
										
											2016-04-16 19:45:56 +01:00
										 |  |  |         return 'file:' + fn if fn != '-' else fn | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FFmpegExtractAudioPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def __init__(self, downloader=None, preferredcodec=None, preferredquality=None, nopostoverwrites=False): | 
					
						
							|  |  |  |         FFmpegPostProcessor.__init__(self, downloader) | 
					
						
							|  |  |  |         if preferredcodec is None: | 
					
						
							|  |  |  |             preferredcodec = 'best' | 
					
						
							|  |  |  |         self._preferredcodec = preferredcodec | 
					
						
							|  |  |  |         self._preferredquality = preferredquality | 
					
						
							|  |  |  |         self._nopostoverwrites = nopostoverwrites | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run_ffmpeg(self, path, out_path, codec, more_opts): | 
					
						
							|  |  |  |         if codec is None: | 
					
						
							|  |  |  |             acodec_opts = [] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             acodec_opts = ['-acodec', codec] | 
					
						
							|  |  |  |         opts = ['-vn'] + acodec_opts + more_opts | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             FFmpegPostProcessor.run_ffmpeg(self, path, out_path, opts) | 
					
						
							|  |  |  |         except FFmpegPostProcessorError as err: | 
					
						
							|  |  |  |             raise AudioConversionError(err.msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run(self, information): | 
					
						
							|  |  |  |         path = information['filepath'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         filecodec = self.get_audio_codec(path) | 
					
						
							|  |  |  |         if filecodec is None: | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             raise PostProcessingError('WARNING: unable to obtain file audio codec with ffprobe') | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         more_opts = [] | 
					
						
							|  |  |  |         if self._preferredcodec == 'best' or self._preferredcodec == filecodec or (self._preferredcodec == 'm4a' and filecodec == 'aac'): | 
					
						
							|  |  |  |             if filecodec == 'aac' and self._preferredcodec in ['m4a', 'best']: | 
					
						
							|  |  |  |                 # Lossless, but in another container | 
					
						
							|  |  |  |                 acodec = 'copy' | 
					
						
							|  |  |  |                 extension = 'm4a' | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                 more_opts = ['-bsf:a', 'aac_adtstoasc'] | 
					
						
							| 
									
										
										
										
											2017-03-16 12:50:45 +01:00
										 |  |  |             elif filecodec in ['aac', 'flac', 'mp3', 'vorbis', 'opus']: | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |                 # Lossless if possible | 
					
						
							|  |  |  |                 acodec = 'copy' | 
					
						
							|  |  |  |                 extension = filecodec | 
					
						
							|  |  |  |                 if filecodec == 'aac': | 
					
						
							|  |  |  |                     more_opts = ['-f', 'adts'] | 
					
						
							|  |  |  |                 if filecodec == 'vorbis': | 
					
						
							|  |  |  |                     extension = 'ogg' | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # MP3 otherwise. | 
					
						
							|  |  |  |                 acodec = 'libmp3lame' | 
					
						
							|  |  |  |                 extension = 'mp3' | 
					
						
							|  |  |  |                 more_opts = [] | 
					
						
							|  |  |  |                 if self._preferredquality is not None: | 
					
						
							|  |  |  |                     if int(self._preferredquality) < 10: | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                         more_opts += ['-q:a', self._preferredquality] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                         more_opts += ['-b:a', self._preferredquality + 'k'] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2017-03-16 12:50:45 +01:00
										 |  |  |             # We convert the audio (lossy if codec is lossy) | 
					
						
							|  |  |  |             acodec = ACODECS[self._preferredcodec] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |             extension = self._preferredcodec | 
					
						
							|  |  |  |             more_opts = [] | 
					
						
							|  |  |  |             if self._preferredquality is not None: | 
					
						
							|  |  |  |                 # The opus codec doesn't support the -aq option | 
					
						
							|  |  |  |                 if int(self._preferredquality) < 10 and extension != 'opus': | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                     more_opts += ['-q:a', self._preferredquality] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                     more_opts += ['-b:a', self._preferredquality + 'k'] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |             if self._preferredcodec == 'aac': | 
					
						
							|  |  |  |                 more_opts += ['-f', 'adts'] | 
					
						
							|  |  |  |             if self._preferredcodec == 'm4a': | 
					
						
							| 
									
										
										
										
											2015-02-06 22:05:11 +01:00
										 |  |  |                 more_opts += ['-bsf:a', 'aac_adtstoasc'] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |             if self._preferredcodec == 'vorbis': | 
					
						
							|  |  |  |                 extension = 'ogg' | 
					
						
							|  |  |  |             if self._preferredcodec == 'wav': | 
					
						
							|  |  |  |                 extension = 'wav' | 
					
						
							|  |  |  |                 more_opts += ['-f', 'wav'] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         prefix, sep, ext = path.rpartition('.')  # not os.path.splitext, since the latter does not work on unicode in all setups | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         new_path = prefix + sep + extension | 
					
						
							| 
									
										
										
										
											2016-10-16 19:16:47 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-15 22:12:03 +03:00
										 |  |  |         information['filepath'] = new_path | 
					
						
							|  |  |  |         information['ext'] = extension | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # If we download foo.mp3 and convert it to... foo.mp3, then don't delete foo.mp3, silly. | 
					
						
							| 
									
										
										
										
											2015-04-17 22:29:30 +02:00
										 |  |  |         if (new_path == path or | 
					
						
							|  |  |  |                 (self._nopostoverwrites and os.path.exists(encodeFilename(new_path)))): | 
					
						
							| 
									
										
										
										
											2015-07-14 12:56:32 +02:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] Post-process file %s exists, skipping' % new_path) | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], information | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2015-11-06 21:56:31 +06:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] Destination: ' + new_path) | 
					
						
							| 
									
										
										
										
											2015-04-17 22:29:30 +02:00
										 |  |  |             self.run_ffmpeg(path, new_path, acodec, more_opts) | 
					
						
							| 
									
										
										
										
											2015-03-27 13:02:20 +01:00
										 |  |  |         except AudioConversionError as e: | 
					
						
							|  |  |  |             raise PostProcessingError( | 
					
						
							|  |  |  |                 'audio conversion failed: ' + e.msg) | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             raise PostProcessingError('error running ' + self.basename) | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Try to update the date time for extracted audio file. | 
					
						
							|  |  |  |         if information.get('filetime') is not None: | 
					
						
							| 
									
										
										
										
											2015-04-08 21:40:31 +06:00
										 |  |  |             self.try_utime( | 
					
						
							|  |  |  |                 new_path, time.time(), information['filetime'], | 
					
						
							|  |  |  |                 errnote='Cannot update utime of audio file') | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |         return [path], information | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-15 01:06:25 +01:00
										 |  |  | class FFmpegVideoConvertorPP(FFmpegPostProcessor): | 
					
						
							| 
									
										
										
										
											2014-11-23 20:41:03 +01:00
										 |  |  |     def __init__(self, downloader=None, preferedformat=None): | 
					
						
							| 
									
										
										
										
											2014-12-15 01:06:25 +01:00
										 |  |  |         super(FFmpegVideoConvertorPP, self).__init__(downloader) | 
					
						
							| 
									
										
										
										
											2014-11-23 20:41:03 +01:00
										 |  |  |         self._preferedformat = preferedformat | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def run(self, information): | 
					
						
							|  |  |  |         path = information['filepath'] | 
					
						
							|  |  |  |         if information['ext'] == self._preferedformat: | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] Not converting video file %s - already is in target format %s' % (path, self._preferedformat)) | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], information | 
					
						
							| 
									
										
										
										
											2015-07-11 22:42:03 +06:00
										 |  |  |         options = [] | 
					
						
							|  |  |  |         if self._preferedformat == 'avi': | 
					
						
							|  |  |  |             options.extend(['-c:v', 'libxvid', '-vtag', 'XVID']) | 
					
						
							|  |  |  |         prefix, sep, ext = path.rpartition('.') | 
					
						
							|  |  |  |         outpath = prefix + sep + self._preferedformat | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         self._downloader.to_screen('[' + 'ffmpeg' + '] Converting video from %s to %s, Destination: ' % (information['ext'], self._preferedformat) + outpath) | 
					
						
							| 
									
										
										
										
											2015-06-09 22:08:16 -03:00
										 |  |  |         self.run_ffmpeg(path, outpath, options) | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         information['filepath'] = outpath | 
					
						
							|  |  |  |         information['format'] = self._preferedformat | 
					
						
							| 
									
										
										
										
											2015-07-11 22:15:16 +06:00
										 |  |  |         information['ext'] = self._preferedformat | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |         return [path], information | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FFmpegEmbedSubtitlePP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, information): | 
					
						
							| 
									
										
										
										
											2016-03-20 04:12:34 +06:00
										 |  |  |         if information['ext'] not in ('mp4', 'webm', 'mkv'): | 
					
						
							|  |  |  |             self._downloader.to_screen('[ffmpeg] Subtitles can only be embedded in mp4, webm or mkv files') | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], information | 
					
						
							| 
									
										
										
										
											2015-02-16 21:12:31 +01:00
										 |  |  |         subtitles = information.get('requested_subtitles') | 
					
						
							|  |  |  |         if not subtitles: | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] There aren\'t any subtitles to embed') | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], information | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         filename = information['filepath'] | 
					
						
							| 
									
										
										
										
											2016-03-20 04:12:34 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ext = information['ext'] | 
					
						
							|  |  |  |         sub_langs = [] | 
					
						
							|  |  |  |         sub_filenames = [] | 
					
						
							|  |  |  |         webm_vtt_warn = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for lang, sub_info in subtitles.items(): | 
					
						
							|  |  |  |             sub_ext = sub_info['ext'] | 
					
						
							|  |  |  |             if ext != 'webm' or ext == 'webm' and sub_ext == 'vtt': | 
					
						
							|  |  |  |                 sub_langs.append(lang) | 
					
						
							|  |  |  |                 sub_filenames.append(subtitles_filename(filename, lang, sub_ext)) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 if not webm_vtt_warn and ext == 'webm' and sub_ext != 'vtt': | 
					
						
							|  |  |  |                     webm_vtt_warn = True | 
					
						
							|  |  |  |                     self._downloader.to_screen('[ffmpeg] Only WebVTT subtitles can be embedded in webm files') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not sub_langs: | 
					
						
							|  |  |  |             return [], information | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 11:44:42 +02:00
										 |  |  |         input_files = [filename] + sub_filenames | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-16 13:29:01 +01:00
										 |  |  |         opts = [ | 
					
						
							| 
									
										
										
											
												Revert "[ffmpeg] Fix embedding subtitles (#9063)"
This reverts commit ccff2c404d7ea9f5b21ede8ae57bb79feec7eb94.
Fixes #10081.
The new approach breaks embedding subtitles into video-only or
audio-only files. FFMpeg provides a trick: add '?' after the argument of
'-map' so that a missing stream is ignored. For example:
opts = [
    '-map', '0:v?',
    '-c:v', 'copy',
    '-map', '0:a?',
    '-c:a', 'copy',
    # other options...
]
Unfortunately, such a format is not implemented in avconv, either.
I guess adding '-ignore_unknown' if self.basename == 'ffmpeg' is the
best solution. However, the example mentioned in #9063 no longer serves
problematic files, so I can't test it. I'll reopen #9063 and wait for
another example so that I can test '-ignore_unknown'.
											
										 
											2016-07-15 19:53:10 +08:00
										 |  |  |             '-map', '0', | 
					
						
							|  |  |  |             '-c', 'copy', | 
					
						
							| 
									
										
										
										
											2015-01-16 13:29:01 +01:00
										 |  |  |             # Don't copy the existing subtitles, we may be running the | 
					
						
							|  |  |  |             # postprocessor a second time | 
					
						
							|  |  |  |             '-map', '-0:s', | 
					
						
							| 
									
										
										
										
											2019-01-28 10:57:14 -05:00
										 |  |  |             # Don't copy Apple TV chapters track, bin_data (see #19042, #19024, | 
					
						
							|  |  |  |             # https://trac.ffmpeg.org/ticket/6016) | 
					
						
							|  |  |  |             '-map', '-0:d', | 
					
						
							| 
									
										
										
										
											2015-01-16 13:29:01 +01:00
										 |  |  |         ] | 
					
						
							| 
									
										
										
										
											2015-04-15 20:27:40 +03:00
										 |  |  |         if information['ext'] == 'mp4': | 
					
						
							|  |  |  |             opts += ['-c:s', 'mov_text'] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         for (i, lang) in enumerate(sub_langs): | 
					
						
							| 
									
										
										
										
											2015-01-16 13:37:37 +01:00
										 |  |  |             opts.extend(['-map', '%d:0' % (i + 1)]) | 
					
						
							| 
									
										
										
										
											2019-01-07 00:57:24 +07:00
										 |  |  |             lang_code = ISO639Utils.short2long(lang) or lang | 
					
						
							|  |  |  |             opts.extend(['-metadata:s:s:%d' % i, 'language=%s' % lang_code]) | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-16 13:37:37 +01:00
										 |  |  |         temp_filename = prepend_extension(filename, 'temp') | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         self._downloader.to_screen('[ffmpeg] Embedding subtitles in \'%s\'' % filename) | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         self.run_ffmpeg_multiple_files(input_files, temp_filename, opts) | 
					
						
							|  |  |  |         os.remove(encodeFilename(filename)) | 
					
						
							|  |  |  |         os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 11:44:42 +02:00
										 |  |  |         return sub_filenames, information | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FFmpegMetadataPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         metadata = {} | 
					
						
							| 
									
										
										
										
											2016-05-01 10:56:54 +06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def add(meta_list, info_list=None): | 
					
						
							|  |  |  |             if not info_list: | 
					
						
							|  |  |  |                 info_list = meta_list | 
					
						
							|  |  |  |             if not isinstance(meta_list, (list, tuple)): | 
					
						
							|  |  |  |                 meta_list = (meta_list,) | 
					
						
							|  |  |  |             if not isinstance(info_list, (list, tuple)): | 
					
						
							|  |  |  |                 info_list = (info_list,) | 
					
						
							|  |  |  |             for info_f in info_list: | 
					
						
							|  |  |  |                 if info.get(info_f) is not None: | 
					
						
							|  |  |  |                     for meta_f in meta_list: | 
					
						
							|  |  |  |                         metadata[meta_f] = info[info_f] | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         add('title', ('track', 'title')) | 
					
						
							|  |  |  |         add('date', 'upload_date') | 
					
						
							|  |  |  |         add(('description', 'comment'), 'description') | 
					
						
							|  |  |  |         add('purl', 'webpage_url') | 
					
						
							|  |  |  |         add('track', 'track_number') | 
					
						
							|  |  |  |         add('artist', ('artist', 'creator', 'uploader', 'uploader_id')) | 
					
						
							|  |  |  |         add('genre') | 
					
						
							|  |  |  |         add('album') | 
					
						
							|  |  |  |         add('album_artist') | 
					
						
							|  |  |  |         add('disc', 'disc_number') | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if not metadata: | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] There isn\'t any metadata to add') | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], info | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							|  |  |  |         temp_filename = prepend_extension(filename, 'temp') | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |         in_filenames = [filename] | 
					
						
							|  |  |  |         options = [] | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         if info['ext'] == 'm4a': | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |             options.extend(['-vn', '-acodec', 'copy']) | 
					
						
							| 
									
										
										
										
											2014-02-22 18:23:30 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |             options.extend(['-c', 'copy']) | 
					
						
							| 
									
										
										
										
											2014-02-22 18:23:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         for (name, value) in metadata.items(): | 
					
						
							|  |  |  |             options.extend(['-metadata', '%s=%s' % (name, value)]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |         chapters = info.get('chapters', []) | 
					
						
							|  |  |  |         if chapters: | 
					
						
							| 
									
										
										
										
											2017-05-25 22:04:12 +08:00
										 |  |  |             metadata_filename = replace_extension(filename, 'meta') | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |             with io.open(metadata_filename, 'wt', encoding='utf-8') as f: | 
					
						
							|  |  |  |                 def ffmpeg_escape(text): | 
					
						
							|  |  |  |                     return re.sub(r'(=|;|#|\\|\n)', r'\\\1', text) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 metadata_file_content = ';FFMETADATA1\n' | 
					
						
							|  |  |  |                 for chapter in chapters: | 
					
						
							|  |  |  |                     metadata_file_content += '[CHAPTER]\nTIMEBASE=1/1000\n' | 
					
						
							|  |  |  |                     metadata_file_content += 'START=%d\n' % (chapter['start_time'] * 1000) | 
					
						
							|  |  |  |                     metadata_file_content += 'END=%d\n' % (chapter['end_time'] * 1000) | 
					
						
							|  |  |  |                     chapter_title = chapter.get('title') | 
					
						
							|  |  |  |                     if chapter_title: | 
					
						
							|  |  |  |                         metadata_file_content += 'title=%s\n' % ffmpeg_escape(chapter_title) | 
					
						
							|  |  |  |                 f.write(metadata_file_content) | 
					
						
							|  |  |  |                 in_filenames.append(metadata_filename) | 
					
						
							|  |  |  |                 options.extend(['-map_metadata', '1']) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         self._downloader.to_screen('[ffmpeg] Adding metadata to \'%s\'' % filename) | 
					
						
							| 
									
										
										
										
											2016-05-05 21:41:30 +01:00
										 |  |  |         self.run_ffmpeg_multiple_files(in_filenames, temp_filename, options) | 
					
						
							|  |  |  |         if chapters: | 
					
						
							|  |  |  |             os.remove(metadata_filename) | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  |         os.remove(encodeFilename(filename)) | 
					
						
							|  |  |  |         os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |         return [], info | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FFmpegMergerPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							| 
									
										
										
										
											2015-04-19 16:56:22 +02:00
										 |  |  |         temp_filename = prepend_extension(filename, 'temp') | 
					
						
							| 
									
										
										
										
											2015-01-04 14:02:17 +01:00
										 |  |  |         args = ['-c', 'copy', '-map', '0:v:0', '-map', '1:a:0'] | 
					
						
							| 
									
										
										
										
											2014-11-26 13:05:11 +01:00
										 |  |  |         self._downloader.to_screen('[ffmpeg] Merging formats into "%s"' % filename) | 
					
						
							| 
									
										
										
										
											2015-04-19 16:56:22 +02:00
										 |  |  |         self.run_ffmpeg_multiple_files(info['__files_to_merge'], temp_filename, args) | 
					
						
							|  |  |  |         os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							| 
									
										
										
										
											2015-04-18 11:52:36 +02:00
										 |  |  |         return info['__files_to_merge'], info | 
					
						
							| 
									
										
										
										
											2014-01-07 05:59:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 02:00:31 +06:00
										 |  |  |     def can_merge(self): | 
					
						
							|  |  |  |         # TODO: figure out merge-capable ffmpeg version | 
					
						
							|  |  |  |         if self.basename != 'avconv': | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         required_version = '10-0' | 
					
						
							|  |  |  |         if is_outdated_version( | 
					
						
							|  |  |  |                 self._versions[self.basename], required_version): | 
					
						
							|  |  |  |             warning = ('Your copy of %s is outdated and unable to properly mux separate video and audio files, ' | 
					
						
							|  |  |  |                        'youtube-dl will download single file media. ' | 
					
						
							|  |  |  |                        'Update %s to version %s or newer to fix this.') % ( | 
					
						
							|  |  |  |                            self.basename, self.basename, required_version) | 
					
						
							|  |  |  |             if self._downloader: | 
					
						
							|  |  |  |                 self._downloader.report_warning(warning) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-22 13:55:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-10 05:45:51 +01:00
										 |  |  | class FFmpegFixupStretchedPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         stretched_ratio = info.get('stretched_ratio') | 
					
						
							|  |  |  |         if stretched_ratio is None or stretched_ratio == 1: | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], info | 
					
						
							| 
									
										
										
										
											2015-01-10 05:45:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							|  |  |  |         temp_filename = prepend_extension(filename, 'temp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         options = ['-c', 'copy', '-aspect', '%f' % stretched_ratio] | 
					
						
							|  |  |  |         self._downloader.to_screen('[ffmpeg] Fixing aspect ratio in "%s"' % filename) | 
					
						
							|  |  |  |         self.run_ffmpeg(filename, temp_filename, options) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         os.remove(encodeFilename(filename)) | 
					
						
							|  |  |  |         os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |         return [], info | 
					
						
							| 
									
										
										
										
											2015-01-23 18:39:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FFmpegFixupM4aPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         if info.get('container') != 'm4a_dash': | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], info | 
					
						
							| 
									
										
										
										
											2015-01-23 18:39:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							|  |  |  |         temp_filename = prepend_extension(filename, 'temp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         options = ['-c', 'copy', '-f', 'mp4'] | 
					
						
							|  |  |  |         self._downloader.to_screen('[ffmpeg] Correcting container in "%s"' % filename) | 
					
						
							|  |  |  |         self.run_ffmpeg(filename, temp_filename, options) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         os.remove(encodeFilename(filename)) | 
					
						
							|  |  |  |         os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |         return [], info | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-01 21:08:50 +01:00
										 |  |  | class FFmpegFixupM3u8PP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |         if self.get_audio_codec(filename) == 'aac': | 
					
						
							|  |  |  |             temp_filename = prepend_extension(filename, 'temp') | 
					
						
							| 
									
										
										
										
											2016-03-01 21:08:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |             options = ['-c', 'copy', '-f', 'mp4', '-bsf:a', 'aac_adtstoasc'] | 
					
						
							| 
									
										
										
										
											2017-07-09 17:09:44 +07:00
										 |  |  |             self._downloader.to_screen('[ffmpeg] Fixing malformed AAC bitstream in "%s"' % filename) | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |             self.run_ffmpeg(filename, temp_filename, options) | 
					
						
							| 
									
										
										
										
											2016-03-01 21:08:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-16 22:06:55 +01:00
										 |  |  |             os.remove(encodeFilename(filename)) | 
					
						
							|  |  |  |             os.rename(encodeFilename(temp_filename), encodeFilename(filename)) | 
					
						
							| 
									
										
										
										
											2016-03-01 21:08:50 +01:00
										 |  |  |         return [], info | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  | class FFmpegSubtitlesConvertorPP(FFmpegPostProcessor): | 
					
						
							|  |  |  |     def __init__(self, downloader=None, format=None): | 
					
						
							|  |  |  |         super(FFmpegSubtitlesConvertorPP, self).__init__(downloader) | 
					
						
							|  |  |  |         self.format = format | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run(self, info): | 
					
						
							|  |  |  |         subs = info.get('requested_subtitles') | 
					
						
							|  |  |  |         filename = info['filepath'] | 
					
						
							|  |  |  |         new_ext = self.format | 
					
						
							|  |  |  |         new_format = new_ext | 
					
						
							|  |  |  |         if new_format == 'vtt': | 
					
						
							|  |  |  |             new_format = 'webvtt' | 
					
						
							|  |  |  |         if subs is None: | 
					
						
							|  |  |  |             self._downloader.to_screen('[ffmpeg] There aren\'t any subtitles to convert') | 
					
						
							| 
									
										
										
										
											2015-04-18 11:36:42 +02:00
										 |  |  |             return [], info | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  |         self._downloader.to_screen('[ffmpeg] Converting subtitles') | 
					
						
							| 
									
										
										
										
											2016-01-31 14:22:36 +01:00
										 |  |  |         sub_filenames = [] | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  |         for lang, sub in subs.items(): | 
					
						
							|  |  |  |             ext = sub['ext'] | 
					
						
							|  |  |  |             if ext == new_ext: | 
					
						
							|  |  |  |                 self._downloader.to_screen( | 
					
						
							| 
									
										
										
										
											2017-02-24 04:56:58 +07:00
										 |  |  |                     '[ffmpeg] Subtitle file for %s is already in the requested format' % new_ext) | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2016-01-31 14:22:36 +01:00
										 |  |  |             old_file = subtitles_filename(filename, lang, ext) | 
					
						
							|  |  |  |             sub_filenames.append(old_file) | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  |             new_file = subtitles_filename(filename, lang, new_ext) | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 20:38:43 +01:00
										 |  |  |             if ext in ('dfxp', 'ttml', 'tt'): | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  |                 self._downloader.report_warning( | 
					
						
							|  |  |  |                     'You have requested to convert dfxp (TTML) subtitles into another format, ' | 
					
						
							|  |  |  |                     'which results in style information loss') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-31 14:22:36 +01:00
										 |  |  |                 dfxp_file = old_file | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  |                 srt_file = subtitles_filename(filename, lang, 'srt') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 12:18:38 +08:00
										 |  |  |                 with open(dfxp_file, 'rb') as f: | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  |                     srt_data = dfxp2srt(f.read()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 with io.open(srt_file, 'wt', encoding='utf-8') as f: | 
					
						
							|  |  |  |                     f.write(srt_data) | 
					
						
							| 
									
										
										
										
											2016-02-06 18:51:05 +01:00
										 |  |  |                 old_file = srt_file | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 subs[lang] = { | 
					
						
							|  |  |  |                     'ext': 'srt', | 
					
						
							|  |  |  |                     'data': srt_data | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if new_ext == 'srt': | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2016-02-06 19:04:18 +01:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     sub_filenames.append(srt_file) | 
					
						
							| 
									
										
										
										
											2015-04-25 23:15:05 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-31 14:22:36 +01:00
										 |  |  |             self.run_ffmpeg(old_file, new_file, ['-f', new_format]) | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             with io.open(new_file, 'rt', encoding='utf-8') as f: | 
					
						
							|  |  |  |                 subs[lang] = { | 
					
						
							| 
									
										
										
										
											2016-02-06 18:58:18 +01:00
										 |  |  |                     'ext': new_ext, | 
					
						
							| 
									
										
										
										
											2015-02-28 14:43:24 +01:00
										 |  |  |                     'data': f.read(), | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-31 14:22:36 +01:00
										 |  |  |         return sub_filenames, info |