SpringCloud-Oauth2 不同类型用户认证

oauth2提供了UserDetailsService类中loadUserByUsername方法来让我们查询数据库,返回查询到的数据,但是只提供了一个参数的方式,

最近在项目中,遇到了小程序端用户和管理端用户分表的情况,那么这种单参数的认证是无法满足的,需要根据不同场景来做不同的令牌发放处理,
百度之后找到了答案,很感谢,
参考参考自此博客
解决之后记录一下

  @Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
        return null;
    }

1.用户数据库的查询是由DaoAuthenticationProvider类中retrieveUser方法来做的,方法中调用userDetailsService 的loadUserByUsername单参方法来进行数据库查询,那么我们自己实现AbstractUserDetailsAuthenticationProvider类,然后改写retrieveUser方式,使用多参,是不是可以呢?答案是可以的,下面来 看具体该怎么做

1.创建一个自定义类去实现userDetailsService ,并增加一个重载的方法,两个参数,来进行多类型用户认证

根据不同的loginType来进行不同方式的数据库查询,令牌发放
下方的代码逻辑仅供参考,具体业务看你自己的

/**
 * @author Sir_小三
 */
@Component
@Slf4j
public class CustomUserDetailsService implements UserDetailsService  {


    @Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
        return null;
    }

   public UserDetails loadUserByUsername(String account, String loginType) throws UsernameNotFoundException{
       //角色集合
       List<GrantedAuthority> grantedAuthorities = new ArrayList<>();
        if(ZcConst.ADMIN_LOGIN_TYPE.equals(loginType)){
            ZcSysUser sysUser = zcSysUserService.getOne(new QueryWrapper<ZcSysUser>().eq("account", account));
            if(ZcConst.IS_DELETED.equals(sysUser.getIsDeleted())){
                throw new UsernameNotFoundException("此用户已被删除");
            }
            //查询用户角色,管理端需要
            List<ZcSysRoleUser> list = zcSysRoleUserService.list( new QueryWrapper<ZcSysRoleUser>().eq("user_id", sysUser.getSysId()));
            for (int i = 0; i < list.size(); i++) {
                ZcSysRole role = zcSysRoleService.getById(list.get(i).getRoleId());
                grantedAuthorities.add(role);
            }
           log.info("UserDetailsService:管理端登陆,查询用户成功,交由oauth2进行下一步密码认证");
            SecurityUser securityUser = new SecurityUser(account, sysUser.getPassword(), grantedAuthorities);
            return securityUser;
        }
        if(ZcConst.APP_LOGIN_TYPE.equals(loginType)){
            ZcWebUser webUser = zcWebUserService.getOne(new QueryWrapper<ZcWebUser>().eq("open_id", account));
            if(ZcConst.IS_DELETED.equals(webUser.getIsDeleted())){
                throw new UsernameNotFoundException("此用户已被删除");
            }
            SecurityUser securityUser = new SecurityUser(account, webUser.getPassword(), grantedAuthorities);
            log.info("UserDetailsService:web用户端登陆,查询用户成功,交由oauth2进行下一步密码认证");
            return securityUser;
        }
       throw new UsernameNotFoundException("没有匹配的登陆类型:loginType");
    }
}

2.我们只需要自定义一个类,比如说A类,实现AbstractUserDetailsAuthenticationProvider类,
然后将DaoAuthenticationProvider类中的所有方法复制出来,放到A类中,并更改retrieveUser方法,在retrieveUser方法中,获取前端传过来的登陆类型,然后调用我们自己的
UserDetailsService类中的重载两个参数的方法
注意!!!!需要A类中的UserDetailsService对象换成上面自定义的
CustomUserDetailsService的,不然你没有两个参数的方法可以掉。。。。

改这里:CustomUserDetailsService 自定义的类

    /**
     * 这里改为自定义的CustomUserDetailsService
     */
    public void setUserDetailsService(CustomUserDetailsService userDetailsService) {
        this.userDetailsService = userDetailsService;
    }

    /**
     * 这里改为自定义的CustomUserDetailsService
     */
    protected CustomUserDetailsService getUserDetailsService() {
        return userDetailsService;
    }
    @Override
    protected final UserDetails retrieveUser(String username,
                                             UsernamePasswordAuthenticationToken authentication)
            throws AuthenticationException {
        prepareTimingAttackProtection();
        // 自定义添加
        Map<String,String> map = (Map<String, String>) authentication.getDetails();
        try {
            // 自定义添加
            String loginType = map.get("loginType");
            // 自定义添加userType参数
            UserDetails loadedUser = this.getUserDetailsService().loadUserByUsername(username,loginType);
            if (loadedUser == null) {
                throw new InternalAuthenticationServiceException(
                        "UserDetailsService returned null, which is an interface contract violation");
            }
            return loadedUser;
        }
        catch (UsernameNotFoundException ex) {
            mitigateAgainstTimingAttack(authentication);
            throw ex;
        }
        catch (InternalAuthenticationServiceException ex) {
            throw ex;
        }
        catch (Exception ex) {
            throw new InternalAuthenticationServiceException(ex.getMessage(), ex);
        }
    

2最后一步在认证服务配置SerurityConfig中配置

    /**
     * 不同类型平台用户登陆,
     * 自定义bean之后会走我们定义得CustomAuthenticationProvider,调用CustomUserDetailsService定义的多参loadUserByUsername方法
     * 需要将CustomAuthenticationProvider中默认的UserDetailsService换为我们自己定义的CustomUserDetailsService
     * 不配置,默认走 DaoAuthenticationProvider,然后调用UserDetailsService中的单参数方法loadUserByUsername
     * 不适用于多种类型用户登陆
     * @return
     */
    @Bean
    public AuthenticationProvider customAuthenticationProvider() {
        CustomAuthenticationProvider customAuthenticationProvider= new CustomAuthenticationProvider();
        customAuthenticationProvider.setUserDetailsService(customUserDetailsService);
        customAuthenticationProvider.setHideUserNotFoundExceptions(false);
        customAuthenticationProvider.setPasswordEncoder(passwordEncoder());
        return customAuthenticationProvider;
    }

    /**
     * 不同类型平台用户登陆
     * @param auth
     */
    @Override
    protected void configure(AuthenticationManagerBuilder auth){
        auth.authenticationProvider(customAuthenticationProvider());
    }

到这里就ok了,通过接收不同类型,进行不同认证处理
SpringCloud-Oauth2 不同类型用户认证_第1张图片

你可能感兴趣的:(spring-security)