DBUnit:NoSuchColumnException ColumnNameToIndexes 缓存映射中的非大写输入列。地图的列名不区分大小写

问题描述 投票:0回答:5

我的java应用程序将您的信息存储在MySql数据库版本8中。用户信息和密码存储在该数据库中。我正在实施集成测试来测试使用 dbunit 验证数据库中用户的方法。测试方法运行后,出现如下错误。

Caused by: org.dbunit.dataset.NoSuchColumnException: USER.USER_ID -  (Non-uppercase input column: USER_ID) in ColumnNameToIndexes cache map. Note that the map's column names are NOT case sensitive.

我通过MySql Workbench检查发现表和列都是大写创建的,所以显示的错误消息没有意义。有谁知道什么可能导致此异常?

testContext.xml

unitils.properties

unitils.module.hibernate.enabled=true
unitils.module.jpa.enabled=false
unitils.module.easymock.enabled=false

database.url=jdbc:mysql://192.168.0.12:3306/db_ca?useUnicode=true&characterEncoding=UTF-8&serverTimezone=America/Sao_Paulo
database.driverClassName=com.mysql.jdbc.Driver
database.userName=root
database.password=9999
database.schemaNames=db_main, db_ca
database.dialect=mysql

unitils.module.database.className=com.myapplication.test.unitils.SingleConnectionDatabaseModule
DatabaseModule.Transactional.value.default=disabled

DbUnitModule.DataSet.loadStrategy.default=org.unitils.dbunit.datasetloadstrategy.impl.CleanInsertLoadStrategy

updateDataBaseSchema.enabled=true
dbMaintainer.script.locations=src/test/dbscripts
dbMaintainer.autoCreateExecutedScriptsTable=true
dbMaintainer.generateDataSetStructure.enabled=false
dbMaintainer.preserve.schemas=

在 src/test/dbscripts 中存在文件 001_SCRIPT_01.00.00

SET character_set_client = utf8mb4 ;
CREATE TABLE db_ca.ROLE (
  ROLE_ID INT(11) NOT NULL AUTO_INCREMENT,
  NAME VARCHAR(200) NOT NULL,
  PRIMARY KEY (ROLE_ID)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

SET character_set_client = utf8mb4 ;
CREATE TABLE db_ca.USER (
  USER_ID INT(11) NOT NULL AUTO_INCREMENT,
  EMAIL VARCHAR(100) NULL DEFAULT NULL,
  NAME VARCHAR(150) NOT NULL,
  PASSWORD VARCHAR(200) NOT NULL,
  PRIMARY KEY (USER_ID)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

SET character_set_client = utf8mb4 ;
CREATE TABLE db_ca.USER_ROLE (
  ROLE_ID INT(11) NOT NULL,
  USER_ID INT(11) NOT NULL,
  PRIMARY KEY (ROLE_ID, USER_ID)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

/数据集/UserServiceTest.xml

<?xml version="1.0" encoding="UTF-8"?>
<dataset xmlns="db_ca">
    <USER
        USER_ID = "1"
        EMAIL = "[email protected]"
        NAME = "TEST"
        PASSWORD = "e8d95a51f3af4a3b134bf6bb680a213a"
   />    
   <ROLE
        ROLE_ID = "1"
        NAME = "ADMIN"
   />   
   <USER_ROLE
        USER_ID = "1"
        ROLE_ID = "1"
   />   
</dataset>

实体

@Entity
@Table(name = "USER", schema="db_ca")
public class User {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name="USER_ID", nullable=false)
    private Long id;

    ...
}

@Entity
@Table(name = "ROLE", schema="db_ca" )
public class Role {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name="ROLE_ID", nullable=false)
    private Long id;    

    ...
}   

类测试AccessServiceIT.java

@DataSet("/datasets/UsuarioServiceTest.xml")
@RunWith(UnitilsJUnit4TestClassRunner.class)
@SpringApplicationContext("testContext.xml")
public class AccessServiceIT {

    @SpringBean("AccessServiceImpl")
    private AccessService accessService;

    @Test
    public void accessTest() {
        CredentialsBean credentialsBean = accessService.confirmLogon("[email protected]", "e8d95a51f3af4a3b134bf6bb680a213a");
        assertEquals("[email protected]", credentialsBean.getLogon());
        assertEquals(true, credentialsBean.isAuthenticated());
    }
}
java mysql junit dbunit spring-test-dbunit
5个回答
3
投票

我找到了问题的解决方案。我刚刚将以下属性放入unitils.properties 中,现在一切工作正常。

org.dbunit.database.IMetadataHandler.implClassName=org.dbunit.ext.mysql.MySqlMetadataHandler

我在这里找到了解决方案https://stackoverflow.com/a/39549867/3554659

我希望它能帮助那些经历同样错误的人。


2
投票

我没有使用 Unitils,而是直接调用 dbunit。我通过设置连接的属性解决了这个问题:

final IDatabaseConnection connection = new DatabaseConnection(dataSource.getConnection());
connection.getConfig()
          .setProperty(DatabaseConfig.PROPERTY_METADATA_HANDLER, new MySqlMetadataHandler());

0
投票

对我来说,我收到此错误,因为该列未在 hibernate 实体中声明,但该列存在于表(oracle 数据库)中。这就是为什么 Dbunit 数据集认为该列可能以小写或大写形式声明,但它根本没有声明。异常消息不准确。


0
投票

就我而言,这个错误是在使用

@DatabaseSetup
模拟数据库的 XML 文件的测试中发生的。

我必须向

companyOffice
实体添加一个新列,但是还有
company
实体,并且全部混合在 XML 文件中。因此,我将该列添加到
company
实体,并且它最终试图在错误的位置找到这个字段。

我不是很专心,但是错误信息根本就不清楚。


0
投票

多个数据集

老问题,但今天遇到了问题。 问题是由于多个数据集包含同一张表上的插入内容,其中一些数据集提供了字段,而另一些则没有。 我必须将这些不同文件中包含的所有数据合并到一个千兆文件中。

具体案例

我在我的

@DatabaseSetup
中导入了三个数据集:

  • customer_underaged.xml
  • customer_company.xml
  • customer_no_birthdate.xml

customer_no_birthdate.xml
中,故意不提供
birthdate
栏。在运行测试时,我遇到了臭名昭著的情况:

Caused by: org.dbunit.dataset.NoSuchColumnException: USER.BIRTHDATE -  (Non-uppercase input column: BIRTHDATE) in ColumnNameToIndexes cache map. Note that the map's column names are NOT case sensitive.

将所有这三个数据集合并为一个

customer_mixed.xml
,问题就消失了。

想法

我发现由于某种原因,仅考虑包含表定义的最后一个数据集来计算每个其他数据集的该表的定义。 我认为在计算缓存的表元数据时存在某种问题(请参阅

org.dbunit.dataset.xml.FlatXmlProducer#createTableMetaData
),其中检查每个文件,并且对于定义的每个表,仅保留最新的定义以供以后插入。

© www.soinside.com 2019 - 2024. All rights reserved.