我使用 Devise 进行身份验证,使用 Rolify 进行角色管理,使用 CanCan 2.0 进行授权。
我试图允许 :admin 角色更改用户的角色,但不允许所有其他用户访问。
这是我尝试过但不起作用的方法:
#ability.rb
class Ability
  include CanCan::Ability
  def initialize(user)
    if user.has_role? :admin
      can :access, :all
    elsif user.has_role? :moderator
      can [:index, :read, :update, :destroy], :users, :user_id => user.id
      cannot :access, :users, [:role_ids]
    end
end
#application_controller.rb
...
rescue_from CanCan::Unauthorized do |exception|
    redirect_to root_url, :alert => exception.message
  end
我故意在我的用户表单中留下了关联:
#_form.html.erb
<%= simple_form_for @user do |f| %>
  <%= f.association :roles, as: :check_boxes %>
  <%#= f.association :roles, as: :check_boxes if can? :update, @user, :roles %>
  <%= f.button :submit %>
<% end %>
控制器
#users_controller.rb
class UsersController < ApplicationController
  before_filter :authenticate_user!
  load_and_authorize_resource
  def index
    @users = User.accessible_by(current_ability)
  end
  def new
    @user = User.new
  end
  def create
    @user = User.new(params[:user])
  end
  def show
    @user = User.find(params[:id])
  end
  def edit
    @user = User.find(params[:id])
  end
  def update
    @user = User.find(params[:id])
    @user.update_without_password(params[:user])
    if successfully_updated
      redirect_to @user
    else
      render :action => "edit"
    end
  end
end
和模型:
#user.rb
    class User < ActiveRecord::Base
      rolify
      attr_accessible :role_ids
    ...
现在,如果具有 :moderator 角色的用户尝试更改其他用户(或他自己)的角色,则会发生以下情况:
- 抛出 CanCan::Unauthorized 异常并将用户重定向到 root_url
- 为用户更改角色
我很困惑。如果发生异常,为什么还要进行更改?我可能做错了什么:)
我已经尝试根据 users_controller.rb 中的用户角色操作查询参数如果我在 def 更新后立即放置日志语句,这是我的输出:
2013-04-24 12:42:21 [4161] DEBUG    (0.1ms)  BEGIN
2013-04-24 12:42:21 [4161] DEBUG    (0.3ms)  INSERT INTO "users_roles" ("user_id", "role_id") VALUES (5, 1)
2013-04-24 12:42:21 [4161] DEBUG    (0.4ms)  COMMIT
2013-04-24 12:42:21 [4161] DEBUG   User Load (0.5ms)  SELECT "users".* FROM "users" WHERE "users"."id" = $1 LIMIT 1  [["id", "5"]]
2013-04-24 12:42:21 [4161] DEBUG {"username"=>"Blabla", "email"=>"bla@bla.com", "password"=>"", "password_confirmation"=>"", "approved"=>"1", "role_ids"=>["1", "2", ""]}
我一定是忽略了什么...