Class: KeycloakRack::Authenticate

Inherits:
Object
  • Object
show all
Defined in:
lib/keycloak_rack/authenticate.rb

Overview

The core service that handles authenticating a request from Keycloak.

Examples:

class ApplicationController < ActionController::API
  before_action :authenticate_user!

  # @return [void]
  def authenticate_user!
    # KeycloakRack::Session#authenticate! implements a Dry::Matcher::ResultMatcher
    request.env["keycloak:session"].authenticate! do |m|
      m.success(:authenticated) do |_, token|
        # this is the case when a user is successfully authenticated

        # token will be a KeycloakRack::DecodedToken instance, a
        # hash-like PORO that maps a number of values from the
        # decoded JWT that can be used to find or upsert a user

        attrs = decoded_token.slice(:keycloak_id, :email, :email_verified, :realm_access, :resource_access)

        result = User.upsert attrs, returning: %i[id], unique_by: %i[keycloak_id]

        @current_user = User.find result.first["id"]
      end

      m.success do
        # When allow_anonymous is true, or
        # a URI is skipped because of skip_paths, this
        # case will be reached. Requests from here on
        # out should be considered anonymous and treated
        # accordingly

        @current_user = AnonymousUser.new
      end

      m.failure do |code, reason|
        # All authentication failures are reached here,
        # assuming halt_on_auth_failure is set to false
        # This allows the application to decide how it
        # wants to respond

        render json: { errors: [{ message: "Auth Failure" }] }, status: :forbidden
      end
    end
  end
end

Instance Method Summary collapse

Instance Method Details

#call(env) ⇒ Dry::Monads::Success(:authenticated, KeycloakRack::DecodedToken), ...

Parameters:

  • env (Hash)

    the rack environment

Returns:

  • (Dry::Monads::Success(:authenticated, KeycloakRack::DecodedToken))
  • (Dry::Monads::Success(:skipped, String))
  • (Dry::Monads::Success(:unauthenticated))
  • (Dry::Monads::Failure(:expired, String, String, Exception))
  • (Dry::Monads::Failure(:decoding_failed, String, Exception))


65
66
67
68
69
70
71
72
73
74
75
76
77
# File 'lib/keycloak_rack/authenticate.rb', line 65

def call(env)
  return Success[:skipped] if yield skip_authentication.call(env)

  token = yield read_token.call env

  return Success[:unauthenticated] if token.blank?

  payload, headers = yield decode_and_verify.call token

  decoded_token = yield wrap.call payload, headers

  Success[:authenticated, decoded_token]
end