我正在从“Grails In Action”一书中学习grails,我正在尝试从示例中运行集成测试。在书中它说每个集成测试函数应该在每个测试结束时回滚它的操作。它不会回滚每个事务(因为当我完成数据库时是脏的)。我试图找出原因,发现找到了一个名为“transactional”的属性。据称你将此属性设置为true,它将使测试用例处于事务状态,但它似乎不会改变行为。我已经在下面列出了单元测试的代码。
我正在使用grails 1.3.7并连接到MySql数据库。测试成功运行,它不会回滚。我是否在此集成测试中做错了它跳过了回滚?
UserIntegrationTests.groovy:
package com.grailsinaction
import framework.TestTools
class UserIntegrationTests extends GroovyTestCase {
static transactional = true
protected void setUp() {
super.setUp()
}
protected void tearDown() {
super.tearDown()
}
void testCreateUser() {
TestTools.banner(log, "testCreateUser()")
def user = new User(userId:"joe", password:"secret")
assertNotNull user.save()
assertNotNull user.id
def foundUser = User.get(user.id)
assertEquals 'joe', foundUser.userId
}
void testSaveAndUpdate() {
TestTools.banner(log, "testSaveAndUpdate()")
def user = new User(userId:"joe2", password:"secret")
assertNotNull user.save()
def foundUser = User.get(user.id)
foundUser.password = 'sesame'
foundUser.save()
def editedUser = User.get(user.id)
assertEquals 'sesame', editedUser.password
}
void testSaveThenDelete() {
TestTools.banner(log, "testSaveThenDelete()")
def user = new User(userId: 'joe3', password: 'secret')
assertNotNull user.save()
def foundUser = User.get(user.id)
foundUser.delete()
assertFalse User.exists(foundUser.id)
}
void testValidation() {
TestTools.banner(log, "testValidation()")
def user = new User(userId: 'chuck-norris', password: 'tiny')
assertFalse user.validate()
assertTrue user.hasErrors()
def errors = user.errors
assertNotNull errors
errors.allErrors.each {
log.info("field: ${it.field}, code=${it.code}, rejected=${it.rejectedValue}")
}
}
}
User.groovy
package com.grailsinaction
class User {
String userId
String password
Date dateCreated
Profile profile
static constraints = {
userId(size: 3..20, unique: true)
password(size: 6..8, validator: {password, user ->
return (password != user.userId)
})
dateCreated()
profile(nullable: true)
}
static mapping = {
profile lazy: false
}
static hasMany = [posts : Post]
}
测试执行日志
Testing started at 8:28 PM ...
Welcome to Grails 1.3.7 - http://grails.org/
Licensed under Apache Standard License 2.0
Grails home is set to: C:\Users\jmquigley\workspace\apps\Grails\grails-1.3.7
Base Directory: C:\Users\jmquigley\workspace\samples\lang-grails\hubbub
Resolving dependencies...
Dependencies resolved in 963ms.
Running script C:\Users\jmquigley\workspace\apps\Grails\grails-1.3.7\scripts\TestApp.groovy
Environment set to test
[groovyc] Compiling 1 source file to C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\classes
[mkdir] Created dir: C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\test-reports\html
[mkdir] Created dir: C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\test-reports\plain
Starting integration test phase ...
[groovyc] Compiling 1 source file to C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\classes
[groovyc] Compiling 1 source file to C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\classes
[INFO ]20110417@20:28:42,959:grails.spring.BeanBuilder: [RuntimeConfiguration] Configuring data source for environment: TEST
[groovyc] Compiling 1 source file to C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\test-classes\integration
-------------------------------------------------------
Running 4 integration tests...
Running test com.grailsinaction.UserIntegrationTests...
--Output from testCreateUser--
[INFO ]20110417@20:28:46,897:groovy.util.GroovyTestCase: Test Case: testCreateUser()
--Output from testSaveAndUpdate--
[INFO ]20110417@20:28:47,534:groovy.util.GroovyTestCase: Test Case: testSaveAndUpdate()
--Output from testSaveThenDelete--
[INFO ]20110417@20:28:47,568:groovy.util.GroovyTestCase: Test Case: testSaveThenDelete()
--Output from testValidation--
[INFO ]20110417@20:28:47,642:groovy.util.GroovyTestCase: Test Case: testValidation()
[INFO ]20110417@20:28:47,668:groovy.util.GroovyTestCase: field: password, code=size.toosmall, rejected=tiny
null
PASSED
Tests Completed in 1173ms ...
-------------------------------------------------------
Tests passed: 4
Tests failed: 0
-------------------------------------------------------
[junitreport] Processing C:\Users\jmquigley\workspace\samples\lang-grails\hubbub\target\test-reports\TESTS-TestSuites.xml to C:\Users\JMQUIG~1\AppData\Local\Temp\null90011239
[junitreport] Loading stylesheet C:\Users\jmquigley\workspace\apps\Grails\grails-1.3.7\lib\junit-frames.xsl
[junitreport] Transform time: 415ms
[junitreport] Deleting: C:\Users\JMQUIG~1\AppData\Local\Temp\null90011239
Tests PASSED - view reports in target\test-reports
Application context shutting down...
Application context shutdown.
Process finished with exit code 0
答案 0 :(得分:9)
默认情况下,测试(和服务)是事务性的,因此您通常只在transactional
时指定静态false
属性。如果你没有指定方言,它可能会自动检测MySQL,但是这些表是使用默认引擎创建的,可能是MyISAM。 MyISAM表不是事务性的。每当使用MySQL时,请务必指定InnoDB方言,例如
test {
dataSource {
dialect= org.hibernate.dialect.MySQLInnoDBDialect
driverClassName = 'com.mysql.jdbc.Driver'
username = '...'
password = '...'
url = '...'
dbCreate = 'update'
}
}
或者如果你在所有环境中都使用MySQL,你可以将它移到顶层,例如
dataSource {
pooled = true
dialect = org.hibernate.dialect.MySQLInnoDBDialect
driverClassName = 'com.mysql.jdbc.Driver'
}
答案 1 :(得分:0)
我有同样的问题。
环境:STS(IDE),mysql(数据库)
grails集成测试必须如您所知回滚(默认)。
'grails integration test'尝试在每次测试完成后回滚。
如果您的数据库不支持交易..?
我将auto create database table更改为直接创建表,选项Type = InnoDB。所以我可以解决它。
创建表格(...)Type = InnoDB;