Django возвращает redirect () с параметрами - PullRequest
69 голосов
/ 09 июля 2010

В моем представлении я хочу вызвать другое представление и передать ему данные:

return redirect('some-view-name', backend, form.cleaned_data)

, где backend представляет собой объект registration.backends, а form.cleaned_data представляет собой данные данных формы (но оба должны быть отправлены как * args или ** kwargs, чтобы избежать появления ошибки Don't mix *args and **kwargs in call to reverse()!) Из того, что я нашел в документах:

def my_view(request):
    ...
    return redirect('some-view-name', foo='bar')

Похоже, мне нужно предоставить аргумент "some-view-name", но это просто имя функции представления или имя url? Поэтому я хотел бы сделать его похожим на способ регистрации django, где:

to, args, kwargs = backend.post_registration_redirect(request, new_user)
return redirect(to, *args, **kwargs)

def post_registration_redirect(self, request, user):
    return ('registration_complete', (), {})

Хорошо, теперь я могу напрямую вызвать функцию просмотра или мне нужно указать ее URL? И что более важно, как должен выглядеть мой вызов funciotn (и URL-адрес при необходимости)? И backend, и cleaned_data просто передаются через это представление для дальнейшего использования. Я пробовал это, но это неправильно:

url(r'^link/$', some-view-name)   
def some-view-name(request, *args):

А также это:

return redirect('some_url', backend=backend, dataform.cleaned_data) 
url(r'^link/$', some-view-name)    
def some-view-name(request, backend, data):

по-прежнему NoReverseMatch. Но в django-регистрации я видел что-то вроде этого:

url(r'^register/$',register,{'backend': 'registration.backends.default.DefaultBackend'}, name='registration_register'),

def register(request, backend, success_url=None, form_class=None,
             disallowed_url='registration_disallowed',
             template_name='user/login_logout_register/registration_form.html',
             extra_context=None):

Ответы [ 2 ]

57 голосов
/ 09 июля 2010

Во-первых, ваше определение URL вообще не принимает никаких параметров.Если вы хотите, чтобы параметры передавались из URL-адреса в представление, вам нужно определить их в urlconf.

Во-вторых, совсем не ясно, что вы ожидаете от словаря cleaned_data.Не забывайте, что вы не можете перенаправить на POST - это ограничение HTTP, а не Django - поэтому ваши cleaned_data должны быть либо параметром URL (ужасно), либо, что немного лучше, серией параметров GET - так что URLбудет в форме:

/link/mybackend/?field1=value1&field2=value2&field3=value3

и так далее.В этом случае field1, field2 и field3 не включены в определение URLconf - они доступны в представлении через request.GET.

Таким образом, ваш urlconf будет:

url(r'^link/(?P<backend>\w+?)/$', my_function)

и вид будет выглядеть следующим образом:

def my_function(request, backend):
   data = request.GET

, и обратное будет (после импорта urllib):

return "%s?%s" % (redirect('my_function', args=(backend,)),
                  urllib.urlencode(form.cleaned_data))

Отредактировано после комментария

Весь смысл использования перенаправления и обратного хода, как вы это делали, заключается в том, что вы переходите на URL-адрес - он возвращает код Http, который заставляет браузер перенаправить на новый URL-адрес, и вызывать его.

Если вы просто хотите вызвать представление из своего кода, просто сделайте это напрямую - вообще не нужно использовать реверс.

При этом, если все, что вы хотите сделать, это сохранитьданные, а затем просто положить его в сеансе:

request.session['temp_data'] = form.cleaned_data
43 голосов
/ 29 мая 2014

urls.py:

#...    
url(r'element/update/(?P<pk>\d+)/$', 'element.views.element_update', name='element_update'),

views.py:

from django.shortcuts import redirect
from .models import Element


def element_info(request):
    # ...
    element = Element.object.get(pk=1)
    return redirect('element_update', pk=element.id)

def element_update(request, pk)
    # ...
...