0

我有一个部署到 Tomcat 的小型示例 servlet,它将回显任何接收到的参数。Tomcat 配置为支持 HTTP/1.1 和 HTTP/2。我可以用来curl向 servlet 发出 GET 请求,证明它可以在 HTTP/1.1 和 HTTP/2 上工作。

使用 Apache HTTP 5.0 客户端我有一个单元测试来命中 servlet 和POST一个带有一些参数(请求内容)的请求,它工作正常 - 服务器接收参数并将它们返回给我可以读取它们的客户端。在这个测试中,我调用客户端使用 HTTP/1.1 或 HTTP/2CloseableHttpAsyncClient.setVersionPolicy()调用。但是,如果我更改为使用 HTTP/2,则客户端不会将请求内容发送到服务器。

下面是成功的 HTTP/1.1 调用的示例代码 - 你可以看到我有setVersionPolicy(HttpVersionPolicy.FORCE_HTTP_1). 如果我将其更改为,FORCE_HTTP_2则通过 HTTP/2.0 进行调用,但不发送请求内容。

谁能建议我做错了什么或我需要做些什么?我没有找到很多关于使用新的 Apache 5.0 客户端库的文档,并且这些示例没有显示发送和 POST 内容(或者至少是我能找到的那些)。

    public void testWithHTTP1() throws Exception {

        final String content = "Code=99&Message=Hello";
        final String contentType = "application/text";

        final CloseableHttpAsyncClient client = HttpAsyncClients.custom()
                .setVersionPolicy(HttpVersionPolicy.FORCE_HTTP_1)
                .build();

        client.start();

        final HttpHost target = new HttpHost("http","localhost",8002);
        final String requestUri = "/VSFTestMT_Test/Test/Test_EchoHttpGet";

        final HttpClientContext clientContext = HttpClientContext.create();

        final SimpleHttpRequest httppost = SimpleHttpRequests.post(target, requestUri);     
        httppost.setBody(content, ContentType.create("application.x_www_form_urlencoded"));
        
        final Future<SimpleHttpResponse> future = client.execute(
                httppost,
                clientContext,
                new FutureCallback<SimpleHttpResponse>() {

                    @Override
                    public void completed(final SimpleHttpResponse response) {
                        System.out.println(requestUri + "->" + response.getCode());
                        System.out.println(response.getBody());
                    }

                    @Override
                    public void failed(final Exception ex) {
                        System.out.println(requestUri + "->" + ex);
                    }

                    @Override
                    public void cancelled() {
                        System.out.println(requestUri + " cancelled");
                    }

                });


        System.out.println("Shutting down");
        client.close(CloseMode.GRACEFUL);
    }
4

2 回答 2

1

HTTP/1 可以通过 HTTP 和 HTTPS 实现,但对于 HTTP/2,它只能在 HTTPS 中实现。

从我可以看到你只调用 http url,这可能是 servlet 无法获取有效负载内容的原因

请看这个更详细

于 2021-03-02T15:03:44.597 回答
0

看起来他们有一个完全不同的 Http/2 构建器来查看他们的源代码:

/**
 * Creates builder object for construction of custom HTTP/2
 * {@link CloseableHttpAsyncClient} instances optimized for HTTP/2 protocol
 * and message multiplexing
 */
public static H2AsyncClientBuilder customHttp2() {
    return H2AsyncClientBuilder.create();
}

/**
 * Creates HTTP/2 {@link CloseableHttpAsyncClient} instance with default configuration and
 * system properties optimized for HTTP/2 protocol and message multiplexing.
 */
public static CloseableHttpAsyncClient createHttp2System() {
    return H2AsyncClientBuilder.create().useSystemProperties().build();
}

因此,我建议您尝试实际的 customHttp2(),而不是 custom()。

https://github.com/apache/httpcomponents-client/blob/master/httpclient5/src/main/java/org/apache/hc/client5/http/impl/async/HttpAsyncClients.java

于 2021-03-02T13:20:27.170 回答