rails基于devise+cancancan+rolify的简单权限控制实现

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 在需要进行权限验证的控制器里里加上load_and_authorize_resource,例如:

class Admins::AdminsController < ApplicationController
  load_and_authorize_resource

这样就可以对整个控制器的所有动作进行验证,而不需要手动的对每个动作进行验证
运行项目后访问admins控制器后,报错

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

因为我的项目中用登录注册的模型为admin,所以current_user这个方法是不存在的,需要把他改掉
在ApplicationController中,将current_user 替换为current_admin:

alias_method :current_user, :current_admin

刷新后,依然报错:

ArgumentError in Admins::AdminsController#index

这是rolify自带的一个bug,暂未修复,解决办法是将:

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

改为

belongs_to :resource,
           :polymorphic => true

NO.3 rolify的主要作用是用来关联用户和角色的。

它封装了2个主要的方法:
add_role :admin,用来添加角色;
has_role? :admin, 判断是否有这个角色;

在新建用户的是时候,将角色和用户关联起来,params[:role]为页面上传过来的参数

  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

角色和用户关联成功。

NO.4 接下来就是在views里面和ablity.rb里面设置具体权限。

例如:如果有创建服务的权限,在服务模板的视图中做如下控制:

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

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

class Ability
  include CanCan::Ability
  def initialize(user)
    if user.blank?
      cannot :manage, :all
    ####超级管理员权限####
    elsif user.has_role? :admin
      can :manage, :all
    ####店长权限####
    elsif user.has_role? "店长"
      #####显示属于自己的店铺#####
      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
      #######切换店铺开关状态#######
      can :switch, Shop do |shop|
      (shop.admin_id == user.id)
      end
      ######管理所有本店铺的理发师######
      can :manage, Barber
      ####管理本店铺的服务####
      can :manage, ShopService
      ####查看所有的服务模板####
      can :read, Service
      ###看到属于自己的订单统计###
      can :stat, Order
    end
  end
end

所以,当角色为店长的用户登录的时候,是视图中,是看不见"添加服务"这个选项的,因为can :read, Service,而视图中需要创建权限才能看到。

NO.5 当控制器中的某个action没有对应的具体model时,该如何处理

在控制中加入load_and_authorize_resource的时候,可以指定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 权限类别说明

权限类别说明 :manage, :all, ..etc.
cancan 里面用了一些自定义缩写,如 :manage、:read、:update、:all,让人不知道在做神马。

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

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

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

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

参考资料:
cancancan
rolify
Cancan 實作角色權限設計的最佳實踐(1)
Cancan 實作角色權限設計的最佳實踐(2)
Cancan 實作角色權限設計的最佳實踐(3)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容