Doorkeeper::OpenidConnect
This library implements an OpenID Connect authentication provider for Rails applications on top of the Doorkeeper OAuth 2.0 framework.
OpenID Connect is a single-sign-on and identity layer with a growing list of server and client implementations. If you're looking for a client in Ruby check out omniauth_openid_connect.
Table of Contents
Status
The following parts of OpenID Connect Core 1.0 are currently supported:
- Authentication using the Authorization Code Flow
- Authentication using the Implicit Flow
- Requesting Claims using Scope Values
- UserInfo Endpoint
- Normal Claims
- OAuth 2.0 Form Post Response Mode
- OAuth 2.0 Dynamic Client Registration Protocol
In addition, we also support most of OpenID Connect Discovery 1.0 for automatic configuration discovery.
Take a look at the DiscoveryController for more details on supported features.
Known Issues
- Doorkeeper's API mode (
Doorkeeper.configuration.api_only) is not properly supported yet
Example Applications
Installation
Make sure your application is already set up with Doorkeeper.
Add this line to your application's Gemfile and run bundle install:
gem 'doorkeeper-openid_connect'
Run the installation generator to update routes and create the initializer:
rails generate doorkeeper:openid_connect:install
Generate a migration for Active Record (other ORMs are currently not supported):
rails generate doorkeeper:openid_connect:migration
rake db:migrate
If you're upgrading from an earlier version, check Migration from old versions wiki and CHANGELOG.md for upgrade instructions.
Configuration
Make sure you've configured Doorkeeper before continuing.
Verify your settings in config/initializers/doorkeeper.rb:
resource_owner_authenticator- This callback needs to returns a falsey value if the current user can't be determined:
resource_owner_authenticator do if current_user current_user else redirect_to(new_user_session_url) nil end endgrant_flows- If you want to use
id_tokenorid_token tokenresponse types you need to addimplicit_oidctogrant_flows:
grant_flows %w(authorization_code implicit_oidc)- If you want to use
The following settings are required in config/initializers/doorkeeper_openid_connect.rb:
issuer- Identifier for the issuer of the response (i.e. your application URL). The value is a case sensitive URL using the
httpsscheme that contains scheme, host, and optionally, port number and path components and no query or fragment components. - You can either pass a string value, or a block to generate the issuer dynamically. The block receives
resource_owner,application, andrequestso that the same configuration can serve both ID token issuance (whereresource_ownerandapplicationare available) and the discovery endpoint (where onlyrequestis available):
# config/initializers/doorkeeper_openid_connect.rb Doorkeeper::OpenidConnect.configure do # ... issuer do |resource_owner, application, request| request&.base_url || "https://default.example.com" end end- For backward compatibility, blocks with arity 0, 1, or 2 are also accepted. An arity-1 block receives
requestfrom the discovery endpoint andresource_ownerfrom the ID token context, while an arity-2 block always receivesresource_ownerandapplication.
- Identifier for the issuer of the response (i.e. your application URL). The value is a case sensitive URL using the
subject- Identifier for the resource owner (i.e. the authenticated user). A locally unique and never reassigned identifier within the issuer for the end-user, which is intended to be consumed by the client. The value is a case-sensitive string and must not exceed 255 ASCII characters in length.
- The database ID of the user is an acceptable choice if you don't mind leaking that information.
- If you want to provide a different subject identifier to each client, use pairwise subject identifier with configurations like below.
# config/initializers/doorkeeper_openid_connect.rb Doorkeeper::OpenidConnect.configure do # ... subject_types_supported [:pairwise] subject do |resource_owner, application| Digest::SHA256.hexdigest("#{resource_owner.id}#{URI.parse(application.redirect_uri).host}#{'your_secret_salt'}") end # ... endsigning_key- Private key to be used for JSON Web Signature.
- You can generate a private key with the
opensslcommand, see e.g. Generate an RSA keypair using OpenSSL. - You should not commit the key to your repository, but use an external file (in combination with
File.read) and/or the dotenv-rails gem (in combination withENV[...]).
signing_algorithm- The signing algorithm used for the ID token, which defaults to
:rs256. The list of supported algorithms can be found here
- The signing algorithm used for the ID token, which defaults to
resource_owner_from_access_token- Defines how to translate the Doorkeeper access token to a resource owner model.
[!Note] Both
signing_keyandsigning_algorithmalso accept callable objects (e.g. a lambda), which are evaluated on each call — useful for multi-tenant setups where the key or algorithm varies per request:signing_key -> { current_tenant.private_key } signing_algorithm -> { current_tenant.algorithm }[!Note]
signing_keyalso accepts an array for key rotation. The first entry is the active key used to sign newly issued ID tokens; the remaining entries are still published in the JWKS so clients can validate tokens signed with a retired key during a rotation window. Callable forms returning an array are also supported.signing_key [ File.read("config/keys/current.pem"), # active, signs new tokens File.read("config/keys/previous.pem"), # retired, exposed in JWKS only ]
The following settings are optional, but recommended for better client compatibility:
auth_time_from_resource_owner- Returns the time of the user's last login, this can be a
Time,DateTime, or any other class that responds toto_i - Used to populate the
auth_timeclaim on the ID Token. - Used as a fallback for
max_ageenforcement whenauth_time_from_sessionis not configured. Note: for multi-session deployments this is insecure (it returns the most recent login on any device), and emits a deprecation warning — preferauth_time_from_sessionbelow.
- Returns the time of the user's last login, this can be a
auth_time_from_session- Returns the time the user authenticated for the current session. Required for correct
max_ageenforcement when the same user can hold multiple concurrent sessions (e.g. PC + smartphone) —auth_time_from_resource_ownercannot distinguish between sessions and would let a stale session inherit a fresh login from another device. - The block is executed in the controller's scope and receives
(session, request). Return value can be aTime,DateTime, or anything responding toto_i. Returnnilto force reauthentication.
# Example: capture auth_time on the session at login, # and surface it here for the OIDC max_age check. auth_time_from_session do |session, _request| session[:auth_time] end- Returns the time the user authenticated for the current session. Required for correct
reauthenticate_resource_owner- Defines how to trigger reauthentication for the current user (e.g. display a password prompt, or sign-out the user and redirect to the login form).
- Required to support the
max_ageandprompt=loginparameters. - The block is executed in the controller's scope, so you have access to methods like
params,redirect_toetc.
select_account_for_resource_owner- Defines how to trigger account selection to choose the current login user.
- Required to support the
prompt=select_accountparameter. - The block is executed in the controller's scope, so you have access to methods like
params,redirect_toetc.
The following settings are optional:
expiration- Expiration time after which the ID Token must not be accepted for processing by clients.
- The default is 120 seconds, it can be configured using a value or block.
ruby # config/initializers/doorkeeper_openid_connect.rb Doorkeeper::OpenidConnect.configure do # ... expiration do |resource_owner, application| # You will have to ensure the application model implements an expiration method application.expiration end # ... end
protocol- The protocol to use when generating URIs for the discovery endpoints.
- The default is
httpsfor production, andhttpfor all other environments - Note that the OIDC specification mandates HTTPS, so you shouldn't change this for production environments unless you have a really good reason!
end_session_endpoint- The URL that the user is redirected to after ending the session on the client.
- Used by implementations like https://github.com/IdentityModel/oidc-client-js.
- The block is executed in the controller's scope, so you have access to your route helpers.
discovery_url_options- The URL options for every available endpoint to use when generating the endpoint URL in the
discovery response. Available endpoints:
authorization,token,revocation,introspection,userinfo,jwks,dynamic_client_registration. - This option requires option keys with an available endpoint and URL options as value.
- The default is to use the request host, just like all the other URLs in the discovery response.
- This is useful when you want endpoints to use a different URL than other requests. For example, if your Doorkeeper server is behind a firewall with other servers, you might want other servers to use an "internal" URL to communicate with Doorkeeper, but you want to present an "external" URL to end-users for authentication requests. Note that this setting does not actually change the URL that your Doorkeeper server responds on - that is outside the scope of Doorkeeper.
# config/initializers/doorkeeper_openid_connect.rb Doorkeeper::OpenidConnect.configure do # ... do |request| { authorization: { host: 'host.example.com' }, jwks: { protocol: request.ssl? ? :https : :http } } end # ... end- The URL options for every available endpoint to use when generating the endpoint URL in the
discovery response. Available endpoints:
apply_prompt_to_non_oidc_requests- Whether to honor the
promptauthorization parameter (none,login,consent,select_account) on plain OAuth requests that do not include theopenidscope. - Defaults to
false, which preserves the historical behavior of silently ignoringpromptoutside of OIDC requests. max_ageenforcement remains OIDC-only regardless of this option, since it is defined by OIDC Core.
# config/initializers/doorkeeper_openid_connect.rb Doorkeeper::OpenidConnect.configure do # ... apply_prompt_to_non_oidc_requests true # ... end- Whether to honor the
Scopes
To perform authentication over OpenID Connect, an OAuth client needs to request the openid scope. This scope needs to be enabled using either optional_scopes in the global Doorkeeper configuration in config/initializers/doorkeeper.rb, or by adding it to any OAuth application's scope attribute.
Note that any application defining its own scopes won't inherit the scopes defined in the initializer, so you might have to update existing applications as well.
See Using Scopes in the Doorkeeper wiki for more information.
offline_access
Per OIDC Core §11, the offline_access scope signals that the client wants a refresh token so it can access the user's resources while the user is offline. Doorkeeper's existing use_refresh_token block already covers the basic flow — issue a refresh token only when the client actually asked for offline_access:
# config/initializers/doorkeeper.rb
Doorkeeper.configure do
optional_scopes :openid, :offline_access
# Issue a refresh token only when the client requests offline_access
use_refresh_token do |context|
context.scopes.exists?("offline_access")
end
end
Note: This does not automatically enforce OIDC Core §11's strict requirements — for example, the OP MUST ignore
offline_accessunlessprompt=consentis present andresponse_typereturns an Authorization Code. If you need that level of enforcement, filter the scope in youruse_refresh_tokenblock or authorization controller override.
Claims
Claims can be defined in a claims block inside config/initializers/doorkeeper_openid_connect.rb:
Doorkeeper::OpenidConnect.configure do
claims do
claim :email do |resource_owner|
resource_owner.email
end
claim :full_name do |resource_owner|
"#{resource_owner.first_name} #{resource_owner.last_name}"
end
claim :preferred_username, scope: :openid do |resource_owner, scopes, access_token|
# Pass the resource_owner's preferred_username if the application has
# `profile` scope access. Otherwise, provide a more generic alternative.
scopes.exists?(:profile) ? resource_owner.preferred_username : "summer-sun-9449"
end
claim :groups, response: [:id_token, :user_info] do |resource_owner|
resource_owner.groups
end
end
end
Each claim block will be passed:
- the
resource_owner, which is the return value ofresource_owner_authenticatorin your initializer - the
scopesgranted by the access token, which is an instance ofDoorkeeper::OAuth::Scopes - the
access_tokenitself, which is an instance ofDoorkeeper::AccessToken
By default all custom claims are only returned from the UserInfo endpoint and not included in the ID token. You can optionally pass a response: keyword with one or both of the symbols :id_token or :user_info to specify where the claim should be returned.
You can also pass a scope: keyword argument on each claim to specify which OAuth scope should be required to access the claim. If you define any of the defined Standard Claims they will by default use their corresponding scopes (profile, email, address and phone), and any other claims will by default use the profile scope. Again, to use any of these scopes you need to enable them as described above.
You can also pass an array of scopes, in which case the claim is returned whenever the access token grants any of the listed scopes. This is useful when you want to expose the same claim under both a standard scope and an aggregate scope:
claim :given_name, scope: [:profile, :all_data] do |user|
user.first_name
end
claim :email, scope: [:email, :all_data] do |user|
user.email
end
Authentication Context (acr) and Methods (amr)
The claim DSL also handles standard top-level ID Token claims such as acr (Authentication Context Class Reference) and amr (Authentication Methods References) — commonly used to expose MFA status to clients:
claims do
claim :acr, response: [:id_token, :user_info], scope: :openid do |resource_owner|
# Single string — e.g. a URI like "urn:mace:incommon:iap:silver",
# or a numeric Level of Assurance "1".."4" per ISO/IEC 29115.
resource_owner.mfa_enabled? ? "2" : "1"
end
claim :amr, response: [:id_token, :user_info], scope: :openid do |resource_owner|
# Array of strings, per RFC 8176.
methods = ["pwd"]
methods << "mfa" if resource_owner.mfa_enabled?
methods << "otp" if resource_owner.last_login_used_totp?
methods
end
end
Two defaults are worth calling out because they bite silently:
response: [:id_token, :user_info]— custom claims default to UserInfo only, but relying parties usually expectacr/amron the ID Token.scope: :openid— without it, non-standard claims fall back to theprofilescope and disappear for clients that only requestedopenid.
Claim names you declare here are automatically advertised under claims_supported in the discovery document. The list of advertised acr values (acr_values_supported) is not currently generated.
Routes
The installation generator will update your config/routes.rb to define all required routes:
Rails.application.routes.draw do
use_doorkeeper_openid_connect
# your routes
end
This will mount the following routes:
GET /oauth/userinfo
POST /oauth/userinfo
GET /oauth/discovery/keys
GET /.well-known/openid-configuration
GET /.well-known/oauth-authorization-server
GET /.well-known/webfinger
With the exception of the hard-coded /.well-known paths (see RFC 5785) you can customize routes in the same way as with Doorkeeper, please refer to this page on their wiki.
Customizing the jwks_uri path in the discovery document
discovery_url_options lets you tweak the host, protocol, or port of the published jwks_uri, but not the path itself. To advertise a custom path — while keeping /oauth/discovery/keys working for existing clients during a rollover — mount the discovery controller at the new path and re-point the oauth_discovery_keys_url helper at it via direct:
# config/routes.rb
Rails.application.routes.draw do
use_doorkeeper_openid_connect
# 1. Mount the custom path under a non-conflicting helper name
get "/-/jwks",
to: "doorkeeper/openid_connect/discovery#keys",
as: :custom_jwks
# 2. Re-point oauth_discovery_keys_url at the new path
direct(:oauth_discovery_keys) { |opts| custom_jwks_url(opts) }
end
After this, .well-known/openid-configuration returns "jwks_uri": "https://example.com/-/jwks", and the original /oauth/discovery/keys route still responds (handy during a rollover).
[!Note] A naive
match "/-/jwks", ..., as: :oauth_discovery_keyswon't work — Rails has refused to reuse a route name since 4.0 and raisesArgumentError: Invalid route name, already in use: 'oauth_discovery_keys'. Thedirecthelper sidesteps this by overriding the URL helper itself rather than re-declaring the route name.
Nonces
To support clients who send nonces you have to tweak Doorkeeper's authorization view so the parameter is passed on.
If you don't already have custom templates, run this generator in your Rails application to add them:
rails generate doorkeeper:views
Then tweak the template as follows:
--- i/app/views/doorkeeper/authorizations/new.html.erb
+++ w/app/views/doorkeeper/authorizations/new.html.erb
@@ -26,6 +26,7 @@
<%= hidden_field_tag :state, @pre_auth.state %>
<%= hidden_field_tag :response_type, @pre_auth.response_type %>
<%= hidden_field_tag :scope, @pre_auth.scope %>
+ <%= hidden_field_tag :nonce, @pre_auth.nonce %>
<%= submit_tag t('doorkeeper.authorizations.buttons.authorize'), class: "btn btn-success btn-lg btn-block" %>
<% end %>
<%= form_tag oauth_authorization_path, method: :delete do %>
@@ -34,6 +35,7 @@
<%= hidden_field_tag :state, @pre_auth.state %>
<%= hidden_field_tag :response_type, @pre_auth.response_type %>
<%= hidden_field_tag :scope, @pre_auth.scope %>
+ <%= hidden_field_tag :nonce, @pre_auth.nonce %>
<%= submit_tag t('doorkeeper.authorizations.buttons.deny'), class: "btn btn-danger btn-lg btn-block" %>
<% end %>
</div>
Internationalization (I18n)
We use Rails locale files for error messages and scope descriptions, see config/locales/en.yml. You can override these by adding them to your own translations in config/locale.
Dynamic Client Registration
This gem supports OpenID Connect Dynamic Client Registration 1.0 based on RFC 7591.
To enable dynamic client registration, add the following to config/initializers/doorkeeper_openid_connect.rb:
Doorkeeper::OpenidConnect.configure do
# ...
dynamic_client_registration true
# ...
end
This exposes a POST /oauth/registration endpoint where OAuth clients can register themselves.
Supported parameters
The registration endpoint currently accepts the following RFC 7591 §2 parameters:
| Parameter | Description |
|---|---|
client_name |
Human-readable name of the client |
redirect_uris |
Array of redirection URIs |
scope |
Space-delimited list of requested scopes |
token_endpoint_auth_method |
Requested authentication method. Defaults to client_secret_basic. none is always allowed (and registers a public client); other allowed values depend on the host application's Doorkeeper client_credentials_methods configuration. Unsupported values are rejected with invalid_client_metadata. |
application_type |
Client type: web (default) or native, per OpenID Connect Discovery 1.0. Unsupported values are rejected with invalid_client_metadata. |
response_types |
Array of OAuth 2.0 response types the client will use (e.g. ["code"]). Must be a subset of the server's supported response types. Defaults to the server's full set when omitted. |
grant_types |
Array of OAuth 2.0 grant types the client will use (e.g. ["authorization_code"]). Must be a subset of the server's supported grant types. Defaults to the server's full set when omitted. |
When token_endpoint_auth_method is set to none, the client is registered as public (i.e. confidential: false). For all other values — or when the parameter is omitted — the client is registered as confidential, matching the RFC 7591 default of client_secret_basic.
Other RFC 7591 parameters (e.g. client_uri, logo_uri, contacts) require schema additions to oauth_applications and are not yet supported.
Authorization
By default, the registration endpoint is open to any request. To require authorization (e.g. an Initial Access Token per RFC 7591 §3.1), configure authorize_dynamic_client_registration:
Doorkeeper::OpenidConnect.configure do
# ...
dynamic_client_registration true
do
provided = request.headers["Authorization"].to_s
expected = "Bearer #{ENV['DCR_INITIAL_ACCESS_TOKEN']}"
# Use a constant-time comparison to avoid leaking the token via timing.
# Digesting first keeps the comparison fixed-length so the token's length
# isn't leaked either.
ActiveSupport::SecurityUtils.secure_compare(
Digest::SHA256.hexdigest(provided),
Digest::SHA256.hexdigest(expected),
)
end
# ...
end
The block is evaluated in the controller scope (with access to request, params, request.headers, etc.). Return a truthy value to allow the request, or a falsy value to reject it with 401 invalid_token.
When not configured (default), the endpoint remains open for backward compatibility.
Development
Run bundle install to setup all development dependencies.
To run all specs:
bundle exec rake spec
To generate and run migrations in the test application:
bundle exec rake migrate
To run the local engine server:
bundle exec rake server
By default, the latest Rails version is used. To use a specific version run:
rails=7.2 bundle update
License
Doorkeeper::OpenidConnect is released under the MIT License.
Sponsors
Initial development of this project was sponsored by PlayOn! Sports.