【问题标题】:Why is this Spock Specification in Grails resulting in 'identifier of an instance of projectname.Event was altered from 1 to 2;'?为什么 Grails 中的 Spock 规范会导致“projectname.Event 实例的标识符从 1 更改为 2;”?
【发布时间】:2024-04-26 18:10:01
【问题描述】:

运行test-app integration:spock 会导致以下错误:

实例的标识符 projectname.Event 从 1 更改 到 2;嵌套异常是 org.hibernate.HibernateException:实例的标识符 projectname.Event 从 1 更改 到 2

org.springframework.orm.hibernate3.HibernateSystemException: 实例的标识符 projectname.Event 从 1 更改 到 2;

嵌套异常是 org.hibernate.HibernateException:实例的标识符 projectname.Event 从 1 更改 到 2 在 项目名称.EventControllerSpec.save: 一个 事件(EventControllerSpec.groovy:74)

原因: org.hibernate.HibernateException:实例的标识符 projectname.Event 从 1 更改 到 2

  1. 此问题源于何处?
  2. 如何解决?
  3. [可选] 有人可以创建标签“spock”吗?

来自 EventControllerSpec.groovy

def "save: an event"() {
    given: "Constraint-conform event properties"
    def eventTitle              = "Being in Beijing"
    controller.params.title     = eventTitle
    controller.params.details   = "Details"
    controller.params.location  = "Beijing"
    controller.params.startDate = "01.09.2030"
    controller.params.startTime = "20:15"
    controller.params.endDate   = "01.09.2030"
    controller.params.endTime   = "21:45"
    controller.params.publisher = getUserObject("someuser")

    when: "I save that event"
    def result = controller.save() // THIS IS LINE #74 AS STATED IN THE ERROR

    then: "The event is successfully saved and the show-view rendered"
    controller.flash.message.args.grep(eventTitle)
    redirectArgs.action             == "show"
    redirectArgs.id                 == result.eventInstance.id
}

private User getUserObject(String name) {
    def user = User.findByUsername(name)

    if (!user) {
        user = new User()
        user.username = name
        user.email = "${name}@example.com"
        user.pw = "barbar"
        user.pwConfirmation = "barbar"
        assert user.save()
    }

    user
}

来自 EventController.groovy

def save = {
    def eventInstance = new Event()

    eventInstance.title     = params.title
    eventInstance.details   = params.details
    eventInstance.location  = params.location
    eventInstance.startDate = DateUtil.createDate(params.startDate, params.startTime)
    eventInstance.endDate   = DateUtil.createDate(params.endDate, params.endTime)
    eventInstance.publisher = session.user

    if (eventInstance.save(flush: true)) {
        flash.message = "${message(code: 'default.created.message', args: [message(code: 'event.label', default: 'Event'), eventInstance.title])}"
        redirect(action: "show", id: eventInstance.id)
    }
    else {
        eventInstance.errors.each { log.warn it }
        render(view: "add", model: [eventInstance: eventInstance])
    }
}

【问题讨论】:

  • 如果您可以发布事件的定义可能会有所帮助。另外,您使用什么数据库进行集成测试?
  • 我都试过了,MySQL 和 HSQLDB;您认为域类定义对此有何影响?

标签: hibernate exception grails integration-testing spock


【解决方案1】:

您的 Spock 测试扩展了什么类?它应该是spock.lang.Specificationgrails.plugin.spock.IntegrationSpec 而不是为单元测试而设计的grails.plugin.spock.ControllerSpec

【讨论】:

  • 嗨彼得,我在开发后期将规范从单元移动到集成(因此仍在扩展 ControllerSpec)。现在,按照您的提示并更改为 IntegrationSpec(或 spock.lang.Specification)删除对“控制器”的访问(没有这样的属性:类的控制器:corilla.EventControllerSpec)。我已经尝试了几件事来恢复它(new ...(),Mock()),但显然我做错了什么......
  • 正确。 IntegrationSpec 没有“控制器”属性。您必须自己实例化控制器并连接任何服务或其他 Spring bean。这是集成测试的标准做法。顺便说一句,您为什么要对控制器使用集成测试?它的价值通常很低,而访问​​数据库的服务的集成测试通常价值更高。