2016-03-05 17:49:18 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2017-01-11 00:17:02 +01:00
|
|
|
# Copyright 2016-2017 Mike Fährmann
|
2016-03-05 17:49:18 +01:00
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
|
|
# published by the Free Software Foundation.
|
|
|
|
|
2017-01-30 19:40:15 +01:00
|
|
|
"""Decorator to keep function results in a in-memory and database cache"""
|
2016-04-20 08:40:41 +02:00
|
|
|
|
2016-03-05 17:49:18 +01:00
|
|
|
import sqlite3
|
|
|
|
import pickle
|
|
|
|
import time
|
|
|
|
import tempfile
|
2016-09-23 08:23:04 +02:00
|
|
|
import os.path
|
2016-04-20 08:40:41 +02:00
|
|
|
import functools
|
2016-03-05 17:49:18 +01:00
|
|
|
from . import config
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
|
2016-03-05 17:49:18 +01:00
|
|
|
class CacheInvalidError(Exception):
|
2016-04-20 08:40:41 +02:00
|
|
|
"""A cache entry is either expired or does not exist"""
|
2016-03-05 17:49:18 +01:00
|
|
|
pass
|
|
|
|
|
2016-03-06 16:05:59 +01:00
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
class CacheModule():
|
|
|
|
"""Base class for cache modules"""
|
|
|
|
def __init__(self):
|
2016-11-22 17:57:41 +01:00
|
|
|
pass
|
2016-04-20 08:40:41 +02:00
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
raise CacheInvalidError()
|
|
|
|
|
|
|
|
def __setitem__(self, key, item):
|
|
|
|
pass
|
|
|
|
|
2016-11-22 17:57:41 +01:00
|
|
|
def __delitem__(self, key):
|
|
|
|
pass
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
def __enter__(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __exit__(self, *exc_info):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class CacheChain(CacheModule):
|
|
|
|
|
|
|
|
def __init__(self, modules=[]):
|
|
|
|
CacheModule.__init__(self)
|
|
|
|
self.modules = modules
|
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
num = 0
|
|
|
|
for module in self.modules:
|
2016-03-05 17:49:18 +01:00
|
|
|
try:
|
2016-04-20 08:40:41 +02:00
|
|
|
value = module[key]
|
|
|
|
break
|
2016-03-05 17:49:18 +01:00
|
|
|
except CacheInvalidError:
|
2016-04-20 08:40:41 +02:00
|
|
|
num += 1
|
|
|
|
else:
|
2016-03-05 17:49:18 +01:00
|
|
|
raise CacheInvalidError()
|
2016-04-20 08:40:41 +02:00
|
|
|
while num:
|
|
|
|
num -= 1
|
|
|
|
self.modules[num][key[0]] = value
|
|
|
|
return value
|
|
|
|
|
|
|
|
def __setitem__(self, key, item):
|
|
|
|
for module in self.modules:
|
|
|
|
module.__setitem__(key, item)
|
|
|
|
|
2016-11-22 17:57:41 +01:00
|
|
|
def __delitem__(self, key):
|
|
|
|
for module in self.modules:
|
|
|
|
module.__delitem__(key)
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
def __exit__(self, exc_type, exc_value, exc_traceback):
|
|
|
|
for module in self.modules:
|
|
|
|
module.__exit__(exc_type, exc_value, exc_traceback)
|
|
|
|
|
|
|
|
|
|
|
|
class MemoryCache(CacheModule):
|
|
|
|
"""In-memory cache module"""
|
|
|
|
def __init__(self):
|
|
|
|
CacheModule.__init__(self)
|
|
|
|
self.cache = {}
|
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
key, timestamp = key
|
|
|
|
try:
|
|
|
|
value, expires = self.cache[key]
|
|
|
|
if timestamp < expires:
|
|
|
|
return value, expires
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
raise CacheInvalidError()
|
|
|
|
|
|
|
|
def __setitem__(self, key, item):
|
|
|
|
self.cache[key] = item
|
|
|
|
|
2016-11-22 17:57:41 +01:00
|
|
|
def __delitem__(self, key):
|
|
|
|
try:
|
|
|
|
del self.cache[key]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
|
|
|
|
class DatabaseCache(CacheModule):
|
|
|
|
"""Database cache module"""
|
|
|
|
def __init__(self):
|
|
|
|
CacheModule.__init__(self)
|
|
|
|
path_default = os.path.join(tempfile.gettempdir(), ".gallery-dl.cache")
|
|
|
|
path = config.get(("cache", "file"), path_default)
|
|
|
|
if path is None:
|
|
|
|
raise RuntimeError()
|
2016-09-23 08:23:04 +02:00
|
|
|
path = os.path.expanduser(os.path.expandvars(path))
|
2016-04-20 08:40:41 +02:00
|
|
|
self.db = sqlite3.connect(path, timeout=30, check_same_thread=False)
|
2017-01-30 19:40:15 +01:00
|
|
|
self.db.execute(
|
|
|
|
"CREATE TABLE IF NOT EXISTS data ("
|
|
|
|
"key TEXT PRIMARY KEY,"
|
|
|
|
"value TEXT,"
|
|
|
|
"expires INTEGER"
|
|
|
|
")"
|
|
|
|
)
|
2016-04-20 08:40:41 +02:00
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
key, timestamp = key
|
|
|
|
try:
|
|
|
|
cursor = self.db.cursor()
|
2017-01-11 00:17:02 +01:00
|
|
|
try:
|
|
|
|
cursor.execute("BEGIN EXCLUSIVE")
|
|
|
|
except sqlite3.OperationalError:
|
|
|
|
"""workaround for python 3.6"""
|
2017-01-30 19:40:15 +01:00
|
|
|
cursor.execute(
|
|
|
|
"SELECT value, expires "
|
|
|
|
"FROM data "
|
|
|
|
"WHERE key=?",
|
|
|
|
(key,)
|
|
|
|
)
|
2016-04-20 08:40:41 +02:00
|
|
|
value, expires = cursor.fetchone()
|
|
|
|
if timestamp < expires:
|
|
|
|
self.commit()
|
|
|
|
return pickle.loads(value), expires
|
|
|
|
except TypeError:
|
|
|
|
pass
|
|
|
|
raise CacheInvalidError()
|
|
|
|
|
|
|
|
def __setitem__(self, key, item):
|
|
|
|
value, expires = item
|
|
|
|
self.db.execute("INSERT OR REPLACE INTO data VALUES (?,?,?)",
|
|
|
|
(key, pickle.dumps(value), expires))
|
|
|
|
|
2016-11-22 17:57:41 +01:00
|
|
|
def __delitem__(self, key):
|
|
|
|
self.db.execute("DELETE FROM data WHERE key=?", (key,))
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
def __exit__(self, *exc_info):
|
|
|
|
self.commit()
|
|
|
|
|
|
|
|
def commit(self):
|
|
|
|
self.db.commit()
|
|
|
|
|
|
|
|
|
|
|
|
class CacheDecorator():
|
|
|
|
|
|
|
|
def __init__(self, func, module, maxage, keyarg):
|
|
|
|
self.func = func
|
|
|
|
self.key = "%s.%s" % (func.__module__, func.__name__)
|
|
|
|
self.cache = module
|
|
|
|
self.maxage = maxage
|
|
|
|
self.keyarg = keyarg
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
timestamp = time.time()
|
|
|
|
if self.keyarg is None:
|
|
|
|
key = self.key
|
|
|
|
else:
|
|
|
|
key = "%s-%s" % (self.key, args[self.keyarg])
|
|
|
|
try:
|
|
|
|
result, _ = self.cache[key, timestamp]
|
|
|
|
except CacheInvalidError:
|
|
|
|
with self.cache:
|
|
|
|
result = self.func(*args, **kwargs)
|
|
|
|
expires = int(timestamp + self.maxage)
|
|
|
|
self.cache[key] = result, expires
|
|
|
|
return result
|
|
|
|
|
|
|
|
def __get__(self, obj, objtype):
|
|
|
|
"""Support instance methods."""
|
|
|
|
return functools.partial(self.__call__, obj)
|
|
|
|
|
2016-11-22 17:57:41 +01:00
|
|
|
def invalidate(self, key=None):
|
|
|
|
if key is None:
|
|
|
|
key = self.key
|
|
|
|
else:
|
|
|
|
key = "%s-%s" % (self.key, key)
|
|
|
|
del self.cache[key]
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
|
|
|
|
def build_cache_decorator(*modules):
|
|
|
|
if len(modules) > 1:
|
|
|
|
module = CacheChain(modules)
|
|
|
|
else:
|
|
|
|
module = modules[0]
|
2017-01-30 19:40:15 +01:00
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
def decorator(maxage=3600, keyarg=None):
|
|
|
|
def wrap(func):
|
|
|
|
return CacheDecorator(func, module, maxage, keyarg)
|
|
|
|
return wrap
|
|
|
|
return decorator
|
2016-03-05 17:49:18 +01:00
|
|
|
|
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
MEMCACHE = MemoryCache()
|
|
|
|
memcache = build_cache_decorator(MEMCACHE)
|
2016-03-05 17:49:18 +01:00
|
|
|
|
2016-04-20 08:40:41 +02:00
|
|
|
try:
|
|
|
|
DBCACHE = DatabaseCache()
|
|
|
|
cache = build_cache_decorator(MEMCACHE, DBCACHE)
|
|
|
|
except RuntimeError():
|
|
|
|
DBCACHE = None
|
|
|
|
cache = memcache
|