219 lines
5.6 KiB
Python
Raw Normal View History

2016-03-05 17:49:18 +01:00
# -*- coding: utf-8 -*-
# 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-04-20 08:40:41 +02:00
class CacheModule():
"""Base class for cache modules"""
def __init__(self):
pass
2016-04-20 08:40:41 +02:00
def __getitem__(self, key):
raise CacheInvalidError()
def __setitem__(self, key, item):
pass
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)
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
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()
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))
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)
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