做婚庆网站的想法,无极网站建设,华蓥网站建设,精美的php个人网站源码存储用户信息的方式
获取用户信息的流程
用户提交账号和密码后,DaoAuthenticationProvider调用UserDetailsService接口实现类的loadUserByUsername()方法,该方法可以接收请求参数username的值,然后根据该值查询用户信息,最后将账号,密码,权限封装到UserDetails对象中并返回给…存储用户信息的方式
获取用户信息的流程
用户提交账号和密码后,DaoAuthenticationProvider调用UserDetailsService接口实现类的loadUserByUsername()方法,该方法可以接收请求参数username的值,然后根据该值查询用户信息,最后将账号,密码,权限封装到UserDetails对象中并返回给Spring security框架
public interface UserDetailsService {// 根据用户名获取用户信息的方法,不同的实现类对应不同的实现方式UserDetails loadUserByUsername(String var1) throws UsernameNotFoundException;
}// UserDetails是用户信息接口
public interface UserDetails extends Serializable {Collection? extends GrantedAuthority getAuthorities();String getPassword();String getUsername();boolean isAccountNonExpired();boolean isAccountNonLocked();boolean isCredentialsNonExpired();boolean isEnabled();
}基于内存的方式存储用户信息
InMemoryUserDetailsManager是UserDetailsService接口的实现类,可以将用户信息以硬编码的方式存储在内存中并封装到UserDetails对象中
package com.xuecheng.auth.config;
EnableWebSecurity
EnableGlobalMethodSecurity(securedEnabled true,prePostEnabled true)
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {Beanpublic UserDetailsService userDetailsService() {// 1. 配置用户信息服务InMemoryUserDetailsManager manager new InMemoryUserDetailsManager();// 2. 创建用户信息这里暂时写死,后面需要从数据库中动态查询, Kyle的权限是p1Lucy的权限是p2// User是Spring Security提供的工具类manager.createUser(User.withUsername(Kyle).password(123).authorities(p1).build());manager.createUser(User.withUsername(Lucy).password(456).authorities(p2).build());return manager;}// 设置密码比对的规则Beanpublic PasswordEncoder passwordEncoder() {// 采用明文的方式直接比对return NoOpPasswordEncoder.getInstance();}protected void configure(HttpSecurity http) throws Exception {http.authorizeRequests().antMatchers(/r/**)// 表示/r/**开头的请求需要认证.authenticated().anyRequest().permitAll().and().formLogin().successForwardUrl(/login-success);// 配置退出登录页面,认证成功后访问/logout可退出登录http.logout().logoutUrl(/logout);}
}基于数据库存储用户信息和密码编码格式
实际开发中认证服务所需要的用户信息都是存储在数据库中,所以我们要从数据库中查询用户信息
第一步屏蔽掉原先的InMemoryUserDetailsManager, 自定义一个UserDetailsService接口的实现类UserDetailsImpl并重写loadUserByUsername()方法该方法可以接收URL中携带的请求参数username的值
我们重写方法的业务逻辑是根据用户的账号名从数据库中查询对应的用户信息,然后将账号密码用户权限信息封装到UserDetails类型的对象中并返回给Spring Security框架不设置权限会报Cannot pass a null GrantedAuthority collection
Service
public class UserDetailsImpl implements UserDetailsService {AutowiredXcUserMapper xcUserMapper;/**** param name 用户输入的登录账号* return UserDetails* throws UsernameNotFoundException*/Overridepublic UserDetails loadUserByUsername(String name) throws UsernameNotFoundException {// 根据username去XcUser表中查询对应的用户信息XcUser user xcUserMapper.selectOne(new LambdaQueryWrapperXcUser().eq(XcUser::getUsername, name));// 返回NULL表示用户不存在SpringSecurity会帮我们处理即抛出异常表示用户不存在if (user null) {return null;}// 设置用户权限// String[] authorities {test};// 取出数据库中用户的密码String password user.getPassword();// 将用户名和密码以及用户权限(权限必须要设置)封装成一个UserDetails对象,然后返回即可return User.withUsername(user.getUsername()).password(password).authorities(test).build();}
}第二部Spring Security框架将URL中的请求参数password的值即用户的登陆密码和UserDetails对象中从数据库中查询到的密码进行比对
数据库中的密码是加密过的,但用户登陆时输入的密码是明文形式,所以我们要修改WebSecurityConfig配置类中的密码格式器PasswordEncoder指定的密码校验方式
密码格式描述NoOpPasswordEncoder表示通过明文的方式比较密码BCryptPasswordEncoder表示先将用户输入的密码编码为BCrypt格式后才与数据库中的密码进行比对
Bean
public PasswordEncoder passwordEncoder() { // return NoOpPasswordEncoder.getInstance();return new BCryptPasswordEncoder();
}BCryptPasswordEncoder原理: 虽然同样的原始密码可以对应不同的Hash值,但是比对密码时如果调用它提供的matches方法即使Hash值不同但校验是正确的
先将数据库表中用户密码的存储方式改为BCrype格式, 申请令牌时输入的密码需要的还是明文形式
public static void main(String[] args) {String password 123456;BCryptPasswordEncoder encoder new BCryptPasswordEncoder();for (int i 0; i 5; i) {// 每个计算出的Hash值都不一样String encodePsw encoder.encode(password);// 虽然Hash值不一样但是校验是可以通过的System.out.println(转换后密码 encodePsw 比对情况 encoder.matches(password, encodePsw));}
}// 转换后密码$2a$10$6hbvtCtgcISvbBHJ.UnhPO1io7StF.ySPkmAvzO/efvNmHVVJZOeK比对情况true
// 转换后密码$2a$10$ufYW9qXSAk0N201B/wCR7uGrzygawnwXtyL2vKpDLAOCOkF33sGnK比对情况true
// 转换后密码$2a$10$DEaVxYHakIE/kDvAU4eC7OZ7c9kqKBJedClVxDPnYH.zwuZvCRnzm比对情况true
// 转换后密码$2a$10$s2qgaKGgULYQ7tce2u6TIeHopap4HqfyghJYu1vdDZ2WcNk70ykFe比对情况true
// 转换后密码$2a$10$XQaQJIfXyd/UvMHC..uBNuDXNVrZHnEGn.tW0oSB6WVjdsZLFpkGq比对情况true由于我们修改了框架密码的校验方式,所以还需要将认证服务批准接入的客户端的密钥也更改为BCrypt的编码格式
// 配置客户端的详情信息
Override
public void configure(ClientDetailsServiceConfigurer clients) throws Exception {clients.inMemory()// 使用in-memory存储.withClient(XcWebApp)// 客户端的Id.secret(new BCryptPasswordEncoder().encode(XcWebApp))//客户端密钥转为BCrypt格式.resourceIds(xuecheng-plus)//资源列表.authorizedGrantTypes(authorization_code, password, client_credentials, implicit, refresh_token)// 允许的授权类型.scopes(all)// 允许的授权范围.autoApprove(false)//false跳转到授权页面 .redirectUris(http://localhost/);//客户端接收授权码的重定向地址
}第三步重启认证服务使用HttpClient通过密码模式申请令牌,要保证登陆的明文密码在数据库中有对应的BCrypt格式的加密密码
###### 密码模式
POST localhost:53070/auth/oauth/token?client_idXcWebAppclient_secretXcWebAppgrant_typepasswordusernameKylepassword111111第四步查看服务端响应的结果
// 输入正确的账号密码即可成功申请令牌
{
access_token: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJhdWQiOlsieHVlY2hlbmctcGx1cyJdLCJ1c2VyX25hbWUiOiJLeWxlIiwic2NvcGUiOlsiYWxsIl0sImV4cCI6MTY3ODQ1MTUxNiwiYXV0aG9yaXRpZXMiOlsidGVzdCJdLCJqdGkiOiJkOWUwYjU0ZS03Zjg4LTQ2NjAtYjFlZS04ZWQzYjYzZmQwNjMiLCJjbGllbnRfaWQiOiJYY1dlYkFwcCJ9.NaS3hmpDtX3zkXvnZWoo9ROEWgYeA6GoxBzy_lOxzvA,
token_type: bearer,
refresh_token: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJhdWQiOlsieHVlY2hlbmctcGx1cyJdLCJ1c2VyX25hbWUiOiJLeWxlIiwic2NvcGUiOlsiYWxsIl0sImF0aSI6ImQ5ZTBiNTRlLTdmODgtNDY2MC1iMWVlLThlZDNiNjNmZDA2MyIsImV4cCI6MTY3ODcwMzUxNiwiYXV0aG9yaXRpZXMiOlsidGVzdCJdLCJqdGkiOiI5YjE4OTUwOS1iYmEzLTRkMTctYTNkNC05OWQwZGI5NjU0MDAiLCJjbGllbnRfaWQiOiJYY1dlYkFwcCJ9.548w5CdQiIU5_k1qRBjzM-PMBqy-XX3zr17tQS6g6CM,
expires_in: 7199,
scope: all,
jti: d9e0b54e-7f88-4660-b1ee-8ed3b63fd063
}// 输入错误的密码申请令牌失败
{
error: invalid_grant,
error_description: 用户名或密码错误
}扩展用户身份信息的方式(UserDetails)
用户表中除了存储用户的账号密码还有其他信息,但是UserDetails接口中只有username、password字段,这样JWT令牌中也只会写入用户的账号和密码信息
在认证阶段DaoAuthenticationProvider会调用我们自定义的UserDetailsService接口实现类的loadUserByUsername()方法查询数据库表中用户全部的信息,但是UserDetails中只能存储账号和密码
public interface UserDetails extends Serializable {Collection? extends GrantedAuthority getAuthorities();String getPassword();String getUsername();boolean isAccountNonExpired();boolean isAccountNonLocked();boolean isCredentialsNonExpired();boolean isEnabled();
}由于JWT令牌中用户身份信息来源于UserDetails, 所以我们只能把UserDetails中包含的用户信息写入JWT令牌中,想要扩展用户的身份信息有两种方式
修改源码: 对框架提供的UserDetails的属性进行扩展,使之包括更多的自定义属性存储Json(推荐): 我们还可以将用户的全部信息转为Json数据存储到UserDetails的username属性(需要脱敏)并不一定要只存储用户账号
Service
public class UserDetailsImpl implements UserDetailsService {AutowiredXcUserMapper xcUserMapper;/**** param name 用户输入的登录账号* return UserDetails* throws UsernameNotFoundException*/Overridepublic UserDetails loadUserByUsername(String name) throws UsernameNotFoundException {// 根据username去XcUser表中查询对应的用户信息XcUser user xcUserMapper.selectOne(new LambdaQueryWrapperXcUser().eq(XcUser::getUsername, name));// 返回NULL表示用户不存在SpringSecurity会帮我们处理即抛出异常表示用户不存在if (user null) {return null;}// 设置用户权限// String[] authorities {test};// 取出数据库中用户的密码String password user.getPassword();// 将查询到的用户信息映射的XcUser对象转为Json,注意对用户的敏感信息要进行脱敏user.setPassword(null);String userString JSON.toJSONString(user);// 将用户名和密码以及用户权限(权限必须要设置)封装成一个UserDetails对象,然后返回即可return User.withUsername(userString).password(password).authorities(test).build();}
}重启认证服务使用密码模式重新获取令牌
// 密码模式
POST localhost:53070/auth/oauth/token?client_idXcWebAppclient_secretXcWebAppgrant_typepasswordusernameKylepassword111111访问校验接口查看JWT令牌中的内容,此时令牌的user_name属性中包含了查询到的用户全部信息(密码为null)
// 访问校验接口
POST localhost:53070/auth/oauth/check_token?tokeneyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJhdWQiOlsieHVlY2hlbmctcGx1cyJdLCJ1c2VyX25hbWUiOiJ7XCJjb21wYW55SWRcIjpcIjEyMzIxNDE0MjVcIixcImNyZWF0ZVRpbWVcIjpcIjIwMjItMDktMjhUMDg6MzI6MDNcIixcImlkXCI6XCI1MlwiLFwibmFtZVwiOlwiS2lraVwiLFwicGFzc3dvcmRcIjpcIiQyYSQxMCQwcHQ3V2xmVGJuUERUY1d0cC8uMk11NUNUWHZvaG5OUWhSNjI4cXE0Um9LU2MwZEdBZEVnbVwiLFwic2V4XCI6XCIxXCIsXCJzdGF0dXNcIjpcIlwiLFwidXNlcm5hbWVcIjpcIkt5bGVcIixcInV0eXBlXCI6XCIxMDEwMDJcIn0iLCJzY29wZSI6WyJhbGwiXSwiZXhwIjoxNjc4NDUyMzU0LCJhdXRob3JpdGllcyI6WyJ0ZXN0Il0sImp0aSI6Ijc2MDc0MDI4LTBiM2MtNDQ4Mi1hN2Y0LTc1NDI3ZTA2OTFjMSIsImNsaWVudF9pZCI6IlhjV2ViQXBwIn0._GKfGE2s5k0n6VC4_RKQrzdzydWY-WtX3Q_Hc4DxQ1g{aud: [xuecheng-plus],user_name: {\companyId\:\1232141425\,\createTime\:\2022-09-28T08:32:03\,\id\:\52\,\name\:\Kiki\,\password\:\$2a$10$0pt7WlfTbnPDTcWtp/.2Mu5CTXvohnNQhR628qq4RoKSc0dGAdEgm\,\sex\:\1\,\status\:\\,\username\:\Kyle\,\utype\:\101002\},scope: [all],active: true,exp: 1678452354,authorities: [test],jti: 76074028-0b3c-4482-a7f4-75427e0691c1,client_id: XcWebApp
}资源服务获取用户身份信息
现在认证服务生成JWT令牌的user_name属性已经存储用户JSON格式的信息,在资源服务中就可以取出该JSON格式的内容转换为用户对象去使用
第一步: 在content-api工程中写一个工具类SecurityUtil用来在各个微服务中获取到当前登录用户信息的对象
Slf4j
public class SecurityUtil {public static XcUser getUser() {try {// 通过SecurityContextHolder获取user_name属性的值即包含用户信息的Json字符串Object principal SecurityContextHolder.getContext().getAuthentication().getPrincipal();if (principal instanceof String) {// 将包含用户信息的Json字符串转换为XcUser对象String userJson principal.toString();XcUser xcUser JSON.parseObject(userJson, XcUser.class);return xcUser;}} catch (Exception e) {log.error(获取当前登录用户身份信息出错{}, e.getMessage());e.printStackTrace();}return null;}// 这里使用内部类是为了不让content工程去依赖auth工程Datapublic static class XcUser implements Serializable {private static final long serialVersionUID 1L;private String id;private String username;private String password;private String salt;private String name;private String nickname;private String wxUnionid;private String companyId;/*** 头像*/private String userpic;private String utype;private LocalDateTime birthday;private String sex;private String email;private String cellphone;private String qq;/*** 用户状态*/private String status;private LocalDateTime createTime;private LocalDateTime updateTime;}
}第二步: 在内容管理服务中的查询课程信息接口中使用工具类获取当前登陆的用户信息
ApiOperation(根据课程id查询课程基础信息)
GetMapping(/course/{courseId})
public CourseBaseInfoDto getCourseBaseById(PathVariable Long courseId) {SecurityUtil.XcUser user SecurityUtil.getUser();System.out.println(当前用户身份为 user);return courseBaseInfoService.getCourseBaseInfo(courseId);
}第三步: 启动认证服务、网关、内容管理服务, 首先访问认证服务生成令牌,然后携带生成的令牌访问内容管理服务的查询课程接口,最后在控制台中查看登陆的用户信息
当前用户身份为SecurityUtil.XcUser(id52, usernameKyle, passwordnull, saltnull, nameKiki, nicknamenull, wxUnionidnull, companyId1232141425, userpicnull, utype101002, birthdaynull, sex1, emailnull, cellphonenull, qqnull, status, createTime2022-09-28T08:32:03, updateTimenull)