How can I use Django permissions without defining a content type or model?

后端 未结 6 1101
小鲜肉
小鲜肉 2021-01-29 18:56

I\'d like to use a permissions based system to restrict certain actions within my Django application. These actions need not be related to a particular model (e.g. access to sec

相关标签:
6条回答
  • 2021-01-29 19:02

    Following Gonzalo's advice, I used a proxy model and a custom manager to handle my "modelless" permissions with a dummy content type.

    from django.db import models
    from django.contrib.auth.models import Permission
    from django.contrib.contenttypes.models import ContentType
    
    
    class GlobalPermissionManager(models.Manager):
        def get_query_set(self):
            return super(GlobalPermissionManager, self).\
                get_query_set().filter(content_type__name='global_permission')
    
    
    class GlobalPermission(Permission):
        """A global permission, not attached to a model"""
    
        objects = GlobalPermissionManager()
    
        class Meta:
            proxy = True
    
        def save(self, *args, **kwargs):
            ct, created = ContentType.objects.get_or_create(
                name="global_permission", app_label=self._meta.app_label
            )
            self.content_type = ct
            super(GlobalPermission, self).save(*args, **kwargs)
    
    0 讨论(0)
  • 2021-01-29 19:03

    This is alternative solution. First ask yourself: Why not create a Dummy-Model which really exists in DB but never ever gets used, except for holding permissions? That's not nice, but I think it is valid and straight forward solution.

    from django.db import models
    
    class Permissions(models.Model):
    
        can_search_blue_flower = 'my_app.can_search_blue_flower'
    
        class Meta:
            permissions = [
                ('can_search_blue_flower', 'Allowed to search for the blue flower'),
            ]
    

    Above solution has the benefit, that you can use the variable Permissions.can_search_blue_flower in your source code instead of using the literal string "my_app.can_search_blue_flower". This means less typos and more autocomplete in IDE.

    0 讨论(0)
  • 2021-01-29 19:06

    For those of you, who are still searching:

    You can create an auxiliary model with no database table. That model can bring to your project any permission you need. There is no need to deal with ContentType or create Permission objects explicitly.

    from django.db import models
            
    class RightsSupport(models.Model):
                
        class Meta:
            
            managed = False  # No database table creation or deletion  \
                             # operations will be performed for this model. 
                    
            default_permissions = () # disable "add", "change", "delete"
                                     # and "view" default permissions
    
            permissions = ( 
                ('customer_rights', 'Global customer rights'),  
                ('vendor_rights', 'Global vendor rights'), 
                ('any_rights', 'Global any rights'), 
            )
    

    Right after manage.py makemigrations and manage.py migrate you can use these permissions like any other.

    # Decorator
    
    @permission_required('app.customer_rights')
    def my_search_view(request):
        …
    
    # Inside a view
    
    def my_search_view(request):
        request.user.has_perm('app.customer_rights')
    
    # In a template
    # The currently logged-in user’s permissions are stored in the template variable {{ perms }}
    
    {% if perms.app.customer_rights %}
        <p>You can do any customer stuff</p>
    {% endif %}
    
    0 讨论(0)
  • 2021-01-29 19:21

    Fix for Chewie's answer in Django 1.8, which as been requested in a few comments.

    It says in the release notes:

    The name field of django.contrib.contenttypes.models.ContentType has been removed by a migration and replaced by a property. That means it’s not possible to query or filter a ContentType by this field any longer.

    So it's the 'name' in reference in ContentType that the uses not in GlobalPermissions.

    When I fix it I get the following:

    from django.db import models
    from django.contrib.auth.models import Permission
    from django.contrib.contenttypes.models import ContentType
    
    
    class GlobalPermissionManager(models.Manager):
        def get_queryset(self):
            return super(GlobalPermissionManager, self).\
                get_queryset().filter(content_type__model='global_permission')
    
    
    class GlobalPermission(Permission):
        """A global permission, not attached to a model"""
    
        objects = GlobalPermissionManager()
    
        class Meta:
            proxy = True
            verbose_name = "global_permission"
    
        def save(self, *args, **kwargs):
            ct, created = ContentType.objects.get_or_create(
                model=self._meta.verbose_name, app_label=self._meta.app_label,
            )
            self.content_type = ct
            super(GlobalPermission, self).save(*args)
    

    The GlobalPermissionManager class is unchanged but included for completeness.

    0 讨论(0)
  • 2021-01-29 19:26

    Django's Permission model requires a ContentType instance.

    I think one way around it is creating a dummy ContentType that isn't related to any model (the app_label and model fields can be set to any string value).

    If you want it all clean and nice, you can create a Permission proxy model that handles all the ugly details of the dummy ContentType and creates "modelless" permission instances. You can also add a custom manager that filters out all Permission instances related to real models.

    0 讨论(0)
  • 2021-01-29 19:26

    You can use the proxy model for this with a dummy content type.

    from django.contrib.auth.models import Permission
    from django.contrib.contenttypes.models import ContentType
    
    
    class CustomPermission(Permission):
    
        class Meta:
            proxy = True
    
        def save(self, *args, **kwargs):
            ct, created = ContentType.objects.get_or_create(
                model=self._meta.verbose_name, app_label=self._meta.app_label,
            )
            self.content_type = ct
            super(CustomPermission, self).save(*args)
    

    Now you can create the permission with just name and codename of the permission from the CustomPermission model.

     CustomPermission.objects.create(name='Can do something', codename='can_do_something')
    

    And you can query and display only the custom permissions in your templates like this.

     CustomPermission.objects.filter(content_type__model='custom permission')
    
    0 讨论(0)
提交回复
热议问题