在将Grails应用程序从5.3.2升级到6.0.0之后,我在集成测试中遇到了解决i18n消息的问题。在我们的110个集成测试中,有26个出现了类似的错误:
org.springframework.context.NoSuchMessageException: No message found under code 'itemDisplayDateTimeFormat' for locale 'en'.
at org.springframework.context.support.AbstractMessageSource.getMessage(AbstractMessageSource.java:161)
at com.hypercision.attproto.device.DeviceManagementController.getDataForDeviceApproval(DeviceManagementController.groovy:93)
at com.hypercision.attproto.device.DeviceManagementController.getDataForDeviceApproval(DeviceManagementController.groovy)
at org.grails.core.DefaultGrailsControllerClass$MethodHandleInvoker.invoke(DefaultGrailsControllerClass.java:223)
这发生在我的本地机器和我们的CircleCI工作流程中。我尝试删除项目根目录下的.gradle
目录,但这并没有修复错误。
我能够让我们的一个服务集成测试再次通过,通过手动创建和注入一个StaticMessageSource
,就像我们在单元测试中已经做的那样:
@Integration
@Rollback
class SubmissionServiceIntegrationSpec extends Specification {
@Autowired
SubmissionService submissionService
StaticMessageSource staticMessageSource = new StaticMessageSource()
def setup() {
staticMessageSource.addMessage("SubmissionService.submit.sessionAlreadySubmittedError",
Locale.US, "session already submitted")
submissionService.setMessageSource(staticMessageSource)
}
def setupData() {
// omitting some domain classes that are created here
}
def "test submitAllSessionsInGroup when the SessionItem is in a group of two and there are errors"() {
given: "the SessionItem is in a SessionItemGroup that has 2 sessionItems"
setupData()
assert sessionItemGroup.getSize() == 2
and: "One of the sessionItems is submitted"
sessionItem1.status = "Submitted"
assert sessionItem2.status == "Not Started"
assert sessionItem2.submitTS == null
when:
submissionService.submitAllSessionsInGroup(sessionItem1)
then:
InvalidSessionItemStatusException exception = thrown()
exception != null
}
}
然而,我不确定我们如何使用控制器集成测试来做到这一点,就像下面这样:
// src/integration-test/groovy/com/hypercision/attproto/device/DeviceManagementControllerIntegrationSpec.groovy
package com.hypercision.attproto.device
import com.hypercision.attproto.instructor.Instructor
import com.hypercision.attproto.serviceaccount.ServiceAccount
import grails.gorm.transactions.Rollback
import grails.testing.mixin.integration.Integration
import groovy.json.JsonSlurper
import io.micronaut.http.HttpRequest
import io.micronaut.http.HttpResponse
import io.micronaut.http.HttpStatus
import io.micronaut.http.client.HttpClient
import io.micronaut.http.client.exceptions.HttpClientResponseException
import io.micronaut.http.uri.UriBuilder
import spock.lang.Shared
import spock.lang.Specification
import spock.lang.Unroll
@Integration
@Rollback
class DeviceManagementControllerIntegrationSpec extends Specification {
@Shared HttpClient client
@Shared String accessToken
@Shared Instructor instructor
@Shared List<Device> devices = []
void setup() {
// Use the ServiceAccount created in Bootstrap.groovy
ServiceAccount serviceAccount = ServiceAccount.first()
assert serviceAccount
instructor = new Instructor(userID: UUID.randomUUID().toString(), fullName: "Wendy",
serviceAccount: serviceAccount).save(failOnError: true)
Device device1 = new Device(deviceID: UUID.randomUUID().toString(), registrationCode: "111",
platform: "iOS", platformVersion: "12.1.0", instructor: instructor).save(failOnError: true)
Device device2 = new Device(deviceID: UUID.randomUUID().toString(), registrationCode: "222",
platform: "Android", platformVersion: "9.0.0", instructor: instructor,
approvalToken: "aaabbb123").save(failOnError: true)
devices = [device1, device2]
}
void cleanup() {
Device.deleteAll(Device.getAll())
instructor.delete()
}
@Unroll
void "test getDataForDeviceApproval returns 200"() {
given:
assert Device.count == 2
final String uriPath = '/deviceManagement/getDataForDeviceApproval'
String uri = UriBuilder.of(uriPath)
.queryParam("token", devices[1].approvalToken)
.queryParam("lang", lang)
.build()
when: "we call the method being tested"
HttpRequest request = HttpRequest.GET(uri)
HttpResponse<String> resp = this.client.toBlocking().exchange(request, String)
Map json = new JsonSlurper().parseText(resp.body()) as Map
then: "we get a success status"
resp.status == HttpStatus.OK
json != null
json.id == devices[1].id
json.deviceID == devices[1].deviceID
json.platform == devices[1].platform
json.approved == devices[1].approved
and: "the response has the correct dateformat for the locale of the request"
json.itemDisplayDateTimeFormat == format
when: "we call the method again with a bad approval token"
uri = UriBuilder.of(uriPath)
.queryParam("token", "this_does_not_match_any_saved_devices")
.queryParam("lang", lang)
.build()
request = HttpRequest.GET(uri)
this.client.toBlocking().exchange(request, String)
then: "we get a 400 status"
HttpClientResponseException ex = thrown()
HttpResponse<String> exResponse = ex.response as HttpResponse<String>
Map exJson = new JsonSlurper().parseText(exResponse.body()) as Map
exResponse.status == HttpStatus.BAD_REQUEST
and:
exJson != null
exJson.errors != null
exJson.errors.size() == 1
exJson.errors[0].message == "Invalid token: device not found with that token."
where:
lang | format
"en" | "MM/dd/yyyy HH:mm z"
"en_GB" | "d MMM yyyy HH:mm z"
}
}
更新:我创建了一个示例Grails 6.0.0项目,还没有能够重现这个问题,所以很可能是我们应用程序的配置有问题,导致了NoSuchMessageException。
1条答案
按热度按时间g2ieeal71#
在出现问题的应用程序中,我注意到使用非英语语言环境的控制器集成测试用例通过了。当调用我的控制器方法时,我通过在URL中设置
lang
参数来指定区域设置。lang=es
。当我添加一个与
messages.properties
文件内容相同的messages_en.properties
文件时,我的所有集成测试都通过了。所以现在,我的解决方法是创建一个指向www.example.com的符号链接messages.properties,并将该文件提交到我们的存储库: