class Six
class NoPackError < StandardError
def message
"No such pack"
end
end
class InvalidPackPassed < StandardError
def message
"Wrong Rule Pack. You must provide correct 'allowed' method"
end
end
class InitializeArgumentError < StandardError
def message
"Six.new require hash as pack argument in format {:name_of_pack => PackRules.new}"
end
end
attr_reader :rules_packs
attr_reader :current_rule_pack
# Initialize ability object
#
# == Parameters:
# packs::
# A Hash or rules to add with initializtion
#
# == Returns:
# self
#
def initialize(packs={})
raise InitializeArgumentError.new unless packs.kind_of?(Hash)
@rules_packs = {}
@current_rule_pack = nil
packs.each { |key, pack| add_pack!(key, pack) }
end
# Set current pack from stored packs by key
#
# == Parameters:
# name::
# A Symbol declaring the key name of stored pack
#
# == Returns:
# self or false
#
def use_pack(name)
if pack_exist?(name)
@current_rule_pack = name.to_sym
self
end
end
# Same as use but raise exception if no pack found
def use_pack!(name)
use_pack(name) ? self : raise_no_such_pack
end
# Add pack to authorization class
#
# == Parameters:
# name::
# A Symbol declaring the key name of stored pack
# pack::
# Any kind of object responding to allowed method
#
# == Returns:
# true or false
#
def add_pack(name, pack)
rules_packs[name.to_sym] = pack if valid_rules_object?(pack)
end
# Same as add_pack but raise exception if pack is invalid
def add_pack!(name, pack)
add_pack(name, pack) || raise_incorrect_pack_object
end
# Add pack to authorization class w/o key
#
# == Parameters:
# pack::
# Any kind of object responding to allowed method
#
# == Returns:
# true or raise exception
#
def <<(pack)
add_pack!(pack.object_id.to_s, pack)
end
# Remove pack from authorization class
#
# == Parameters:
# name::
# A Symbol declaring the key name of stored pack
#
# == Returns:
# true or false
#
def remove_pack(name)
if pack_exist?(name)
@current_rule_pack = nil if rules_packs[name.to_sym] == @current_rule_pack
rules_packs.delete(name.to_sym)
end
end
# Same as remove_pack but raise exception if pack wasnt found
def remove_pack!(name)
remove_pack(name) || raise_no_such_pack
end
# Check if object for rule pack is valid
#
# == Parameters:
# pack::
# Any kind of object responding to allowed method
#
# == Returns:
# true or false
#
def valid_rules_object?(object)
object.respond_to?(:allowed) &&
object.send(:allowed, nil, nil).kind_of?(Array)
rescue
false
end
# Check if authorization class has pack with such name
#
# == Parameters:
# name::
# A Symbol declaring the key name of stored pack
#
# == Returns:
# true or false
#
def pack_exist?(name)
rules_packs.has_key?(name.to_sym)
end
# Check if authorization class allow access for object to subject
# using selected pack or all stored.
# Basically this method
# 1. send :allowed for every stored object in packs and pass object & subject
# 2. check if any of results include allowed action
#
# == Parameters:
# action::
# Action name to check for access
# object::
# object trying to access resource
# subject::
# resource
#
# == Returns:
# true or false
#
def allowed?(object, actions, subject)
# if multiple actions passed
# check all actions to be allowed
if actions.respond_to?(:each)
actions.all? { |action| action_included?(object, action, subject) }
else
# single action check
action_included?(object, actions, subject)
end
end
# Reset current used rule pack so auth class use
# global allowed? for new request
def reset_use
@current_rule_pack = nil
end
protected
def action_included?(object, action, subject)
if current_rule_pack
rules_packs[current_rule_pack].allowed(object, subject).include?(action)
else
rules_packs.values.map { |rp| rp.allowed(object, subject) }.flatten.include?(action)
end
end
def raise_no_such_pack
raise Six::NoPackError.new
end
def raise_incorrect_pack_object
raise Six::InvalidPackPassed.new
end
# shotcuts for long methods
alias_method :use, :use_pack
alias_method :use!, :use_pack!
alias_method :add, :add_pack
alias_method :add!, :add_pack!
alias_method :remove, :remove_pack
alias_method :remove!, :remove_pack!
alias_method :reset, :reset_use
alias_method :exist?, :pack_exist?
end
上面是源代码,这是一个非常简单而有趣的插件,小巧灵活的同时也需要自己动手编写一些代码,才能和 cancan 比较,我们不知道作者为什么不用 cancan, 而是自己动手。
还好作者给了我们这几行代码,基本上也可以像 cancan 一样玩了。
helper_method :abilities, :can?
protected
def abilities
@abilities ||= Six.new
end
# simple delegate method for controller & view
def can?(object, action, subject)
abilities.allowed?(object, action, subject)
end
最后值得一提的是插件名为 Six, 但实际我们从源代码却可以用这 8 个方法
alias_method :use, :use_pack
alias_method :use!, :use_pack!
alias_method :add, :add_pack
alias_method :add!, :add_pack!
alias_method :remove, :remove_pack
alias_method :remove!, :remove_pack!
alias_method :reset, :reset_use
alias_method :exist?, :pack_exist?