2013-03-09 79 views
1

對此感到沮喪了幾天,並想知道是否有人可以提供幫助。我是新來的Ajax,並嘗試從登錄表單發佈數據以便登錄到Django(Django Userena)。但是,當我嘗試發佈數據時,alert()將數據中的錯誤顯示爲[對象對象]。我甚至無法看到網絡錯誤代碼,因爲POST在運行之前被取消。將Ajax POST表單發送到Django數據問題

有沒有辦法發送數據,而不是我認爲它發送的整個JSON數組,還是我需要在Django視圖後端解析它?明智地完成這個代碼的最好方法是什麼?非常感謝你!

下面是相關代碼:

Ajax代碼:對的Index.html

$('#login').submit(function(){ 

    $.ajax({ 

     url: 'http://127.0.0.1:8000/accounts/signin/', 
     type: 'POST', 
     data: { 
      csrfmiddlewaretoken: '{{csrf_token}}', 
      identification: $("#id_identification").val(), 
      password: $("#id_password").val(), 
      }, 
     success: function() { 
      alert('Test'); 
      $('#datadisplay').append("<h2>It worked</h2>"); 
     }, 
     error: function(errorThrown){ 
      console.log(errorThrown); 
      alert('Error'); 
      alert(errorThrown); 
     } 
    }); 
}); 

<form name="login" id="login" action=""> 
    <fieldset> 
      <label for="id_identification">Email or username</label> 
      <input class="required" id="id_identification" maxlength="75" name="identification" type="text" /> 

      <label for="id_password">Password</label> 
      <input class="required" id="id_password" name="password" type="password" /> 

      <input type="submit" name="submit" class="loginbutton" value="Login" /> 
     </fieldset> 
    </form> 

從Django的Userena

views.py
@secure_required 
def signin(request, auth_form=AuthenticationForm, 
      template_name='userena/signin_form.html', 
      redirect_field_name=REDIRECT_FIELD_NAME, 
      redirect_signin_function=signin_redirect, extra_context=None): 
    """ 
    Signin using email or username with password. 

    Signs a user in by combining email/username with password. If the 
    combination is correct and the user :func:`is_active` the 
    :func:`redirect_signin_function` is called with the arguments 
    ``REDIRECT_FIELD_NAME`` and an instance of the :class:`User` who is is 
    trying the login. The returned value of the function will be the URL that 
    is redirected to. 

    A user can also select to be remembered for ``USERENA_REMEMBER_DAYS``. 

    :param auth_form: 
     Form to use for signing the user in. Defaults to the 
     :class:`AuthenticationForm` supplied by userena. 

    :param template_name: 
     String defining the name of the template to use. Defaults to 
     ``userena/signin_form.html``. 

    :param redirect_field_name: 
     Form field name which contains the value for a redirect to the 
     succeeding page. Defaults to ``next`` and is set in 
     ``REDIRECT_FIELD_NAME`` setting. 

    :param redirect_signin_function: 
     Function which handles the redirect. This functions gets the value of 
     ``REDIRECT_FIELD_NAME`` and the :class:`User` who has logged in. It 
     must return a string which specifies the URI to redirect to. 

    :param extra_context: 
     A dictionary containing extra variables that should be passed to the 
     rendered template. The ``form`` key is always the ``auth_form``. 

    **Context** 

    ``form`` 
     Form used for authentication supplied by ``auth_form``. 

    """ 
    form = auth_form() 

    if request.method == 'POST': 
     form = auth_form(request.POST, request.FILES) 
     if form.is_valid(): 
      #identification, password, remember_me = (form.cleaned_data['identification'], 
                #form.cleaned_data['password'], 
                #form.cleaned_data['remember_me']) 
      identification, password = (form.cleaned_data['identification'], form.cleaned_data['password'])  

      user = authenticate(identification=identification, 
           password=password) 
      if user.is_active: 
       login(request, user) 
       if remember_me: 
        request.session.set_expiry(userena_settings.USERENA_REMEMBER_ME_DAYS[1] * 86400) 
       else: request.session.set_expiry(0) 

       if userena_settings.USERENA_USE_MESSAGES: 
        messages.success(request, _('You have been signed in.'), 
            fail_silently=True) 

       # Whereto now? 
       redirect_to = redirect_signin_function(
        request.REQUEST.get(redirect_field_name), user) 
       return HttpResponseRedirect(redirect_to) 
      else: 
       return redirect(reverse('userena_disabled', 
             kwargs={'username': user.username})) 

    if not extra_context: extra_context = dict() 
    extra_context.update({ 
     'form': form, 
     'next': request.REQUEST.get(redirect_field_name), 
    }) 
    return ExtraContextTemplateView.as_view(template_name=template_name, 
              extra_context=extra_context)(request) 

AuthenticationForm

class AuthenticationForm(forms.Form): 
    """ 
    A custom form where the identification can be a e-mail address or username. 

    """ 
    identification = identification_field_factory(_(u"Email or username"), 
                _(u"Either supply us with your email or username.")) 
    password = forms.CharField(label=_("Password"), 
           widget=forms.PasswordInput(attrs=attrs_dict, render_value=False)) 
    remember_me = forms.BooleanField(widget=forms.CheckboxInput(), 
            required=False, 
            label=_(u'Remember me for %(days)s') % {'days': _(userena_settings.USERENA_REMEMBER_ME_DAYS[0])}) 

    def __init__(self, *args, **kwargs): 
     """ A custom init because we need to change the label if no usernames is used """ 
     super(AuthenticationForm, self).__init__(*args, **kwargs) 
     # Dirty hack, somehow the label doesn't get translated without declaring 
     # it again here. 
     self.fields['remember_me'].label = _(u'Remember me for %(days)s') % {'days': _(userena_settings.USERENA_REMEMBER_ME_DAYS[0])} 
     if userena_settings.USERENA_WITHOUT_USERNAMES: 
      self.fields['identification'] = identification_field_factory(_(u"Email"), 
                     _(u"Please supply your email.")) 

    def clean(self): 
     """ 
     Checks for the identification and password. 

     If the combination can't be found will raise an invalid sign in error. 

     """ 
     identification = self.cleaned_data.get('identification') 
     password = self.cleaned_data.get('password') 

     if identification and password: 
      user = authenticate(identification=identification, password=password) 
      if user is None: 
       raise forms.ValidationError(_(u"Please enter a correct username or email and password. Note that both fields are case-sensitive.")) 
     return self.cleaned_data 
+1

我沒有看到任何代碼易受POST POST JSON數組影響。 AJAX請求的響應代碼是什麼?也許'{{csrf_token}}'是空的? – jpic 2013-03-09 23:01:58

+0

您可以嘗試使用@csrf_exempt臨時刪除csrf驗證,以查看是否還有其他問題 – 2013-03-09 23:07:12

+0

感謝您的回覆。我無法獲取響應代碼,因爲它在完成之前出錯。上面代碼的控制檯打印出「Object {readyState:0,getResponseHeader:function,getAllResponseHeaders:function,set RequestHeader:function,overrideMimeType:function ...}」,但是一旦我關閉了最後一個對話框,它就會退出並清除控制檯。 \t 當窗體不是通過Django顯示而是外部html窗體顯示時,如何將csrf_token放入窗體中? – Jack 2013-03-09 23:08:39

回答

1

這些都是錯誤的(不是全部,我想有更多的)我看到,當我測試你的代碼:

  1. 你怎麼可以登錄,如果你沒有在您的模板中使用您的AuthenticationForm()。你在表格中的代碼是不同的。
  2. 您沒有在您的表單中放入method =「POST」,這會導致GET響應而不是POST。
  3. 你忘了把{%csrf_token%}在你的模板形式
  4. 在你views.py,用戶認證=(標識=識別,密碼=密碼)必須用戶認證=(用戶名=識別,密碼=密碼)
  5. 在你forms.py,用戶認證=(標識=識別,密碼=密碼)必須用戶認證=(用戶名=識別,密碼=密碼)
  6. 你的AJAX代碼沒有效果,這就是爲什麼你可以沒有得到拋出的錯誤。我認爲你的ajax中缺少一些東西。
  7. 要獲得拋出的錯誤的詳細信息:

    error: function(ts){ 
        //console.log(errorThrown); 
        //alert('Error'); 
        alert(ts.responseText); 
        } 
    
  8. 你的Ajax數據必須':

    data: { 
        'csrfmiddlewaretoken': '{{csrf_token}}', 
        'identification': $("#id_identification").val(), 
        'password': $("#id_password").val(), 
        }, 
    
  9. 你不必詳細闡述您的網址像這樣的

    url: 'http://127.0.0.1:8000/accounts/signin/', 
    

    這將導致

    "http://127.0.0.1:8000/http://127.0.0.1:8000/accounts/signin/" 
    

    它必須是

    url: '/accounts/signin/', 
    

    ,不也忘了把

    contentType: "application/json;charset=utf-8", 
    dataType: "json", 
    

有更多的錯誤,到現在爲止我無法成功登錄。

+0

非常感謝您的幫助。我正在處理這個問題,現在發佈POST沒有錯誤,但我認爲它沒有做任何事情,頁面只是引用了沒有發送的數據。我的目標是用Django作爲後端製作移動應用程序。隨着PhoneGap的前端只需要HTML/CSS/JS ** **,所以沒有Django模板和Django標籤沒有意​​義。你知道是否有一個更簡單的解決方案來實現我的目標,即讓用戶登錄/註冊?我認爲TastyPie可能是最好的選擇,但是我嘗試了這一點,但是也找不到它。任何見解都會非常有幫助。謝謝! – Jack 2013-03-10 03:34:36

+0

我不知道任何移動設備。我也從來沒有使用TastyPie。如果我能得到一個信息,我會馬上告訴你:) – catherine 2013-03-10 07:56:10