Django:如何在开发环境中获取静态文件的文件路径?

时间:2012-02-22 08:10:49

标签: python django static-files

首先是一些背景知识。我正在使用以下“技巧”来防止不希望的浏览器缓存静态文件(CSS,JS等):

<script src="{{ STATIC_URL }}js/utils.js?version=1302983029"></script>

当后续页面加载时版本字符串发生更改时,它会使浏览器从服务器重新获取静态文件。 (谷歌“css缓存”有关此技巧的文章。)

我希望浏览器获取最新版本的静态文件,但我还希望在文件未更改时允许浏览器缓存。换句话说,当且仅当静态文件已更新时,我希望更改版本字符串。我也想自动生成版本字符串。

为此,我使用静态文件的上次修改时间作为版本字符串。我正在制作一个自定义模板标签:

<script src="{% versioned_static 'js/utils.js' %}"></script>

以下是模板标签的工作原理:

import os.path
from django import template
from django.conf import settings

class VersionedStaticNode(template.Node):
    ...
    def render(self, context):
        # With the example above, self.path_string is "js/utils.js"
        static_file_path = os.path.join(settings.STATIC_ROOT, self.path_string)
        return '%s?version=%s' % (
            os.path.join(settings.STATIC_URL, self.path_string),
            int(os.path.getmtime(static_file_path))
            )

要获取静态文件的上次修改时间,我需要知道它在系统上的文件路径。我通过加入settings.STATIC_ROOT和该静态根目录中的文件相对路径来获取此文件路径。这对于生产服务器来说都很好,因为所有静态文件都是在STATIC_ROOT收集的。

但是,在开发服务器上(使用manage.py runserver命令),不会在STATIC_ROOT收集静态文件。那么在开发中运行时如何获取静态文件的文件路径?

(为了澄清我的目的:我想避免的缓存情况是浏览器使用新HTML和旧CSS / JS的不匹配。在生产中,这会极大地混淆用户;在开发中,这会让我和其他开发人员感到困惑,并强制我们经常刷新页面/清除浏览器缓存。)

1 个答案:

答案 0 :(得分:13)

如果使用django.contrib.staticfiles,这里有findstatic命令(django / contrib / staticfiles / management / commands / findstatic.py)的摘录应该有所帮助。它使用finders.find来定位文件。

    from django.contrib.staticfiles import finders

    result = finders.find(path, all=options['all'])

    path = smart_unicode(path)
    if result:
        if not isinstance(result, (list, tuple)):
            result = [result]
        output = u'\n  '.join(
            (smart_unicode(os.path.realpath(path)) for path in result))
        self.stdout.write(
            smart_str(u"Found '%s' here:\n  %s\n" % (path, output)))