2022-04-17 20:58:28 +00:00
|
|
|
import contextlib
|
2014-09-03 10:41:05 +00:00
|
|
|
import errno
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import shutil
|
|
|
|
import traceback
|
|
|
|
|
2022-08-27 01:52:48 +00:00
|
|
|
from .utils import expand_path, traverse_obj, version_tuple, write_json_file
|
|
|
|
from .version import __version__
|
2014-09-03 10:41:05 +00:00
|
|
|
|
|
|
|
|
2022-04-11 15:10:28 +00:00
|
|
|
class Cache:
|
2014-09-03 10:41:05 +00:00
|
|
|
def __init__(self, ydl):
|
|
|
|
self._ydl = ydl
|
|
|
|
|
|
|
|
def _get_root_dir(self):
|
|
|
|
res = self._ydl.params.get('cachedir')
|
|
|
|
if res is None:
|
2022-06-24 08:10:17 +00:00
|
|
|
cache_root = os.getenv('XDG_CACHE_HOME', '~/.cache')
|
2021-02-24 18:45:56 +00:00
|
|
|
res = os.path.join(cache_root, 'yt-dlp')
|
2017-03-25 19:31:16 +00:00
|
|
|
return expand_path(res)
|
2014-09-03 10:41:05 +00:00
|
|
|
|
|
|
|
def _get_cache_fn(self, section, key, dtype):
|
2014-09-04 02:47:56 +00:00
|
|
|
assert re.match(r'^[a-zA-Z0-9_.-]+$', section), \
|
|
|
|
'invalid section %r' % section
|
|
|
|
assert re.match(r'^[a-zA-Z0-9_.-]+$', key), 'invalid key %r' % key
|
2014-09-03 10:41:05 +00:00
|
|
|
return os.path.join(
|
2022-04-11 15:10:28 +00:00
|
|
|
self._get_root_dir(), section, f'{key}.{dtype}')
|
2014-09-03 10:41:05 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def enabled(self):
|
|
|
|
return self._ydl.params.get('cachedir') is not False
|
|
|
|
|
|
|
|
def store(self, section, key, data, dtype='json'):
|
|
|
|
assert dtype in ('json',)
|
|
|
|
|
|
|
|
if not self.enabled:
|
|
|
|
return
|
|
|
|
|
|
|
|
fn = self._get_cache_fn(section, key, dtype)
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
os.makedirs(os.path.dirname(fn))
|
|
|
|
except OSError as ose:
|
|
|
|
if ose.errno != errno.EEXIST:
|
|
|
|
raise
|
2021-09-17 18:23:55 +00:00
|
|
|
self._ydl.write_debug(f'Saving {section}.{key} to cache')
|
2022-08-27 01:52:48 +00:00
|
|
|
write_json_file({'yt-dlp_version': __version__, 'data': data}, fn)
|
2014-09-03 10:41:05 +00:00
|
|
|
except Exception:
|
|
|
|
tb = traceback.format_exc()
|
2022-04-11 15:10:28 +00:00
|
|
|
self._ydl.report_warning(f'Writing cache to {fn!r} failed: {tb}')
|
2014-09-03 10:41:05 +00:00
|
|
|
|
2022-08-27 01:52:48 +00:00
|
|
|
def _validate(self, data, after):
|
|
|
|
version = traverse_obj(data, 'yt-dlp_version')
|
|
|
|
if not version: # Backward compatibility
|
|
|
|
data, version = {'data': data}, '2022.08.19'
|
|
|
|
if not after or version_tuple(version) > version_tuple(after):
|
|
|
|
return data['data']
|
|
|
|
self._ydl.write_debug(f'Discarding old cache from version {version} (need {after})')
|
|
|
|
|
|
|
|
def load(self, section, key, dtype='json', default=None, *, after=None):
|
2014-09-03 10:41:05 +00:00
|
|
|
assert dtype in ('json',)
|
|
|
|
|
|
|
|
if not self.enabled:
|
|
|
|
return default
|
|
|
|
|
|
|
|
cache_fn = self._get_cache_fn(section, key, dtype)
|
2022-04-17 20:58:28 +00:00
|
|
|
with contextlib.suppress(OSError):
|
2014-09-03 10:41:05 +00:00
|
|
|
try:
|
2022-04-11 15:10:28 +00:00
|
|
|
with open(cache_fn, encoding='utf-8') as cachef:
|
2021-09-17 18:23:55 +00:00
|
|
|
self._ydl.write_debug(f'Loading {section}.{key} from cache')
|
2022-08-27 01:52:48 +00:00
|
|
|
return self._validate(json.load(cachef), after)
|
|
|
|
except (ValueError, KeyError):
|
2014-09-03 10:41:05 +00:00
|
|
|
try:
|
|
|
|
file_size = os.path.getsize(cache_fn)
|
2022-04-11 15:10:28 +00:00
|
|
|
except OSError as oe:
|
2014-09-03 10:41:05 +00:00
|
|
|
file_size = str(oe)
|
2022-04-11 15:10:28 +00:00
|
|
|
self._ydl.report_warning(f'Cache retrieval from {cache_fn} failed ({file_size})')
|
2014-09-03 10:41:05 +00:00
|
|
|
|
|
|
|
return default
|
|
|
|
|
|
|
|
def remove(self):
|
|
|
|
if not self.enabled:
|
|
|
|
self._ydl.to_screen('Cache is disabled (Did you combine --no-cache-dir and --rm-cache-dir?)')
|
|
|
|
return
|
|
|
|
|
|
|
|
cachedir = self._get_root_dir()
|
|
|
|
if not any((term in cachedir) for term in ('cache', 'tmp')):
|
|
|
|
raise Exception('Not removing directory %s - this does not look like a cache dir' % cachedir)
|
|
|
|
|
|
|
|
self._ydl.to_screen(
|
|
|
|
'Removing cache dir %s .' % cachedir, skip_eol=True)
|
|
|
|
if os.path.exists(cachedir):
|
|
|
|
self._ydl.to_screen('.', skip_eol=True)
|
|
|
|
shutil.rmtree(cachedir)
|
|
|
|
self._ydl.to_screen('.')
|