Django自定义模板标签中的访问请求


73

我在myapp_extras.py中的代码:

from django import template

register = template.Library()

@register.inclusion_tag('new/userinfo.html')
def address():
    address = request.session['address']
    return {'address':address}

在“ settings.py”中:

TEMPLATE_CONTEXT_PROCESSORS =(
    "django.core.context_processors.auth",
    "django.core.context_processors.debug",
    "django.core.context_processors.i18n",
    "django.core.context_processors.media",
    'django.core.context_processors.request'
)

但我得到一个错误:

TemplateSyntaxError at /items/

Caught an exception while rendering: global name 'request' is not defined

Original Traceback (most recent call last):
  File "C:\Python25\lib\site-packages\django\template\debug.py", line 71, in render_node
    result = node.render(context)
  File "C:\Python25\lib\site-packages\django\template\__init__.py", line 915, in render
    dict = func(*args)
  File "C:\p4\projects\myproject\..\myproject\invoice\templatetags\myapp_extras.py", line 9, in address
    address = request.session['address']
NameError: global name 'request' is not defined

我引用了这一点。 在Django中,是否可以从自定义标签中访问当前用户会话?


1
嗯...这个问题的答案似乎是错误的...
Dominic Rodger 2010年

如果您要访问请求,请检查此答案stackoverflow.com/a/2567234/1153703
Bikesh M 2016年

Answers:


161

request在该范围内不是变量。您必须首先从上下文中获取它。传递takes_context给装饰器,并添加context到tag参数

像这样:

@register.inclusion_tag('new/userinfo.html', takes_context=True)
def address(context):
    request = context['request']
    address = request.session['address']
    return {'address':address}

似乎我仍然在/渲染时遇到错误TemplateSyntaxError /渲染时捕获了异常:“请求”原始回溯(最近一次调用为最新):文件“ C:\ Python25 \ lib \ site-packages \ django \ template \ debug.py”,行71,在render_node结果= node.render(上下文)文件“C:\ Python25 \ lib中\站点包\ django的\ TEMPLATE_ init_ py”为,线936,在渲染的dict = FUNC(*参数)文件“C:\ ... \ myapp_extras.py”,第7行,在登录请求=上下文[ '请求']文件“C:\ Python25 \ lib中\站点包\ django的\模板\ context.py”,第44行,在的GetItem 加注KeyError(key)
KeyError

5
我在此页面中注意到docs.djangoproject.com/en/dev/ref/templates/api “ DJANGO.CORE.CONTEXT_PROCESSORS.REQUEST如果TEMPLATE_CONTEXT_PROCESSORS包含此处理器,则每个RequestContext将包含一个变量请求,即当前的HttpRequest。请注意默认情况下,此处理器未启用;您必须激活它。” 我需要配置以激活请求处理器的地方
icn

1
为什么采用include_tag是这种情况?
aehlke 2012年

1
Django中的MVP分隔可防止其看到有关请求本身的任何信息,因此必须将其馈送到标记。
Ignacio Vazquez-Abrams'4

2
不要忘记在您的settings.py中添加django.template.context_processors.request,否则请求将不会包含在上下文中TEMPLATES
Zat42

12

我从上面尝试了解决方案(来自Ignacio Vazquez-Abrams),直到我发现上下文处理器仅适用于RequestContext包装器类,它才真正起作用。

因此,在主视图方法中,您应该添加以下行:

from django.template import RequestContext        
return render_to_response('index.html', {'form': form, }, 
                              context_instance = RequestContext(request))

实际上,context_instance = RequestContext(request)是缺少的部分。用它!
Yauhen Yakimovich

4
如果将响应呈现为string / html,则可以不使用RequestContext。@register.simple_tag(takes_context = True) ,然后def my_tag(context, ...) ... return render_to_response('index.html', {'form': form, }, context_instance = context)
谢尔盖·奥尔尚斯基

8

我这样做是这样的:

from django import template
register = template.Library()

def do_test_request(parser,token):
    try:
        tag_name = token.split_contents() # Not really useful
    except ValueError:
        raise template.TemplateSyntaxError("%r error" % token.contents.split()[0])
    return RequestTestNode()

class RequestTestNode(template.Node):
    def __init__(self,):
        self.request = template.Variable('request')
    def render(self, context):
        rqst = self.request.resolve(context)
        return "The URL is: %s" % rqst.get_full_path()

register.tag('test_request', do_test_request)

还有一个名为的函数resolve_variable,但已弃用。

希望能帮助到你!

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.