#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)
), также отменяется. Это ошибка.