通过 Memcache 的 Django 会话:无法手动找到会话密钥

noa*_*ale 2 django session memcached caching pylibmc

我最近使用pylibmc从数据库支持的会话迁移到通过 memcached 存储的会话。

这是我的 settings.py 中的缓存、SESSION_CACHE_ALIAS 和 SESSION_ENGINE

CACHES = {
    'default': {
        'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
        'LOCATION': ['127.0.0.1:11211'],
    }
}


SESSION_CACHE_ALIAS = 'default'


SESSION_ENGINE = "django.contrib.sessions.backends.cache"
Run Code Online (Sandbox Code Playgroud)

在幕后一切正常,我可以看到它正在使用新的缓存系统。从 pylibmc 运行 get_stats() 方法向我显示缓存中当前项目的数量,我可以看到它增加了 1。

问题是我无法使用 pylibmc 手动获取会话。

在检查 views.py 中的请求会话数据时:

def my_view(request):
    if request.user.is_authenticated():

        print request.session.session_key
        # the above prints something like this: "1ay2kcv7axb3nu5fwnwoyf85wkwsttz9"

        print request.session.cache_key
        # the above prints something like this: "django.contrib.sessions.cache1ay2kcv7axb3nu5fwnwoyf85wkwsttz9"

        return HttpResponse(status=200)
    else:
        return HttpResponse(status=401)
Run Code Online (Sandbox Code Playgroud)

我注意到在打印 cache_key 时,它使用默认的 KEY_PREFIX 打印,而对于 session_key 则没有。看看代码中的注释,看看我的意思。

所以我想,“好吧,这些关键名称之一应该有效。让我尝试手动抓取会话数据,仅用于教育目的”:

import pylibmc

mc = pylibmc.Client(['127.0.0.1:11211'])

# Let's try key "1ay2kcv7axb3nu5fwnwoyf85wkwsttz9"

mc.get("1ay2kcv7axb3nu5fwnwoyf85wkwsttz9")
Run Code Online (Sandbox Code Playgroud)

嗯,没有任何反应,该名称不存在任何键。好的,不用担心,让我们尝试使用 cache_key ,那肯定可以正常工作吗?

mc.get("django.contrib.sessions.cache1ay2kcv7axb3nu5fwnwoyf85wkwsttz9")
Run Code Online (Sandbox Code Playgroud)

什么?我怎么还是拿不回来?当我测试时,我决定设置并获取一个随机键值,以查看它是否有效。我再次运行 get_stats() 只是为了确保密钥确实存在。我还测试了网络应用程序以查看我的会话是否确实有效并且确实有效。所以这让我得出结论,有一个我不知道的不同命名方案。

如果是这样,正确的命名方案是什么?

Mar*_*den 5

是的,Django 内部使用的缓存键通常不同于发送到缓存后端的键(在本例中为 pylibmc / memcached)。让我们将这两个键分别称为 django 缓存键和最终缓存键。

给出的django 缓存键request.session.cache_key用于 Django 的低级缓存 API,例如:

>>> from django.core.cache import cache
>>> cache.get(request.session.cache_key)
{'_auth_user_hash': '1ay2kcv7axb3nu5fwnwoyf85wkwsttz9', '_auth_user_id': u'1', '_auth_user_backend': u'django.contrib.auth.backends.ModelBackend'}
Run Code Online (Sandbox Code Playgroud)

最后的缓存键,另一方面,是的一个组成关键前缀,Django的缓存键,并且高速缓存的版本号。下面的make_key函数(来自 Django文档)演示了如何组合这三个值来生成此密钥:

def make_key(key, key_prefix, version):
    return ':'.join([key_prefix, str(version), key])
Run Code Online (Sandbox Code Playgroud)

默认情况下,key_prefix为空字符串且version为 1。

最后,通过检查make_key我们发现要传递给的正确最终缓存键mc.get

:1:django.contrib.sessions.cache1ay2kcv7axb3nu5fwnwoyf85wkwsttz9
Run Code Online (Sandbox Code Playgroud)

其形式为<KEY_PREFIX>:<VERSION>:<KEY>

注意:可以通过KEY_FUNCTION在缓存设置中定义来更改最终缓存键。