在grails中单元测试RestfulController时使用默认索引操作的奇怪行为

时间:2015-05-05 05:28:02

标签: unit-testing grails-2.4

使用grails 2.4.4(和2.4.5)我创建了以下微小的应用程序。

grails create-app example
cd example
grails create-domain-class Book

然后编辑Book.groovy看起来像

package example
class Book {
    String title;
    static constraints = {
    }
}  

然后添加了一个基本的控制器

grails> create-controller example.book
| Created file grails-app/controllers/example/BookController.groovy
| Created file grails-app/views/book
| Created file test/unit/example/BookControllerSpec.groovy

并修改控制器以扩展RestfulController。

package example
import grails.rest.*
class BookController extends RestfulController<Book> {
    static responseFormats=['json','xml']
    BookController() {
        super(Book)
    }
}

将UrlMappings连接起来作为/ api / books的资源提供书籍。

class UrlMappings {
  static mappings = {
    "/api/books"(resources: "book")
    "/"(view:"/index")
    "500"(view:'/error')
  }
}

最后但并非最不重要的是,在BootStrap.groovy中建了几本书。

import example.*
class BootStrap {
    def init = { servletContext ->
        new Book(title: "Book 1").save(failOnError:true);
        new Book(title: "Book 2").save(failOnError:true);
        new Book(title: "Book 3").save(failOnError:true);
    }
    def destroy = {
    }
}

然后grails run-app

一切看起来都不错。 example.Book控制器显示在索引页面中。这三本书可以被视为json或xml。

现在进行单元测试。

编辑BookControllerSpec看起来像这样

package example
import grails.test.mixin.TestFor
import grails.test.mixin.Mock
import spock.lang.Specification

@TestFor(BookController)
@Mock(Book)
class BookControllerSpec extends Specification {

    def setup() {
        new Book(title: "Unit Test Book 1").save(failOnError:true);
        new Book(title: "Unit Test Book 2").save(failOnError:true);
        new Book(title: "Unit Test Book 3").save(failOnError:true);
    }
    void "My Setup Worked"() {
        given: "My setup ran"
        when: "I ask for the count"
        then: "I should get back 3"
        Book.count() == 3;
    }

    void "I can access my Controller index method"() {
        given: "My setup ran"
        when:  "I call index"
        request.method="GET"
        response.format="xml"
        controller.index();

        then: "I get an XML object back with 3 books"
        println response.contentAsString
        response.xml.book*.title.size()==3
    }
}

第一次测试通过,第二次测试失败。失败测试中println的输出是一个空的xml书籍列表。

<?xml version="1.0" encoding="UTF-8"?><list />

在调查发生了什么时,我查看了父类的索引方法(RestfulController)。

通过将该方法复制到BookController中,单元测试将开始传递。

- 带有复制索引方法的新版BookController -

package example
import grails.rest.*
class BookController extends RestfulController<Book> {
    static responseFormats=['json','xml']
    BookController() {
        super(Book)
    }

    def index(Integer max) {
        params.max = Math.min(max ?: 10, 100)
        respond listAllResources(params), model: [("${resourceName}Count".toString()): countResources()]
    }
}

为了让RestfulController的索引方法无需将索引方法复制到BookController中,我还需要添加到Spec中吗?

任何想法为什么在直接从BookController执行时对listAllResources的调用有效,但在从RestfulController执行时不返回任何行。

上面的单元测试是Grails In Action一书中描述的其余单元测试的修改版本,该书是为grails 2.3编写的。 http://www.manning.com/gsmith2/

1 个答案:

答案 0 :(得分:0)

使用grails进行单元测试会非常棘手。我从未确定究竟是什么导致了这种失败,但我确实发现了另一种奇怪的现象。

将失败的测试插入spec文件两次将导致它第一次失败,但是第二次失败。

Additional information: The path to the driver executable must be set by the webdriver.ie.driver
  system property; for more information, see http://code.google.com/p/selenium/wiki/InternetExplorerDriver.
  The latest version can be downloaded from http://code.google.com/p/selenium/downloads/list

Grails测试框架内部的某些东西在第一次调用它时并没有成功地连接该索引方法。

我没有进一步挖掘,而是将其重写为集成测试,并且开始表现正常。

使用grails单元测试可以免费获得很多魔法,但是当魔法不起作用时,尝试不同的测试阶段可能会更好。