Merge remote-tracking branch 'epitron/metadata-pp'
authorPhilipp Hagemeister <phihag@phihag.de>
Tue, 7 Jan 2014 04:44:44 +0000 (05:44 +0100)
committerPhilipp Hagemeister <phihag@phihag.de>
Tue, 7 Jan 2014 04:44:44 +0000 (05:44 +0100)
Conflicts:
youtube_dl/PostProcessor.py

1  2 
README.md
youtube_dl/PostProcessor.py
youtube_dl/__init__.py
youtube_dl/utils.py

diff --cc README.md
Simple merge
index f6be275ff04183d105282ead290e0a46ef3d2a1d,da95f1a87dbe63fc648e061aa5671bb3dba1d740..481c07a9448276e599aeb7becfa4e2bbe4b868e8
@@@ -513,9 -514,118 +516,125 @@@ class FFmpegMetadataPP(FFmpegPostProces
          return True, info
  
  
 +class FFmpegMergerPP(FFmpegPostProcessor):
 +    def run(self, info):
 +        filename = info['filepath']
 +        args = ['-c', 'copy']
 +        self.run_ffmpeg_multiple_files(info['__files_to_merge'], filename, args)
 +        return True, info
++
+ class XAttrMetadataPP(PostProcessor):
+     #
+     # More info about extended attributes for media:
+     #   http://freedesktop.org/wiki/CommonExtendedAttributes/
+     #   http://www.freedesktop.org/wiki/PhreedomDraft/
+     #   http://dublincore.org/documents/usageguide/elements.shtml
+     #
+     # TODO:
+     #  * capture youtube keywords and put them in 'user.dublincore.subject' (comma-separated)
+     #  * figure out which xattrs can be used for 'duration', 'thumbnail', 'resolution'
+     #
+     def run(self, info):
+         """ Set extended attributes on downloaded file (if xattr support is found). """
+         from .utils import hyphenate_date
+         # This mess below finds the best xattr tool for the job and creates a
+         # "write_xattr" function.
+         try:
+             # try the pyxattr module...
+             import xattr
+             def write_xattr(path, key, value):
+                 return xattr.setxattr(path, key, value)
+         except ImportError:
+             if os.name == 'posix':
+                 def which(bin):
+                     for dir in os.environ["PATH"].split(":"):
+                         path = os.path.join(dir, bin)
+                         if os.path.exists(path):
+                             return path
+                 user_has_setfattr = which("setfattr")
+                 user_has_xattr    = which("xattr")
+                 if user_has_setfattr or user_has_xattr:
+                     def write_xattr(path, key, value):
+                         import errno
+                         potential_errors = {
+                             # setfattr: /tmp/blah: Operation not supported
+                             "Operation not supported": errno.EOPNOTSUPP,
+                             # setfattr: ~/blah: No such file or directory
+                             # xattr: No such file: ~/blah
+                             "No such file": errno.ENOENT,
+                         }
+                         if user_has_setfattr:
+                             cmd = ['setfattr', '-n', key, '-v', value, path]
+                         elif user_has_xattr:
+                             cmd = ['xattr', '-w', key, value, path]
+                         try:
+                             output = subprocess.check_output(cmd, stderr=subprocess.STDOUT)
+                         except subprocess.CalledProcessError as e:
+                             errorstr = e.output.strip().decode()
+                             for potential_errorstr, potential_errno in potential_errors.items():
+                                 if errorstr.find(potential_errorstr) > -1:
+                                     e = OSError(potential_errno, potential_errorstr)
+                                     e.__cause__ = None
+                                     raise e
+                             raise # Reraise unhandled error
+                 else:
+                     # On Unix, and can't find pyxattr, setfattr, or xattr.
+                     if sys.platform.startswith('linux'):
+                         self._downloader.report_error("Couldn't find a tool to set the xattrs. Install either the python 'pyxattr' or 'xattr' modules, or the GNU 'attr' package (which contains the 'setfattr' tool).")
+                     elif sys.platform == 'darwin':
+                         self._downloader.report_error("Couldn't find a tool to set the xattrs. Install either the python 'xattr' module, or the 'xattr' binary.")
+             else:
+                 # Write xattrs to NTFS Alternate Data Streams: http://en.wikipedia.org/wiki/NTFS#Alternate_data_streams_.28ADS.29
+                 def write_xattr(path, key, value):
+                     assert(key.find(":") < 0)
+                     assert(path.find(":") < 0)
+                     assert(os.path.exists(path))
+                     f = open(path+":"+key, "w")
+                     f.write(value)
+                     f.close()
+         # Write the metadata to the file's xattrs
+         self._downloader.to_screen('[metadata] Writing metadata to file\'s xattrs...')
+         filename = info['filepath']
+         try:
+             xattr_mapping = {
+                 'user.xdg.referrer.url':       'webpage_url',
+                 # 'user.xdg.comment':            'description',
+                 'user.dublincore.title':       'title',
+                 'user.dublincore.date':        'upload_date',
+                 'user.dublincore.description': 'description',
+                 'user.dublincore.contributor': 'uploader',
+                 'user.dublincore.format':      'format',
+             }
+             for xattrname, infoname in xattr_mapping.items():
+                 value = info.get(infoname)
+                 if value:
+                     if infoname == "upload_date":
+                         value = hyphenate_date(value)
+                     write_xattr(filename, xattrname, value)
+             return True, info
+         except OSError:
+             self._downloader.report_error("This filesystem doesn't support extended attributes. (You may have to enable them in your /etc/fstab)")
+             return False, info
Simple merge
Simple merge