Grails,自定义验证器,Hibernate断言错误

时间:2015-05-27 13:11:38

标签: grails gorm

我尝试在我的域类中使用服务进行自定义验证。在其他域类中,这很好用,但在这里我收到了一个Hibernate断言错误。

域类自定义验证:

end validator: { val, obj ->

    if (obj.userWorkingTimeRegulationService.validateit() == false){
        return ["error.workingregulation"]
    }

我使用以下方式注入了服务:

def userWorkingTimeRegulationService

该服务仅在测试时返回true:

package usermanagement

import grails.transaction.Transactional

@Transactional
class UserWorkingTimeRegulationService {

    def serviceMethod() {

    }

    def validateit(){
        return true
    }
}

堆栈跟踪:

| Error 2015-05-27 15:07:54,909 [localhost-startStop-1] ERROR hibernate.AssertionFailure  - HHH000099: an assertion failure occured (this may indicate a bug in Hibernate, but is more likely due to unsafe use of the session): org.hibernate.AssertionFailure: null id in usermanagement.UserWorkingTimeRegulation entry (don't flush the Session after an exception occurs)
| Error 2015-05-27 15:07:54,927 [localhost-startStop-1] ERROR context.GrailsContextLoaderListener  - Error initializing the application: null id in usermanagement.UserWorkingTimeRegulation entry (don't flush the Session after an exception occurs)
Message: null id in usermanagement.UserWorkingTimeRegulation entry (don't flush the Session after an exception occurs)
    Line | Method
->>   24 | doCall                           in usermanagement.UserWorkingTimeRegulation$__clinit__closure4$_closure8
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 
|     47 | onApplicationEvent               in org.grails.datastore.mapping.engine.event.AbstractPersistenceEventListener
|     52 | create . . . . . . . . . . . . . in usermanagement.UserWorkingTimeRegulation
|     47 | doCall                           in BootStrap$_closure1
|    327 | evaluateEnvironmentSpecificBlock in grails.util.Environment
|    320 | executeForEnvironment            in     ''
|    296 | executeForCurrentEnvironment . . in     ''
|    266 | run                              in java.util.concurrent.FutureTask
|   1142 | runWorker . . . . . . . . . . .  in java.util.concurrent.ThreadPoolExecutor
|    617 | run                              in java.util.concurrent.ThreadPoolExecutor$Worker
^    745 | run . . . . . . . . . . . . . .  in java.lang.Thread
Error |
Forked Grails VM exited with error

如果我使用在域类中声明的方法都有效,那么该服务必定​​存在问题。在另一个域类中,我在自定义验证中使用了没有问题的服务,我不知道这个有什么特别之处。

编辑:

我在我的Bootstrap.groovy中创建了一个UserWorkingTimeRegulation,如果我在我的域类中注释自定义验证,由于服务返回true并且验证无效与false合作。

自举:

UserWorkingTimeRegulation.create adminUser, workingTimeRegulation, new LocalDate(), new LocalDate().plusMonths(1)

1 个答案:

答案 0 :(得分:1)

默认情况下,服务是事务性的 - 使服务非事务性的唯一方法是添加static transactional = false并删除所有@Transactional注释。 Spring / Hibernate事务管理器总是在成功事务结束时刷新Hibernate会话,所以你看到的是在调用服务方法之后(即使是像你这样没有持久性的服务方法),所有修改过的持久对象都是从Hibernate缓存到数据库,但其中一个存在问题。

错误消息相当神秘,但是它说usermanagement.UserWorkingTimeRegulation没有id。这意味着您调用了save()但未验证,因此您需要通过调用hasErrors和/或getErrors来检查哪些值丢失或无效。没有id的问题是这个实例是另一个域对象的属性,而id被用作将它们链接在一起的外键。

P.S。随意删除未使用的自动生成的serviceMethod方法。