From ef7b11ac01d7bfd92a217faf0cc195197b1aa54f Mon Sep 17 00:00:00 2001 From: DingHao Date: Fri, 22 Nov 2024 07:38:44 +0800 Subject: [PATCH] Delay initialization UserDetailsService in Global Authentication --- ...alizeUserDetailsBeanManagerConfigurer.java | 59 ++++--------------- 1 file changed, 11 insertions(+), 48 deletions(-) diff --git a/config/src/main/java/org/springframework/security/config/annotation/authentication/configuration/InitializeUserDetailsBeanManagerConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/authentication/configuration/InitializeUserDetailsBeanManagerConfigurer.java index 3383a5f20f..c2a47b7610 100644 --- a/config/src/main/java/org/springframework/security/config/annotation/authentication/configuration/InitializeUserDetailsBeanManagerConfigurer.java +++ b/config/src/main/java/org/springframework/security/config/annotation/authentication/configuration/InitializeUserDetailsBeanManagerConfigurer.java @@ -16,8 +16,7 @@ package org.springframework.security.config.annotation.authentication.configuration; -import java.util.ArrayList; -import java.util.List; +import java.util.Arrays; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -66,9 +65,10 @@ class InitializeUserDetailsBeanManagerConfigurer extends GlobalAuthenticationCon @Override public void configure(AuthenticationManagerBuilder auth) throws Exception { - List> userDetailsServices = getBeansWithName(UserDetailsService.class); + String[] beanNames = InitializeUserDetailsBeanManagerConfigurer.this.context + .getBeanNamesForType(UserDetailsService.class); if (auth.isConfigured()) { - if (!userDetailsServices.isEmpty()) { + if (beanNames.length > 0) { this.logger.warn("Global AuthenticationManager configured with an AuthenticationProvider bean. " + "UserDetailsService beans will not be used for username/password login. " + "Consider removing the AuthenticationProvider bean. " @@ -78,19 +78,18 @@ class InitializeUserDetailsBeanManagerConfigurer extends GlobalAuthenticationCon return; } - if (userDetailsServices.isEmpty()) { + if (beanNames.length == 0) { return; } - else if (userDetailsServices.size() > 1) { - List beanNames = userDetailsServices.stream().map(BeanWithName::getName).toList(); + else if (beanNames.length > 1) { this.logger.warn(LogMessage.format("Found %s UserDetailsService beans, with names %s. " + "Global Authentication Manager will not use a UserDetailsService for username/password login. " - + "Consider publishing a single UserDetailsService bean.", userDetailsServices.size(), - beanNames)); + + "Consider publishing a single UserDetailsService bean.", beanNames.length, + Arrays.toString(beanNames))); return; } - UserDetailsService userDetailsService = userDetailsServices.get(0).getBean(); - String userDetailsServiceBeanName = userDetailsServices.get(0).getName(); + UserDetailsService userDetailsService = InitializeUserDetailsBeanManagerConfigurer.this.context + .getBean(beanNames[0], UserDetailsService.class); PasswordEncoder passwordEncoder = getBeanOrNull(PasswordEncoder.class); UserDetailsPasswordService passwordManager = getBeanOrNull(UserDetailsPasswordService.class); CompromisedPasswordChecker passwordChecker = getBeanOrNull(CompromisedPasswordChecker.class); @@ -111,8 +110,7 @@ class InitializeUserDetailsBeanManagerConfigurer extends GlobalAuthenticationCon provider.afterPropertiesSet(); auth.authenticationProvider(provider); this.logger.info(LogMessage.format( - "Global AuthenticationManager configured with UserDetailsService bean with name %s", - userDetailsServiceBeanName)); + "Global AuthenticationManager configured with UserDetailsService bean with name %s", beanNames[0])); } /** @@ -127,41 +125,6 @@ class InitializeUserDetailsBeanManagerConfigurer extends GlobalAuthenticationCon return InitializeUserDetailsBeanManagerConfigurer.this.context.getBean(beanNames[0], type); } - /** - * @return a list of beans of the requested class, along with their names. If - * there are no registered beans of that type, the list is empty. - */ - private List> getBeansWithName(Class type) { - List> beanWithNames = new ArrayList<>(); - String[] beanNames = InitializeUserDetailsBeanManagerConfigurer.this.context.getBeanNamesForType(type); - for (String beanName : beanNames) { - T bean = InitializeUserDetailsBeanManagerConfigurer.this.context.getBean(beanName, type); - beanWithNames.add(new BeanWithName(bean, beanName)); - } - return beanWithNames; - } - - static class BeanWithName { - - private final T bean; - - private final String name; - - BeanWithName(T bean, String name) { - this.bean = bean; - this.name = name; - } - - T getBean() { - return this.bean; - } - - String getName() { - return this.name; - } - - } - } }