Source code for aiocache.factory

import logging
import urllib
from contextlib import suppress
from copy import deepcopy
from typing import Dict

from aiocache import AIOCACHE_CACHES
from aiocache.base import BaseCache
from aiocache.exceptions import InvalidCacheType

with suppress(ImportError):
    import redis.asyncio as redis


logger = logging.getLogger(__name__)


def _class_from_string(class_path):
    class_name = class_path.split(".")[-1]
    module_name = class_path.rstrip(class_name).rstrip(".")
    return getattr(__import__(module_name, fromlist=[class_name]), class_name)


def _create_cache(cache, serializer=None, plugins=None, **kwargs):
    kwargs = deepcopy(kwargs)
    if serializer is not None:
        cls = serializer.pop("class")
        cls = _class_from_string(cls) if isinstance(cls, str) else cls
        serializer = cls(**serializer)

    plugins_instances = []
    if plugins is not None:
        for plugin in plugins:
            cls = plugin.pop("class")
            cls = _class_from_string(cls) if isinstance(cls, str) else cls
            plugins_instances.append(cls(**plugin))
    cache = _class_from_string(cache) if isinstance(cache, str) else cache
    if cache == AIOCACHE_CACHES.get("redis"):
        return cache(
            serializer=serializer,
            plugins=plugins_instances,
            namespace=kwargs.pop('namespace', ''),
            ttl=kwargs.pop('ttl', None),
            client=redis.Redis(**kwargs)
        )
    else:
        return cache(serializer=serializer, plugins=plugins_instances, **kwargs)


[docs] class Cache: """ This class is just a proxy to the specific cache implementations like :class:`aiocache.SimpleMemoryCache`, :class:`aiocache.RedisCache` and :class:`aiocache.MemcachedCache`. It is the preferred method of instantiating new caches over using the backend specific classes. You can instatiate a new one using the ``cache_type`` attribute like: >>> from aiocache import Cache >>> Cache(Cache.REDIS) RedisCache (127.0.0.1:6379) If you don't specify anything, ``Cache.MEMORY`` is used. Only ``Cache.MEMORY``, ``Cache.REDIS`` and ``Cache.MEMCACHED`` types are allowed. If the type passed is invalid, it will raise a :class:`aiocache.exceptions.InvalidCacheType` exception. """ MEMORY = AIOCACHE_CACHES["memory"] REDIS = AIOCACHE_CACHES.get("redis") MEMCACHED = AIOCACHE_CACHES.get("memcached") def __new__(cls, cache_class=MEMORY, **kwargs): if not issubclass(cache_class, BaseCache): raise InvalidCacheType( "Invalid cache type, you can only use {}".format(list(AIOCACHE_CACHES.keys())) ) instance = cache_class.__new__(cache_class, **kwargs) instance.__init__(**kwargs) return instance @classmethod def _get_cache_class(cls, scheme): return AIOCACHE_CACHES[scheme] @classmethod def get_scheme_class(cls, scheme): try: return cls._get_cache_class(scheme) except KeyError as e: raise InvalidCacheType( "Invalid cache type, you can only use {}".format(list(AIOCACHE_CACHES.keys())) ) from e
[docs] @classmethod def from_url(cls, url): """ Given a resource uri, return an instance of that cache initialized with the given parameters. An example usage: >>> from aiocache import Cache >>> Cache.from_url('memory://') <aiocache.backends.memory.SimpleMemoryCache object at 0x1081dbb00> a more advanced usage using queryparams to configure the cache: >>> from aiocache import Cache >>> cache = Cache.from_url('redis://localhost:10/1?pool_max_size=1') >>> cache RedisCache (localhost:10) >>> cache.db 1 >>> cache.pool_max_size 1 :param url: string identifying the resource uri of the cache to connect to """ parsed_url = urllib.parse.urlparse(url) kwargs = dict(urllib.parse.parse_qsl(parsed_url.query)) cache_class = Cache.get_scheme_class(parsed_url.scheme) if parsed_url.path: kwargs.update(cache_class.parse_uri_path(parsed_url.path)) if parsed_url.hostname: kwargs["host"] = parsed_url.hostname if parsed_url.port: kwargs["port"] = parsed_url.port if parsed_url.password: kwargs["password"] = parsed_url.password for arg in ['max_connections', 'socket_connect_timeout']: if arg in kwargs: kwargs[arg] = int(kwargs[arg]) if cache_class == cls.REDIS: return Cache(cache_class, client=redis.Redis(**kwargs)) else: return Cache(cache_class, **kwargs)
class CacheHandler: _config: Dict[str, Dict[str, object]] = { "default": { "cache": "aiocache.SimpleMemoryCache", "serializer": {"class": "aiocache.serializers.StringSerializer"}, } } def __init__(self): self._caches = {} def add(self, alias: str, config: Dict[str, object]) -> None: """ Add a cache to the current config. If the key already exists, it will overwrite it:: >>> caches.add('default', { 'cache': "aiocache.SimpleMemoryCache", 'serializer': { 'class': "aiocache.serializers.StringSerializer" } }) :param alias: The alias for the cache :param config: Mapping containing the cache configuration """ self._config[alias] = config def get(self, alias: str) -> object: """ Retrieve cache identified by alias. Will return always the same instance If the cache was not instantiated yet, it will do it lazily the first time this is called. :param alias: str cache alias :return: cache instance """ try: return self._caches[alias] except KeyError: pass config = self.get_alias_config(alias) cache = _create_cache(**deepcopy(config)) self._caches[alias] = cache return cache def create(self, alias, **kwargs): """Create a new cache. You can use kwargs to pass extra parameters to configure the cache. :param alias: alias to pull configuration from :return: New cache instance """ config = self.get_alias_config(alias) # TODO(PY39): **config | kwargs return _create_cache(**{**config, **kwargs}) def get_alias_config(self, alias): config = self.get_config() if alias not in config: raise KeyError( "Could not find config for '{0}', ensure you include {0} when calling" "caches.set_config specifying the config for that cache".format(alias) ) return config[alias] def get_config(self): """ Return copy of current stored config """ return deepcopy(self._config) def set_config(self, config): """ Set (override) the default config for cache aliases from a dict-like structure. The structure is the following:: { 'default': { 'cache': "aiocache.SimpleMemoryCache", 'serializer': { 'class': "aiocache.serializers.StringSerializer" } }, 'redis_alt': { 'cache': "aiocache.RedisCache", 'host': "127.0.0.10", 'port': 6378, 'serializer': { 'class': "aiocache.serializers.PickleSerializer" }, 'plugins': [ {'class': "aiocache.plugins.HitMissRatioPlugin"}, {'class': "aiocache.plugins.TimingPlugin"} ] } } 'default' key must always exist when passing a new config. Default configuration is:: { 'default': { 'cache': "aiocache.SimpleMemoryCache", 'serializer': { 'class': "aiocache.serializers.StringSerializer" } } } You can set your own classes there. The class params accept both str and class types. All keys in the config are optional, if they are not passed the defaults for the specified class will be used. If a config key already exists, it will be updated with the new values. """ if "default" not in config: raise ValueError("default config must be provided") for config_name in config.keys(): self._caches.pop(config_name, None) self._config = config caches = CacheHandler()