rabarber 3.0.1 → 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 5dd689c4149da8e0f11b3e959b116be20b8364ca9546986a6064454b2eb25c44
4
- data.tar.gz: c5f15810b3434ab5275573f8d154c1dd765d5136f5a9d42c6f7cdd28a27c0133
3
+ metadata.gz: 445c88aed39f5026737bdd811e6988b4f688e3b34231709140903a5086563c22
4
+ data.tar.gz: 9668ce42145bf83fb70f7177110d8ab2ddc01aff39b3f9078bd2ea6034d8dc40
5
5
  SHA512:
6
- metadata.gz: b53349cbf3d1fffc9a8a5acc0183eb18a3c6e928b87d613deb6502f60e1ee0ef4b1f49538c51249f07b27666562654f51a70713c7d30d7b41031d0610d340956
7
- data.tar.gz: 44930f95235e429db06e8deca63148a7b89187d9d79724cd0708e42bb46b7486552e0d459cc73fc9a8ab6b248b1e6e2d69cd6a647220275f08a8a7fecc2526f6
6
+ metadata.gz: dc26142a700acb2cc3350a6635c52762056151cd95a442082fd2638dce8fabf6de395d296bd9e47b06013e264c0c5370346a46803a39692682ce169e50fbd59a
7
+ data.tar.gz: c2341a85e1ffa79ef81fd1aeb3ee3a5b53053942e6f07fca927f2ec47755883af114ce44ce6e58764ab6acb3207e8e0f9a0be569f7cf80894bd5663c3cf7fc2c
data/CHANGELOG.md CHANGED
@@ -1,3 +1,21 @@
1
+ ## v4.0.0
2
+
3
+ ### Breaking:
4
+
5
+ - Dropped support for Ruby 3.0
6
+ - Dropped support for Rails 6.1
7
+
8
+ ### Misc:
9
+
10
+ - Added support for Rails 8.0
11
+
12
+ ## v3.0.2
13
+
14
+ ### Misc:
15
+
16
+ - Improved performance for authorization checks
17
+ - Refactored codebase for better maintainability
18
+
1
19
  ## v3.0.1
2
20
 
3
21
  ### Misc:
@@ -15,16 +15,10 @@ module Rabarber
15
15
  private
16
16
 
17
17
  def initialize(roleable, specifics)
18
- raise ArgumentError, "Roleable is required for #{self.class} event" if roleable.nil? && !nil_roleable_allowed?
19
-
20
18
  @roleable = roleable
21
19
  @specifics = specifics
22
20
  end
23
21
 
24
- def nil_roleable_allowed?
25
- raise NotImplementedError
26
- end
27
-
28
22
  def log
29
23
  Rabarber::Audit::Logger.log(log_level, message)
30
24
  end
@@ -38,21 +32,14 @@ module Rabarber
38
32
  end
39
33
 
40
34
  def identity
41
- if roleable.is_a?(Rabarber::Core::NullRoleable)
42
- "Unauthenticated #{Rabarber::HasRoles.roleable_class.model_name.human.downcase}"
43
- else
44
- model_name = roleable.model_name.human
45
- roleable_id = roleable.public_send(roleable.class.primary_key)
46
-
47
- "#{model_name}##{roleable_id}"
48
- end
35
+ roleable.log_identity
49
36
  end
50
37
 
51
38
  def human_context
52
39
  case context
53
40
  in { context_type: nil, context_id: nil } then "Global"
54
- in { context_type: context_type, context_id: nil } then context_type
55
- in { context_type: context_type, context_id: context_id } then "#{context_type}##{context_id}"
41
+ in { context_type:, context_id: nil } then context_type
42
+ in { context_type:, context_id: } then "#{context_type}##{context_id}"
56
43
  else raise "Unexpected context: #{context}"
57
44
  end
58
45
  end
@@ -6,10 +6,6 @@ module Rabarber
6
6
  class RolesAssigned < Base
7
7
  private
8
8
 
9
- def nil_roleable_allowed?
10
- false
11
- end
12
-
13
9
  def log_level
14
10
  :info
15
11
  end
@@ -6,10 +6,6 @@ module Rabarber
6
6
  class RolesRevoked < Base
7
7
  private
8
8
 
9
- def nil_roleable_allowed?
10
- false
11
- end
12
-
13
9
  def log_level
14
10
  :info
15
11
  end
@@ -6,10 +6,6 @@ module Rabarber
6
6
  class UnauthorizedAttempt < Base
7
7
  private
8
8
 
9
- def nil_roleable_allowed?
10
- true
11
- end
12
-
13
9
  def log_level
14
10
  :warn
15
11
  end
@@ -9,22 +9,18 @@ module Rabarber
9
9
  attr_reader :audit_trail_enabled, :cache_enabled, :current_user_method, :must_have_roles
10
10
 
11
11
  def initialize
12
- @audit_trail_enabled = default_audit_trail_enabled
13
- @cache_enabled = default_cache_enabled
14
- @current_user_method = default_current_user_method
15
- @must_have_roles = default_must_have_roles
12
+ @audit_trail_enabled = true
13
+ @cache_enabled = true
14
+ @current_user_method = :current_user
15
+ @must_have_roles = false
16
16
  end
17
17
 
18
- def audit_trail_enabled=(value)
19
- @audit_trail_enabled = Rabarber::Input::Types::Boolean.new(
20
- value, Rabarber::ConfigurationError, "Configuration 'audit_trail_enabled' must be a Boolean"
21
- ).process
22
- end
23
-
24
- def cache_enabled=(value)
25
- @cache_enabled = Rabarber::Input::Types::Boolean.new(
26
- value, Rabarber::ConfigurationError, "Configuration 'cache_enabled' must be a Boolean"
27
- ).process
18
+ [:audit_trail_enabled, :cache_enabled, :must_have_roles].each do |method_name|
19
+ define_method(:"#{method_name}=") do |value|
20
+ instance_variable_set(:"@#{method_name}", Rabarber::Input::Types::Boolean.new(
21
+ value, Rabarber::ConfigurationError, "Configuration '#{method_name}' must be a Boolean"
22
+ ).process)
23
+ end
28
24
  end
29
25
 
30
26
  def current_user_method=(method_name)
@@ -32,29 +28,5 @@ module Rabarber
32
28
  method_name, Rabarber::ConfigurationError, "Configuration 'current_user_method' must be a Symbol or a String"
33
29
  ).process
34
30
  end
35
-
36
- def must_have_roles=(value)
37
- @must_have_roles = Rabarber::Input::Types::Boolean.new(
38
- value, Rabarber::ConfigurationError, "Configuration 'must_have_roles' must be a Boolean"
39
- ).process
40
- end
41
-
42
- private
43
-
44
- def default_audit_trail_enabled
45
- true
46
- end
47
-
48
- def default_cache_enabled
49
- true
50
- end
51
-
52
- def default_current_user_method
53
- :current_user
54
- end
55
-
56
- def default_must_have_roles
57
- false
58
- end
59
31
  end
60
32
  end
@@ -3,12 +3,9 @@
3
3
  module Rabarber
4
4
  module Authorization
5
5
  extend ActiveSupport::Concern
6
-
7
6
  include Rabarber::Core::Roleable
8
7
 
9
- included do
10
- before_action :verify_access
11
- end
8
+ included { before_action :verify_access }
12
9
 
13
10
  class_methods do
14
11
  def skip_authorization(options = {})
@@ -16,15 +13,13 @@ module Rabarber
16
13
  end
17
14
 
18
15
  def grant_access(action: nil, roles: nil, context: nil, if: nil, unless: nil)
19
- dynamic_rule, negated_dynamic_rule = binding.local_variable_get(:if), binding.local_variable_get(:unless)
20
-
21
16
  Rabarber::Core::Permissions.add(
22
17
  self,
23
18
  Rabarber::Input::Action.new(action).process,
24
19
  Rabarber::Input::Roles.new(roles).process,
25
20
  Rabarber::Input::AuthorizationContext.new(context).process,
26
- Rabarber::Input::DynamicRule.new(dynamic_rule).process,
27
- Rabarber::Input::DynamicRule.new(negated_dynamic_rule).process
21
+ Rabarber::Input::DynamicRule.new(binding.local_variable_get(:if)).process,
22
+ Rabarber::Input::DynamicRule.new(binding.local_variable_get(:unless)).process
28
23
  )
29
24
  end
30
25
  end
@@ -9,7 +9,7 @@ module Rabarber
9
9
 
10
10
  def controller_accessible?(roleable, controller_instance)
11
11
  controller_rules.any? do |rule_controller, rule|
12
- controller_instance.class <= rule_controller && rule.verify_access(roleable, controller_instance)
12
+ controller_instance.is_a?(rule_controller) && rule.verify_access(roleable, controller_instance)
13
13
  end
14
14
  end
15
15
 
@@ -5,22 +5,22 @@ require "digest/sha2"
5
5
  module Rabarber
6
6
  module Core
7
7
  module Cache
8
- module_function
9
-
10
8
  CACHE_PREFIX = "rabarber"
11
9
  private_constant :CACHE_PREFIX
12
10
 
11
+ module_function
12
+
13
13
  def fetch(roleable_id, context:, &block)
14
- if enabled?
15
- Rails.cache.fetch(key_for(roleable_id, context), expires_in: 1.hour, race_condition_ttl: 5.seconds, &block)
16
- else
17
- yield
18
- end
14
+ return yield unless enabled?
15
+
16
+ Rails.cache.fetch(key_for(roleable_id, context), expires_in: 1.hour, race_condition_ttl: 5.seconds, &block)
19
17
  end
20
18
 
21
19
  def delete(*roleable_ids, context:)
20
+ return unless enabled?
21
+
22
22
  keys = roleable_ids.map { |roleable_id| key_for(roleable_id, context) }
23
- Rails.cache.delete_multi(keys) if enabled? && keys.any?
23
+ Rails.cache.delete_multi(keys) if keys.any?
24
24
  end
25
25
 
26
26
  def enabled?
@@ -38,9 +38,10 @@ module Rabarber
38
38
  end
39
39
 
40
40
  module Cache
41
+ module_function
42
+
41
43
  def clear
42
44
  Rabarber::Core::Cache.clear
43
45
  end
44
- module_function :clear
45
46
  end
46
47
  end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rabarber
4
+ module Core
5
+ class NullRoleable
6
+ def roles(context:) # rubocop:disable Lint/UnusedMethodArgument
7
+ []
8
+ end
9
+
10
+ def has_role?(*role_names, context: nil) # rubocop:disable Lint/UnusedMethodArgument
11
+ false
12
+ end
13
+
14
+ def log_identity
15
+ "Unauthenticated #{Rabarber::HasRoles.roleable_class.model_name.human.downcase}"
16
+ end
17
+ end
18
+ end
19
+ end
@@ -2,14 +2,12 @@
2
2
 
3
3
  require_relative "access"
4
4
  require_relative "rule"
5
-
6
5
  require "singleton"
7
6
 
8
7
  module Rabarber
9
8
  module Core
10
9
  class Permissions
11
10
  include Singleton
12
-
13
11
  extend Access
14
12
 
15
13
  attr_reader :storage
@@ -21,12 +19,7 @@ module Rabarber
21
19
  class << self
22
20
  def add(controller, action, roles, context, dynamic_rule, negated_dynamic_rule)
23
21
  rule = Rabarber::Core::Rule.new(action, roles, context, dynamic_rule, negated_dynamic_rule)
24
-
25
- if action
26
- instance.storage[:action_rules][controller] += [rule]
27
- else
28
- instance.storage[:controller_rules][controller] = rule
29
- end
22
+ action ? action_rules[controller] += [rule] : controller_rules[controller] = rule
30
23
  end
31
24
 
32
25
  def controller_rules
@@ -28,11 +28,8 @@ module Rabarber
28
28
  end
29
29
 
30
30
  def action_rules
31
- if controller
32
- Rabarber::Core::Permissions.action_rules.slice(controller)
33
- else
34
- Rabarber::Core::Permissions.action_rules
35
- end
31
+ rules = Rabarber::Core::Permissions.action_rules
32
+ controller ? rules.slice(controller) : rules
36
33
  end
37
34
  end
38
35
  end
@@ -1,5 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require_relative "null_roleable"
4
+
3
5
  module Rabarber
4
6
  module Core
5
7
  module Roleable
@@ -8,13 +10,7 @@ module Rabarber
8
10
  end
9
11
 
10
12
  def roleable_roles(context: nil)
11
- roleable.roles(context: context)
12
- end
13
- end
14
-
15
- class NullRoleable
16
- def roles(context:) # rubocop:disable Lint/UnusedMethodArgument
17
- []
13
+ roleable.roles(context:)
18
14
  end
19
15
  end
20
16
  end
@@ -9,44 +9,32 @@ module Rabarber
9
9
  @action = action
10
10
  @roles = Array(roles)
11
11
  @context = context
12
- @dynamic_rule = dynamic_rule
13
- @negated_dynamic_rule = negated_dynamic_rule
12
+ @dynamic_rule = dynamic_rule || -> { true }
13
+ @negated_dynamic_rule = negated_dynamic_rule || -> { false }
14
14
  end
15
15
 
16
16
  def verify_access(roleable, controller_instance)
17
- roles_permitted?(roleable, controller_instance) && dynamic_rule_followed?(controller_instance)
17
+ roles_permitted?(roleable, controller_instance) && dynamic_rules_followed?(controller_instance)
18
18
  end
19
19
 
20
20
  def roles_permitted?(roleable, controller_instance)
21
- processed_context = get_context(controller_instance)
22
- roleable_roles = roleable.roles(context: processed_context)
21
+ resolved_context = resolve_context(controller_instance)
22
+ return false if Rabarber::Configuration.instance.must_have_roles && roleable.roles(context: resolved_context).empty?
23
23
 
24
- return false if Rabarber::Configuration.instance.must_have_roles && roleable_roles.empty?
25
-
26
- roles.empty? || roles.intersection(roleable_roles).any?
24
+ roles.empty? || roleable.has_role?(*roles, context: resolved_context)
27
25
  end
28
26
 
29
- def dynamic_rule_followed?(controller_instance)
30
- !!(execute_dynamic_rule(controller_instance, false) && execute_dynamic_rule(controller_instance, true))
27
+ def dynamic_rules_followed?(controller_instance)
28
+ execute_rule(controller_instance, dynamic_rule) && !execute_rule(controller_instance, negated_dynamic_rule)
31
29
  end
32
30
 
33
31
  private
34
32
 
35
- def execute_dynamic_rule(controller_instance, is_negated)
36
- rule = is_negated ? negated_dynamic_rule : dynamic_rule
37
-
38
- return true if rule.nil?
39
-
40
- result = if rule.is_a?(Proc)
41
- controller_instance.instance_exec(&rule)
42
- else
43
- controller_instance.send(rule)
44
- end
45
-
46
- is_negated ? !result : result
33
+ def execute_rule(controller_instance, rule)
34
+ !!(rule.is_a?(Proc) ? controller_instance.instance_exec(&rule) : controller_instance.send(rule))
47
35
  end
48
36
 
49
- def get_context(controller_instance)
37
+ def resolve_context(controller_instance)
50
38
  case context
51
39
  when Proc then Rabarber::Input::Context.new(controller_instance.instance_exec(&context)).process
52
40
  when Symbol then Rabarber::Input::Context.new(controller_instance.send(context)).process
@@ -5,15 +5,11 @@ module Rabarber
5
5
  include Rabarber::Core::Roleable
6
6
 
7
7
  def visible_to(*roles, context: nil, &block)
8
- return if roleable_roles(context: Rabarber::Input::Context.new(context).process).intersection(Rabarber::Input::Roles.new(roles).process).none?
9
-
10
- capture(&block)
8
+ capture(&block) if roleable.has_role?(*roles, context:)
11
9
  end
12
10
 
13
11
  def hidden_from(*roles, context: nil, &block)
14
- return if roleable_roles(context: Rabarber::Input::Context.new(context).process).intersection(Rabarber::Input::Roles.new(roles).process).any?
15
-
16
- capture(&block)
12
+ capture(&block) unless roleable.has_role?(*roles, context:)
17
13
  end
18
14
  end
19
15
  end
@@ -10,10 +10,7 @@ module Rabarber
10
10
  private
11
11
 
12
12
  def processed_value
13
- case value
14
- when String, Symbol then value.to_sym
15
- when nil then value
16
- end
13
+ value&.to_sym
17
14
  end
18
15
 
19
16
  def default_error_message
@@ -11,8 +11,8 @@ module Rabarber
11
11
 
12
12
  def processed_value
13
13
  case value
14
- when Symbol, String then value.to_sym
15
- when Proc then value
14
+ when String then value.to_sym
15
+ when Symbol, Proc then value
16
16
  else Rabarber::Input::Context.new(value).process
17
17
  end
18
18
  end
@@ -5,10 +5,10 @@ module Rabarber
5
5
  class Base
6
6
  attr_reader :value, :error_type, :error_message
7
7
 
8
- def initialize(value, error_type = Rabarber::InvalidArgumentError, error_message = default_error_message)
8
+ def initialize(value, error_type = Rabarber::InvalidArgumentError, error_message = nil)
9
9
  @value = value
10
10
  @error_type = error_type
11
- @error_message = error_message
11
+ @error_message = error_message || default_error_message
12
12
  end
13
13
 
14
14
  def process
@@ -10,10 +10,7 @@ module Rabarber
10
10
  private
11
11
 
12
12
  def processed_value
13
- case value
14
- when String, Symbol then value.to_sym
15
- when Proc, nil then value
16
- end
13
+ value.is_a?(String) ? value.to_sym : value
17
14
  end
18
15
 
19
16
  def default_error_message
@@ -16,7 +16,7 @@ module Rabarber
16
16
  end
17
17
 
18
18
  def default_error_message
19
- "Role name must be a Symbol or a String and may only contain lowercase letters, numbers and underscores"
19
+ "Role name must be a Symbol or a String and may only contain lowercase letters, numbers, and underscores"
20
20
  end
21
21
  end
22
22
  end
@@ -18,7 +18,7 @@ module Rabarber
18
18
  end
19
19
 
20
20
  def default_error_message
21
- "Role names must be Symbols or Strings and may only contain lowercase letters, numbers and underscores"
21
+ "Role names must be Symbols or Strings and may only contain lowercase letters, numbers, and underscores"
22
22
  end
23
23
  end
24
24
  end
@@ -21,7 +21,8 @@ module Rabarber
21
21
 
22
22
  def has_role?(*role_names, context: nil)
23
23
  processed_context = process_context(context)
24
- roles(context: processed_context).intersection(process_role_names(role_names)).any?
24
+ processed_roles = process_role_names(role_names)
25
+ roles(context: processed_context).any? { |role_name| processed_roles.include?(role_name) }
25
26
  end
26
27
 
27
28
  def assign_roles(*role_names, context: nil, create_new: true)
@@ -72,6 +73,10 @@ module Rabarber
72
73
  roles(context: processed_context)
73
74
  end
74
75
 
76
+ def log_identity
77
+ "#{model_name.human}##{roleable_id}"
78
+ end
79
+
75
80
  def roleable_class
76
81
  @@included.constantize
77
82
  end
@@ -80,7 +85,7 @@ module Rabarber
80
85
  private
81
86
 
82
87
  def create_new_roles(role_names, context:)
83
- new_roles = role_names - Rabarber::Role.names(context: context)
88
+ new_roles = role_names - Rabarber::Role.names(context:)
84
89
  new_roles.each { |role_name| Rabarber::Role.create!(name: role_name, **context) }
85
90
  end
86
91
 
@@ -93,7 +98,7 @@ module Rabarber
93
98
  end
94
99
 
95
100
  def delete_roleable_cache(context:)
96
- Rabarber::Core::Cache.delete(roleable_id, context: context)
101
+ Rabarber::Core::Cache.delete(roleable_id, context:)
97
102
  end
98
103
 
99
104
  def roleable_id
@@ -13,16 +13,16 @@ module Rabarber
13
13
 
14
14
  class << self
15
15
  def names(context: nil)
16
- where(Rabarber::Input::Context.new(context).process).pluck(:name).map(&:to_sym)
16
+ where(process_context(context)).pluck(:name).map(&:to_sym)
17
17
  end
18
18
 
19
19
  def add(name, context: nil)
20
20
  name = process_role_name(name)
21
21
  processed_context = process_context(context)
22
22
 
23
- return false if exists?(name: name, **processed_context)
23
+ return false if exists?(name:, **processed_context)
24
24
 
25
- !!create!(name: name, **processed_context)
25
+ !!create!(name:, **processed_context)
26
26
  end
27
27
 
28
28
  def rename(old_name, new_name, context: nil, force: false)
@@ -30,11 +30,11 @@ module Rabarber
30
30
  role = find_by(name: process_role_name(old_name), **processed_context)
31
31
  name = process_role_name(new_name)
32
32
 
33
- return false if !role || exists?(name: name, **processed_context) || assigned_to_roleables(role).any? && !force
33
+ return false if !role || exists?(name:, **processed_context) || assigned_to_roleables(role).any? && !force
34
34
 
35
35
  delete_roleables_cache(role, context: processed_context)
36
36
 
37
- role.update!(name: name)
37
+ role.update!(name:)
38
38
  end
39
39
 
40
40
  def remove(name, context: nil, force: false)
@@ -50,14 +50,14 @@ module Rabarber
50
50
 
51
51
  def assignees(name, context: nil)
52
52
  Rabarber::HasRoles.roleable_class.joins(:rabarber_roles).where(
53
- rabarber_roles: { name: Rabarber::Input::Role.new(name).process, **process_context(context) }
53
+ rabarber_roles: { name: process_role_name(name), **process_context(context) }
54
54
  )
55
55
  end
56
56
 
57
57
  private
58
58
 
59
59
  def delete_roleables_cache(role, context:)
60
- Rabarber::Core::Cache.delete(*assigned_to_roleables(role), context: context)
60
+ Rabarber::Core::Cache.delete(*assigned_to_roleables(role), context:)
61
61
  end
62
62
 
63
63
  def assigned_to_roleables(role)
@@ -6,7 +6,7 @@ module Rabarber
6
6
  class Railtie < Rails::Railtie
7
7
  initializer "rabarber.after_initialize" do |app|
8
8
  app.config.after_initialize do
9
- Rabarber::Core::PermissionsIntegrityChecker.new.run! if Rails.configuration.eager_load
9
+ Rabarber::Core::PermissionsIntegrityChecker.new.run! if app.config.eager_load
10
10
  end
11
11
  end
12
12
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Rabarber
4
- VERSION = "3.0.1"
4
+ VERSION = "4.0.0"
5
5
  end
data/rabarber.gemspec CHANGED
@@ -14,7 +14,7 @@ Gem::Specification.new do |spec|
14
14
  spec.metadata["rubygems_mfa_required"] = "true"
15
15
  spec.summary = "Simple role-based authorization library for Ruby on Rails"
16
16
  spec.license = "MIT"
17
- spec.required_ruby_version = ">= 3.0", "< 3.4"
17
+ spec.required_ruby_version = ">= 3.1", "< 3.4"
18
18
 
19
19
  spec.files = [
20
20
  "rabarber.gemspec", "README.md", "CHANGELOG.md", "LICENSE.txt"
@@ -22,5 +22,5 @@ Gem::Specification.new do |spec|
22
22
 
23
23
  spec.require_paths = ["lib"]
24
24
 
25
- spec.add_dependency "rails", ">= 6.1", "< 7.3"
25
+ spec.add_dependency "rails", ">= 7.0", "< 8.1"
26
26
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rabarber
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.0.1
4
+ version: 4.0.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - enjaku4
@@ -9,7 +9,7 @@ authors:
9
9
  autorequire:
10
10
  bindir: bin
11
11
  cert_chain: []
12
- date: 2024-08-14 00:00:00.000000000 Z
12
+ date: 2024-11-09 00:00:00.000000000 Z
13
13
  dependencies:
14
14
  - !ruby/object:Gem::Dependency
15
15
  name: rails
@@ -17,20 +17,20 @@ dependencies:
17
17
  requirements:
18
18
  - - ">="
19
19
  - !ruby/object:Gem::Version
20
- version: '6.1'
20
+ version: '7.0'
21
21
  - - "<"
22
22
  - !ruby/object:Gem::Version
23
- version: '7.3'
23
+ version: '8.1'
24
24
  type: :runtime
25
25
  prerelease: false
26
26
  version_requirements: !ruby/object:Gem::Requirement
27
27
  requirements:
28
28
  - - ">="
29
29
  - !ruby/object:Gem::Version
30
- version: '6.1'
30
+ version: '7.0'
31
31
  - - "<"
32
32
  - !ruby/object:Gem::Version
33
- version: '7.3'
33
+ version: '8.1'
34
34
  description:
35
35
  email:
36
36
  - rabarber_gem@icloud.com
@@ -53,6 +53,7 @@ files:
53
53
  - lib/rabarber/controllers/concerns/authorization.rb
54
54
  - lib/rabarber/core/access.rb
55
55
  - lib/rabarber/core/cache.rb
56
+ - lib/rabarber/core/null_roleable.rb
56
57
  - lib/rabarber/core/permissions.rb
57
58
  - lib/rabarber/core/permissions_integrity_checker.rb
58
59
  - lib/rabarber/core/roleable.rb
@@ -89,7 +90,7 @@ required_ruby_version: !ruby/object:Gem::Requirement
89
90
  requirements:
90
91
  - - ">="
91
92
  - !ruby/object:Gem::Version
92
- version: '3.0'
93
+ version: '3.1'
93
94
  - - "<"
94
95
  - !ruby/object:Gem::Version
95
96
  version: '3.4'
@@ -99,7 +100,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
99
100
  - !ruby/object:Gem::Version
100
101
  version: '0'
101
102
  requirements: []
102
- rubygems_version: 3.2.33
103
+ rubygems_version: 3.3.27
103
104
  signing_key:
104
105
  specification_version: 4
105
106
  summary: Simple role-based authorization library for Ruby on Rails