From 64ced6b8fcd1bd3de719b9eeb4176f418f7b2366 Mon Sep 17 00:00:00 2001 From: Mark Manes Date: Thu, 20 Jun 2024 12:43:27 -0500 Subject: [PATCH] Add support for OAuth scope handling (#282) * Add OAuth configuration support for: * scope_handling_policy * unknown_scope_policy * relationship * consent_mode Add userinfo_populate_lambda_id * Add docs for new fields * Update docs/resources/application.md Co-authored-by: Spencer Witt * Require new scope attributes --------- Co-authored-by: Spencer Witt --- docs/resources/application.md | 15 +++++ fusionauth/resource_fusionauth_application.go | 56 +++++++++++++++++-- ...resource_fusionauth_application_helpers.go | 54 +++++++++++++++--- 3 files changed, 112 insertions(+), 13 deletions(-) diff --git a/docs/resources/application.md b/docs/resources/application.md index 7388bf7..0aa078f 100644 --- a/docs/resources/application.md +++ b/docs/resources/application.md @@ -140,6 +140,7 @@ resource "fusionauth_application" "Forum" { - `id_token_populate_id` - (Optional) The Id of the Lambda that will be invoked when an Id token is generated for this application during an OpenID Connect authentication request. - `samlv2_populate_id` - (Optional) The Id of the Lambda that will be invoked when a a SAML response is generated during a SAML authentication request. - `self_service_registration_validation_id` - (Optional) The unique Id of the lambda that will be used to perform additional validation on registration form steps. + - `userinfo_populate_id` - (Optional) The Id of the Lambda that will be invoked when a UserInfo response is generated for this application. * `login_configuration` - (Optional) - `allow_token_refresh` - (Optional) Indicates if a JWT may be refreshed using a Refresh Token for this application. This configuration is separate from issuing new Refresh Tokens which is controlled by the generateRefreshTokens parameter. This configuration indicates specifically if an existing Refresh Token may be used to request a new JWT using the Refresh API. - `generate_refresh_tokens` - (Optional) Indicates if a Refresh Token should be issued from the Login API @@ -156,6 +157,10 @@ resource "fusionauth_application" "Forum" { - `authorized_url_validation_policy` - (Optional) Determines whether wildcard expressions will be allowed in the authorized_redirect_urls and authorized_origin_urls. - `client_secret` - (Optional) The OAuth 2.0 client secret. If you leave this blank during a POST, a secure secret will be generated for you. If you leave this blank during PUT, the previous value will be maintained. For both POST and PUT you can provide a value and it will be stored. - `client_authentication_policy` - (Optional) Determines the client authentication requirements for the OAuth 2.0 Token endpoint. + - `consent_mode` (Optional) Controls the policy for prompting a user to consent to requested OAuth scopes. This configuration only takes effect when `application.oauthConfiguration.relationship` is `ThirdParty`. The possible values are: + - `AlwaysPrompt` - Always prompt the user for consent. + - `RememberDecision` - Remember previous consents; only prompt if the choice expires or if the requested or required scopes have changed. The duration of this persisted choice is controlled by the Tenant’s `externalIdentifierConfiguration.rememberOAuthScopeConsentChoiceTimeToLiveInSeconds` value. + - `NeverPrompt` - The user will be never be prompted to consent to requested OAuth scopes. Permission will be granted implicitly as if this were a `FirstParty` application. This configuration is meant for testing purposes only and should not be used in production. - `debug` - (Optional) Whether or not FusionAuth will log a debug Event Log. This is particular useful for debugging the authorization code exchange with the Token endpoint during an Authorization Code grant." - `device_verification_url` - (Optional) The device verification URL to be used with the Device Code grant type, this field is required when device_code is enabled. - `enabled_grants` - (Optional) The enabled grants for this application. In order to utilize a particular grant with the OAuth 2.0 endpoints you must have enabled the grant. @@ -163,6 +168,9 @@ resource "fusionauth_application" "Forum" { - `logout_behavior` - (Optional) Behavior when /oauth2/logout is called. - `logout_url` - (Optional) The logout URL for the Application. FusionAuth will redirect to this URL after the user logs out of OAuth. - `proof_key_for_code_exchange_policy` - (Optional) Determines the PKCE requirements when using the authorization code grant. + - `relationship` (Optional) The application’s relationship to the OAuth server. The possible values are: + - `FirstParty` - The application has the same owner as the authorization server. Consent to requested OAuth scopes is granted implicitly. + - `ThirdParty` - The application is external to the authorization server. Users will be prompted to consent to requested OAuth scopes based on the application object’s `oauthConfiguration.consentMode` value. Note: An Essentials or Enterprise plan is required to utilize third-party applications. - `require_client_authentication` - (Optional) Determines if the OAuth 2.0 Token endpoint requires client authentication. If this is enabled, the client must provide client credentials when using the Token endpoint. The client_id and client_secret may be provided using a Basic Authorization HTTP header, or by sending these parameters in the request body using POST data. - `require_registration` - (Optional) When enabled the user will be required to be registered, or complete registration before redirecting to the configured callback in the authorization code grant or the implicit grant. This configuration does not currently apply to any other grant. - `provided_scope_policy` - (Optional) Configures which of the default scopes are enabled and required. @@ -178,6 +186,13 @@ resource "fusionauth_application" "Forum" { * `profile` * `enabled` - (Optional) * `required` - (Optional) + - `unknown_scope_policy` Controls the policy for handling unknown scopes on an OAuth request. The possible values are: + - `Allow` - Unknown scopes will be allowed on the request, passed through the OAuth workflow, and written to the resulting tokens without consent. + - `Remove` - Unknown scopes will be removed from the OAuth workflow, but the workflow will proceed without them. + - `Reject` - Unknown scopes will be rejected and cause the OAuth workflow to fail with an error. + - `scope_handling_policy` Controls the policy for handling of OAuth scopes when populating JWTs and the UserInfo response. The possible values are: + - `Compatibility` - OAuth workflows will populate JWT and UserInfo claims in a manner compatible with versions of FusionAuth before version 1.50.0. + - `Strict` - OAuth workflows will populate token and UserInfo claims according to the OpenID Connect 1.0 specification based on requested and consented scopes. * `registration_configuration` - (Optional) - `birth_date` - (Optional) * `enabled` - (Optional) diff --git a/fusionauth/resource_fusionauth_application.go b/fusionauth/resource_fusionauth_application.go index b02de78..1c6d55f 100644 --- a/fusionauth/resource_fusionauth_application.go +++ b/fusionauth/resource_fusionauth_application.go @@ -151,6 +151,11 @@ func newApplication() *schema.Resource { Optional: true, Description: "The unique Id of the lambda that will be used to perform additional validation on registration form steps.", }, + "userinfo_populate_id": { + Type: schema.TypeString, + Optional: true, + Description: "The Id of the Lambda that will be invoked when a UserInfo response is generated for this application.", + }, }, }, }, @@ -625,6 +630,17 @@ func newOAuthConfiguration() *schema.Resource { Description: "The OAuth 2.0 client id. If you leave this blank during a POST, a client id will be generated for you. If you leave this blank during PUT, the previous value will be maintained. For both POST and PUT you can provide a value and it will be stored.", Computed: true, }, + "consent_mode": { + Type: schema.TypeString, + Optional: true, + Default: fusionauth.OAuthScopeConsentMode_AlwaysPrompt.String(), + Description: "Controls the policy for prompting a user to consent to requested OAuth scopes. This configuration only takes effect when `application.oauthConfiguration.relationship` is `ThirdParty`. The possible values are: `AlwaysPrompt` - Always prompt the user for consent. `RememberDecision` - Remember previous consents; only prompt if the choice expires or if the requested or required scopes have changed. The duration of this persisted choice is controlled by the Tenant’s `externalIdentifierConfiguration.rememberOAuthScopeConsentChoiceTimeToLiveInSeconds` value. `NeverPrompt` - The user will be never be prompted to consent to requested OAuth scopes. Permission will be granted implicitly as if this were a `FirstParty` application. This configuration is meant for testing purposes only and should not be used in production.", + ValidateFunc: validation.StringInSlice([]string{ + fusionauth.OAuthScopeConsentMode_AlwaysPrompt.String(), + fusionauth.OAuthScopeConsentMode_RememberDecision.String(), + fusionauth.OAuthScopeConsentMode_NeverPrompt.String(), + }, false), + }, "debug": { Type: schema.TypeBool, Optional: true, @@ -673,6 +689,22 @@ func newOAuthConfiguration() *schema.Resource { }, false), Description: "Determines the PKCE requirements when using the authorization code grant.", }, + "provided_scope_policy": { + Type: schema.TypeList, + Optional: true, + Computed: true, + Elem: newOAuthConfigurationProvidedScopePolicy(), + }, + "relationship": { + Type: schema.TypeString, + Optional: true, + Default: fusionauth.OAuthApplicationRelationship_FirstParty.String(), + Description: "The application’s relationship to the OAuth server. The possible values are: `FirstParty` - The application has the same owner as the authorization server. Consent to requested OAuth scopes is granted implicitly. `ThirdParty` - The application is external to the authorization server. Users will be prompted to consent to requested OAuth scopes based on the application object’s `oauthConfiguration.consentMode` value. Note: An Essentials or Enterprise plan is required to utilize third-party applications.", + ValidateFunc: validation.StringInSlice([]string{ + fusionauth.OAuthApplicationRelationship_FirstParty.String(), + fusionauth.OAuthApplicationRelationship_ThirdParty.String(), + }, false), + }, "require_client_authentication": { Type: schema.TypeBool, Optional: true, @@ -686,10 +718,24 @@ func newOAuthConfiguration() *schema.Resource { Default: false, Description: "When enabled the user will be required to be registered, or complete registration before redirecting to the configured callback in the authorization code grant or the implicit grant. This configuration does not currently apply to any other grant.", }, - "provided_scope_policy": { - Type: schema.TypeList, - Optional: true, - Elem: newOAuthConfigurationScopePolicy(), + "scope_handling_policy": { + Type: schema.TypeString, + Required: true, + Description: "Controls the policy for handling of OAuth scopes when populating JWTs and the UserInfo response. The possible values are: `Compatibility` - OAuth workflows will populate JWT and UserInfo claims in a manner compatible with versions of FusionAuth before version 1.50.0. `Strict` - OAuth workflows will populate token and UserInfo claims according to the OpenID Connect 1.0 specification based on requested and consented scopes.", + ValidateFunc: validation.StringInSlice([]string{ + fusionauth.OAuthScopeHandlingPolicy_Compatibility.String(), + fusionauth.OAuthScopeHandlingPolicy_Strict.String(), + }, false), + }, + "unknown_scope_policy": { + Type: schema.TypeString, + Required: true, + Description: "Controls the policy for handling unknown scopes on an OAuth request. The possible values are: `Allow` - Unknown scopes will be allowed on the request, passed through the OAuth workflow, and written to the resulting tokens without consent. `Remove` - Unknown scopes will be removed from the OAuth workflow, but the workflow will proceed without them. `Reject` - Unknown scopes will be rejected and cause the OAuth workflow to fail with an error.", + ValidateFunc: validation.StringInSlice([]string{ + fusionauth.UnknownScopePolicy_Allow.String(), + fusionauth.UnknownScopePolicy_Remove.String(), + fusionauth.UnknownScopePolicy_Reject.String(), + }, false), }, }, } @@ -860,7 +906,7 @@ func newRegistrationConfiguration() *schema.Resource { } } -func newOAuthConfigurationScopePolicy() *schema.Resource { +func newOAuthConfigurationProvidedScopePolicy() *schema.Resource { requireable := func() *schema.Resource { return &schema.Resource{ Schema: map[string]*schema.Schema{ diff --git a/fusionauth/resource_fusionauth_application_helpers.go b/fusionauth/resource_fusionauth_application_helpers.go index f755ba0..31b6954 100644 --- a/fusionauth/resource_fusionauth_application_helpers.go +++ b/fusionauth/resource_fusionauth_application_helpers.go @@ -44,6 +44,7 @@ func buildApplication(data *schema.ResourceData) fusionauth.Application { IdTokenPopulateId: data.Get("lambda_configuration.0.id_token_populate_id").(string), Samlv2PopulateId: data.Get("lambda_configuration.0.samlv2_populate_id").(string), SelfServiceRegistrationValidationId: data.Get("lambda_configuration.0.self_service_registration_validation_id").(string), + UserinfoPopulateId: data.Get("lambda_configuration.0.userinfo_populate_id").(string), }, LoginConfiguration: fusionauth.LoginConfiguration{ AllowTokenRefresh: data.Get("login_configuration.0.allow_token_refresh").(bool), @@ -67,21 +68,25 @@ func buildApplication(data *schema.ResourceData) fusionauth.Application { AuthorizedURLValidationPolicy: fusionauth.Oauth2AuthorizedURLValidationPolicy(data.Get("oauth_configuration.0.authorized_url_validation_policy").(string)), ClientAuthenticationPolicy: fusionauth.ClientAuthenticationPolicy(data.Get("oauth_configuration.0.client_authentication_policy").(string)), ClientSecret: data.Get("oauth_configuration.0.client_secret").(string), + ConsentMode: fusionauth.OAuthScopeConsentMode(data.Get("oauth_configuration.0.consent_mode").(string)), Debug: data.Get("oauth_configuration.0.debug").(bool), DeviceVerificationURL: data.Get("oauth_configuration.0.device_verification_url").(string), + EnabledGrants: buildGrants("oauth_configuration.0.enabled_grants", data), GenerateRefreshTokens: data.Get("oauth_configuration.0.generate_refresh_tokens").(bool), + LogoutBehavior: fusionauth.LogoutBehavior(data.Get("oauth_configuration.0.logout_behavior").(string)), LogoutURL: data.Get("oauth_configuration.0.logout_url").(string), ProofKeyForCodeExchangePolicy: fusionauth.ProofKeyForCodeExchangePolicy(data.Get("oauth_configuration.0.proof_key_for_code_exchange_policy").(string)), - RequireClientAuthentication: data.Get("oauth_configuration.0.require_client_authentication").(bool), - LogoutBehavior: fusionauth.LogoutBehavior(data.Get("oauth_configuration.0.logout_behavior").(string)), - EnabledGrants: buildGrants("oauth_configuration.0.enabled_grants", data), - RequireRegistration: data.Get("oauth_configuration.0.require_registration").(bool), ProvidedScopePolicy: fusionauth.ProvidedScopePolicy{ Address: buildRequireable("oauth_configuration.0.provided_scope_policy.0.address", data), Email: buildRequireable("oauth_configuration.0.provided_scope_policy.0.email", data), Phone: buildRequireable("oauth_configuration.0.provided_scope_policy.0.phone", data), Profile: buildRequireable("oauth_configuration.0.provided_scope_policy.0.profile", data), }, + Relationship: fusionauth.OAuthApplicationRelationship(data.Get("oauth_configuration.0.relationship").(string)), + RequireClientAuthentication: data.Get("oauth_configuration.0.require_client_authentication").(bool), + RequireRegistration: data.Get("oauth_configuration.0.require_registration").(bool), + ScopeHandlingPolicy: fusionauth.OAuthScopeHandlingPolicy(data.Get("oauth_configuration.0.scope_handling_policy").(string)), + UnknownScopePolicy: fusionauth.UnknownScopePolicy(data.Get("oauth_configuration.0.unknown_scope_policy").(string)), }, PasswordlessConfiguration: fusionauth.PasswordlessConfiguration{ Enableable: buildEnableable("passwordless_configuration_enabled", data), @@ -256,6 +261,7 @@ func buildResourceDataFromApplication(a fusionauth.Application, data *schema.Res "id_token_populate_id": a.LambdaConfiguration.IdTokenPopulateId, "samlv2_populate_id": a.LambdaConfiguration.Samlv2PopulateId, "self_service_registration_validation_id": a.LambdaConfiguration.SelfServiceRegistrationValidationId, + "userinfo_populate_id": a.LambdaConfiguration.UserinfoPopulateId, }, }) if err != nil { @@ -297,15 +303,47 @@ func buildResourceDataFromApplication(a fusionauth.Application, data *schema.Res "client_authentication_policy": a.OauthConfiguration.ClientAuthenticationPolicy, "client_secret": a.OauthConfiguration.ClientSecret, "client_id": a.OauthConfiguration.ClientId, + "consent_mode": a.OauthConfiguration.ConsentMode, "debug": a.OauthConfiguration.Debug, "device_verification_url": a.OauthConfiguration.DeviceVerificationURL, + "enabled_grants": a.OauthConfiguration.EnabledGrants, "generate_refresh_tokens": a.OauthConfiguration.GenerateRefreshTokens, - "logout_url": a.OauthConfiguration.LogoutURL, - "require_client_authentication": a.OauthConfiguration.RequireClientAuthentication, "logout_behavior": a.OauthConfiguration.LogoutBehavior, - "enabled_grants": a.OauthConfiguration.EnabledGrants, - "require_registration": a.OauthConfiguration.RequireRegistration, + "logout_url": a.OauthConfiguration.LogoutURL, "proof_key_for_code_exchange_policy": a.OauthConfiguration.ProofKeyForCodeExchangePolicy, + "provided_scope_policy": []map[string]interface{}{ + { + "address": []map[string]interface{}{ + { + "enabled": a.OauthConfiguration.ProvidedScopePolicy.Address.Enabled, + "required": a.OauthConfiguration.ProvidedScopePolicy.Address.Required, + }, + }, + "email": []map[string]interface{}{ + { + "enabled": a.OauthConfiguration.ProvidedScopePolicy.Email.Enabled, + "required": a.OauthConfiguration.ProvidedScopePolicy.Email.Required, + }, + }, + "phone": []map[string]interface{}{ + { + "enabled": a.OauthConfiguration.ProvidedScopePolicy.Phone.Enabled, + "required": a.OauthConfiguration.ProvidedScopePolicy.Phone.Required, + }, + }, + "profile": []map[string]interface{}{ + { + "enabled": a.OauthConfiguration.ProvidedScopePolicy.Profile.Enabled, + "required": a.OauthConfiguration.ProvidedScopePolicy.Profile.Required, + }, + }, + }, + }, + "relationship": a.OauthConfiguration.Relationship, + "require_client_authentication": a.OauthConfiguration.RequireClientAuthentication, + "require_registration": a.OauthConfiguration.RequireRegistration, + "scope_handling_policy": a.OauthConfiguration.ScopeHandlingPolicy, + "unknown_scope_policy": a.OauthConfiguration.UnknownScopePolicy, }, }) if err != nil {