我使用djangorestframework创建了一个API,但是当我使用 python manage.py runserver 命令在终端中运行它时,它显示" HTTP / 1.0 301永久移动"错误。This the Message which it shows
但很酷的是,当我在浏览器中运行它时会显示数据。 This the result in Chrome
我想知道发生过这件事。
此时POST调用也无效。
以下是代码:
serializers.py
from rest_framework import serializers
from snippets.models import Snippet
class SnippetSerializer(serializers.ModelSerializer):
class Meta:
model = Snippet
fields = ('title','code',)
def create(self, validated_data):
return Snippet.objects.create(**validated_data)
def update(self, instance, validated_data):
instance.title = validated_data.get('title', instance.title)
instance.code = validated_data.get('code', instance.code)
instance.save()
return instance
Views.py
from django.shortcuts import render
from django.http import HttpResponse
from django.views.decorators.csrf import csrf_exempt
from rest_framework.renderers import JSONRenderer
from rest_framework.parsers import JSONParser
from snippets.models import Snippet
from snippets.serializers import SnippetSerializer
# Create your views here.
class JSONResponse(HttpResponse):
def __init__(self,data,**kwargs):
content = JSONRenderer().render(data)
kwargs['content_type']='application/json'
super(JSONResponse, self).__init__(content, **kwargs)
@csrf_exempt
def snippet_list(request):
"""
List all code snippets, or create a new snippet.
"""
if request.method == 'GET':
snippets = Snippet.objects.all()
serializer = SnippetSerializer(snippets, many=True)
return JSONResponse(serializer.data)
elif request.method == 'POST':
data = JSONParser().parse(request)
serializer = SnippetSerializer(data=data)
if serializer.is_valid():
serializer.save()
return JSONResponse(serializer.data, status=201)
return JSONResponse(serializer.errors, status=400)
@csrf_exempt
def snippet_detail(request, pk):
"""
Retrieve, update or delete a code snippet.
"""
try:
snippet = Snippet.objects.get(pk=pk)
except Snippet.DoesNotExist:
return HttpResponse(status=404)
if request.method == 'GET':
serializer = SnippetSerializer(snippet)
return JSONResponse(serializer.data)
elif request.method == 'PUT':
data = JSONParser().parse(request)
serializer = SnippetSerializer(snippet, data=data)
if serializer.is_valid():
serializer.save()
return JSONResponse(serializer.data)
return JSONResponse(serializer.errors, status=400)
elif request.method == 'DELETE':
snippet.delete()
return HttpResponse(status=204)
Models.py
from __future__ import unicode_literals
from django.db import models
# Create your models here.
class Snippet(models.Model):
created = models.DateTimeField(auto_now_add=True)
title = models.CharField(max_length=100, blank=True, default='')
code = models.TextField()
class Meta:
ordering = ('created',)
答案 0 :(得分:8)
301不是错误,而是重定向。从命令行,您请求的URL没有斜杠,但您的URL模式希望斜杠在那里。在这种情况下,Django有一个很好的convenience feature来返回重定向。
与命令行HTTP客户端不同,您的浏览器会识别重定向,然后再向第二个请求发送新URL,包括尾部斜杠。
答案 1 :(得分:1)
您看到的效果是因为您的网址已使用结尾/
进行映射,而您请求的链接没有结尾/
。 Django默认情况下会尝试结束/
的网址,如果它找不到与请求的网址匹配的内容。
这由APPEND_SLASH
设置控制,默认情况下设置为True
:
设置为True时,如果请求URL与任何一个都不匹配 URLconf中的模式并不以斜杠(HTTP)结尾 重定向发送到相同的URL,并附加斜杠。注意 重定向可能导致POST请求中提交的任何数据 丢失。
要解决此问题,请将命令更改为:
http http://127.0.0.1:8006/snippets/
答案 2 :(得分:0)
看起来您将APPEND_SLASH
设置为true。当您在没有斜杠的情况下转到端点时,它会自动将其重定向到带有斜杠的同一URL,从而产生301.
答案 3 :(得分:0)
https://www.django-rest-framework.org/api-guide/routers/
"默认情况下,SimpleRouter 创建的 URL 会附加一个斜杠。可以通过在实例化路由器时将 trailing_slash 参数设置为 False 来修改此行为。例如:
router = SimpleRouter(trailing_slash=False)