国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

使用django-guardian實現django-admin的行級權限控制的方法

2020-02-15 23:28:50
字體:
來源:轉載
供稿:網友

用django框架來做一些后臺管理的web頁面簡直太方便了,django自帶模塊級的權限系統,用來做一些內部的系統非常合適,可以大大的減少開發量。但是django自帶的權限系統還不支持行級的權限控制,如果要實現行級的權限控制,需要依賴第三方的app來開發,或者自己重新寫一個。

需求描述

我們項目組開發的一些系統通常會用mysql數據庫來存儲一些配置,但是如果每次有配置修改的時候都去手動修改mysql數據的話,會挺麻煩的,同時也比較容易出錯。django-admin能夠根據定義的model自動的生成相應的頁面,同時還能提供權限的管理,所以我們就把一些系統到的配置放到django中。但是到現在,隨著系統的需求越來越多,該系統已經不止我們自己項目組的人員使用,也要開放給其他項目組的同事使用,所以就產生了一些更細粒度的權限需求。因此,我們要在現有的系統上支持行級的權限控制。

解決方案

當然可以自己寫一套權限系統了,但是自己寫的成本比較高,而且自己寫的不一定比較好。所以我就先在網上找了一些現成的解決方案, https://djangopackages.org/grids/g/perms/ 該鏈接列出了現有的一些第三方的權限系統解決方案。從該頁面來看,django-guardian是最受歡迎的第三方權限系統,而且支持行級的權限系統,同時還可以整合到django-admin里面,所以我就選擇了django-guardian。

關鍵步驟

安裝配置django-guardian

安裝配置django-guardian比較簡單,按照她項目提供的 文檔 進行安裝就可以了,安裝完成后會在數據庫里面創建兩張權限相關的表。

把django-guardian整合到django-admin

首先把admin.py文件里面需要用到行級權限的類,由原來的繼承admin.ModelAdmin,改成繼承GuardedModelAdmin,這時候打開某個數據行的頁面的時候,在該頁面的右上角的歷史旁邊會顯示編輯對象權限的按鈕,點擊該按鈕進去相應的頁面就可以編輯該行數據的具體權限。

配置完權限的時候,用一個新的用戶測試的話,會發現該用戶沒有權限來訪問任何的數據,這是因為GuardedModelAdmin還有很多事情沒有幫我們做,我們還需要重寫一些函數來實現admin后臺頁面的顯示。具體的信息看下面的代碼注釋。

from guardian.admin import GuardedModelAdminfrom guardian.shortcuts import get_objects_for_user, assign_perm, remove_perm, get_users_with_perms, /  get_groups_with_perms  # 需改前@admin.register(DataAssistantJob)class DataAssistantJobAdmin(admin.ModelAdmin):  pass# 修改后@admin.register(DataAssistantJob)class DataAssistantJobAdmin(GuardedModelAdmin):  # app是否在主頁面中顯示的話由該函數決定  def has_module_permission(self, request):    if super().has_module_permission(request):      return True    return self.get_model_objs(request).exists()  # 在顯示數據列表額時候,哪些數據顯示,哪些不顯示,由該函數控制  def get_queryset(self, request):    if request.user.is_superuser:      return super().get_queryset(request)    data = self.get_model_objs(request)    return data      # 內部用來獲取某個用戶有權限訪問的數據行  def get_model_objs(self, request, action=None, klass=None):    opts = self.opts    actions = [action] if action else ['view', 'change', 'delete']    klass = klass if klass else opts.model    model_name = klass._meta.model_name    return get_objects_for_user(user=request.user, perms=[f'{perm}_{model_name}' for perm in actions],                  klass=klass, any_perm=True)  # 用來判斷某個用戶是否有某個數據行的權限  def has_perm(self, request, obj, action):    opts = self.opts    codename = f'{action}_{opts.model_name}'    if obj:      return request.user.has_perm(f'{opts.app_label}.{codename}', obj)    else:      return self.get_model_objs(request, action).exists()  # 是否有查看某個數據行的權限  def has_view_permission(self, request, obj=None):    return self.has_perm(request, obj, 'view')  # 是否有修改某個數據行的權限  def has_change_permission(self, request, obj=None):    return self.has_perm(request, obj, 'change')  # 是否有刪除某個數據行的權限  def has_delete_permission(self, request, obj=None):    return self.has_perm(request, obj, 'delete')  # 用戶應該擁有他新增的數據行的所有權限  def save_model(self, request, obj, form, change):    result = super().save_model(request, obj, form, change)    if not request.user.is_superuser and not change:      opts = self.opts      actions = ['view', 'add', 'change', 'delete']      [assign_perm(f'{opts.app_label}.{action}_{opts.model_name}', request.user, obj) for action in actions]    return result            
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 偃师市| 宜兴市| 特克斯县| 靖安县| 花莲市| 曲水县| 二手房| 夏邑县| 疏附县| 吉木萨尔县| 揭西县| 宣汉县| 北碚区| 鄱阳县| 隆林| 都江堰市| 南漳县| 定安县| 阳城县| 华安县| 榆林市| 德庆县| 平利县| 济南市| 介休市| 定日县| 怀集县| 新宾| 苏尼特左旗| 邯郸市| 易门县| 桐梓县| 邵武市| 广安市| 永嘉县| 安庆市| 资溪县| 绥阳县| 余干县| 清镇市| 徐水县|