Ошибка формы Django: объект ‘QueryDict’ не имеет атрибута ‘метод’ при назначении POST

#python #django #forms

#python #django #формы

Вопрос:

У меня возникли проблемы с конкретной формой, которая не позволяет мне отправлять данные в функцию, которую я определил. Я очень смущен тем, почему это выдает мне ошибки, потому что я использую почти идентичную форму для выполнения другого действия на том же веб-сайте.

Когда я отправляю эти данные, Django выдает «объект’QueryDict’ не имеет атрибута ‘method'», хотя я назначил метод как POST. Он отлично отображает форму, когда она не публикуется, только когда я отправляю, у меня проблема.

Ошибка, которую выдает Django, указывает на эту строку в представлениях: if request.method == «POST»: это первая строка в функции «отмена» ниже.

Кто-нибудь может определить, что я здесь сделал не так? Я в недоумении.

Вот функция в view.py:

 from django.shortcuts import render
from django.http import HttpResponseRedirect
from django.shortcuts import redirect
from .apps.instant import orderlist, orderdetails, cancel
from .forms import cancelorder

def cancel(request):
    if request.method == "POST":
        form = cancel(request.POST or None)
        if form.is_valid():
            response = cancel(request.session['token'],form.cleaned_data['orderId'], form.cleaned_data['reason'])
            return render(request, 'instant/review.html', {'data':response})
        else:
            form = cancelorder()
            return render(request, 'instant/cancel.html', {'form':form})
    else:
        form = cancelorder()
        return render(request, 'instant/cancel.html', {'form':form})
  

вот forms.py:

 from django import forms

class cancelorder(forms.Form):
    orderId = forms.CharField(label='Order Id', max_length=5)
    reason = forms.CharField(label='reason', widget=forms.Textarea)
  

вот шаблон, который он отображает (instant/cancel.html ):

 {% extends "main/header.html" %}
{% load widget_tweaks %}
{% block content %}
<div class = "row">
  <div class ="col-sm-3">
  </div>
  <div class ="col-sm-6">
    <div class="alert alert-danger alert-dismissible" role="alert">
      <button type="button" class="close" data-dismiss="alert" aria-label="Close"><span aria-hidden="true">amp;times;</span></button>
      <strong>Warning!</strong> When you cancel, the host gets an email notifiaction! Also, there are no confirmations. Only click cancel if you're sure!
    </div>
  </div>
</div>
<div class = "row">
  <div class ="col-sm-3">
  </div>
  <div class ="col-sm-6">
    <div class="well-lg" style="background-color:#efefef">
      <center>
      <h3> cancel an order</h3>
    </center><br>
      <!-- <h3>Login</h3> -->
      <form action="/instant/cancel/" method="POST" class="post-form">{% csrf_token %}
        <div class="form-group">
          <label for="login_email">Order Id</label>
          {{ form.orderId|add_class:"form-control"|attr:"placeholder:Enter numbers only"}}
        </div>
        <div class="form-group">
          <label for="login_password">Reason</label>
          {{ form.reason|add_class:"form-control"|attr:"placeholder:Why are you cancelling?"}}
        </div>
        <button type="submit" class="btn btn-danger btn-lg">Send Cancellation</button>
      </form>
    </div>
  </div>
</div>
{% endblock %}
  

И, наконец, вот функция, для которой форма собирает данные. .apps.instant (я знаю, что API, который я вызываю, работает):

 from time import sleep
import requests
import logging
import json

LOG_FILENAME = 'transaction_log.log' #Production Log
logging.basicConfig(filename=LOG_FILENAME,level=logging.INFO,format='%(asctime)s %(message)s')

url = APIURL (removed for stackoverflow)

def cancel(token,orderId,reason):
    auth_token = {"Authorization" : "bearer "   str(token)}
    raw = {'instabookingId':orderId,"reason":reason}
    info = json.dumps(raw)
    cancellation = requests.post(url, headers=auth_token,data=info)
    response = cancellation.json()
    return response
  

Я ценю любую помощь, которую вы можете предложить, действительно хотел бы знать, почему это не работает.

вот ошибка для Django:

 Environment:


Request Method: POST
Request URL: http://164.132.48.154:8000/instant/cancel/

Django Version: 1.10.2
Python Version: 2.7.9
Installed Applications:
['main',
 'instant',
 'widget_tweaks',
 'django.contrib.admin',
 'django.contrib.auth',
 'django.contrib.contenttypes',
 'django.contrib.sessions',
 'django.contrib.messages',
 'django.contrib.staticfiles']
Installed Middleware:
['django.middleware.security.SecurityMiddleware',
 'django.contrib.sessions.middleware.SessionMiddleware',
 'django.middleware.common.CommonMiddleware',
 'django.middleware.csrf.CsrfViewMiddleware',
 'django.contrib.auth.middleware.AuthenticationMiddleware',
 'django.contrib.messages.middleware.MessageMiddleware',
 'django.middleware.clickjacking.XFrameOptionsMiddleware']



Traceback:

File "/usr/local/lib/python2.7/dist-packages/django/core/handlers/exception.py" in inner
  39.             response = get_response(request)

File "/usr/local/lib/python2.7/dist-packages/django/core/handlers/base.py" in _get_response
  187.                 response = self.process_exception_by_middleware(e, request)

File "/usr/local/lib/python2.7/dist-packages/django/core/handlers/base.py" in _get_response
  185.                 response = wrapped_callback(request, *callback_args, **callback_kwargs)

File "/root/mysite/instant/views.py" in cancel
  24.         form = cancel(request.POST or None)

File "/root/mysite/instant/views.py" in cancel
  23.     if request.method == "POST":

Exception Type: AttributeError at /instant/cancel/
Exception Value: 'QueryDict' object has no attribute 'method'
  

Комментарии:

1. «Ошибка, которую выдает Django» почему бы вам не поделиться этим с нами. Пожалуйста, выполните полную трассировку стека

2. @e4c5 Спасибо, это хороший момент. Я добавил это выше.

3. У вас есть view, а также какой-либо другой метод импорта с тем же именем, что и cancel

4. @Rohan Да, я вижу это сейчас, иногда дополнительные глаза помогают вам слишком долго смотреть на одно и то же. Я ценю помощь.

Ответ №1:

Ошибка находится в этой строке

 ...
def cancel(request):
    if request.method == "POST":
        form = cancel(request.POST or None) # you call the function, its the same name, you would call your cancelform
        if form.is_valid():
            response = cancel(request.session['token'],form.cleaned_data['orderId'], form.cleaned_data['reason'])
            return render(request, 'instant/review.html', {'data':response})
        else:
           ...
  

выполняется рекурсивно, во второй раз, в функции cancel, request == запрос.POST и запрос.POST не имеет атрибута ‘метод’

Комментарии:

1. Спасибо, иногда требуется просто посторонний взгляд, чтобы уловить эти вещи. Это доставляло мне настоящую головную боль.

Ответ №2:

 def cancel(request):
    if request.method == "POST":
        form = cancel(request.POST or None) # you call the function, its the same name, you would call your cancelform
        if form.is_valid():
            response = cancel(request.session['token'],form.cleaned_data['orderId'], form.cleaned_data['reason'])
            return render(request, 'instant/review.html', {'data':response})
        else:
  

В этом случае имя функции в представлениях равно cancel. Кроме того, имя формы, которое упоминается в 3-й строке ( cancel(request.POST or None) ), также отменяется. Это ошибка.