ManageIQ/manageiq

View on GitHub
lib/rbac/authorizer.rb

Summary

Maintainability
A
0 mins
Test Coverage
A
100%
module Rbac
  class Authorizer
    include Vmdb::Logging

    def self.role_allows?(**options)
      new.role_allows?(**options)
    end

    # @options option any         - any of the identifiers can match (default: false)
    # @options option identifier  - comes from the back end (ex: User#role_allows?)
    # @options option feature     - same as identifier, comes from the front end (ex: ApplicationHelper#role_allows?)
    # @options option identifiers - same as others
    #                               This comes from the former User#role_allows_any? API
    #                               there may be only ONE caller using 'identifiers'
    #                               in Menu::Section, so this option may be changed shortly.
    def role_allows?(user:, identifier: nil, feature: nil, identifiers: nil, any: false)
      identifier ||= feature

      tenant_identifier = MiqProductFeature.current_tenant_identifier(identifier)

      auth = if any.present?
               user_role_allows_any?(user, :identifiers => (identifiers || [identifier]))
             elsif tenant_identifier
               [identifier, tenant_identifier].any? { |i| user_role_allows?(user, :identifier => i) }
             else
               user_role_allows?(user, :identifier => identifier)
             end
      _log.debug("Auth #{auth ? "successful" : "failed"} for user '#{user.userid}', role '#{user.miq_user_role.try(:name)}', feature identifier '#{identifier}'")

      auth
    end

    private

    def user_role_allows?(user, **options)
      return false if user.miq_user_role.nil?
      return true if user.miq_user_role.allows?(**options)

      ident = options[:identifier]
      parent = MiqProductFeature.feature_parent(ident)
      return false if parent.nil?

      if MiqProductFeature.feature_hidden(ident)
        # return true for common features that are hidden and are under hidden parent
        # return true if any visible siblings are entitled
        if MiqProductFeature.feature_hidden(parent)
          true
        else
          user.miq_user_role.allows_any?(:identifiers => MiqProductFeature.feature_children(parent))
        end
      end
    end

    def user_role_allows_any?(user, **options)
      return false if user.miq_user_role.nil?

      user.miq_user_role.allows_any?(**options)
    end
  end
end