Spring Boot Security OAuth2 JWT代码
时间: 2024-02-22 09:47:19 浏览: 92
以下是一个基于Spring Boot Security、OAuth2和JWT实现的示例代码:
1. 添加Maven依赖
在pom.xml文件中添加以下依赖:
```
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-security</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.security</groupId>
<artifactId>spring-security-oauth2</artifactId>
<version>2.3.7.RELEASE</version>
</dependency>
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt</artifactId>
<version>0.9.1</version>
</dependency>
```
2. 配置Spring Security
在Spring Boot应用程序中,您可以通过@Configuration配置类来配置Spring Security。以下是一个示例配置类:
```
@Configuration
@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {
@Autowired
private UserDetailsService userDetailsService;
@Autowired
private PasswordEncoder passwordEncoder;
@Override
protected void configure(HttpSecurity http) throws Exception {
http.csrf().disable()
.authorizeRequests()
.antMatchers("/oauth/**").permitAll()
.antMatchers("/api/**").authenticated()
.and()
.formLogin().permitAll()
.and()
.logout().permitAll();
}
@Override
protected void configure(AuthenticationManagerBuilder auth) throws Exception {
auth.userDetailsService(userDetailsService).passwordEncoder(passwordEncoder);
}
}
```
在上面的配置类中,我们禁用了CSRF保护,并配置了访问权限。具体来说:
- /oauth/** URL模式应该允许所有人访问,因为我们将使用OAuth2协议进行身份验证和授权。
- /api/** URL模式应该需要身份验证。
- 我们还配置了表单登录和注销。
3. 配置OAuth2
在Spring Boot应用程序中,您可以使用@Configuration配置类来配置OAuth2。以下是一个示例配置类:
```
@Configuration
@EnableAuthorizationServer
public class OAuth2Config extends AuthorizationServerConfigurerAdapter {
@Autowired
private AuthenticationManager authenticationManager;
@Autowired
private UserDetailsService userDetailsService;
@Autowired
private PasswordEncoder passwordEncoder;
@Bean
public JwtAccessTokenConverter jwtAccessTokenConverter() {
JwtAccessTokenConverter converter = new JwtAccessTokenConverter();
converter.setSigningKey("my-signing-key"); // 设置JWT签名密钥
return converter;
}
@Bean
public JwtTokenStore jwtTokenStore() {
return new JwtTokenStore(jwtAccessTokenConverter());
}
@Override
public void configure(ClientDetailsServiceConfigurer clients) throws Exception {
clients.inMemory()
.withClient("my-client-id")
.secret(passwordEncoder.encode("my-client-secret"))
.authorizedGrantTypes("password", "refresh_token")
.scopes("read", "write")
.accessTokenValiditySeconds(3600)
.refreshTokenValiditySeconds(86400);
}
@Override
public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
endpoints.authenticationManager(authenticationManager)
.userDetailsService(userDetailsService)
.accessTokenConverter(jwtAccessTokenConverter())
.tokenStore(jwtTokenStore());
}
}
```
在上面的配置类中,我们使用@EnableAuthorizationServer注释启用OAuth2,并实现了AuthorizationServerConfigurer接口以配置客户端和端点。具体来说:
- 我们配置了一个内存中的客户端,使用密码和刷新令牌授权类型,以及读写作用域。
- 我们还配置了JWT令牌转换器和令牌存储。
- 我们将身份验证管理器、用户详细信息服务、JWT令牌转换器和令牌存储配置为端点。
4. 实现用户详细信息服务
我们需要实现UserDetailsService接口来加载用户详细信息。以下是一个示例实现:
```
@Service
public class UserDetailsServiceImpl implements UserDetailsService {
@Autowired
private UserRepository userRepository;
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
User user = userRepository.findByUsername(username);
if (user == null) {
throw new UsernameNotFoundException("User not found with username: " + username);
}
return new org.springframework.security.core.userdetails.User(
user.getUsername(), user.getPassword(), Collections.emptyList());
}
}
```
在上面的实现中,我们使用Spring Data JPA从数据库中加载用户,并创建一个SimpleGrantedAuthority对象列表作为用户的权限。
5. 实现密码编码器
我们需要实现PasswordEncoder接口,以便在创建用户时对密码进行编码。以下是一个示例实现:
```
@Service
public class PasswordEncoderImpl implements PasswordEncoder {
@Override
public String encode(CharSequence rawPassword) {
return BCrypt.hashpw(rawPassword.toString(), BCrypt.gensalt());
}
@Override
public boolean matches(CharSequence rawPassword, String encodedPassword) {
return BCrypt.checkpw(rawPassword.toString(), encodedPassword);
}
}
```
在上面的实现中,我们使用BCrypt编码算法对密码进行编码和验证。
6. 实现控制器
最后,我们需要实现一个控制器来测试OAuth2和JWT。以下是一个示例实现:
```
@RestController
@RequestMapping("/api")
public class ApiController {
@GetMapping("/hello")
public String hello() {
return "Hello, World!";
}
@GetMapping("/user")
public Principal user(Principal principal) {
return principal;
}
}
```
在上面的实现中,我们有两个端点:/api/hello和/api/user。前者返回一个简单的字符串,后者返回当前用户的Principal对象。
7. 测试应用程序
现在,您可以启动应用程序并使用以下步骤测试OAuth2和JWT:
- 获取访问令牌。使用以下curl命令以密码授权方式获取访问令牌:
```
curl -X POST \
-d 'grant_type=password&username=my-username&password=my-password' \
-H 'Authorization: Basic bXktY2xpZW50LWlkOm15LWNsaWVudC1zZWNyZXQ=' \
http://localhost:8080/oauth/token
```
在上面的命令中,my-username和my-password应该是您的用户名和密码,bXktY2xpZW50LWlkOm15LWNsaWVudC1zZWNyZXQ=应该是Base64编码的客户端ID和客户端密钥。
- 使用访问令牌访问API。使用以下curl命令使用JWT访问/api/hello端点:
```
curl -H 'Authorization: Bearer <access-token>' \
http://localhost:8080/api/hello
```
在上面的命令中,<access-token>应该是您在第一步中获取的访问令牌。
- 获取用户信息。使用以下curl命令使用JWT访问/api/user端点:
```
curl -H 'Authorization: Bearer <access-token>' \
http://localhost:8080/api/user
```
在上面的命令中,<access-token>应该是您在第一步中获取的访问令牌。
阅读全文