Django - CSRF令牌生成 - render_to_response vs render

时间:2017-01-12 06:46:02

标签: django render csrf render-to-response

我正在Django中尝试一个简单的用户注册表单。使用render_to_response

时,未生成CSRF令牌
  

返回render_to_response('registration / register.html',RequestContext(request,{'form':RegistrationForm()}))

当我使用render

时生成CSRF令牌
  

return render(request,'registration / register.html',{'form':RegistrationForm()})

我对render_to_response

做错了什么

以下是相关代码块

views.py

@csrf_protect
def register(request):
    if request.method == 'POST':
        form = RegistrationForm(request.POST)
        if form.is_valid():
            user = User.objects.create_user(
                username=form.cleaned_data['username'],
                password=form.cleaned_data['password1'],
                email=form.cleaned_data['email']
            )
            return HttpResponseRedirect('/register_success/')
    else:
        return render_to_response('registration/register.html', RequestContext(request, {'form': RegistrationForm()}))

register.html

{% extends "base.html" %}
{% block title %}User Registration{% endblock %}

{% block content %}
    <form method="post" action=".">
        {% csrf_token %}
        <table border="0">
            {{ form.as_table }}
        </table>
        <input type="submit" value="Register" />
    </form>
{% endblock %}

1 个答案:

答案 0 :(得分:8)

建议的方法是使用render代替render_to_response。代码更简单,CSRF令牌将起作用,因为render将使用请求上下文来呈现模板。

return render(request, 'registration/register.html', {'form': RegistrationForm()})

The docs recommend that you don't use render_to_response。在Django 1.10之前,您可以手动传递RequestContext作为第三个参数,但这在Django 1.10+中是不可能的。您使用RequestContext作为第二个参数是不正确的 - 第二个参数应该是常规字典。

最后,请注意,当表单无效时,您不会返回响应。您可以通过稍微调整代码来解决此问题:

if request.method == 'POST':
    form = RegistrationForm(request.POST)
    if form.is_valid():
        ...
        return HttpResponseRedirect('/register_success/')
else:
    form = RegistrationForm()
return render(request, 'registration/register.html', {'form': form})