Ошибка создания бина с именем authenticationTokenFilterBean: неудовлетворенная зависимость выражается через поле tokenUtils - PullRequest
0 голосов
/ 01 января 2019

Я хочу реализовать Spring Security в моем проекте.Но я продолжаю получать ту же ошибку независимо от того, что я пытаюсь.

Я создал необходимые классы (TokenUtils, AuthenticationTokenFilter, SpringConfiguration).Они находятся в одном пакете, но я получаю следующую ошибку:

Это часть моего Stack Trace, где возникает проблема

2018-12-31 23:58: 10.616 INFO 9952 --- [main] j.LocalContainerEntityManagerFactoryBean: Инициализированный JPA EntityManagerFactory для единицы сохраняемости 'default' 2018-12-31 23: 58: 11.444 ОШИБКА 9952 --- [main] osbweb.embedded.tomcat.TomcatStarter:Ошибка запуска контекста Tomcat.Исключение: org.springframework.beans.factory.UnsatisfiedDependencyException.Сообщение: Ошибка создания компонента с именем authenticationTokenFilterBean: Неудовлетворенная зависимость, выраженная через поле «tokenUtils»;вложенное исключение - org.springframework.beans.factory.NoSuchBeanDefinitionException: нет доступного квалифицирующего компонента типа 'com.sbvtransport.sbvtransport.security.TokenUtils': ожидается как минимум один компонент, который считается кандидатом на автоматическое подключение.Аннотации зависимостей: {@ org.springframework.beans.factory.annotation.Autowired (обязательно = true)} 2018-12-31 23: 58: 11.479 INFO 9952 --- [main] o.apache.catalina.core.StandardService:Остановка службы [Tomcat] 2018-12-31 23: 58: 11.495 WARN 9952 --- [main] oacloader.WebappClassLoaderBase: веб-приложение [ROOT], по-видимому, запустило поток с именем [HikariPool-1 housekeeper], но не смоглопрекрати этоЭто очень вероятно, чтобы создать утечку памяти.Трассировка стека потока: java.base@11.0.1/jdk.internal.misc.Unsafe.park (собственный метод) java.base@11.0.1/java.util.concurrent.locks.LockSupport.parkNanos (LockSupport.java:234) java.base@11.0.1/java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.awaitNanos (AbstractQueuedSynchronizer.java:2123) java.base@11.0.1/java.util.concurrent.ScheedueorkScheduledThreadPoolExecutor.java:1182) java.base@11.0.1/java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take (ScheduledThreadPoolExecutor.java:899) java.base@11.0.1/javaurExk.ThreadPoolExecutor.java:1054) java.base@11.0.1/java.util.concurrent.ThreadPoolExecutor.runWorker (ThreadPoolExecutor.java:1114) java.base@11.0.1/java.util.concurrent.ThreadPoolExecrunWorkThreadPoolExecutor.java:628) java.base@11.0.1/java.lang.Thread.run (Thread.java:834) ПРЕДУПРЕЖДЕНИЕ: произошла недопустимая операция доступа с отражением ПРЕДУПРЕЖДЕНИЕ: незаконный доступ с отражениемs by org.apache.catalina.loader.WebappClassLoaderBase (файл: / C: /Users/Danijela/.m2/repository/org/apache/tomcat/embed/tomcat-embed-core/9.0.12/tomcat-embed-core-9.0.12.jar) в поле java.io.ObjectStreamClass $ Caches.localDescs ПРЕДУПРЕЖДЕНИЕ. Пожалуйста, рассмотрите возможность сообщения об этом сопровождающим org.apache.catalina.loader.WebappClassLoaderBase ПРЕДУПРЕЖДЕНИЕ: используйте --illegal-access = warn, чтобы включить предупреждениядальнейших недопустимых операций доступа с отражением ПРЕДУПРЕЖДЕНИЕ. Все недопустимые операции доступа будут отклонены в будущем выпуске 2018-12-31 23: 58: 11.502 WARN 9952 --- [main] ConfigServletWebServerApplicationContext: во время инициализации контекста возникла исключительная ситуация - отмена попытки обновления: org.springframework.context.ApplicationContextException: Невозможно запустить веб-сервер;вложенное исключение: org.springframework.boot.web.server.WebServerException: невозможно запустить встроенный Tomcat 2018-12-31 23: 58: 11.502 INFO 9952 --- [main] j.LocalContainerEntityManagerFactoryBean: закрытие JPA EntityManagerFactory для модуля сохранения по умолчанию ''2018-12-31 23: 58: 11.502 ИНФОРМАЦИЯ 9952 --- [main] com.zaxxer.hikari.HikariDataSource: HikariPool-1 - Завершение работы инициировано ...2018-12-31 23: 58: 11.517 INFO 9952 --- [main] com.zaxxer.hikari.HikariDataSource: HikariPool-1 - Завершение работы завершено.2018-12-31 23: 58: 11.533 INFO 9952 --- [main] ConditionEvaluationReportLoggingListener:

Error starting ApplicationContext. To display the conditions report re-run your application with 'debug' enabled.
2018-12-31 23:58:11.845 ERROR 9952 --- [           main] o.s.b.d.LoggingFailureAnalysisReporter   : 

***************************
APPLICATION FAILED TO START
***************************

Description:

   Field tokenUtils in      com.sbvtransport.sbvtransport.security.AuthenticationTokenFilter required a bean of type 'com.sbvtransport.sbvtransport.security.TokenUtils' that could not be found.

   The injection point has the following annotations:
- @org.springframework.beans.factory.annotation.Autowired(required=true)


Action:

Consider defining a bean of type 'com.sbvtransport.sbvtransport.security.TokenUtils' in your configuration.    
package com.sbvtransport.sbvtransport.security;

Это мои классы безопасности:

AuthenticationTokenFilter

import java.io.IOException;

import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.security.core.userdetails.UserDetails;
import org.springframework.security.core.userdetails.UserDetailsService;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;
import org.springframework.security.web.authentication.WebAuthenticationDetailsSource;

public class AuthenticationTokenFilter  extends UsernamePasswordAuthenticationFilter{

@Autowired
TokenUtils tokenUtils;

@Autowired
private UserDetailsService userDetailsService;

@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
        throws IOException, ServletException {
    HttpServletRequest httpRequest = (HttpServletRequest) request;
    String authToken = httpRequest.getHeader("Authorization");
    String username = tokenUtils.getUsernameFromToken(authToken);

    if (username != null && SecurityContextHolder.getContext().getAuthentication() == null) {
        UserDetails userDetails = this.userDetailsService.loadUserByUsername(username);
        if (tokenUtils.validateToken(authToken, userDetails)) {
            UsernamePasswordAuthenticationToken authentication = new UsernamePasswordAuthenticationToken(
                    userDetails, null, userDetails.getAuthorities());
            authentication.setDetails(new WebAuthenticationDetailsSource().buildDetails(httpRequest));
            SecurityContextHolder.getContext().setAuthentication(authentication);
        }
    }

    chain.doFilter(request, response);
}

}

TokenUtils

package com.sbvtransport.sbvtransport.security;

import java.util.Date;
import java.util.HashMap;
import java.util.Map;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.security.core.userdetails.UserDetails;

import io.jsonwebtoken.Claims;
import io.jsonwebtoken.Jwts;
import io.jsonwebtoken.SignatureAlgorithm;

public class TokenUtils {

@Value("SuperSecreat")
private String secret;

@Value("86400") // 24 hours(in seconds)
private Long expiration;

public String getUsernameFromToken(String token) {
    String username;
    try {
        Claims claims = this.getClaimsFromToken(token);
        username = claims.getSubject();
    } catch (Exception e) {
        username = null;
    }
    return username;
}

private Claims getClaimsFromToken(String token) {
    Claims claims;
    try {
        claims = Jwts.parser().setSigningKey(this.secret).parseClaimsJws(token).getBody();
    } catch (Exception e) {
        claims = null;
    }
    return claims;
}

public Date getExpirationDateFromToken(String token) {
    Date expirationDate;
    try {
        final Claims claims = this.getClaimsFromToken(token);
        expirationDate = claims.getExpiration();
    } catch (Exception e) {
        expirationDate = null;
    }
    return expirationDate;
}

private boolean isTokenExpired(String token) {
    final Date expirationDate = this.getExpirationDateFromToken(token);
    return expirationDate.before(new Date(System.currentTimeMillis()));
}

public boolean validateToken(String token, UserDetails userDetails) {
    final String username = getUsernameFromToken(token);
    return username.equals(userDetails.getUsername()) && !isTokenExpired(token);
}

public String generateToken(UserDetails userDetails) {
    Map<String, Object> claims = new HashMap<String, Object>();
    claims.put("sub", userDetails.getUsername());
    claims.put("created", new Date(System.currentTimeMillis()));
    return Jwts.builder().setClaims(claims).setExpiration(new Date(System.currentTimeMillis() + expiration * 1000))
            .signWith(SignatureAlgorithm.HS512, secret).compact();
}

}

SecurityConfiguration

package com.sbvtransport.sbvtransport.security;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.authentication.AuthenticationManager;
import org.springframework.security.config.annotation.authentication.builders.AuthenticationManagerBuilder;
import org.springframework.security.config.annotation.method.configuration.EnableGlobalMethodSecurity;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity;
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;
import org.springframework.security.config.http.SessionCreationPolicy;
import org.springframework.security.core.userdetails.UserDetailsService;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
import org.springframework.security.crypto.password.PasswordEncoder;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;

@SpringBootApplication
@Configuration
@EnableWebSecurity
@EnableAutoConfiguration
@EnableGlobalMethodSecurity(prePostEnabled = true)
public class SecurityConfiguration extends WebSecurityConfigurerAdapter {
@Autowired
private UserDetailsService userDetailsService;

@Autowired
public void configureAuthentication(AuthenticationManagerBuilder authenticationManagerBuilder) throws Exception {

    authenticationManagerBuilder.userDetailsService(this.userDetailsService).passwordEncoder(passwordEncoder());
}

@Bean
public PasswordEncoder passwordEncoder() {
    return new BCryptPasswordEncoder();
}

@Bean
@Override
public AuthenticationManager authenticationManagerBean() throws Exception {
    return super.authenticationManagerBean();
}

@Bean
public AuthenticationTokenFilter authenticationTokenFilterBean() throws Exception {
    AuthenticationTokenFilter authenticationTokenFilter = new AuthenticationTokenFilter();
    authenticationTokenFilter.setAuthenticationManager(authenticationManagerBean());
    return authenticationTokenFilter;
}

@Override
protected void configure(HttpSecurity httpSecurity) throws Exception {
    httpSecurity.csrf().disable().sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS).and()
            .authorizeRequests().antMatchers("").permitAll();

    httpSecurity.addFilterBefore(authenticationTokenFilterBean(), UsernamePasswordAuthenticationFilter.class);
}

  }

Кажется, есть проблема с автопроводкойTokenUtils внутри AuthenticationTokenFilter, но я тоже могу ошибаться.

Надеюсь, мой вопрос достаточно ясен, если я могу что-то сделать для его улучшения, предложите его.

1 Ответ

0 голосов
/ 01 января 2019

Ошибка: В сообщении об ошибке четко указано, что TokenUtils бин не найден

'com.sbvtransport.sbvtransport.security.TokenUtils', который не может быть найден.

Вам не хватает @Component на TokenUtils, при объявлении класса с помощью @Component выполняется компонентное сканирование для создания одноэлементного компонента этого класса в контексте приложения

@Component
public class TokenUtils {

И @ConfigurationAuthenticationTokenFilter

@Configuration
public class AuthenticationTokenFilter  extends UsernamePasswordAuthenticationFilter{

@ SpringBootApplication включает проверку компонентов в объявленном пакете

@ EnableAutoConfiguration: включить механизм автоконфигурации Spring Boot

@ ComponentScan: включить сканирование @Component для пакета, в котором находится приложение (см. Рекомендации)

@ Конфигурация: разрешить регистрировать дополнительные компоненты в контексте или импортировать дополнительные классы конфигурации

Аннотация @SpringBootApplication эквивалентна использованию @Configuration, @EnableAutoConfiguration и @ComponentScan с их атрибутами по умолчанию,

...