zoukankan      html  css  js  c++  java
  • Django REST framework 第四章 Authentication

    到目前为止,撰写的API没有任何限制关于谁能更新、删除snippet. 我们更想要一些高级行为来确保:

    1、代码段总是跟创建者有关联

    2、只要认证通过的用户才能创建

    3、只有创建者有权限更新或者删除

    4、没有认证的请求应该有且只有完全的只读权限

    Adding information to our model

    我们打算在Snippet模型类上做一些改变。首先,添加一些字段,其中之一用来代表创建这个code的用户。其他的字段将用于存储代码中突出显示的HTML表示形式。

    添加下面两个字段到Snippet类在models.py.

    owner = models.ForeignKey('auth.User', related_name='snippets', on_delete=models.CASCADE)
    highlighted = models.TextField()

    我们还需要确保当模型被保存时,我们使用pygments代码突出显示库填充突出显示的字段。此外还需要导入些其他的:

    from pygments.lexers import get_lexer_by_name
    from pygments.formatters.html import HtmlFormatter
    from pygments import highlight

    现在需要往model类里面添加一个save方法:

    def save(self, *args, **kwargs):
        """
        Use the `pygments` library to create a highlighted HTML
        representation of the code snippet.
        """
        lexer = get_lexer_by_name(self.language)
        linenos = 'table' if self.linenos else False
        options = {'title': self.title} if self.title else {}
        formatter = HtmlFormatter(style=self.style, linenos=linenos,
                                  full=True, **options)
        self.highlighted = highlight(self.code, lexer, formatter)
        super(Snippet, self).save(*args, **kwargs)

    做好上面的操作后,需要同步数据库

    shuais-MacBook-Pro:TestApp dandyzhang$ rm -f db.sqlite3
    shuais-MacBook-Pro:TestApp dandyzhang$ rm -r app01/migrations
    shuais-MacBook-Pro:TestApp dandyzhang$ python3 manage.py makemigrations
    shuais-MacBook-Pro:TestApp dandyzhang$ python3 manage.py migrate

    为了测试API,还需要创建一下超级用户:

    shuais-MacBook-Pro:TestApp dandyzhang$ python3 manage.py createsuperuser

    Adding endpoints for our User models

    现在我们已经有一些工作的用户了,还需要添加这些用户的表示到API中。

    在serializers.py文件中添加:

    from django.contrib.auth.models import User
    
    class UserSerializer(serializers.ModelSerializer):
        snippets = serializers.PrimaryKeyRelatedField(many=True, queryset=Snippet.objects.all())
    
        class Meta:
            model = User
            fields = ('id', 'username', 'snippets')

    snippets在 User model上是一种反转关系,在继承ModelSerializer类不能使用默认值。所以需要为它添加一个明确的字段

    同样,也需要在views.py文件内添加一些东西。我们更倾向于为用户表示使用只读权限,所以我们将使用ListAPIViewRetrieveAPIView generic CBV.

    from django.contrib.auth.models import User
    
    
    class UserList(generics.ListAPIView):
        queryset = User.objects.all()
        serializer_class = UserSerializer
    
    
    class UserDetail(generics.RetrieveAPIView):
        queryset = User.objects.all()
        serializer_class = UserSerializer

    确认下导入的UserSerializer

    from app01.serializers import UserSerializer

    最后,我们需要把这些视图添加进API,通过url设置,在urls.py文件中

        path('users/', views.UserList.as_view()),
        path('users/<int:pk>/', views.UserDetail.as_view()),

    Associating Snippets with Users

    现在,如果我们创建snippet,会发现没有办法通过snippet实例关联到创建的用户。用户不是作为序列化表示的一部分发送的,而是即将到来的请求的属性。

    解决这个问题的方式是重写一个.perform_create()方法在views里面,它允许我们修改管理实例如何保存,处理传入请求或者被请求URL的任何信息

    SnippetList视图类添加如下的方法:

    def perform_create(self, serializer):
        serializer.save(owner=self.request.user)

    一个额外添加的owner字段现在将会被serializer里面的create方法通过,跟请求里面的验证数据一起。

    Updating our serializer

    现在snippets和用户的关联建好了,更新一下SnippetSerializer来反应它。在serializers.py文件添加:

            owner = serializers.ReadOnlyField(source='owner.username')

    注意将它添加在Meta类内部的字段列表中.

    class SnippetSerializer(serializers.ModelSerializer):
        owner = serializers.ReadOnlyField(source='owner.username')
    
        class Meta:
            model = Snippet
            fields = ('id', 'title', 'code', 'linenos', 'language', 'style', 'owner')

    这个字段做了些很有趣的事情。source参数控制一种属性用来跟某个字段关联对应,可以指向序列化实例的任何属性。它也可以采用上面所显示的实例的表示法,将会遍历给定的属性,跟Django的模版语言很类似。

    我们刚刚添加的这个字段是一个非类型化的ReadOnlyField类,与其他类型字段相反,比如CharFieldBooleanField等...非类型话的ReadOnlyField总是只读的,并将用于序列化表示,但是在反序列化的时候不能被用来更新模型实例,也可以用CharField(read_only=True)

    Adding required permissions to views

    现在snippet已经跟用户有联系了,我们想确定只有通过验证的用户可以创建,更新,删除snippet。

    REST framework包含了一串权限类供用来限制谁能访问一个给定的视图。在这里,我们想要寻找的是IsAuthenticatedOrReadOnly这个类用来确保通过验证的请求获取到读写权限,没有通过验证的请求获得只读权限。

    首先在views里面导入模块

    from rest_framework import permissions

    然后添加下面的属性到SnippetList和SnippetDetail两个视图类中。

    permission_classes = (permissions.IsAuthenticatedOrReadOnly,)

    Adding login to Browable API

    如果你打开浏览器,导航到了可浏览的API,你会发现你不能再创建新的spnippet了。为了能这样做,需要先登陆。

    我们可以再根目录的urls文件内加入下面的路由,它包含可浏览的API的登陆和登出。

        path('api/', include('rest_framework.urls', namespace='rest_framework')),

    路由的名称可以自定义。

    现在打开浏览器,你会在网页的右上角看到login的图标。如果你用之前创建的用户登陆了,你就可以再一次创建snippet了。

    创建好一些snippet后,你就可以在子路由users下面查看到

    Object level permissions 

    我们想让所有的snippet被任何人看到,但必须保证只有创建的人才可以增删改。

    想要这么做,需要创建一个定制化的权限。创建一个新的文件在app内permissions.py

    from rest_framework import permissions
    
    
    class IsOwnerOrReadOnly(permissions.BasePermission):
        """
        Custom permission to only allow owners of an object to edit it.
        """
    
        def has_object_permission(self, request, view, obj):
            # Read permissions are allowed to any request,
            # so we'll always allow GET, HEAD or OPTIONS requests.
            if request.method in permissions.SAFE_METHODS:
                return True
    
            # Write permissions are only allowed to the owner of the snippet.
            return obj.owner == request.user

    现在可以添加自定义的权限到snippet实例终端,通过编辑在SnippetDetail视图类里面的permission_classes属性

    from app01.permissions import IsOwnerOrReadOnly  # 先导入
    
    
    permission_classes = (permissions.IsAuthenticatedOrReadOnly,
                          IsOwnerOrReadOnly,)

    现在重新运行一下此项目,你会发现delete和put按钮出现在你创建的一个snippet实例的终端。

    Authenticating with the API

    因为现在已经有一批权限在API上,如果需要编辑任何的snippet需要认证请求。但是我们还没有设置任何的认证类,所以此时是使用的默认的类SessionAuthenticationBasicAuthentication

    当我们通过浏览器跟API进行交互,我们可以登陆,然后浏览器的session会提供必要的认证给请求。如果我们以变成的方式跟API交互,就需要提供明确的认证凭据在每一次请求上。如果尝试创建一个新的snippet不带验证,将会得到报错:

    带上认证凭据:

  • 相关阅读:
    dpkg安装deb缺少依赖包的解决方法
    一个linux命令之grep---1
    win10快捷键
    Windows Server 2008 R2遗忘管理员密码后的解决方案
    手工释放linux内存
    oracle数据库用户加锁和解锁
    完全卸载Oracle数据库软件
    Linux上VNC 启动和关闭常见问题
    Linux 开启VNCSERVER
    RedHat 简易配置 VNC Server 与VNC View详细说明
  • 原文地址:https://www.cnblogs.com/wuzdandz/p/9460508.html
Copyright © 2011-2022 走看看