|
|
@ -28,6 +28,7 @@ from ..utils import (
|
|
|
|
shell_quote,
|
|
|
|
shell_quote,
|
|
|
|
traverse_obj,
|
|
|
|
traverse_obj,
|
|
|
|
variadic,
|
|
|
|
variadic,
|
|
|
|
|
|
|
|
write_json_file,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -636,10 +637,11 @@ class FFmpegEmbedSubtitlePP(FFmpegPostProcessor):
|
|
|
|
|
|
|
|
|
|
|
|
class FFmpegMetadataPP(FFmpegPostProcessor):
|
|
|
|
class FFmpegMetadataPP(FFmpegPostProcessor):
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, downloader, add_metadata=True, add_chapters=True):
|
|
|
|
def __init__(self, downloader, add_metadata=True, add_chapters=True, add_infojson='if_exists'):
|
|
|
|
FFmpegPostProcessor.__init__(self, downloader)
|
|
|
|
FFmpegPostProcessor.__init__(self, downloader)
|
|
|
|
self._add_metadata = add_metadata
|
|
|
|
self._add_metadata = add_metadata
|
|
|
|
self._add_chapters = add_chapters
|
|
|
|
self._add_chapters = add_chapters
|
|
|
|
|
|
|
|
self._add_infojson = add_infojson
|
|
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@staticmethod
|
|
|
|
def _options(target_ext):
|
|
|
|
def _options(target_ext):
|
|
|
@ -652,13 +654,23 @@ class FFmpegMetadataPP(FFmpegPostProcessor):
|
|
|
|
@PostProcessor._restrict_to(images=False)
|
|
|
|
@PostProcessor._restrict_to(images=False)
|
|
|
|
def run(self, info):
|
|
|
|
def run(self, info):
|
|
|
|
filename, metadata_filename = info['filepath'], None
|
|
|
|
filename, metadata_filename = info['filepath'], None
|
|
|
|
options = []
|
|
|
|
files_to_delete, options = [], []
|
|
|
|
if self._add_chapters and info.get('chapters'):
|
|
|
|
if self._add_chapters and info.get('chapters'):
|
|
|
|
metadata_filename = replace_extension(filename, 'meta')
|
|
|
|
metadata_filename = replace_extension(filename, 'meta')
|
|
|
|
options.extend(self._get_chapter_opts(info['chapters'], metadata_filename))
|
|
|
|
options.extend(self._get_chapter_opts(info['chapters'], metadata_filename))
|
|
|
|
|
|
|
|
files_to_delete.append(metadata_filename)
|
|
|
|
if self._add_metadata:
|
|
|
|
if self._add_metadata:
|
|
|
|
options.extend(self._get_metadata_opts(info))
|
|
|
|
options.extend(self._get_metadata_opts(info))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if self._add_infojson:
|
|
|
|
|
|
|
|
if info['ext'] in ('mkv', 'mka'):
|
|
|
|
|
|
|
|
infojson_filename = info.get('infojson_filename')
|
|
|
|
|
|
|
|
options.extend(self._get_infojson_opts(info, infojson_filename))
|
|
|
|
|
|
|
|
if not infojson_filename:
|
|
|
|
|
|
|
|
files_to_delete.append(info.get('infojson_filename'))
|
|
|
|
|
|
|
|
elif self._add_infojson is True:
|
|
|
|
|
|
|
|
self.to_screen('The info-json can only be attached to mkv/mka files')
|
|
|
|
|
|
|
|
|
|
|
|
if not options:
|
|
|
|
if not options:
|
|
|
|
self.to_screen('There isn\'t any metadata to add')
|
|
|
|
self.to_screen('There isn\'t any metadata to add')
|
|
|
|
return [], info
|
|
|
|
return [], info
|
|
|
@ -668,8 +680,8 @@ class FFmpegMetadataPP(FFmpegPostProcessor):
|
|
|
|
self.run_ffmpeg_multiple_files(
|
|
|
|
self.run_ffmpeg_multiple_files(
|
|
|
|
(filename, metadata_filename), temp_filename,
|
|
|
|
(filename, metadata_filename), temp_filename,
|
|
|
|
itertools.chain(self._options(info['ext']), *options))
|
|
|
|
itertools.chain(self._options(info['ext']), *options))
|
|
|
|
if metadata_filename:
|
|
|
|
for file in filter(None, files_to_delete):
|
|
|
|
os.remove(metadata_filename)
|
|
|
|
os.remove(file) # Don't obey --keep-files
|
|
|
|
os.replace(temp_filename, filename)
|
|
|
|
os.replace(temp_filename, filename)
|
|
|
|
return [], info
|
|
|
|
return [], info
|
|
|
|
|
|
|
|
|
|
|
@ -741,15 +753,26 @@ class FFmpegMetadataPP(FFmpegPostProcessor):
|
|
|
|
yield ('-metadata:s:%d' % (stream_idx + i), 'language=%s' % lang)
|
|
|
|
yield ('-metadata:s:%d' % (stream_idx + i), 'language=%s' % lang)
|
|
|
|
stream_idx += stream_count
|
|
|
|
stream_idx += stream_count
|
|
|
|
|
|
|
|
|
|
|
|
if ('no-attach-info-json' not in self.get_param('compat_opts', [])
|
|
|
|
def _get_infojson_opts(self, info, infofn):
|
|
|
|
and '__infojson_filename' in info and info['ext'] in ('mkv', 'mka')):
|
|
|
|
if not infofn or not os.path.exists(infofn):
|
|
|
|
old_stream, new_stream = self.get_stream_number(info['filepath'], ('tags', 'mimetype'), 'application/json')
|
|
|
|
if self._add_infojson is not True:
|
|
|
|
if old_stream is not None:
|
|
|
|
return
|
|
|
|
yield ('-map', '-0:%d' % old_stream)
|
|
|
|
infofn = infofn or '%s.temp' % (
|
|
|
|
new_stream -= 1
|
|
|
|
self._downloader.prepare_filename(info, 'infojson')
|
|
|
|
|
|
|
|
or replace_extension(self._downloader.prepare_filename(info), 'info.json', info['ext']))
|
|
|
|
|
|
|
|
if not self._downloader._ensure_dir_exists(infofn):
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
self.write_debug(f'Writing info-json to: {infofn}')
|
|
|
|
|
|
|
|
write_json_file(self._downloader.sanitize_info(info, self.get_param('clean_infojson', True)), infofn)
|
|
|
|
|
|
|
|
info['infojson_filename'] = infofn
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
old_stream, new_stream = self.get_stream_number(info['filepath'], ('tags', 'mimetype'), 'application/json')
|
|
|
|
|
|
|
|
if old_stream is not None:
|
|
|
|
|
|
|
|
yield ('-map', '-0:%d' % old_stream)
|
|
|
|
|
|
|
|
new_stream -= 1
|
|
|
|
|
|
|
|
|
|
|
|
yield ('-attach', info['__infojson_filename'],
|
|
|
|
yield ('-attach', infofn,
|
|
|
|
'-metadata:s:%d' % new_stream, 'mimetype=application/json')
|
|
|
|
'-metadata:s:%d' % new_stream, 'mimetype=application/json')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FFmpegMergerPP(FFmpegPostProcessor):
|
|
|
|
class FFmpegMergerPP(FFmpegPostProcessor):
|
|
|
|