我有简单的集成测试
@Test
public void shouldReturnErrorMessageToAdminWhenCreatingUserWithUsedUserName() throws Exception {
mockMvc.perform(post("/api/users").header("Authorization", base64ForTestUser).contentType(MediaType.APPLICATION_JSON)
.content("{\"userName\":\"testUserDetails\",\"firstName\":\"xxx\",\"lastName\":\"xxx\",\"password\":\"xxx\"}"))
.andDo(print())
.andExpect(status().isBadRequest())
.andExpect(?);
}
在最后一行中,我想将响应体中收到的字符串与预期字符串
进行比较作为回应,我得到了:
MockHttpServletResponse:
Status = 400
Error message = null
Headers = {Content-Type=[application/json]}
Content type = application/json
Body = "Username already taken"
Forwarded URL = null
Redirected URL = null
尝试了一些内容(),body(),但没有任何效果。
答案 0 :(得分:274)
您可以致电andReturn()
并使用返回的MvcResult
对象将内容设为String
。见下文:
MvcResult result = mockMvc.perform(post("/api/users").header("Authorization", base64ForTestUser).contentType(MediaType.APPLICATION_JSON)
.content("{\"userName\":\"testUserDetails\",\"firstName\":\"xxx\",\"lastName\":\"xxx\",\"password\":\"xxx\"}"))
.andDo(MockMvcResultHandlers.print())
.andExpect(status().isBadRequest())
.andReturn();
String content = result.getResponse().getContentAsString();
// do what you will
答案 1 :(得分:80)
@Sotirios Delimanolis回答这项工作,但我正在寻找比较这个mockMvc断言中的字符串
所以这是
.andExpect(content().string("\"Username already taken - please try with different username\""));
当然我的断言失败了:
java.lang.AssertionError: Response content expected:
<"Username already taken - please try with different username"> but was:<"Something gone wrong">
,因为:
MockHttpServletResponse:
Body = "Something gone wrong"
所以这证明它有效!
答案 2 :(得分:52)
Spring MockMvc现在直接支持JSON。所以你只要说:
.andExpect(content().json("{'message':'ok'}"));
与字符串比较不同,它会说“缺少字段xyz”或“消息预期'确定'得到'nok'。
此方法是在Spring 4.1中引入的。
答案 3 :(得分:38)
阅读这些答案,我可以看到很多与Spring版本4.x相关的内容,我出于各种原因使用版本3.2.0。所以json支持直接来自content()
是不可能的。
我发现使用MockMvcResultMatchers.jsonPath
非常简单并且很有效。以下是测试post方法的示例。
这个解决方案的好处是你仍然匹配属性,而不是依赖于完整的json字符串比较。
(使用org.springframework.test.web.servlet.result.MockMvcResultMatchers
)
String expectedData = "some value";
mockMvc.perform(post("/endPoint")
.contentType(MediaType.APPLICATION_JSON)
.content(mockRequestBodyAsString.getBytes()))
.andExpect(status().isOk())
.andExpect(MockMvcResultMatchers.jsonPath("$.data").value(expectedData));
请求正文只是一个json字符串,如果你愿意,你可以从一个真正的json模拟数据文件中轻松加载,但是我没有把它包含在这里,因为它会偏离问题。
返回的实际json看起来像这样:
{
"data":"some value"
}
答案 4 :(得分:16)
Spring security的$('.someInput').keypress(function (e) {
if( e.which==188){
alert('coma')
}
else if( e.which==190){
alert('dot')
}
});
和hamcrest的@WithMockUser
匹配器提供了一个简单而优雅的解决方案:
containsString
答案 5 :(得分:16)
取自春天的tutorial
mockMvc.perform(get("/" + userName + "/bookmarks/"
+ this.bookmarkList.get(0).getId()))
.andExpect(status().isOk())
.andExpect(content().contentType(contentType))
.andExpect(jsonPath("$.id", is(this.bookmarkList.get(0).getId().intValue())))
.andExpect(jsonPath("$.uri", is("http://bookmark.com/1/" + userName)))
.andExpect(jsonPath("$.description", is("A description")));
is
可从import static org.hamcrest.Matchers.*;
jsonPath
可从import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
和jsonPath
引用可以找到here
答案 6 :(得分:4)
一种可能的方法是简单地包含gson
依赖项:
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
</dependency>
并解析值以进行验证:
@RunWith(SpringRunner.class)
@WebMvcTest(HelloController.class)
public class HelloControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private HelloService helloService;
@Before
public void before() {
Mockito.when(helloService.message()).thenReturn("hello world!");
}
@Test
public void testMessage() throws Exception {
MvcResult mvcResult = mockMvc.perform(MockMvcRequestBuilders.get("/"))
.andExpect(status().isOk())
.andExpect(content().contentTypeCompatibleWith(MediaType.APPLICATION_JSON_VALUE))
.andReturn();
String responseBody = mvcResult.getResponse().getContentAsString();
ResponseDto responseDto
= new Gson().fromJson(responseBody, ResponseDto.class);
Assertions.assertThat(responseDto.message).isEqualTo("hello world!");
}
}
答案 7 :(得分:2)
这是一种更优雅的方式
mockMvc.perform(post("/retrieve?page=1&countReg=999999")
.header("Authorization", "Bearer " + validToken))
.andExpect(status().isOk())
.andExpect(content().string(containsString("regCount")));
答案 8 :(得分:2)
这里是一个示例,该示例说明如何解析JSON响应,甚至如何使用JSON形式的bean发送请求:
@Autowired
protected MockMvc mvc;
private static final ObjectMapper MAPPER = new ObjectMapper()
.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false)
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
.registerModule(new JavaTimeModule());
public static String requestBody(Object request) {
try {
return MAPPER.writeValueAsString(request);
} catch (JsonProcessingException e) {
throw new RuntimeException(e);
}
}
public static <T> T parseResponse(MvcResult result, Class<T> responseClass) {
try {
String contentAsString = result.getResponse().getContentAsString();
return MAPPER.readValue(contentAsString, responseClass);
} catch (IOException e) {
throw new RuntimeException(e);
}
}
@Test
public void testUpdate() {
Book book = new Book();
book.setTitle("1984");
book.setAuthor("Orwell");
MvcResult requestResult = mvc.perform(post("http://example.com/book/")
.contentType(MediaType.APPLICATION_JSON)
.content(requestBody(book)))
.andExpect(status().isOk())
.andReturn();
UpdateBookResponse updateBookResponse = parseResponse(requestResult, UpdateBookResponse.class);
assertEquals("1984", updateBookResponse.getTitle());
assertEquals("Orwell", updateBookResponse.getAuthor());
}
如您在此处看到的,Book
是请求DTO,而UpdateBookResponse
是从JSON解析的响应对象。您可能需要更改Jakson的ObjectMapper
配置。
答案 9 :(得分:2)
您可以使用getContentAsString
方法以字符串形式获取响应数据。
String payload = "....";
String apiToTest = "....";
MvcResult mvcResult = mockMvc.
perform(post(apiToTest).
content(payload).
contentType(MediaType.APPLICATION_JSON)).
andReturn();
String responseData = mvcResult.getResponse().getContentAsString();
您可以将此link用于测试应用程序。
答案 10 :(得分:1)
String body = mockMvc.perform(bla... bla).andReturn().getResolvedException().getMessage()
这应该会给你回复的正文。 &#34;用户名已经采取&#34;在你的情况下。
答案 11 :(得分:0)
另一个选择是:
when:
def response = mockMvc.perform(
get('/path/to/api')
.header("Content-Type", "application/json"))
then:
response.andExpect(status().isOk())
response.andReturn().getResponse().getContentAsString() == "what you expect"