mirror of
https://github.com/github/awesome-copilot.git
synced 2026-03-20 08:05:12 +00:00
- Introduced MockMvcTester for AssertJ-style assertions in Spring MVC testing. - Added @RestClientTest for testing REST clients with MockRestServiceServer. - Implemented RestTestClient as a modern alternative to TestRestTemplate. - Documented migration steps from Spring Boot 3.x to 4.0, including dependency and annotation changes. - Created an overview of test slices to guide testing strategies. - Included Testcontainers setup for JDBC testing with PostgreSQL and MySQL. - Enhanced @WebMvcTest documentation with examples for various HTTP methods and validation.
7.5 KiB
7.5 KiB
MockMvcTester
AssertJ-style testing for Spring MVC controllers (Spring Boot 3.2+).
Overview
MockMvcTester provides fluent, AssertJ-style assertions for web layer testing. More readable and type-safe than traditional MockMvc.
Recommended Pattern: Convert JSON to real objects and assert with AssertJ:
assertThat(mvc.get().uri("/orders/1"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(OrderResponse.class)
.satisfies(response -> {
assertThat(response.getTotalToPay()).isEqualTo(expectedAmount);
assertThat(response.getItems()).isNotEmpty();
});
Basic Usage
@WebMvcTest(OrderController.class)
class OrderControllerTest {
@Autowired
private MockMvcTester mvc;
@MockitoBean
private OrderService orderService;
}
Recommended: Object Conversion Pattern
Single Object Response
@Test
void shouldGetOrder() {
given(orderService.findById(1L)).willReturn(new Order(1L, "PENDING", 99.99));
assertThat(mvc.get().uri("/orders/1"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(OrderResponse.class)
.satisfies(response -> {
assertThat(response.getId()).isEqualTo(1L);
assertThat(response.getStatus()).isEqualTo("PENDING");
assertThat(response.getTotalToPay()).isEqualTo(new BigDecimal("99.99"));
});
}
List Response
@Test
void shouldGetAllOrders() {
given(orderService.findAll()).willReturn(Arrays.asList(
new Order(1L, "PENDING"),
new Order(2L, "COMPLETED")
));
assertThat(mvc.get().uri("/orders"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(new TypeReference<List<OrderResponse>>() {})
.satisfies(orders -> {
assertThat(orders).hasSize(2);
assertThat(orders.get(0).getStatus()).isEqualTo("PENDING");
assertThat(orders.get(1).getStatus()).isEqualTo("COMPLETED");
});
}
Nested Objects
@Test
void shouldGetOrderWithCustomer() {
assertThat(mvc.get().uri("/orders/1"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(OrderResponse.class)
.satisfies(response -> {
assertThat(response.getCustomer()).isNotNull();
assertThat(response.getCustomer().getName()).isEqualTo("John Doe");
assertThat(response.getCustomer().getAddress().getCity()).isEqualTo("Berlin");
});
}
Complex Assertions
@Test
void shouldCalculateOrderTotal() {
assertThat(mvc.get().uri("/orders/1/calculate"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(CalculationResponse.class)
.satisfies(calc -> {
assertThat(calc.getSubtotal()).isEqualTo(new BigDecimal("100.00"));
assertThat(calc.getTax()).isEqualTo(new BigDecimal("19.00"));
assertThat(calc.getTotalToPay()).isEqualTo(new BigDecimal("119.00"));
assertThat(calc.getItems()).allMatch(item -> item.getPrice().compareTo(BigDecimal.ZERO) > 0);
});
}
HTTP Methods
POST with Request Body
@Test
void shouldCreateOrder() {
given(orderService.create(any())).willReturn(1L);
assertThat(mvc.post().uri("/orders")
.contentType(MediaType.APPLICATION_JSON)
.content("{\"product\": \"Laptop\", \"quantity\": 2}"))
.hasStatus(HttpStatus.CREATED)
.hasHeader("Location", "/orders/1");
}
PUT Request
@Test
void shouldUpdateOrder() {
assertThat(mvc.put().uri("/orders/1")
.contentType(MediaType.APPLICATION_JSON)
.content("{\"status\": \"COMPLETED\"}"))
.hasStatus(HttpStatus.OK);
}
DELETE Request
@Test
void shouldDeleteOrder() {
assertThat(mvc.delete().uri("/orders/1"))
.hasStatus(HttpStatus.NO_CONTENT);
}
Status Assertions
assertThat(mvc.get().uri("/orders/1"))
.hasStatusOk() // 200
.hasStatus(HttpStatus.OK) // 200
.hasStatus2xxSuccessful() // 2xx
.hasStatusBadRequest() // 400
.hasStatusNotFound() // 404
.hasStatusUnauthorized() // 401
.hasStatusForbidden() // 403
.hasStatus(HttpStatus.CREATED); // 201
Content Type Assertions
assertThat(mvc.get().uri("/orders/1"))
.hasContentType(MediaType.APPLICATION_JSON)
.hasContentTypeCompatibleWith(MediaType.APPLICATION_JSON);
Header Assertions
assertThat(mvc.post().uri("/orders"))
.hasHeader("Location", "/orders/123")
.hasHeader("X-Request-Id", matchesPattern("[a-z0-9-]+"));
Alternative: JSON Path (Use Sparingly)
Only use when you cannot convert to a typed object:
assertThat(mvc.get().uri("/orders/1"))
.hasStatusOk()
.bodyJson()
.extractingPath("$.customer.address.city")
.asString()
.isEqualTo("Berlin");
Request Parameters
// Query parameters
assertThat(mvc.get().uri("/orders?status=PENDING&page=0"))
.hasStatusOk();
// Path parameters
assertThat(mvc.get().uri("/orders/{id}", 1L))
.hasStatusOk();
// Headers
assertThat(mvc.get().uri("/orders/1")
.header("X-Api-Key", "secret"))
.hasStatusOk();
Request Body with JacksonTester
@Autowired
private JacksonTester<OrderRequest> json;
@Test
void shouldCreateOrder() {
OrderRequest request = new OrderRequest("Laptop", 2);
assertThat(mvc.post().uri("/orders")
.contentType(MediaType.APPLICATION_JSON)
.content(json.write(request).getJson()))
.hasStatus(HttpStatus.CREATED);
}
Error Responses
@Test
void shouldReturnValidationErrors() {
given(orderService.findById(999L))
.willThrow(new OrderNotFoundException(999L));
assertThat(mvc.get().uri("/orders/999"))
.hasStatus(HttpStatus.NOT_FOUND)
.bodyJson()
.convertTo(ErrorResponse.class)
.satisfies(error -> {
assertThat(error.getMessage()).isEqualTo("Order 999 not found");
assertThat(error.getCode()).isEqualTo("ORDER_NOT_FOUND");
});
}
Validation Error Testing
@Test
void shouldRejectInvalidOrder() {
OrderRequest invalidRequest = new OrderRequest("", -1);
assertThat(mvc.post().uri("/orders")
.contentType(MediaType.APPLICATION_JSON)
.content(json.write(invalidRequest).getJson()))
.hasStatus(HttpStatus.BAD_REQUEST)
.bodyJson()
.convertTo(ValidationErrorResponse.class)
.satisfies(errors -> {
assertThat(errors.getFieldErrors()).hasSize(2);
assertThat(errors.getFieldErrors())
.extracting("field")
.contains("product", "quantity");
});
}
Comparison: MockMvcTester vs Classic MockMvc
| Feature | MockMvcTester | Classic MockMvc |
|---|---|---|
| Style | AssertJ fluent | MockMvc matchers |
| Readability | High | Medium |
| Type Safety | Better | Less |
| IDE Support | Excellent | Good |
| Object Conversion | Native | Manual |
Migration from Classic MockMvc
Before (Classic)
mvc.perform(get("/orders/1"))
.andExpect(status().isOk())
.andExpect(jsonPath("$.status").value("PENDING"))
.andExpect(jsonPath("$.totalToPay").value(99.99));
After (Tester with Object Conversion)
assertThat(mvc.get().uri("/orders/1"))
.hasStatus(HttpStatus.OK)
.bodyJson()
.convertTo(OrderResponse.class)
.satisfies(response -> {
assertThat(response.getStatus()).isEqualTo("PENDING");
assertThat(response.getTotalToPay()).isEqualTo(new BigDecimal("99.99"));
});
Key Points
- Prefer
convertTo()overextractingPath()- Type-safe, refactorable - Use
satisfies()for multiple assertions - Keeps tests readable - Import static
org.assertj.core.api.Assertions.assertThat - Works with generics via
TypeReference- ForList<T>responses - IDE refactoring friendly - Rename fields, IDE updates tests