Zugriffsanforderung in benutzerdefinierten Django-Vorlagen-Tags

73

Mein Code in 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}

in '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'
)

aber ich habe einen Fehler bekommen:

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

Ich habe auf diese verwiesen. Ist es in Django möglich, über ein benutzerdefiniertes Tag auf die aktuelle Benutzersitzung zuzugreifen? .

icn
quelle
1
Hmmm ... Die Antworten auf diese verknüpfte Frage scheinen falsch zu sein ...
Dominic Rodger
Wenn Sie auf die Anfrage zugreifen möchten, überprüfen Sie diese Antwort. stackoverflow.com/a/2567234/1153703
Bikesh M

Antworten:

160

requestist keine Variable in diesem Bereich. Sie müssen es zuerst aus dem Kontext abrufen. Übergeben Sie es takes_contextan den Dekorateur und fügen Sie contextes den Tag-Argumenten hinzu .

So was:

@register.inclusion_tag('new/userinfo.html', takes_context=True)
def address(context):
    request = context['request']
    address = request.session['address']
    return {'address':address}
Ignacio Vazquez-Abrams
quelle
Es scheint, dass ich immer noch Fehler habe. TemplateSyntaxError bei / Beim Rendern ist eine Ausnahme aufgetreten: 'request' Original Traceback (letzter Aufruf zuletzt): Datei "C: \ Python25 \ lib \ site-packages \ django \ template \ debug.py", Zeile 71, in render_node result = node.render (Kontext) Datei "C: \ Python25 \ lib \ site-packages \ django \ template_ init_ .py", Zeile 936, in render dict = func (* args) Datei "c: \ ... \ myapp_extras.py ", Zeile 7, in login request = context ['request'] Datei" C: \ Python25 \ lib \ site-packages \ django \ template \ context.py ", Zeile 44, in getitem raise KeyError (Schlüssel) KeyError: 'Anfrage'
icn
5
Ich habe auf dieser Seite festgestellt, dass docs.djangoproject.com/de/dev/ref/templates/api "DJANGO.CORE.CONTEXT_PROCESSORS.REQUEST Wenn TEMPLATE_CONTEXT_PROCESSORS diesen Prozessor enthält, enthält jeder RequestContext eine variable Anforderung, die die aktuelle HttpRequest ist Dieser Prozessor ist standardmäßig nicht aktiviert. Sie müssen ihn aktivieren. " Irgendwo muss ich konfigurieren, um den Anforderungsprozessor zu aktivieren
icn
1
Warum ist dies bei include_tag der Fall?
Aehlke
1
Die MVP-Trennung in Django verhindert, dass etwas über die Anforderung selbst angezeigt wird. Daher muss es dem Tag zugeführt werden.
Ignacio Vazquez-Abrams
2
Vergessen Sie nicht django.template.context_processors.request, TEMPLATESin Ihren settings.py hinzuzufügen , sonst wird die Anfrage nicht in den Kontext aufgenommen
Zat42
12

Ich habe eine Lösung von oben (von Ignacio Vazquez-Abrams) ausprobiert und es hat tatsächlich nicht funktioniert, bis ich herausgefunden habe, dass Kontextprozessoren nur mit der RequestContextWrapper-Klasse funktionieren .

In der Hauptansichtsmethode sollten Sie also die folgende Zeile hinzufügen:

from django.template import RequestContext        
return render_to_response('index.html', {'form': form, }, 
                              context_instance = RequestContext(request))
oben
quelle
In der Tat fehlt context_instance = RequestContext (Anfrage). Benutze es!
Yauhen Yakimovich
4
Wenn Sie die Antwort auf string / html rendern, können Sie auf RequestContext verzichten. @register.simple_tag(takes_context = True) und dann def my_tag(context, ...) ... return render_to_response('index.html', {'form': form, }, context_instance = context).
Sergey Orshanskiy
8

Ich habe es so gemacht:

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)

Es gibt auch eine Funktion namens resolve_variable, die jedoch veraltet ist.

Ich hoffe es hilft!

Santiagobasulto
quelle