Skip to content

Design for Enhance AAD token authentication converter to customized granted authorities converter

Shili Chen edited this page Dec 7, 2022 · 3 revisions

1 Context

The customer reported the issue AAD braking changes blocked the SCA upgrade from 3.6 to 4.0, not support a custom granted author converter anymore in the AAD token authentication converter.

Let's see the related classes structure and what happened:

1.1 Class diagram relationship before version 3.7

![class-diagram-relationship](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/class-diagram-relationship.png)

  • The AADJwtBearerTokenAuthenticationConverter was used in the default configuration AADResourceServerWebSecurityConfigurerAdapter as a custom JWT authentication converter for the Resource Server scenario.

  • The AADB2CJwtBearerTokenAuthenticationConverter can be used in a customer Resource Server configuration, as a custom JWT authentication converter for the Resource Server scenario. SCA does not provide a default configuration to use for Azure AD B2C side.

    The customer wants the feature implemented in class AbstractJwtBearerTokenAuthenticationConverter.

1.2 Breaking changes since 3.8

The PR Deprecate AADB2CJwtBearerTokenAuthenticationConverter deleted the class AbstractJwtBearerTokenAuthenticationConverter,

The below methods are removed, and they are not added back to the subclass AADJwtBearerTokenAuthenticationConverter.

![breaking-changes](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/breaking-changes.png)

2 Cause analysis

This PR Deprecate AADB2CJwtBearerTokenAuthenticationConverter has removed the class AbstractJwtBearerTokenAuthenticationConverter and hardcoded the Aad JWT granted authorities converter AADJwtGrantedAuthoritiesConverter, this is the blocker for the customer upgrade to 3.8 or 4.0.

2.1 Analyze the PR motivations

  • Reduce code redundancy(AADJwtBearerTokenAuthenticationConverter and AADB2CJwtBearerTokenAuthenticationConverter).
  • Simplify the class AADJwtBearerTokenAuthenticationConverter.

New Class diagram relationship:

![new-class-diagram-relationship](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/new-class-diagram-relationship.png)

2.2 Subjective reasons

  • There was no design review to ensure the rationality and accuracy of this modification
  • The PR reviewer did not check carefully.
  • The SCA release pipeline has not set up an API review process to monitor and do approval.

3 Solution design

3.1 Goal

  • Keep the API unchanged.
  • Enhance the deprecated token authentication converter to add back the customized JWT granted authorities converter support.
  • Enhance the configurer to support the Jwt-granted authorities converter.

3.2 Token authentication Converter

A token authentication converter is required to define a security configurer JwtConfigurer, which is a part of the security configurer OAuth2ResourceServerConfigurer.

![token-authentication-converter](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/token-authentication-converter.png)

3.2.1 Enhance the deprecated AadJwtBearerTokenAuthenticationConverter for 4.x

At present, the converter AadJwtBearerTokenAuthenticationConverter has missing function and does not support customized JWT granted authorities converter.

![enhance-deprecated-converter](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/enhance-deprecated-converter.png)

Solution Make the class AadJwtBearerTokenAuthenticationConverter support customized JWT granted authorities converter, not only the converter AadJwtGrantedAuthoritiesConverter.

![solution-enhance-deprecated-converter](https://github.com/Azure/azure-sdk-for-java/wiki/spring/design-docs/resources/design-for-enhance-aad- token-authentication-converter-to-customized-granted-authorities-converter/solution-enhance-deprecated-converter.png)

3.2.2 Use Spring Security JwtAuthenticationConverter for 6.x

❌ AadJwtBearerTokenAuthenticationConverter: this converter has been deleted in 6.x (PR) ✅ JwtAuthenticationConverter: recommend user use this Spring security built-in converter for the resource server. 🛎️ The JwtAuthenticationConverter already supports setPrincipalClaimName() and setJwtGrantedAuthoritiesConverter.

Solution No changes needed.

3.3 Security configurer for Resource Server

3.3.1 How a security configurer is used in a Resource Server application

3.3.1.1 Spring Boot 2.x

Sample code for using WebSecurityConfigurerAdapter

@EnableWebSecurity
@EnableGlobalMethodSecurity(prePostEnabled = true)
public static class ResourceServerWebSecurityConfigurerAdapter extends
    WebSecurityConfigurerAdapter {

    @Override
    protected void configure(HttpSecurity http) throws Exception {
      http.oauth2ResourceServer()
            .jwt()
              .jwtAuthenticationConverter(new JwtAuthenticationConverter());
    }
}

Sample code for using Azure AD configurer adapter:

@EnableWebSecurity
@EnableGlobalMethodSecurity(prePostEnabled = true)
public static class EnhancedResourceServerWebSecurityConfigurerAdapter extends
	AadResourceServerWebSecurityConfigurerAdapter {

	@Override
	protected void configure(HttpSecurity http) throws Exception {
	    super.configure(http);
	}
}

3.3.1.2 Spring Boot 3.x

Sample code for using AbstractHttpConfigurer:

@EnableWebSecurity
@EnableMethodSecurity
static class EnhancedResourceServerConfiguration {
    
    @Bean
    SecurityFilterChain enhancedResourceServerFilterChain(HttpSecurity http) throws Exception {
        http.apply(EnhancedResourceServerHttpSecurityConfigurer.enhancedResourceServer());
        return http.build();
    }
}

public class EnhancedResourceServerHttpSecurityConfigurer extends AbstractHttpConfigurer<EnhancedResourceServerHttpSecurityConfigurer, HttpSecurity> {

    @Override
    public void init(HttpSecurity builder) throws Exception {
        super.init(builder);
        builder.oauth2ResourceServer()
                 .jwt()
                   .jwtAuthenticationConverter(new JwtAuthenticationConverter());
    }
    
    public static EnhancedResourceServerHttpSecurityConfigurer enhancedResourceServer() {
        return new EnhancedResourceServerHttpSecurityConfigurer();
    }
}

Sample code for using Azure AD Security Configurer:

@EnableWebSecurity
@EnableMethodSecurity
static class EnhancedResourceServerConfiguration {

    @Bean
    SecurityFilterChain enhancedResourceServerFilterChain(HttpSecurity http) throws Exception {
        http.apply(AadResourceServerHttpSecurityConfigurer.aadResourceServer());
        return http.build();
    }
}

3.3.2 Enhance configurer for 4.x

Make the default configurer AadResourceServerWebSecurityConfigurerAdapter support the customized JWT granted authorities converter.

Solution:

class AadResourceServerWebSecurityConfigurerAdapter extends WebSecurityConfigurerAdapter {

  public AadResourceServerWebSecurityConfigurerAdapter(AadResourceServerProperties properties,
					Converter<Jwt, Collection<GrantedAuthority>> jwtGrantedAuthoritiesConverter) {}

  protected Converter<Jwt, Collection<GrantedAuthority>> jwtGrantedAuthoritiesConverter() {}
} 

3.3.3 Enhance configurer for 6.x

Make the default configurer AadResourceServerHttpSecurityConfigurer support the customized JWT granted authorities converter through the custom DSL.

Solution:

public class AadResourceServerHttpSecurityConfigurer extends AbstractHttpConfigurer<AadResourceServerHttpSecurityConfigurer, HttpSecurity> {

    public AadResourceServerHttpSecurityConfigurer jwtGrantedAuthoritiesConverter(
        Converter<Jwt, Collection<GrantedAuthority>> jwtGrantedAuthoritiesConverter) {
    }
} 
Clone this wiki locally