简体   繁体   English

Spring Boot JPA Starter的Spring Boot启动问题

[英]Spring Boot startup issue with Spring Boot JPA Starter

I'm getting this peculiar error when trying to start up my spring boot project with spring-boot-starter-data-jpa . 尝试使用spring-boot-starter-data-jpa启动我的spring boot项目时,出现了这个特殊错误。 What is weird is that my application would start before I added a couple repositories and services but I can't seem to narrow down why spring cannot initialize a repository that was working before the additions. 奇怪的是,我的应用程序会在添加几个存储库和服务之前启动,但是我似乎无法缩小为什么spring无法初始化在添加存储库之前可以使用的存储库。

Here is the related errors: 这是相关的错误:

11:38:42.313 INFO  org.hibernate.Version.logVersion() @37 [localhost-startStop-1] - HHH000412: Hibernate Core {5.0.11.Final}
11:38:42.316 INFO  org.hibernate.cfg.Environment.<clinit>() @213 [localhost-startStop-1] - HHH000206: hibernate.properties not found 
11:38:42.319 INFO  org.hibernate.cfg.Environment.buildBytecodeProvider() @317 [localhost-startStop-1] - HHH000021: Bytecode provider name : javassist
11:38:42.430 INFO  org.hibernate.annotations.common.Version.<clinit>() @66 [localhost-startStop-1] - HCANN000001: Hibernate Commons Annotations {5.0.1.Final}
11:38:43.165 INFO  org.hibernate.dialect.Dialect.<init>() @156 [localhost-startStop-1] - HHH000400: Using dialect: org.hibernate.dialect.MySQL5Dialect
11:38:43.583 ERROR org.springframework.boot.context.embedded.tomcat.TomcatStarter.onStartup() @63 [localhost-startStop-1] - Error starting Tomcat context. Exception: org.springframework.beans.factory.BeanCreationException. Message: Error creating bean with name 'emailAuthenticationFilter' defined in class path resource [gg/leet/security/WebSecurityConfig.class]: Bean instantiation via factory method failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [gg.leet.security.EmailAuthenticationFilter]: Factory method 'emailAuthenticationFilter' threw exception; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'authenticationManager' defined in class path resource [gg/leet/security/WebSecurityConfig.class]: Bean instantiation via factory method failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.springframework.security.authentication.AuthenticationManager]: Factory method 'authenticationManager' threw exception; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'emailAuthenticationProvider': Unsatisfied dependency expressed through field 'userService'; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userServiceImpl' defined in file [/Users/andrew/Programs/leet-tournaments/backend/build/classes/main/gg/leet/service/UserServiceImpl.class]: Unsatisfied dependency expressed through constructor parameter 0; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'userRepository': Cannot create inner bean '(inner bean)#76212c93' of type [org.springframework.orm.jpa.SharedEntityManagerCreator] while setting bean property 'entityManager'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name '(inner bean)#76212c93': Cannot resolve reference to bean 'entityManagerFactory' while setting constructor argument; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'entityManagerFactory' defined in class path resource [org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfiguration.class]: Invocation of init method failed; nested exception is javax.persistence.PersistenceException: [PersistenceUnit: default] Unable to build Hibernate SessionFactory
11:38:43.617 WARN  org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext.refresh() @550 [restartedMain] - Exception encountered during context initialization - cancelling refresh attempt: org.springframework.context.ApplicationContextException: Unable to start embedded container; nested exception is org.springframework.boot.context.embedded.EmbeddedServletContainerException: Unable to start embedded Tomcat
11:38:43.631 ERROR org.springframework.boot.SpringApplication.reportFailure() @839 [restartedMain] - Application startup failed

The most interesting being the line: 最有趣的是该行:

11:38:43.583 ERROR org.springframework.boot.context.embedded.tomcat.TomcatStarter.onStartup() @63 [localhost-startStop-1] - Error starting Tomcat context. 11:38:43.583错误org.springframework.boot.context.embedded.tomcat.TomcatStarter.onStartup()@ 63 [localhost-startStop-1]-启动Tomcat上下文时出错。 Exception: org.springframework.beans.factory.BeanCreationException. 异常:org.springframework.beans.factory.BeanCreationException。 Message: Error creating bean with name 'emailAuthenticationFilter' defined in class path resource [gg/leet/security/WebSecurityConfig.class]: Bean instantiation via factory method failed; 消息:创建在类路径资源[gg / leet / security / WebSecurityConfig.class]中定义的名称为'emailAuthenticationFilter'的bean时出错:通过工厂方法的Bean实例化失败。 nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [gg.leet.security.EmailAuthenticationFilter]: Factory method 'emailAuthenticationFilter' threw exception; 嵌套的异常是org.springframework.beans.BeanInstantiationException:无法实例化[gg.leet.security.EmailAuthenticationFilter]:工厂方法'emailAuthenticationFilter'引发了异常; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'authenticationManager' defined in class path resource [gg/leet/security/WebSecurityConfig.class]: Bean instantiation via factory method failed; 嵌套的异常是org.springframework.beans.factory.BeanCreationException:创建在类路径资源[gg / leet / security / WebSecurityConfig.class]中定义的名称为'authenticationManager'的bean时出错:通过工厂方法的Bean实例化失败; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.springframework.security.authentication.AuthenticationManager]: Factory method 'authenticationManager' threw exception; 嵌套的异常是org.springframework.beans.BeanInstantiationException:无法实例化[org.springframework.security.authentication.AuthenticationManager]:工厂方法'authenticationManager'引发了异常; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'emailAuthenticationProvider': Unsatisfied dependency expressed through field 'userService'; 嵌套的异常是org.springframework.beans.factory.UnsatisfiedDependencyException:创建名称为'emailAuthenticationProvider'的bean时出错:通过字段'userService'表示的不满足的依赖关系; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userServiceImpl' defined in file [/Users/andrew/Programs/leet-tournaments/backend/build/classes/main/gg/leet/service/UserServiceImpl.class]: Unsatisfied dependency expressed through constructor parameter 0; 嵌套的异常是org.springframework.beans.factory.UnsatisfiedDependencyException:在文件[/ Users / andrew / Programs / leet-tournaments / backend / build / classes / main / gg / leet / service /中创建名称为'userServiceImpl'的bean时出错[UserServiceImpl.class]:通过构造函数参数0表示的不满足的依赖关系; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'userRepository': Cannot create inner bean '(inner bean)#76212c93' of type [org.springframework.orm.jpa.SharedEntityManagerCreator] while setting bean property 'entityManager'; 嵌套异常是org.springframework.beans.factory.BeanCreationException:创建名称为'userRepository'的bean时出错:设置bean时无法创建类型为[org.springframework.orm.jpa.SharedEntityManagerCreator]的内部bean'(inner bean)#76212c93'属性“ entityManager”; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name '(inner bean)#76212c93': Cannot resolve reference to bean 'entityManagerFactory' while setting constructor argument; 嵌套的异常是org.springframework.beans.factory.BeanCreationException:创建名称为“((内部bean)#76212c93”)的bean时出错:设置构造函数参数时无法解析对bean“ entityManagerFactory”的引用; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'entityManagerFactory' defined in class path resource [org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaAutoConfiguration.class]: Invocation of init method failed; 嵌套的异常是org.springframework.beans.factory.BeanCreationException:创建在类路径资源[org / springframework / boot / autoconfigure / orm / jpa / HibernateJpaAutoConfiguration.class]中定义的名称为'entityManagerFactory'的bean时出错:调用init方法失败; nested exception is javax.persistence.PersistenceException: [PersistenceUnit: default] Unable to build Hibernate SessionFactory 嵌套的异常是javax.persistence.PersistenceException:[PersistenceUnit:默认]无法构建Hibernate SessionFactory

So the offending class that's in parameter 0 of userServiceImpl which is the userRepository . 因此,在userServiceImpl的参数0中有问题的类是userRepository

This is what my UserRepository looks like: 这是我的UserRepository的样子:

package gg.leet.repository;

import gg.leet.model.User;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.stereotype.Repository;

import java.util.List;
import java.util.Optional;

@Repository
public interface UserRepository extends JpaRepository<User, Long> {
    Optional<User> findByEmail(String email);
    @Query("select u from User u where u.id = ?1")
    Optional<User> findOneById(Long id);
    User save(User user);
    Optional<User> findOneByUsername(String username);
    List<User> findByEmailContaining(String search);
    List<User> findByFirstNameContaining(String search);
    List<User> findByLastNameContaining(String search);
    List<User> findByUsernameContaining(String search);
    @Query("select u from User u where u.username like ?1 or u.firstName like ?1 or u.lastName like ?1")
    Page<User> findByUsernameOrLastNameOrFirstNameContaining(String search, Pageable pageable);
}

UserService implementation: UserService的实现:

package gg.leet.service;

import gg.leet.model.User;
import gg.leet.repository.UserRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.stereotype.Service;

import java.util.*;

/**
 * Implementation of the UserService over the mysql repository
 */
@SuppressWarnings("unused")
@Service
public class UserServiceImpl implements UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserServiceImpl(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    @Override
    public Optional<User> getByUsername(String username) {
        return this.userRepository.findOneByUsername(username);
    }

    @Override
    public Optional<User> getByEmail(String email) { return this.userRepository.findByEmail(email); }

    @Override
    public Optional<User> getById(Long id) {
        return this.userRepository.findOneById(id);
    }

    @Override
    public User save(User user) {
        return userRepository.save(user);
    }

    @Override
    public Page<User> findByContaining(String search, Pageable pageable) {
        return this.userRepository.findByUsernameOrLastNameOrFirstNameContaining(search, pageable);
    }
}

Nothing too fancy but I don't understand why we cannot instantiate a the bean for the authentication filter. 太花哨了,但是我不明白为什么我们不能为身份验证过滤器实例化一个bean。 The error is somewhat generic so I don't know where I should be looking. 该错误有些笼统,所以我不知道应该去哪里。 I tried to manually initialize the entityManagerFactory bean within a configuration class and reference it in the @EnableJpaRepositories annotation to no avail. 我试图在配置类中手动初始化entityManagerFactory bean,并在@EnableJpaRepositories批注中对其进行引用无济于事。 Any advice on where I should also look to fix this issue? 关于在哪里也应该解决此问题的任何建议?

----- EDIT 1 ----- -----编辑1 -----

Here is my datasource configuration, fairly minimal but has worked in the past: 这是我的数据源配置,相当少,但过去一直有效:

spring:
  profiles: default
  datasource:
    url: jdbc:mysql://localhost:3306/leettournaments?autoReconnect=true&useSSL=false&useTimezone=true&serverTimezone=UTC
    username: root
    password:
  jpa:
    hibernate:
      ddl-auto: create-drop


spring:
  datasource:
    driver-class-name: com.mysql.jdbc.Driver
    max-active: 25
    max-idle: 10
  jpa:
      hibernate:
        dialect: org.hibernate.dialect.MySQLDialect

------ EDIT 2 ------ ------编辑2 ------

@SuppressWarnings("unused")
@Configuration
@EnableRedisHttpSession(maxInactiveIntervalInSeconds=Integer.MAX_VALUE, redisFlushMode=RedisFlushMode.IMMEDIATE)
@EnableWebSecurity
@EnableGlobalMethodSecurity(securedEnabled = true)
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {
    private static final Logger LOGGER = LogManager.getLogger(WebSecurityConfig.class);

    @Value("${gg.leet.debug}")
    private boolean debug;

    /**
     * Establishes role hierarchy for user roles.
     * @return the RoleHierarchy
     */
    @Bean
    public RoleHierarchyImpl roleHierarchy() {
        RoleHierarchyImpl roleHierarchy = new RoleHierarchyImpl();
        roleHierarchy.setHierarchy(
                "ROLE_SUPER_ADMIN > ROLE_LOCATION_ADMIN " +
                        "ROLE_LOCATION_ADMIN > ROLE_REGULAR_USER ");
        return roleHierarchy;
    }

    @Bean
    public RoleHierarchyVoter roleVoter() {
        return new RoleHierarchyVoter(roleHierarchy());
    }


    @Bean
    HeaderHttpSessionStrategy sessionStrategy() {
        return new HeaderHttpSessionStrategy();
    }


    @Bean
    public AuthenticationProvider emailAuthenticationProvider() {
        return new EmailAuthenticationProvider();
    }

    @Bean
    public AuthenticationManager authenticationManager() {
        return new ProviderManager(Arrays.asList(
                emailAuthenticationProvider()
        ));
    }

    @Bean
    public AuthenticationSuccess authenticationSuccess() {
        return new AuthenticationSuccess();
    }

    @Bean
    public AuthenticationFailure authenticationFailure() {
        return new AuthenticationFailure();
    }

    @Bean
    public EmailAuthenticationFilter emailAuthenticationFilter() {
        EmailAuthenticationFilter filter = new EmailAuthenticationFilter(new AntPathRequestMatcher("/login-process", "POST"));
        applyFilterAuthRules(filter);
        return filter;
    }

    private void applyFilterAuthRules(AbstractAuthenticationProcessingFilter filter) {
        filter.setAuthenticationManager(authenticationManager());
        filter.setAuthenticationSuccessHandler(authenticationSuccess());
        filter.setAuthenticationFailureHandler(authenticationFailure());
        filter.setAllowSessionCreation(true);
    }

    @Override
    public void configure(WebSecurity web) throws Exception {
        web
                .ignoring()
                .antMatchers("/", "/**/*.css", "/**/*.js", "/index.html", "/ws/**", "/assets/**/*");
    }

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        if(debug) {
            LOGGER.warn("Allowing preflight options requests to all");
            // For the pre-flight request for options
            http
                    .authorizeRequests()
                    .antMatchers(HttpMethod.OPTIONS, "/**")
                    .permitAll();
            // Disable csrf on debug dev
            LOGGER.warn("Allowing no CSRF protection");
            http
                    .csrf()
                    .disable();
        } else {
            http
                    .csrf()
                    .csrfTokenRepository(CookieCsrfTokenRepository.withHttpOnlyFalse());
        }

        http
                .authorizeRequests()
                .antMatchers( "/**")
                .permitAll()
                .anyRequest()
                .authenticated()
                .and()
                .logout()
                .logoutUrl("/logout")
                .logoutSuccessHandler((new HttpStatusReturningLogoutSuccessHandler(HttpStatus.OK)))
                .invalidateHttpSession(true)
                .permitAll()
                .and()
                .addFilterBefore(emailAuthenticationFilter(), FilterSecurityInterceptor.class);



        http.requestCache().requestCache(new NullRequestCache());
        http.headers().cacheControl();
        http.headers().httpStrictTransportSecurity().disable();
        // .headers().contentSecurityPolicy(
        // .and().antMatcher("/bingo/card").;
    }

    @Bean
    protected BCryptPasswordEncoder passwordEncoder() {
        return new BCryptPasswordEncoder();
    }
}

------ EDIT 3 ------ ------编辑3 ------

buildscript {
    repositories {
        mavenCentral()
    }
    dependencies {
        classpath 'org.springframework.boot:spring-boot-gradle-plugin:1.4.3.RELEASE'
        // classpath 'org.springframework:springloaded:1.2.6.RELEASE'
    }
}

apply plugin: 'java'
apply plugin: 'idea'
apply plugin: 'org.springframework.boot'
apply plugin: 'io.spring.dependency-management'

repositories {
    mavenCentral()
}

idea {
    module {
        inheritOutputDirs = false
        outputDir = file("$buildDir/classes/main/")
    }
}

jar {
    baseName = 'leet-tournaments'
    version = '0.1.0'
}

sourceCompatibility = 1.8
targetCompatibility = 1.8

configurations {
    dev
    all*.exclude module: 'spring-boot-starter-logging'
}

dependencies {

    // Spring Boot Starter Framework
    compile(
            [group: 'org.springframework.boot', name: 'spring-boot-starter-data-jpa', version: '1.4.3.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-starter-log4j2', version: '1.2.0.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-starter-security', version: '1.4.3.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-starter-web', version: '1.3.2.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-starter-websocket', version: '1.0.0.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-starter-data-redis', version: '1.4.3.RELEASE'],
            [group: 'org.springframework.boot', name: 'spring-boot-devtools', version: '1.4.3.RELEASE'],
            [group: 'org.springframework.session', name: 'spring-session', version: '1.3.0.RELEASE'],
            [group: 'org.springframework', name: 'spring-messaging', version: '4.3.6.RELEASE'],
    )

    compile(
            [group: 'javax.mail', name: 'mail', version: '1.4.7'],
            [group: 'com.fasterxml.jackson.core', name: 'jackson-core', version: '2.8.5'],
            [group: 'mysql', name: 'mysql-connector-java', version: '5.1.35'],
            [group: 'org.projectlombok', name: 'lombok', version: '1.16.12'],
            [group: 'com.goebl', name: 'david-webb', version: '1.3.0'],
            [group: 'com.amazonaws', name: 'aws-java-sdk', version: '1.11.77'],
            [group: 'com.fasterxml.jackson.datatype', name: 'jackson-datatype-hibernate5', version: '2.8.8'],
            [group: 'com.google.guava', name: 'guava', version: '22.0-rc1'],
    )

    compile(
            [group: 'io.netty',                 name: 'netty-all',              version: '4.1.7.Final'],
            [group: 'io.projectreactor',        name: 'reactor-core',           version: '2.0.7.RELEASE'],
            [group: 'io.projectreactor',        name: 'reactor-net',            version: '2.0.7.RELEASE'],
            [group: 'io.projectreactor.spring', name: 'reactor-spring-core',    version: '2.0.7.RELEASE'],
            [group: 'io.projectreactor.spring', name: 'reactor-spring-context', version: '2.0.7.RELEASE'],
    )

    // testing
    testCompile('org.springframework.boot:spring-boot-starter-test')
    testCompile('junit:junit')
}

// gradle wrapper
task wrapper(type: Wrapper) {
    gradleVersion = '3.0'
}

// run spring boot app
bootRun {
    addResources = true
    classpath = sourceSets.main.runtimeClasspath + configurations.dev
}

Main class: 主班:

@SpringBootApplication
@EnableConfigurationProperties
@EnableAutoConfiguration
@EnableScheduling
@ComponentScan
public class LeetTournaments {
    private static final Logger LOGGER = LogManager.getLogger(LeetTournaments.class);

    public static void main(String[] args) {
        SpringApplication.run(LeetTournaments.class, args);
        // LeetTournaments.initDB();
    }

    static void initDB() {
        Webb webb = Webb.create();
        webb.get("http://127.0.0.1:8080/init").asString();
    }
}

Please note the text marked with the bold font: 请注意标有粗体的文本:

Spring Data repositories usually extend from the Repository or CrudRepository interfaces. Spring Data存储库通常从RepositoryCrudRepository接口扩展。 If you are using auto-configuration, repositories will be searched from the package containing your main configuration class (the one annotated with @EnableAutoConfiguration or @SpringBootApplication ) down. 如果您使用的是自动配置, 则会从包含您的主要配置类 (以@EnableAutoConfiguration@SpringBootApplication注释的类)的包中搜索存储库。

Spring Boot Reference Guide, 29.3.2 Spring Data JPA Repositories . Spring Boot参考指南,29.3.2 Spring Data JPA存储库

Actually, I think that this is the solution to the problem (the text marked with the bold font): 实际上,我认为这是解决问题的方法(用粗体标记的文本):

By default, Spring Boot will enable JPA repository support and look in the package (and its subpackages) where @SpringBootApplication is located. 默认情况下,Spring Boot将启用JPA存储库支持,并在@SpringBootApplication所在的包(及其子包)中@SpringBootApplication If your configuration has JPA repository interface definitions located in a package not visible, you can point out alternate packages using @EnableJpaRepositories and its type-safe basePackageClasses=MyRepository.class parameter . 如果您的配置在不可见的软件包中有JPA仓库接口定义,则可以使用@EnableJpaRepositories及其类型安全的basePackageClasses=MyRepository.class参数指出备用软件包

Getting Started · Accessing Data with JPA . 入门·使用JPA访问数据

All in all, consider updating the implementation of the LeetTournaments class as follows: 总而言之,考虑如下更新LeetTournaments类的实现:

@EnableJpaRepositories(basePackageClasses = UserRepository.class)
public class LeetTournaments {
    ...
}

Hope this helps. 希望这可以帮助。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM