HTTP / 1.0 301永久移动-Django

时间:2016-12-13 10:28:46

标签: django python-2.7 django-models django-views django-rest-framework

我使用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',)

4 个答案:

答案 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.

Documentation is here.

答案 3 :(得分:0)

https://www.django-rest-framework.org/api-guide/routers/

"默认情况下,SimpleRouter 创建的 URL 会附加一个斜杠。可以通过在实例化路由器时将 trailing_slash 参数设置为 False 来修改此行为。例如:

router = SimpleRouter(trailing_slash=False)