Spring Security 创建一个名为 USERS 的表

Posted

技术标签:

【中文标题】Spring Security 创建一个名为 USERS 的表【英文标题】:Spring Security creates a table named USERS 【发布时间】:2015-10-28 22:11:03 【问题描述】:

当启动我的应用程序时,Spring Security 正在我的 h2 数据库中创建两个表“用户”和“权限”。我的问题是,当我第二次启动应用程序时,数据库已经填充了这两个表,但是 Spring Security 直到想要创建它们。 我如何告诉 Spring Secutiry 仅在表不存在时才创建表?还是我应该改变其他东西来解决这个问题?

Spring Secutiry 正在创建我从博客中获得的两个表,这些表描述了我查看过的一些代码。它说:“关于数据模型的一些注释。没有真正的用户名和密码存储在 USERS 表中。该表由 Spring Security 定义用于一般用途,而不是特定于社交登录用例。”

Link to the blog

我得到的错误消息,一些 java 堆栈跟踪,是:

Caused by: org.springframework.jdbc.datasource.init.ScriptStatementFailedException: Failed to execute SQL script statement at line 1 of resource class path resource [org/springframework/security/core/userdetails/jdbc/users.ddl]: 
create table users(username varchar_ignorecase(50) not null primary key,password varchar_ignorecase(500) not null,enabled boolean not null); nested exception is org.h2.jdbc.JdbcSQLException: Table "USERS" already exists; 
SQL statement: create table users(username varchar_ignorecase(50) not null primary key,password varchar_ignorecase(500) not null,enabled boolean not null) [42101-175]

Caused by: org.springframework.jdbc.datasource.init.ScriptStatementFailedException: Failed to execute SQL script statement at line 1 of resource class path resource [org/springframework/security/core/userdetails/jdbc/users.ddl]:    
create table users(username varchar_ignorecase(50) not null primary key,password varchar_ignorecase(500) not null,enabled boolean not null); nested exception is org.h2.jdbc.JdbcSQLException: Table "USERS" already exists;
SQL statement: create table users(username varchar_ignorecase(50) not null primary key,password varchar_ignorecase(500) not null,enabled boolean not null) [42101-175]

我不确定这发生在代码中的什么位置。我有一个像这样的 entityManageFactory

@Autowired DataSource dataSource;
@Bean
public EntityManagerFactory entityManagerFactory() 
    HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
    vendorAdapter.setGenerateDdl(true);
    Properties props = new Properties();
    props.put("hibernate.dialect", H2Dialect.class.getName());
    props.put("hibernate.format_sql", "true");
    props.put("spring.jpa.properties.hibernate.hbm2ddl.auto", "update");
    props.put("spring.datasource.dbCreate", "update");
    LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean();
    factory.setJpaVendorAdapter(vendorAdapter);
    factory.setPackagesToScan("mypackage.demo");
    factory.setDataSource(dataSource);
    factory.setJpaProperties(props);
    factory.afterPropertiesSet();

    return factory.getObject();

【问题讨论】:

【参考方案1】:

在我的案例中,在配置 Spring Security 时创建了 USERS 表(和其他一些表)。这是该代码的一部分:

public class SecurityConfiguration extends WebSecurityConfigurerAdapter 

@Autowired
private DataSource dataSource;

@Autowired
public void configureGlobal(AuthenticationManagerBuilder auth) throws Exception 
    auth.
        jdbcAuthentication()
        .dataSource(dataSource)
        .withDefaultSchema(); 

如果我从已配置的现有数据库开始,并在其中部署数据,则删除该行:

      .withDefaultSchema(); 

Spring Security 将使用数据库中现有的表。

【讨论】:

【参考方案2】:

我是这样处理的:

    JdbcUserDetailsManagerConfigurer<AuthenticationManagerBuilder> configurer =
            auth.jdbcAuthentication().dataSource(dataSource);

    // Check if the auth schema has been populated (from a persistent source); if not, set it to populate
    if (!dataSource.getConnection().getMetaData().getTables(null, "", "USERS", null).first()) 
        configurer = configurer.withDefaultSchema();
    

【讨论】:

以上是关于Spring Security 创建一个名为 USERS 的表的主要内容,如果未能解决你的问题,请参考以下文章

创建名为 AuthenticationManager 的 bean 时出错

Spring security-org.springframework.beans.factory.BeanCreationException:创建名为“org.springframework.sec

使用 Spring Security 的自动装配依赖注入

Spring Security @PreAuthorize基于自定义布尔属性值[关闭]

基于自定义布尔属性值的 Spring Security @PreAuthorize [关闭]

基于Spring Security Oauth2的SSO单点登录+JWT权限控制实践