cache.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. # coding: utf-8
  2. from __future__ import unicode_literals
  3. import errno
  4. import json
  5. import os
  6. import re
  7. import shutil
  8. import traceback
  9. from .compat import (
  10. compat_getenv,
  11. compat_open as open,
  12. compat_os_makedirs,
  13. )
  14. from .utils import (
  15. error_to_compat_str,
  16. escape_rfc3986,
  17. expand_path,
  18. is_outdated_version,
  19. traverse_obj,
  20. write_json_file,
  21. )
  22. from .version import __version__
  23. class Cache(object):
  24. _YTDL_DIR = 'youtube-dl'
  25. _VERSION_KEY = _YTDL_DIR + '_version'
  26. _DEFAULT_VERSION = '2021.12.17'
  27. def __init__(self, ydl):
  28. self._ydl = ydl
  29. def _write_debug(self, *args, **kwargs):
  30. self._ydl.write_debug(*args, **kwargs)
  31. def _report_warning(self, *args, **kwargs):
  32. self._ydl.report_warning(*args, **kwargs)
  33. def _to_screen(self, *args, **kwargs):
  34. self._ydl.to_screen(*args, **kwargs)
  35. def _get_param(self, k, default=None):
  36. return self._ydl.params.get(k, default)
  37. def _get_root_dir(self):
  38. res = self._get_param('cachedir')
  39. if res is None:
  40. cache_root = compat_getenv('XDG_CACHE_HOME', '~/.cache')
  41. res = os.path.join(cache_root, self._YTDL_DIR)
  42. return expand_path(res)
  43. def _get_cache_fn(self, section, key, dtype):
  44. assert re.match(r'^[\w.-]+$', section), \
  45. 'invalid section %r' % section
  46. key = escape_rfc3986(key, safe='').replace('%', ',') # encode non-ascii characters
  47. return os.path.join(
  48. self._get_root_dir(), section, '%s.%s' % (key, dtype))
  49. @property
  50. def enabled(self):
  51. return self._get_param('cachedir') is not False
  52. def store(self, section, key, data, dtype='json'):
  53. assert dtype in ('json',)
  54. if not self.enabled:
  55. return
  56. fn = self._get_cache_fn(section, key, dtype)
  57. try:
  58. compat_os_makedirs(os.path.dirname(fn), exist_ok=True)
  59. self._write_debug('Saving {section}.{key} to cache'.format(section=section, key=key))
  60. write_json_file({self._VERSION_KEY: __version__, 'data': data}, fn)
  61. except Exception:
  62. tb = traceback.format_exc()
  63. self._report_warning('Writing cache to {fn!r} failed: {tb}'.format(fn=fn, tb=tb))
  64. def clear(self, section, key, dtype='json'):
  65. if not self.enabled:
  66. return
  67. fn = self._get_cache_fn(section, key, dtype)
  68. self._write_debug('Clearing {section}.{key} from cache'.format(section=section, key=key))
  69. try:
  70. os.remove(fn)
  71. except Exception as e:
  72. if getattr(e, 'errno') == errno.ENOENT:
  73. # file not found
  74. return
  75. tb = traceback.format_exc()
  76. self._report_warning('Clearing cache from {fn!r} failed: {tb}'.format(fn=fn, tb=tb))
  77. def _validate(self, data, min_ver):
  78. version = traverse_obj(data, self._VERSION_KEY)
  79. if not version: # Backward compatibility
  80. data, version = {'data': data}, self._DEFAULT_VERSION
  81. if not is_outdated_version(version, min_ver or '0', assume_new=False):
  82. return data['data']
  83. self._write_debug('Discarding old cache from version {version} (needs {min_ver})'.format(version=version, min_ver=min_ver))
  84. def load(self, section, key, dtype='json', default=None, **kw_min_ver):
  85. assert dtype in ('json',)
  86. min_ver = kw_min_ver.get('min_ver')
  87. if not self.enabled:
  88. return default
  89. cache_fn = self._get_cache_fn(section, key, dtype)
  90. try:
  91. with open(cache_fn, encoding='utf-8') as cachef:
  92. self._write_debug('Loading {section}.{key} from cache'.format(section=section, key=key), only_once=True)
  93. return self._validate(json.load(cachef), min_ver)
  94. except (ValueError, KeyError):
  95. try:
  96. file_size = 'size: %d' % os.path.getsize(cache_fn)
  97. except (OSError, IOError) as oe:
  98. file_size = error_to_compat_str(oe)
  99. self._report_warning('Cache retrieval from %s failed (%s)' % (cache_fn, file_size))
  100. except Exception as e:
  101. if getattr(e, 'errno') == errno.ENOENT:
  102. # no cache available
  103. return
  104. self._report_warning('Cache retrieval from %s failed' % (cache_fn,))
  105. return default
  106. def remove(self):
  107. if not self.enabled:
  108. self._to_screen('Cache is disabled (Did you combine --no-cache-dir and --rm-cache-dir?)')
  109. return
  110. cachedir = self._get_root_dir()
  111. if not any((term in cachedir) for term in ('cache', 'tmp')):
  112. raise Exception('Not removing directory %s - this does not look like a cache dir' % (cachedir,))
  113. self._to_screen(
  114. 'Removing cache dir %s .' % (cachedir,), skip_eol=True, ),
  115. if os.path.exists(cachedir):
  116. self._to_screen('.', skip_eol=True)
  117. shutil.rmtree(cachedir)
  118. self._to_screen('.')