The logic about missing tokens implying logged in users (and that
all logged in users have access to any method protected by a token
capability) is correct. However, I believe it is both confusing and
brittle, and leaves a security-related door ajar for future foot-gun
incidents.
Instead, apply Abilities as normal, and keep the Capabilities
involvement only for situations where a token is provided. This
reduces the cognitive burden when considering Abilities in isolation.
can :welcome, :site
can [:create, :edit, :comment, :subscribe, :unsubscribe], DiaryEntry
can [:new, :create], Report
can :welcome, :site
can [:create, :edit, :comment, :subscribe, :unsubscribe], DiaryEntry
can [:new, :create], Report
+ can [:read, :read_one, :update, :update_one, :delete_one], UserPreference
if user.moderator?
can [:index, :show, :resolve, :ignore, :reopen], Issue
if user.moderator?
can [:index, :show, :resolve, :ignore, :reopen], Issue
if user
can [:read, :read_one], UserPreference if capability?(token, :allow_read_prefs)
can [:update, :update_one, :delete_one], UserPreference if capability?(token, :allow_write_prefs)
if user
can [:read, :read_one], UserPreference if capability?(token, :allow_read_prefs)
can [:update, :update_one, :delete_one], UserPreference if capability?(token, :allow_write_prefs)
- # If a user provides no tokens, they've authenticated via a non-oauth method
- # and permission to access to all capabilities is assumed.
def capability?(token, cap)
def capability?(token, cap)
- token.nil? || token.read_attribute(cap)
+ token&.read_attribute(cap)
assert ability.can?(action, DiaryComment), "should be able to #{action} DiaryComment"
end
end
assert ability.can?(action, DiaryComment), "should be able to #{action} DiaryComment"
end
end
-
- test "administrator does not auto-grant user preferences" do
- ability = Ability.new create(:administrator_user)
-
- [:read, :read_one, :update, :update_one, :delete_one].each do |act|
- assert ability.cannot? act, UserPreference
- end
- end
# a user with no tokens
capability = Capability.new create(:user), nil
[:read, :read_one, :update, :update_one, :delete_one].each do |act|
# a user with no tokens
capability = Capability.new create(:user), nil
[:read, :read_one, :update, :update_one, :delete_one].each do |act|
- assert capability.can? act, UserPreference
+ assert capability.cannot? act, UserPreference
end
# A user with empty tokens
end
# A user with empty tokens