使用 MockRestServiceServer 测试 HttpRequestExecutingMessageHandler
Testing HttpRequestExecutingMessageHandler using MockRestServiceServer
我正在开发一个使用 Spring 集成的应用程序,该集成使用 HttpRequestExecutingMessageHandler
class 定期向 REST 服务发出后端请求。我想在测试中模拟 REST 服务器,而不是构建一个模拟服务器,我更愿意使用 MockRestServiceServer
来执行此操作。然而,MockRestServiceServer
似乎并没有拦截 RestTemplate
调用,而是它们正在通过(到 http://example.com/
)并引发 java.net.ConnectException: Connection refused
。有没有办法强制 HttpRequestExecutingMessageHandler
调用 MockRestServiceServer
,或者我应该重新考虑这个测试策略?
应用配置:
@Configuration
public class RestClientTestApplicationConfig {
@Bean
@Qualifier("httpRequestChannel")
public MessageChannel httpRequestChannel() { return new QueueChannel(); }
@Bean
@Qualifier("httpReplyChannel")
public MessageChannel httpReplyChannel() { return new QueueChannel(); }
@Bean
public RestTemplate restTemplate() { return new RestTemplate(); }
@Bean
@InboundChannelAdapter(value="httpRequestChannel", poller=@Poller(fixedDelay = "1000"))
public MessageSource<String> httpRequestTrigger() { return new ExpressionEvaluatingMessageSource<>(new LiteralExpression(""), String.class); }
@Bean
@ServiceActivator(inputChannel="httpRequestChannel", poller=@Poller(fixedDelay = "1000"))
public MessageHandler messageHandler(
RestTemplate restTemplate,
@Qualifier("httpReplyChannel") MessageChannel messageChannel,
@Value("${url}") String url
) {
HttpRequestExecutingMessageHandler messageHandler = new HttpRequestExecutingMessageHandler(url, restTemplate);
messageHandler.setOutputChannel(messageChannel);
return messageHandler;
}
}
(url
在 application-test.properties
中被定义为 http://example.com
在测试中和真正的 URL 否则)
测试:
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest
public class RestClientIntegrationTest {
@Autowired
private RestTemplate restTemplate;
private MockRestServiceServer mockServer;
@Before
public void setup() {
mockServer = MockRestServiceServer.createServer(restTemplate);
}
@Test
public void makesBackendRequest() {
mockServer.expect(ExpectedCount.once(), MockRestRequestMatchers.requestTo("http://example.com/"))
.andExpect(MockRestRequestMatchers.method(HttpMethod.GET));
mockServer.verify();
}
}
测试结果:
2016-12-29 16:14:36.902 ERROR 16665 --- [ask-scheduler-2] o.s.integration.handler.LoggingHandler : org.springframework.messaging.MessageHandlingException: HTTP request execution failed for URI [http://example.com]; nested exception is org.springframework.web.client.ResourceAccessException: I/O error on POST request for "http://example.com": Connection refused (Connection refused); nested exception is java.net.ConnectException: Connection refused (Connection refused)
at org.springframework.integration.http.outbound.HttpRequestExecutingMessageHandler.handleRequestMessage(HttpRequestExecutingMessageHandler.java:409)
java.lang.AssertionError: Further request(s) expected leaving 1 unsatisfied expectation(s).
0 request(s) executed.
at org.springframework.test.web.client.AbstractRequestExpectationManager.verify(AbstractRequestExpectationManager.java:103)
at org.springframework.test.web.client.MockRestServiceServer.verify(MockRestServiceServer.java:117)
at com.restclienttest.RestClientIntegrationTest.makesBackendRequest(RestClientIntegrationTest.java:35)
更新
根据 Artem Bilan 的评论,对测试代码进行了如下调整:
mockServer.expect(ExpectedCount.once(), MockRestRequestMatchers.requestTo("http://example.com/"))
.andExpect(MockRestRequestMatchers.method(HttpMethod.GET))
.andRespond(MockRestResponseCreators.withSuccess("example reply", MediaType.TEXT_PLAIN));
Message<?> message = httpReplyChannel.receive(1001);
assertNotNull(message);
assertThat(((ResponseEntity<String>) message.getPayload()).getBody(), is("example reply"));
仍然收到 ConnectException
并且 MockRestServiceServer
发送的示例回复似乎没有通过,因为 ResponseEntity
的正文为空。
我觉得你在这里很好。唯一的问题是您忽略了您的应用程序是 async 的事实。 @InboundChannelAdapter
定期向 QueueChannel
发送消息,依此类推。但它是在轮询器的线程中执行的,而不是在您等待验证的线程中执行的。
作为修复,我认为您真的应该等待 httpReplyChannel
通过其 .receive(10000)
方法回复。只有在那之后调用 mockServer.verify()
.
更新
嗯。我会说我们已经为您准备了一个测试用例:
<bean id="restTemplate" class="org.springframework.web.client.RestTemplate"/>
<int-http:outbound-gateway url="/testApps/httpMethod"
request-channel="requestChannel"
reply-channel="replyChannel"
rest-template="restTemplate"
expected-response-type="java.lang.String"
http-method-expression="payload"/>
<int:channel id="replyChannel">
<int:queue/>
</int:channel>
@Autowired
private RestTemplate restTemplate;
private MockRestServiceServer mockServer;
@Before
public void setup() {
this.mockServer = MockRestServiceServer.createServer(this.restTemplate);
}
@Test
public void testDefaultMethod() throws Exception {
this.mockServer.expect(requestTo("/testApps/httpMethod"))
.andExpect(method(HttpMethod.POST))
.andRespond(withSuccess(HttpMethod.POST.name(), MediaType.TEXT_PLAIN));
this.defaultChannel.send(new GenericMessage<String>("Hello"));
Message<?> message = this.replyChannel.receive(5000);
assertNotNull(message);
assertEquals("POST", message.getPayload());
this.mockServer.verify();
}
我正在开发一个使用 Spring 集成的应用程序,该集成使用 HttpRequestExecutingMessageHandler
class 定期向 REST 服务发出后端请求。我想在测试中模拟 REST 服务器,而不是构建一个模拟服务器,我更愿意使用 MockRestServiceServer
来执行此操作。然而,MockRestServiceServer
似乎并没有拦截 RestTemplate
调用,而是它们正在通过(到 http://example.com/
)并引发 java.net.ConnectException: Connection refused
。有没有办法强制 HttpRequestExecutingMessageHandler
调用 MockRestServiceServer
,或者我应该重新考虑这个测试策略?
应用配置:
@Configuration
public class RestClientTestApplicationConfig {
@Bean
@Qualifier("httpRequestChannel")
public MessageChannel httpRequestChannel() { return new QueueChannel(); }
@Bean
@Qualifier("httpReplyChannel")
public MessageChannel httpReplyChannel() { return new QueueChannel(); }
@Bean
public RestTemplate restTemplate() { return new RestTemplate(); }
@Bean
@InboundChannelAdapter(value="httpRequestChannel", poller=@Poller(fixedDelay = "1000"))
public MessageSource<String> httpRequestTrigger() { return new ExpressionEvaluatingMessageSource<>(new LiteralExpression(""), String.class); }
@Bean
@ServiceActivator(inputChannel="httpRequestChannel", poller=@Poller(fixedDelay = "1000"))
public MessageHandler messageHandler(
RestTemplate restTemplate,
@Qualifier("httpReplyChannel") MessageChannel messageChannel,
@Value("${url}") String url
) {
HttpRequestExecutingMessageHandler messageHandler = new HttpRequestExecutingMessageHandler(url, restTemplate);
messageHandler.setOutputChannel(messageChannel);
return messageHandler;
}
}
(url
在 application-test.properties
中被定义为 http://example.com
在测试中和真正的 URL 否则)
测试:
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest
public class RestClientIntegrationTest {
@Autowired
private RestTemplate restTemplate;
private MockRestServiceServer mockServer;
@Before
public void setup() {
mockServer = MockRestServiceServer.createServer(restTemplate);
}
@Test
public void makesBackendRequest() {
mockServer.expect(ExpectedCount.once(), MockRestRequestMatchers.requestTo("http://example.com/"))
.andExpect(MockRestRequestMatchers.method(HttpMethod.GET));
mockServer.verify();
}
}
测试结果:
2016-12-29 16:14:36.902 ERROR 16665 --- [ask-scheduler-2] o.s.integration.handler.LoggingHandler : org.springframework.messaging.MessageHandlingException: HTTP request execution failed for URI [http://example.com]; nested exception is org.springframework.web.client.ResourceAccessException: I/O error on POST request for "http://example.com": Connection refused (Connection refused); nested exception is java.net.ConnectException: Connection refused (Connection refused)
at org.springframework.integration.http.outbound.HttpRequestExecutingMessageHandler.handleRequestMessage(HttpRequestExecutingMessageHandler.java:409)
java.lang.AssertionError: Further request(s) expected leaving 1 unsatisfied expectation(s).
0 request(s) executed.
at org.springframework.test.web.client.AbstractRequestExpectationManager.verify(AbstractRequestExpectationManager.java:103)
at org.springframework.test.web.client.MockRestServiceServer.verify(MockRestServiceServer.java:117)
at com.restclienttest.RestClientIntegrationTest.makesBackendRequest(RestClientIntegrationTest.java:35)
更新 根据 Artem Bilan 的评论,对测试代码进行了如下调整:
mockServer.expect(ExpectedCount.once(), MockRestRequestMatchers.requestTo("http://example.com/"))
.andExpect(MockRestRequestMatchers.method(HttpMethod.GET))
.andRespond(MockRestResponseCreators.withSuccess("example reply", MediaType.TEXT_PLAIN));
Message<?> message = httpReplyChannel.receive(1001);
assertNotNull(message);
assertThat(((ResponseEntity<String>) message.getPayload()).getBody(), is("example reply"));
仍然收到 ConnectException
并且 MockRestServiceServer
发送的示例回复似乎没有通过,因为 ResponseEntity
的正文为空。
我觉得你在这里很好。唯一的问题是您忽略了您的应用程序是 async 的事实。 @InboundChannelAdapter
定期向 QueueChannel
发送消息,依此类推。但它是在轮询器的线程中执行的,而不是在您等待验证的线程中执行的。
作为修复,我认为您真的应该等待 httpReplyChannel
通过其 .receive(10000)
方法回复。只有在那之后调用 mockServer.verify()
.
更新
嗯。我会说我们已经为您准备了一个测试用例:
<bean id="restTemplate" class="org.springframework.web.client.RestTemplate"/>
<int-http:outbound-gateway url="/testApps/httpMethod"
request-channel="requestChannel"
reply-channel="replyChannel"
rest-template="restTemplate"
expected-response-type="java.lang.String"
http-method-expression="payload"/>
<int:channel id="replyChannel">
<int:queue/>
</int:channel>
@Autowired
private RestTemplate restTemplate;
private MockRestServiceServer mockServer;
@Before
public void setup() {
this.mockServer = MockRestServiceServer.createServer(this.restTemplate);
}
@Test
public void testDefaultMethod() throws Exception {
this.mockServer.expect(requestTo("/testApps/httpMethod"))
.andExpect(method(HttpMethod.POST))
.andRespond(withSuccess(HttpMethod.POST.name(), MediaType.TEXT_PLAIN));
this.defaultChannel.send(new GenericMessage<String>("Hello"));
Message<?> message = this.replyChannel.receive(5000);
assertNotNull(message);
assertEquals("POST", message.getPayload());
this.mockServer.verify();
}