~launchpad-pqm/launchpad/devel

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
# Copyright 2010 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

"""Storm/memcached implementation of lazr.restful's representation cache."""

from lazr.restful.simple import BaseRepresentationCache
from lazr.restful.utils import get_current_web_service_request
import storm
from zope.component import getUtility
from zope.security.proxy import removeSecurityProxy
from zope.traversing.browser import absoluteURL

from canonical.config import config
from lp.services.memcache.interfaces import IMemcacheClient


__metaclass__ = type
__all__ = [
    'MemcachedStormRepresentationCache',
]


class MemcachedStormRepresentationCache(BaseRepresentationCache):
    """Caches lazr.restful representations of Storm objects in memcached."""

    def __init__(self):
        """Initialize with the memcached client."""
        self.client = getUtility(IMemcacheClient)

    def key_for(self, obj, media_type, version):
        """See `BaseRepresentationCache`."""
        obj = removeSecurityProxy(obj)
        try:
            storm_info = storm.info.get_obj_info(obj)
        except storm.exceptions.ClassInfoError, e:
            # There's no Storm data for this object. Don't cache it,
            # since we don't know how to invalidate the cache.
            return self.DO_NOT_CACHE
        table_name = storm_info.cls_info.table
        primary_key = tuple(var.get() for var in storm_info.primary_vars)
        identifier = table_name + repr(primary_key)

        key = (identifier
               + ',' + config._instance_name
               + ',' + media_type + ',' + str(version)).replace(' ', '.')
        return key

    def get_by_key(self, key, default=None):
        """See `BaseRepresentationCache`."""
        value = self.client.get(key)
        if value is None:
            value = default
        return value

    def set_by_key(self, key, value):
        """See `BaseRepresentationCache`."""
        self.client.set(
            key, value,
            time=config.vhost.api.representation_cache_expiration_time)

    def delete_by_key(self, key):
        """See `BaseRepresentationCache`."""
        self.client.delete(key)