Следуя примеру, я реализовал авторизацию с помощью jwt, мои пользователи добавлены, но у меня возникают проблемы во время авторизации, авторизация не происходит, почта всегда будет уникальной в моем проекте, что мне делать? В этом случае используйте тот же метод UsernamePasswordAuthenticationToken?
@RestController
@RequestMapping(value = "/api/v1/auth/")
public class AuthenticationRestControllerV1 {
private final AuthenticationManager authenticationManager;
private final JwtTokenProvider jwtTokenProvider;
private final UserService userService;
@Autowired
public AuthenticationRestControllerV1(AuthenticationManager authenticationManager, JwtTokenProvider jwtTokenProvider, UserService userService) {
this.authenticationManager = authenticationManager;
this.jwtTokenProvider = jwtTokenProvider;
this.userService = userService;
}
@PostMapping("/login")
public ResponseEntity login(@RequestBody AuthenticationRequestDto requestDto) {
System.out.println(requestDto.getUsername());
System.out.println(requestDto.getPassword());
try {
String username = requestDto.getUsername();
authenticationManager.authenticate(new UsernamePasswordAuthenticationToken(username, requestDto.getPassword()));
User user = userService.findByUsername(username);
if (user == null) {
throw new UsernameNotFoundException("User with username: " + username + " not found");
}
String token = jwtTokenProvider.createToken(username, user.getRoles());
Map<Object, Object> response = new HashMap<>();
response.put("username", username);
response.put("token", token);
return ResponseEntity.ok(response);
} catch (AuthenticationException e) {
throw new BadCredentialsException("Invalid username or password");
}
}
}
В этот момент возникает проблема authenticationManager.authenticate (новый UsernamePasswordAuthenticationToken (username, requestDto.getPassword ()));
public class JwtConfigurer extends SecurityConfigurerAdapter<DefaultSecurityFilterChain, HttpSecurity> {
private JwtTokenProvider jwtTokenProvider;
public JwtConfigurer(JwtTokenProvider jwtTokenProvider) {
this.jwtTokenProvider = jwtTokenProvider;
}
@Override
public void configure(HttpSecurity httpSecurity) throws Exception {
JwtTokenFilter jwtTokenFilter = new JwtTokenFilter(jwtTokenProvider);
httpSecurity.addFilterBefore(jwtTokenFilter, UsernamePasswordAuthenticationFilter.class);
}
}
public class JwtTokenFilter extends GenericFilterBean {
@Autowired
private JwtTokenProvider jwtTokenProvider;
public JwtTokenFilter(JwtTokenProvider jwtTokenProvider) {
this.jwtTokenProvider = jwtTokenProvider;
}
@Override
public void doFilter(ServletRequest req, ServletResponse res, FilterChain filterChain)
throws IOException, ServletException {
String token = jwtTokenProvider.resolveToken((HttpServletRequest) req);
if (token != null && jwtTokenProvider.validateToken(token)) {
Authentication auth = jwtTokenProvider.getAuthentication(token);
if (auth != null) {
SecurityContextHolder.getContext().setAuthentication(auth);
}
}
filterChain.doFilter(req, res);
}
}
@Component
public class JwtTokenProvider {
@Value("${jwt.token.secret}")
private String secret;
@Value("${jwt.token.expired}")
private long validityInMilliseconds;
@Autowired
private UserDetailsService userDetailsService;
@Bean
public BCryptPasswordEncoder passwordEncoder() {
BCryptPasswordEncoder bCryptPasswordEncoder = new BCryptPasswordEncoder();
return bCryptPasswordEncoder;
}
@PostConstruct
protected void init() {
secret = Base64.getEncoder().encodeToString(secret.getBytes());
}
public String createToken(String username, Set<Role> roles) {
Claims claims = Jwts.claims().setSubject(username);
claims.put("roles", getRolesName(roles));
Date now = new Date();
Date validity = new Date(now.getTime() + validityInMilliseconds);
return Jwts.builder()//
.setClaims(claims)//
.setIssuedAt(now)//
.setExpiration(validity)//
.signWith(SignatureAlgorithm.HS256, secret)//
.compact();
}
/*
Метод возвращает аутентификацию по токену
*/
public Authentication getAuthentication(String token) {
UserDetails userDetails = this.userDetailsService.loadUserByUsername(getUsername(token));
return new UsernamePasswordAuthenticationToken(userDetails, "", userDetails.getAuthorities());
}
public String getUsername(String token) {
return Jwts.parser().setSigningKey(secret).parseClaimsJws(token).getBody().getSubject();
}
public String resolveToken(HttpServletRequest req) {
String bearerToken = req.getHeader("Authorization");
if (bearerToken != null && bearerToken.startsWith("Bearer_")) {
return bearerToken.substring(7, bearerToken.length());
}
return null;
}
public boolean validateToken(String token) {
try {
Jws<Claims> claims = Jwts.parser().setSigningKey(secret).parseClaimsJws(token);
if (claims.getBody().getExpiration().before(new Date())) {
return false;
}
return true;
} catch (JwtException | IllegalArgumentException e) {
throw new JwtAuthenticationException("JWT token is expired or invalid");
}
}
private List<String> getRolesName(Set<Role> roles) {
List<String> result = new ArrayList<>();
roles.forEach(role -> {
result.add(role.name());
});
return result;
}
}
public class JwtUser implements UserDetails {
private final Integer id;
private final String username;
private final String surname;
private final String patronymic;
private final String email;
private final String password;
private final Date birthDay;
private final Boolean active;
private final String mobileNumber;
private final Collection<? extends GrantedAuthority> authorities;
private final String activationCode;
public JwtUser(Integer id,
String username,
String surname,
String patronymic,
String email,
String password,
Date birthDay,
Boolean active,
String mobileNumber,
Collection<? extends GrantedAuthority> authorities,
String activationCode) {
this.id = id;
this.username = username;
this.surname = surname;
this.patronymic = patronymic;
this.email = email;
this.password = password;
this.birthDay = birthDay;
this.active = active;
this.mobileNumber = mobileNumber;
this.authorities = authorities;
this.activationCode = activationCode;
}
public final class JwtUserFactory {
public JwtUserFactory() {
}
public static JwtUser create(User user) {
return new JwtUser(
user.getId(),
user.getUsername(),
user.getSurname(),
user.getPatronymic(),
user.getEmail(),
user.getPassword(),
user.getBirthDay(),
user.getActive(),
user.getMobileNumber(),
mapToGA(user.getRoles()),
user.getActivationCode()
);
}
public static Set<GrantedAuthority> mapToGA(Set<Role> roles) {
return roles.stream().map(role ->
new SimpleGrantedAuthority(role.name())).
collect(Collectors.toSet());
}
}
@Service
public class JwtUserDetailsService implements UserDetailsService {
@Autowired
private UserService userService;
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
User user = userService.findByUsername(username);
if (user == null) {
throw new UsernameNotFoundException("User with this username" + username + " not found");
}
JwtUser jwtUser = JwtUserFactory.create(user);
return jwtUser;
}
}
Авторизация не работает. Я получаю ошибку org.springframework.security.authentication.BadCredentialsException: Invalid username or password
в строке authenticationManager.authenticate (new UsernamePasswordAuthenticationToken (username, requestDto.getPassword ())) ;
. И, как я заметил в моем случае, это уникальная почта, как это сделать с помощью ее авторизации? что изменить?
Это то, что я отслеживаю при отладке org.springframework.security.authentication. UsernamePasswordAuthenticationToken@4138b2c7: Принципал: xxx@gmail.com; Учетные данные: [PROTECTED]; Аутентифицировано: ложно; Детали: ноль; Не предоставлено никаких полномочий
public enum Role implements GrantedAuthority {
ADMIN,
USER,
WATCHER;
public static Role lookup(String name) {
for (Role role : Role.values()) {
if (role.name().equals(name)) {
return role;
}
}
throw new IllegalArgumentException("No matching constant for " + Role.class.getSimpleName() + " name = " + name);
}
@Override
public String getAuthority() {
return null;
}
}