2014-11-26 20:01:20 +01:00
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2021-06-13 01:32:19 +05:30
|
|
|
import functools
|
2015-04-08 21:40:31 +06:00
|
|
|
import os
|
|
|
|
|
2021-01-20 21:37:40 +05:30
|
|
|
from ..compat import compat_str
|
2015-04-08 21:40:31 +06:00
|
|
|
from ..utils import (
|
2021-08-24 03:15:44 +05:30
|
|
|
_configuration_args,
|
2015-04-08 21:40:31 +06:00
|
|
|
encodeFilename,
|
2021-01-23 15:13:51 +05:30
|
|
|
PostProcessingError,
|
2015-04-08 21:40:31 +06:00
|
|
|
)
|
2014-01-07 05:59:22 +01:00
|
|
|
|
|
|
|
|
|
|
|
class PostProcessor(object):
|
|
|
|
"""Post Processor class.
|
|
|
|
|
|
|
|
PostProcessor objects can be added to downloaders with their
|
|
|
|
add_post_processor() method. When the downloader has finished a
|
|
|
|
successful download, it will take its internal chain of PostProcessors
|
|
|
|
and start calling the run() method on each one of them, first with
|
|
|
|
an initial argument and then with the returned value of the previous
|
|
|
|
PostProcessor.
|
|
|
|
|
|
|
|
The chain will be stopped if one of them ever returns None or the end
|
|
|
|
of the chain is reached.
|
|
|
|
|
|
|
|
PostProcessor objects follow a "mutual registration" process similar
|
2015-07-11 22:41:33 +06:00
|
|
|
to InfoExtractor objects.
|
|
|
|
|
|
|
|
Optionally PostProcessor can use a list of additional command-line arguments
|
|
|
|
with self._configuration_args.
|
2014-01-07 05:59:22 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
_downloader = None
|
|
|
|
|
2015-07-01 20:12:26 -03:00
|
|
|
def __init__(self, downloader=None):
|
2014-01-07 05:59:22 +01:00
|
|
|
self._downloader = downloader
|
2021-01-20 21:37:40 +05:30
|
|
|
self.PP_NAME = self.pp_key()
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def pp_key(cls):
|
|
|
|
name = cls.__name__[:-2]
|
|
|
|
return compat_str(name[6:]) if name[:6].lower() == 'ffmpeg' else name
|
2021-01-08 00:58:41 +05:30
|
|
|
|
2021-01-21 01:37:02 +05:30
|
|
|
def to_screen(self, text, prefix=True, *args, **kwargs):
|
|
|
|
tag = '[%s] ' % self.PP_NAME if prefix else ''
|
2021-01-10 19:14:54 +05:30
|
|
|
if self._downloader:
|
2021-01-21 01:37:02 +05:30
|
|
|
return self._downloader.to_screen('%s%s' % (tag, text), *args, **kwargs)
|
2021-01-10 19:14:54 +05:30
|
|
|
|
|
|
|
def report_warning(self, text, *args, **kwargs):
|
|
|
|
if self._downloader:
|
|
|
|
return self._downloader.report_warning(text, *args, **kwargs)
|
|
|
|
|
|
|
|
def report_error(self, text, *args, **kwargs):
|
|
|
|
if self._downloader:
|
|
|
|
return self._downloader.report_error(text, *args, **kwargs)
|
|
|
|
|
2021-05-14 13:15:29 +05:30
|
|
|
def write_debug(self, text, *args, **kwargs):
|
|
|
|
if self._downloader:
|
|
|
|
return self._downloader.write_debug(text, *args, **kwargs)
|
2021-01-10 19:14:54 +05:30
|
|
|
|
|
|
|
def get_param(self, name, default=None, *args, **kwargs):
|
|
|
|
if self._downloader:
|
|
|
|
return self._downloader.params.get(name, default, *args, **kwargs)
|
|
|
|
return default
|
2014-01-07 05:59:22 +01:00
|
|
|
|
|
|
|
def set_downloader(self, downloader):
|
|
|
|
"""Sets the downloader for this PP."""
|
|
|
|
self._downloader = downloader
|
|
|
|
|
2021-06-13 01:32:19 +05:30
|
|
|
@staticmethod
|
|
|
|
def _restrict_to(*, video=True, audio=True, images=True):
|
|
|
|
allowed = {'video': video, 'audio': audio, 'images': images}
|
|
|
|
|
|
|
|
def decorator(func):
|
|
|
|
@functools.wraps(func)
|
|
|
|
def wrapper(self, info):
|
|
|
|
format_type = (
|
2021-06-14 02:05:57 +05:30
|
|
|
'video' if info.get('vcodec') != 'none'
|
|
|
|
else 'audio' if info.get('acodec') != 'none'
|
2021-06-13 01:32:19 +05:30
|
|
|
else 'images')
|
|
|
|
if allowed[format_type]:
|
2021-06-13 14:36:13 +05:30
|
|
|
return func(self, info)
|
2021-06-13 01:32:19 +05:30
|
|
|
else:
|
|
|
|
self.to_screen('Skipping %s' % format_type)
|
|
|
|
return [], info
|
|
|
|
return wrapper
|
|
|
|
return decorator
|
|
|
|
|
2014-01-07 05:59:22 +01:00
|
|
|
def run(self, information):
|
|
|
|
"""Run the PostProcessor.
|
|
|
|
|
|
|
|
The "information" argument is a dictionary like the ones
|
|
|
|
composed by InfoExtractors. The only difference is that this
|
|
|
|
one has an extra field called "filepath" that points to the
|
|
|
|
downloaded file.
|
|
|
|
|
2015-04-18 11:36:42 +02:00
|
|
|
This method returns a tuple, the first element is a list of the files
|
|
|
|
that can be deleted, and the second of which is the updated
|
|
|
|
information.
|
2014-01-07 05:59:22 +01:00
|
|
|
|
|
|
|
In addition, this method may raise a PostProcessingError
|
|
|
|
exception if post processing fails.
|
|
|
|
"""
|
2015-04-18 11:36:42 +02:00
|
|
|
return [], information # by default, keep file and do nothing
|
2014-01-07 05:59:22 +01:00
|
|
|
|
2015-04-08 21:40:31 +06:00
|
|
|
def try_utime(self, path, atime, mtime, errnote='Cannot update utime of file'):
|
|
|
|
try:
|
|
|
|
os.utime(encodeFilename(path), (atime, mtime))
|
|
|
|
except Exception:
|
2021-01-10 19:14:54 +05:30
|
|
|
self.report_warning(errnote)
|
2015-04-08 21:40:31 +06:00
|
|
|
|
2021-08-24 03:15:44 +05:30
|
|
|
def _configuration_args(self, exe, *args, **kwargs):
|
|
|
|
return _configuration_args(
|
|
|
|
self.pp_key(), self.get_param('postprocessor_args'), exe, *args, **kwargs)
|
2015-07-11 22:41:33 +06:00
|
|
|
|
2014-01-07 05:59:22 +01:00
|
|
|
|
|
|
|
class AudioConversionError(PostProcessingError):
|
|
|
|
pass
|