4

我尝试使用带有 angularjs 的 django-rest 构建的 rest API,但我遇到了问题。当我发送 DELETE 请求时,django-rest 看到 OPTIONS。

这就是我所做的:

我的 Django 视图.py

@api_view(['GET', 'PUT', 'DELETE'])
def node_detail(request, pk):
    """
    Retrieve, update or delete a node. 
    """
    try:
        node = Node.objects.get(pk=pk)
    except Node.DoesNotExist:
        return Response(status=status.HTTP_404_NOT_FOUND)

    if request.method == 'GET':
        serializer = NodeSerializer(node)
        return Response(serializer.data)

    elif request.method == 'PUT':
        data = JSONParser().parse(request)
        serializer = NodeSerializer(node, data=data)
        if serializer.is_valid():
            serializer.save()
            return Response(serializer.data)
        else:
            return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

    elif request.method == 'DELETE':
        node.delete()
        return Response(status=status.HTTP_204_NO_CONTENT)

我已经添加了带有 django-cors-headers 的 CORS,并且所有的 Origins 都被接受(CORS_ORIGIN_ALLOW_ALL = True)

我有一个角度服务来管理这个 $resource

angular.module('akilio.nodes.services', ['ngResource'])
.factory('NodeServices', function($resource){
    return $resource('http://127.0.0.1:8000/nodes/:nodeId/', {nodeId: '@nodeId'});
});

现在,使用 ng-click="deleteNode()" 的按钮,我称这个控制器为

controller('DetailCtrl', ['$scope', '$routeParams', 'NodeServices', function($scope, $routeParams, NodeServices) {
        $scope.node = NodeServices.get({nodeId: $routeParams.pk});

        $scope.deleteNode = function(){
            $scope.node.$delete(function(){
                console.log('ok');
            }, function(){
                console.log('nok');
            });
        };
    }]);

console.log() 每次都返回“nok”,在我的 django 控制台中,我看到了

[08/Nov/2013 09:39:10] "OPTIONS /nodes HTTP/1.1" 301 0

仅供参考,我的角度版本是 v1.2.0-rc.3 然后我添加到我的应用程序中

config(['$sceDelegateProvider', function($sceDelegateProvider) {
     $sceDelegateProvider.resourceUrlWhitelist(['self', 'http://127.0.0.1']);
 }])

但没有任何改变:(

一个主意 ?

编辑

我认为问题来自$resource,也许我在实施中错了。我试图用 $http 而不是 $resource 来删除一个节点

$http.delete('http://127.0.0.1:8000/nodes/4/');

一切都很好。

4

2 回答 2

0

您可能需要相应地调整您的 django 设置:

CORS_ALLOW_CREDENTIALS = True
CORS_ORIGIN_WHITELIST = ('127.0.0.1:8000', )

SESSION_COOKIE_DOMAIN = '127.0.0.1'
CSRF_COOKIE_DOMAIN = '127.0.0.1'

另外,如果您使用 csrf 令牌,请不要忘记:

app.run(function($http) {
    $http.defaults.headers.post['X-CSRFToken'] = $.cookie('csrftoken');
});
于 2013-11-08T12:14:26.067 回答
0

事实上,问题来自于 Django,它增加了一个试用空间。然后当 angular 尝试获取http://test/url但 django 将其重定向到http://test/url/ This redirect 意味着数据丢失。

我的解决方案是将此修改为 resource.js 文件。

其他解决方案是可能的,Django 端的 APPEND_SLASH 为 False

于 2013-11-14T14:04:28.237 回答