rails基于devise+cancancan+rolify的簡(jiǎn)單權(quán)限控制實(shí)現(xiàn)

NO.1 引入cancancan和rolify

gem 'cancancan', '~>1.10'
gem "rolify"
bundle
生成ability模型

rails g cancan:ability
rails g rolify Role Admin

Admin為之前devise用登錄模型

NO.2 在需要進(jìn)行權(quán)限驗(yàn)證的控制器里里加上load_and_authorize_resource,例如:

class Admins::AdminsController < ApplicationController
  load_and_authorize_resource

這樣就可以對(duì)整個(gè)控制器的所有動(dòng)作進(jìn)行驗(yàn)證,而不需要手動(dòng)的對(duì)每個(gè)動(dòng)作進(jìn)行驗(yàn)證
運(yùn)行項(xiàng)目后訪問admins控制器后,報(bào)錯(cuò)

undefined local variable or method `current_user' for #<Admins::AdminsController:0x000000060c9e40>

因?yàn)槲业捻?xiàng)目中用登錄注冊(cè)的模型為admin,所以current_user這個(gè)方法是不存在的,需要把他改掉
在ApplicationController中,將current_user 替換為current_admin:

alias_method :current_user, :current_admin

刷新后,依然報(bào)錯(cuò):

ArgumentError in Admins::AdminsController#index

這是rolify自帶的一個(gè)bug,暫未修復(fù),解決辦法是將:

belongs_to :resource,
           :polymorphic => true,
           :optional => true

改為

belongs_to :resource,
           :polymorphic => true

NO.3 rolify的主要作用是用來關(guān)聯(lián)用戶和角色的。

它封裝了2個(gè)主要的方法:
add_role :admin,用來添加角色;
has_role? :admin, 判斷是否有這個(gè)角色;

在新建用戶的是時(shí)候,將角色和用戶關(guān)聯(lián)起來,params[:role]為頁(yè)面上傳過來的參數(shù)

  def create
    admin = Admin.new(admin_params)
    if admin.save
      admin.add_role params[:role]
      respond_to do |format|
        format.html
        format.js
      end
    else
      render 'new'
    end
  end

角色和用戶關(guān)聯(lián)成功。

NO.4 接下來就是在views里面和ablity.rb里面設(shè)置具體權(quán)限。

例如:如果有創(chuàng)建服務(wù)的權(quán)限,在服務(wù)模板的視圖中做如下控制:

<ul>
<% if can? :create, Service %>
  <li><%= link_to '添加服務(wù)',  new_admins_service_path %></li>
<% end %>
  <li><%= link_to '服務(wù)列表',  admins_services_path %></li>
</ul>

在ablity.rb中,做如下控制:

class Ability
  include CanCan::Ability
  def initialize(user)
    if user.blank?
      cannot :manage, :all
    ####超級(jí)管理員權(quán)限####
    elsif user.has_role? :admin
      can :manage, :all
    ####店長(zhǎng)權(quán)限####
    elsif user.has_role? "店長(zhǎng)"
      #####顯示屬于自己的店鋪#####
      can :show, Shop do |shop|
      (shop.admin_id == user.id)
      end
      #####顯示屬于自己店鋪列表#####
      can :my_shop, Shop do |shop|
      (shop.admin_id == user.id)
      end
      ####編輯修改屬于自己的店鋪####
      can :update, Shop do |shop|
      (shop.admin_id == user.id)
      end
      #######切換店鋪開關(guān)狀態(tài)#######
      can :switch, Shop do |shop|
      (shop.admin_id == user.id)
      end
      ######管理所有本店鋪的理發(fā)師######
      can :manage, Barber
      ####管理本店鋪的服務(wù)####
      can :manage, ShopService
      ####查看所有的服務(wù)模板####
      can :read, Service
      ###看到屬于自己的訂單統(tǒng)計(jì)###
      can :stat, Order
    end
  end
end

所以,當(dāng)角色為店長(zhǎng)的用戶登錄的時(shí)候,是視圖中,是看不見"添加服務(wù)"這個(gè)選項(xiàng)的,因?yàn)?code>can :read, Service,而視圖中需要?jiǎng)?chuàng)建權(quán)限才能看到。

NO.5 當(dāng)控制器中的某個(gè)action沒有對(duì)應(yīng)的具體model時(shí),該如何處理

在控制中加入load_and_authorize_resource的時(shí)候,可以指定class

class Admins::StatsController < ApplicationController
   include ::Wechat::OrdersHelper
   load_and_authorize_resource :class => "Order"
  def stat
     @orders = initialize_grid(
        Order,
        include: :barber,
        conditions: ['shop_id = ?', current_admin.shop],
        per_page: 20
      )
  end

在ablity.rb中,做如下控制:
can :stat, Order

NO.6 權(quán)限類別說明

權(quán)限類別說明 :manage, :all, ..etc.
cancan 里面用了一些自定義縮寫,如 :manage、:read、:update、:all,讓人不知道在做神馬。

:manage: 是指這個(gè) controller 內(nèi)所有的 action
:read : 指 :index 和 :show
:update: 指 :edit 和 :update
:destroy: 指 :destroy
:create: 指 :new 和 :crate
而 :all 是指所有 object (resource)

當(dāng)然,不只是 CRUD 的 method 才可以被列上去,如果你有其他非 RESTful 的 method 如 :search,也是可以寫上去..,只是要一條一條列上去,有點(diǎn)麻煩就是了。

組合技:alias_action
cancan 還提供了組合技,要是嫌原先的 :update, :read 這種組合包不夠用。還可以用 alias_action 自己另外再組。例如把 :update 和 :destroy 組成 :modify。

alias_action :update, :destroy, :to => :modify
   can :modify, Comment

參考資料:
cancancan
rolify
Cancan 實(shí)作角色權(quán)限設(shè)計(jì)的最佳實(shí)踐(1)
Cancan 實(shí)作角色權(quán)限設(shè)計(jì)的最佳實(shí)踐(2)
Cancan 實(shí)作角色權(quán)限設(shè)計(jì)的最佳實(shí)踐(3)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容