5

我想测试一个使用 grails 电子邮件插件发送电子邮件的 Grails 控制器。我不知道如何模拟sendMail闭包以使交互起作用。这是我最新版本的测试代码:

def 'controller should send a multipart email'() {
    given: 'a mocked mailService'
        controller.mailService = Mock(grails.plugin.mail.MailService)
        controller.mailService.sendMail(*_) >> Mock(org.springframework.mail.MailMessage)
    when:
        controller.sendNow()
    then:
        1* _.multipart(true)
}

控制器代码看起来像您期望的那样,例如:

def mailService
def sendNow() {
    mailService.sendMail {
        multipart true
        to 'example@example.org'
        from 'me@here.com'
        subject 'a subject'
        body 'a body'
    }
}

如果我运行这个测试,我会得到 0multipart次交互调用而不是 1 次。该given:块的第二行对我来说似乎很可疑,但如果我尝试模拟 aClosure而不是org.springframework.mail.MailMessage我的测试崩溃。我还应该提到控制器本身按预期工作(它等不及我先弄清楚单元测试)。

已编辑

啊哈,几个小时后重新看代码,我明白为什么上面的代码不起作用了;为了让我捕捉multipart和其他 DSL 调用,我必须模拟闭包本身,而不是 sendMail 方法(我不能这样做,因为闭包是在控制器本身内部定义的)。我可能能做的是检查方法的参数sendMail看看所有必要的东西都传递给它。

4

4 回答 4

4

我能够通过以下方式在 Spock 中实现这一点:

def messageBuilder
def bodyParams
def setup(){
    def mockMailService = new MockFor(MailService)
    mockMailService.ignore.sendMail{ callable ->
        messageBuilder = new MailMessageBuilder(null, new ConfigObject())
        messageBuilder.metaClass.body = { Map params ->
            bodyParams = params
        }
        callable.delegate = messageBuilder
        callable.resolveStrategy = Closure.DELEGATE_FIRST
        callable.call()
    }
    service.mailService = mockMailService.proxyInstance()
}

还有一个示例测试:

def "sendEmailReceipt_passesCorrectParams"(){
    when:
        def receiptItems = [] << [item: "item1", price: 100]
        service.sendEmailReceipt(receiptItems, "some@email.com")

    then:
        messageBuilder.message.to[0] == "some@email.com"
        messageBuilder.message.subject == "My subject"
        bodyParams.view == "/mailtemplates/emailReceipt"
        bodyParams.model.receiptItems == data
}
于 2013-08-27T04:24:13.183 回答
3

您可以安装greenMail插件,并在集成测试中使用它:

从 greenmail 插件主页:

import com.icegreen.greenmail.util.*

class GreenmailTests extends GroovyTestCase {
    def mailService
    def greenMail    

    void testSendMail() {
        Map mail = [message:'hello world', from:'from@piragua.com', to:'to@piragua.com', subject:'subject']        

        mailService.sendMail {
            to mail.to
            from mail.from
            subject mail.subject
            body mail.message
        }        

        assertEquals(1, greenMail.getReceivedMessages().length)        
        def message = greenMail.getReceivedMessages()[0]        
        assertEquals(mail.message, GreenMailUtil.getBody(message))
        assertEquals(mail.from, GreenMailUtil.getAddressList(message.from))
        assertEquals(mail.subject, message.subject)
    }    

    void tearDown() {
        greenMail.deleteAllMessages()
    }
}

我不是 Spock 专家,但您应该能够将此 junit 测试翻译成 spock 风格。

来源: http: //grails.org/plugin/greenmail

更新,通过模拟 sendMail 替代

这是对 Gregor 更新的回答。在我看来,你必须模拟 sendMail 方法,并且在这个方法内部有一个存根,它实现了闭包中使用的不同属性和方法。让我们称之为评估者。您将初始化闭包的委托给 evaluatro,并执行闭包。评估者应该有断言。你看我在这里使用了更多的junit概念。我不知道你能多么容易地将它转化为 spock 概念。您可能可以使用 spock 的行为检查工具。

class MailVerifier {
    void multiPart(boolean v){
        //...
    }

    void to(String address){
        //...
    }

    boolean isVerified() {
        //check internal state obtained by the appropriate invocation of the methods
    }
}

def sendMail(Closure mailDefintion) {
    def evaluator = createMailVerifier()
    mailDefinition.delegate = evaluator

    mailDefinition()

    assert evaluator.verified
}
于 2012-05-23T08:31:49.440 回答
1

在此处查看插件测试:插件集成测试和此处:插件单元测试。在我看来,你很难模拟所有 MailService 依赖项——构建邮件消息的工厂和构建器。只有当我的控制器的 sendNow 被调用时,我才会进行测试。

编辑

我找到了这个答案。根据它,您可以尝试:

def 'controller should send a multipart email'() {
    given: 'a mocked mailService'
        def mockMailService = new Object()
        def mockMessageBuilder = Mock(MessageBuilder)
        mockMailService.metaClass.sendMail = { callable ->
            callable.delegate = mockMessageBuilder
            callable.resolveStrategy = Closure.DELEGATE_FIRST
            callable.call()
        }
        controller.mailService = mockMailService
    when:
        controller.sendNow()
    then:
        1* mockMessageBuilder.multipart(true)

}

于 2012-05-23T09:13:09.807 回答
1
def mailService = Mock(MailService)
mockMailService.metaClass.sendMail = { ... your logic ... }
controller.mailService = mailService
于 2012-05-24T15:57:40.200 回答