我正在编写REST API春季启动的单元测试。

@RunWith(SpringRunner.class)
@SpringBootTest(classes = WorkflowEngineApiApplication.class, webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public class ActionControllerTest {
    @LocalServerPort
    public int port;

    private final TestRestTemplate testRestTemplate = new TestRestTemplate();

    private final HttpHeaders headers = new HttpHeaders();

    private long actionId;

    @BeforeClass
    public static void init() throws Exception{
        String url="/api/v1/actions";
        JSONObject action = new JSONObject();
        action.put("name", "Name Test");
        action.put("actionTypeId", 4L);
        action.put("description", null);
        action.put("createdBy", 2L);
        action.put("processId", 2L);
        headers.setContentType(MediaType.APPLICATION_JSON_UTF8);
        ResponseEntity<ApiResponse> response = getResponse(url, HttpMethod.POST, action.toString());
    }

    private static ResponseEntity<ApiResponse> getResponse(String url, HttpMethod method, Object action){
        HttpEntity<Object> entity = new HttpEntity<>(action, headers);
        return testRestTemplate.exchange(createURL(url), method, entity, ApiResponse.class);
    }

    /**
     * create url http://localhost:...
     * @param uri String
     * @return String
     */
    public static String createURL(String uri) {
        return "http://localhost:" + port + uri;
    }
}


我想在测试运行前向Db添加1行。但是,当我将@BeforeClass与静态方法一起使用时,端口变量必须是静态的,但不会生成值。还是0。
错误显示:


  org.springframework.web.client.ResourceAccessException:I / O错误
  POST请求“ http://localhost:0/api/v1/actions


我的问题:是否可以将@BeforeClasswebEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT一起使用,这意味着@LocalServerPort可以通过static生成值?

最佳答案

我想在测试运行前向Db添加1行。但是,当我将@BeforeClass与静态方法一起使用时,端口变量必须是静态的,但不会生成值。仍然是0。错误显示:


因此,据我了解,您想写下一个集成测试,该测试将从控制器流向DB。这是我的建议。

@RunWith(SpringRunner.class)
@SpringBootTest
@WebAppConfiguration
@Transactional
public class MyIntTest {

    private MockMvc mockMvc;

    @Autowired
    private WebApplicationContext ctx;

    @Before
    public void setUp() {
        this.mockMvc = MockMvcBuilders.webAppContextSetup(ctx).build();

        MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.post(**Your URL").accept(MediaType.APPLICATION_JSON)
                .contentType(MediaType.APPLICATION_JSON);

        builder.setContent("Your JSON request data");

        // This will invoke POST request to "URL"
        this.mockMvc.perform(builder);

        // If you want to verify your result
       MvcResult result = this.mockMvc.perform(builder).andReturn();

      // Read response. If its custom class then just change return type.
      List responses = getPojoFromMvcResult(result, List.class);
    }

    public static <T> T getPojoFromMvcResult(MvcResult result, Class<T> valueType) throws IOException {
        try {
            MockHttpServletResponse response = result.getResponse();
            String responseJson = response.getContentAsString();
            return convertJsonToObject(responseJson, valueType);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    public static <T> T convertJsonToObject(String content, Class<T> valueType) throws IOException {
        return new ObjectMapper().readValue(content, valueType);
    }
}

07-24 09:47