From 4cdc08a661a17bad417b69aaabf403f9e2286847 Mon Sep 17 00:00:00 2001 From: Arturo Bernal Date: Thu, 19 Sep 2024 21:58:28 +0200 Subject: [PATCH] [HTTPCLIENT-1843] - Delegate compression handling to Apache Commons Compress * Integrated Apache Commons Compress into CompressorFactory to handle compression and decompression of HTTP entities using supported algorithms (gzip, deflate, etc.). --- .../sync/StandardTestClientBuilder.java | 9 + .../extension/sync/TestClientBuilder.java | 4 + .../extension/sync/TestClientResources.java | 5 + .../sync/AbstractIntegrationTestBase.java | 4 + .../testing/sync/TestContentCodings.java | 6 +- .../client5/testing/sync/TestRedirects.java | 4 +- .../CompressedResponseHandlingExample.java | 113 +++++++ httpclient5/pom.xml | 4 + .../entity/BrotliDecompressingEntity.java | 2 + .../http/entity/BrotliInputStreamFactory.java | 2 + .../http/entity/CompressingEntity.java | 132 ++++++++ .../http/entity/CompressorFactory.java | 282 ++++++++++++++++++ .../client5/http/entity/DecompressEntity.java | 149 +++++++++ .../http/entity/DecompressingEntity.java | 2 + .../entity/DeflateDecompressingEntity.java | 2 + .../http/entity/DeflateInputStream.java | 2 + .../entity/DeflateInputStreamFactory.java | 2 + .../hc/client5/http/entity/EntityBuilder.java | 30 +- .../http/entity/GZIPInputStreamFactory.java | 2 + .../http/entity/GzipCompressingEntity.java | 2 + .../http/entity/GzipDecompressingEntity.java | 2 + .../http/entity/InputStreamFactory.java | 2 + .../entity/LazyDecompressInputStream.java | 214 +++++++++++++ .../entity/LazyDecompressingInputStream.java | 2 + .../impl/classic/ContentCompressionExec.java | 68 ++--- .../http/impl/classic/HttpClientBuilder.java | 80 +++-- .../hc/client5/http/entity/TestBrotli.java | 2 +- .../http/entity/TestDecompressingEntity.java | 38 ++- .../hc/client5/http/entity/TestDeflate.java | 2 +- .../http/entity/TestEntityBuilder.java | 21 +- .../hc/client5/http/entity/TestGZip.java | 65 ++-- .../classic/TestContentCompressionExec.java | 10 +- pom.xml | 6 + 33 files changed, 1159 insertions(+), 111 deletions(-) create mode 100644 httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/compress/CompressedResponseHandlingExample.java create mode 100644 httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressingEntity.java create mode 100644 httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressorFactory.java create mode 100644 httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressEntity.java create mode 100644 httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressInputStream.java diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/StandardTestClientBuilder.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/StandardTestClientBuilder.java index f75c81b79c..7d9602cd88 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/StandardTestClientBuilder.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/StandardTestClientBuilder.java @@ -57,6 +57,8 @@ final class StandardTestClientBuilder implements TestClientBuilder { private HttpClientConnectionManager connectionManager; + private boolean noWrap; + public StandardTestClientBuilder() { this.clientBuilder = HttpClientBuilder.create(); } @@ -72,6 +74,12 @@ public TestClientBuilder setTimeout(final Timeout timeout) { return this; } + @Override + public TestClientBuilder setNoWrap(final boolean noWrap) { + this.noWrap = noWrap; + return this; + } + @Override public TestClientBuilder setConnectionManager(final HttpClientConnectionManager connectionManager) { this.connectionManager = connectionManager; @@ -165,6 +173,7 @@ public TestClient build() throws Exception { final CloseableHttpClient client = clientBuilder .setConnectionManager(connectionManagerCopy) + .setNoWrap(noWrap) .build(); return new TestClient(client, connectionManagerCopy); } diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientBuilder.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientBuilder.java index 4489b265fc..2760a0b194 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientBuilder.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientBuilder.java @@ -98,6 +98,10 @@ default TestClientBuilder addExecInterceptorLast(String name, ExecChainHandler throw new UnsupportedOperationException("Operation not supported by " + getProtocolLevel()); } + default TestClientBuilder setNoWrap(boolean noWrap){ + return this; + } + TestClient build() throws Exception; } diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientResources.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientResources.java index c38c7fba1b..db4aaa5904 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientResources.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/extension/sync/TestClientResources.java @@ -118,4 +118,9 @@ public TestClient client() throws Exception { return client; } + public TestClient client(final boolean noWrap) throws Exception { + clientBuilder.setNoWrap(noWrap); + return client(); + } + } diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/AbstractIntegrationTestBase.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/AbstractIntegrationTestBase.java index daa5836ffe..e3f9fe2eae 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/AbstractIntegrationTestBase.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/AbstractIntegrationTestBase.java @@ -78,4 +78,8 @@ public TestClient client() throws Exception { return testResources.client(); } + public TestClient client(final boolean noWrap) throws Exception { + return testResources.client(noWrap); + } + } diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestContentCodings.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestContentCodings.java index 0ec2f70c83..3f9aac5e57 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestContentCodings.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestContentCodings.java @@ -108,7 +108,7 @@ void testDeflateSupportForServerReturningRfc1950Stream() throws Exception { final HttpHost target = startServer(); - final TestClient client = client(); + final TestClient client = client(true); final HttpGet request = new HttpGet("/some-resource"); client.execute(target, request, response -> { @@ -133,7 +133,7 @@ void testDeflateSupportForServerReturningRfc1951Stream() throws Exception { final HttpHost target = startServer(); - final TestClient client = client(); + final TestClient client = client(false); final HttpGet request = new HttpGet("/some-resource"); client.execute(target, request, response -> { @@ -289,7 +289,7 @@ void deflateResponsesWorkWithBasicResponseHandler() throws Exception { final HttpHost target = startServer(); - final TestClient client = client(); + final TestClient client = client(true); final HttpGet request = new HttpGet("/some-resource"); final String response = client.execute(target, request, new BasicHttpClientResponseHandler()); diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestRedirects.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestRedirects.java index 7d27efb00c..43ee335ed1 100644 --- a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestRedirects.java +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/TestRedirects.java @@ -639,8 +639,8 @@ public void handle(final ClassicHttpRequest request, Assertions.assertEquals(new URIBuilder().setHttpHost(target).setPath("/random/100").build(), reqWrapper.getUri()); - assertThat(values.poll(), CoreMatchers.equalTo("gzip, x-gzip, deflate")); - assertThat(values.poll(), CoreMatchers.equalTo("gzip, x-gzip, deflate")); + assertThat(values.poll(), CoreMatchers.equalTo("snappy-raw, xz, snappy-framed, bzip2, lz4-framed, deflate64, br, lzma, zstd, lz4-block, gz, deflate, z, pack200")); + assertThat(values.poll(), CoreMatchers.equalTo("snappy-raw, xz, snappy-framed, bzip2, lz4-framed, deflate64, br, lzma, zstd, lz4-block, gz, deflate, z, pack200")); assertThat(values.poll(), CoreMatchers.nullValue()); } diff --git a/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/compress/CompressedResponseHandlingExample.java b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/compress/CompressedResponseHandlingExample.java new file mode 100644 index 0000000000..099439bacd --- /dev/null +++ b/httpclient5-testing/src/test/java/org/apache/hc/client5/testing/sync/compress/CompressedResponseHandlingExample.java @@ -0,0 +1,113 @@ +/* + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + * + * This software consists of voluntary contributions made by many + * individuals on behalf of the Apache Software Foundation. For more + * information on the Apache Software Foundation, please see + * . + * + */ + +package org.apache.hc.client5.testing.sync.compress; + +import java.util.Arrays; +import java.util.List; + +import org.apache.hc.client5.http.entity.CompressorFactory; +import org.apache.hc.client5.http.impl.classic.CloseableHttpClient; +import org.apache.hc.client5.http.impl.classic.HttpClients; +import org.apache.hc.core5.http.ClassicHttpRequest; +import org.apache.hc.core5.http.ContentType; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.HttpHeaders; +import org.apache.hc.core5.http.HttpHost; +import org.apache.hc.core5.http.io.entity.EntityUtils; +import org.apache.hc.core5.http.io.entity.StringEntity; +import org.apache.hc.core5.http.io.support.ClassicRequestBuilder; +import org.apache.hc.core5.io.CloseMode; +import org.apache.hc.core5.testing.classic.ClassicTestServer; + + +/** + * Demonstrates handling of HTTP responses with content compression using Apache HttpClient. + *

+ * This example sets up a local test server that simulates compressed HTTP responses. It then + * creates a custom HttpClient configured to handle compression. The client makes a request to + * the test server, receives a compressed response, and decompresses the content to verify the + * process. + *

+ * The main focus of this example is to illustrate the use of a custom HttpClient that can + * handle compressed HTTP responses transparently, simulating a real-world scenario where + * responses from a server might be compressed to reduce bandwidth usage. + */ +public class CompressedResponseHandlingExample { + + public static void main(final String[] args) { + + final ClassicTestServer server = new ClassicTestServer(); + try { + server.register("/compressed", (request, response, context) -> { + final String uncompressedContent = "This is the uncompressed response content"; + response.setEntity(compress(uncompressedContent, "gzip")); + response.addHeader(HttpHeaders.CONTENT_ENCODING, "gzip"); + }); + + server.start(); + + final HttpHost target = new HttpHost("localhost", server.getPort()); + + final List encodingList = Arrays.asList("gz", "deflate"); + + try (final CloseableHttpClient httpclient = HttpClients + .custom() + .setEncodings(encodingList) + .build()) { + final ClassicHttpRequest httpGet = ClassicRequestBuilder.get() + .setHttpHost(target) + .setPath("/compressed") + .build(); + + System.out.println("Executing request " + httpGet.getMethod() + " " + httpGet.getUri()); + httpclient.execute(httpGet, response -> { + System.out.println("----------------------------------------"); + System.out.println(httpGet + "->" + response.getCode() + " " + response.getReasonPhrase()); + + final HttpEntity responseEntity = response.getEntity(); + final String responseBody = EntityUtils.toString(responseEntity); + System.out.println("Response content: " + responseBody); + + return null; + }); + } + + } catch (final Exception e) { + e.printStackTrace(); + } finally { + server.shutdown(CloseMode.GRACEFUL); + } + } + + + private static HttpEntity compress(final String data, final String name) { + final StringEntity originalEntity = new StringEntity(data, ContentType.TEXT_PLAIN); + return CompressorFactory.INSTANCE.compressEntity(originalEntity, name); + } + +} diff --git a/httpclient5/pom.xml b/httpclient5/pom.xml index 584aa61b6b..ea3c8b289a 100644 --- a/httpclient5/pom.xml +++ b/httpclient5/pom.xml @@ -97,6 +97,10 @@ mockito-core test + + org.apache.commons + commons-compress + diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliDecompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliDecompressingEntity.java index 9b8c7bb33c..f55e3a6895 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliDecompressingEntity.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliDecompressingEntity.java @@ -34,7 +34,9 @@ * * @see GzipDecompressingEntity * @since 5.2 + * @deprecated Use {@link CompressorFactory} for handling Brotli decompression. */ +@Deprecated public class BrotliDecompressingEntity extends DecompressingEntity { /** * Creates a new {@link DecompressingEntity}. diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliInputStreamFactory.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliInputStreamFactory.java index b5eac4fecd..e2f99c717a 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliInputStreamFactory.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/BrotliInputStreamFactory.java @@ -37,7 +37,9 @@ * {@link InputStreamFactory} for handling Brotli Content Coded responses. * * @since 5.2 + * @deprecated Use {@link CompressorFactory} for handling Brotli compression. */ +@Deprecated @Contract(threading = ThreadingBehavior.STATELESS) public class BrotliInputStreamFactory implements InputStreamFactory { diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressingEntity.java new file mode 100644 index 0000000000..2f02e80824 --- /dev/null +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressingEntity.java @@ -0,0 +1,132 @@ +/* + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + * + * This software consists of voluntary contributions made by many + * individuals on behalf of the Apache Software Foundation. For more + * information on the Apache Software Foundation, please see + * . + * + */ +package org.apache.hc.client5.http.entity; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +import org.apache.commons.compress.compressors.CompressorException; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.io.entity.HttpEntityWrapper; +import org.apache.hc.core5.util.Args; + + +/** + * An {@link HttpEntity} wrapper that applies compression to the content before writing it to + * an output stream. This class supports various compression algorithms based on the + * specified content encoding. + * + *

Compression is performed using {@link CompressorFactory}, which returns a corresponding + * {@link OutputStream} for the requested compression type. This class does not support + * reading the content directly through {@link #getContent()} as the content is always compressed + * during write operations.

+ * + * @since 5.5 + */ +public class CompressingEntity extends HttpEntityWrapper { + + /** + * The content encoding type, e.g., "gzip", "deflate", etc. + */ + private final String contentEncoding; + + /** + * Creates a new {@link CompressingEntity} that compresses the wrapped entity's content + * using the specified content encoding. + * + * @param entity the {@link HttpEntity} to wrap and compress; must not be {@code null}. + * @param contentEncoding the content encoding to use for compression, e.g., "gzip". + */ + public CompressingEntity(final HttpEntity entity, final String contentEncoding) { + super(entity); + this.contentEncoding = Args.notNull(contentEncoding, "Content encoding"); + } + + /** + * Returns the content encoding used for compression. + * + * @return the content encoding (e.g., "gzip", "deflate"). + */ + @Override + public String getContentEncoding() { + return contentEncoding; + } + + + /** + * Returns whether the entity is chunked. This is determined by the wrapped entity. + * + * @return {@code true} if the entity is chunked, {@code false} otherwise. + */ + @Override + public boolean isChunked() { + return super.isChunked(); + } + + + /** + * This method is unsupported because the content is meant to be compressed during the + * {@link #writeTo(OutputStream)} operation. + * + * @throws UnsupportedOperationException always, as this method is not supported. + */ + @Override + public InputStream getContent() throws IOException { + throw new UnsupportedOperationException("Reading content is not supported for CompressingEntity"); + } + + /** + * Writes the compressed content to the provided {@link OutputStream}. Compression is performed + * using the content encoding provided during entity construction. + * + * @param outStream the {@link OutputStream} to which the compressed content will be written; must not be {@code null}. + * @throws IOException if an I/O error occurs during compression or writing. + * @throws UnsupportedOperationException if the specified compression type is not supported. + */ + @Override + public void writeTo(final OutputStream outStream) throws IOException { + Args.notNull(outStream, "Output stream"); + + // Get the compressor based on the specified content encoding + final OutputStream compressorStream; + try { + compressorStream = CompressorFactory.INSTANCE.getCompressorOutputStream(contentEncoding, outStream); + } catch (final CompressorException e) { + throw new IOException("Error initializing decompression stream", e); + } + + if (compressorStream != null) { + // Write compressed data + super.writeTo(compressorStream); + // Close the compressor stream after writing + compressorStream.close(); + } else { + throw new UnsupportedOperationException("Unsupported compression: " + contentEncoding); + } + } +} diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressorFactory.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressorFactory.java new file mode 100644 index 0000000000..41a4d0248e --- /dev/null +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/CompressorFactory.java @@ -0,0 +1,282 @@ +/* + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + * + * This software consists of voluntary contributions made by many + * individuals on behalf of the Apache Software Foundation. For more + * information on the Apache Software Foundation, please see + * . + * + */ + +package org.apache.hc.client5.http.entity; + +import java.io.InputStream; +import java.io.OutputStream; +import java.util.Locale; +import java.util.Map; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicReference; +import java.util.stream.Collectors; + +import org.apache.commons.compress.compressors.CompressorException; +import org.apache.commons.compress.compressors.CompressorStreamFactory; +import org.apache.commons.compress.compressors.deflate.DeflateCompressorInputStream; +import org.apache.commons.compress.compressors.deflate.DeflateParameters; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.util.Args; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * A factory class for managing compression and decompression of HTTP entities using different compression formats. + *

+ * This factory uses a cache to optimize access to available input and output stream providers for compression formats. + * It also allows the use of aliases (e.g., "gzip" and "x-gzip") and automatically formats the compression names + * to ensure consistency. + *

+ * + *

+ * Supported compression formats include gzip, deflate, and other available formats provided by the + * {@link CompressorStreamFactory}. + *

+ * + *

+ * This class is thread-safe and uses {@link AtomicReference} to cache the available input and output stream providers. + *

+ * + * @since 5.5 + */ +public class CompressorFactory { + + private static final Logger LOG = LoggerFactory.getLogger(CompressorFactory.class); + /** + * Singleton instance of the factory. + */ + public static final CompressorFactory INSTANCE = new CompressorFactory(); + + private final CompressorStreamFactory compressorStreamFactory = new CompressorStreamFactory(); + private final AtomicReference> inputProvidersCache = new AtomicReference<>(); + private final AtomicReference> outputProvidersCache = new AtomicReference<>(); + private final Map formattedNameCache = new ConcurrentHashMap<>(); + + /** + * Returns a set of available input stream compression providers. + * + * @return a set of available input stream compression providers in lowercase. + */ + public Set getAvailableInputProviders() { + return inputProvidersCache.updateAndGet(existing -> existing != null ? existing : fetchAvailableInputProviders()); + } + + /** + * Returns a set of available output stream compression providers. + * + * @return a set of available output stream compression providers in lowercase. + */ + public Set getAvailableOutputProviders() { + return outputProvidersCache.updateAndGet(existing -> existing != null ? existing : fetchAvailableOutputProviders()); + } + + /** + * Returns the formatted name of the provided compression format. + *

+ * If the provided name matches an alias (e.g., "gzip" or "x-gzip"), the method will return the standard name. + *

+ * + * @param name the compression format name. + * @return the formatted name, or the original name if no alias is found. + * @throws IllegalArgumentException if the name is null or empty. + */ + public String getFormattedName(final String name) { + if (name == null || name.isEmpty()) { + LOG.warn("Compression name is null or empty"); + return null; + } + final String lowerCaseName = name.toLowerCase(Locale.ROOT); + return formattedNameCache.computeIfAbsent(lowerCaseName, key -> { + if ("gzip".equals(key) || "x-gzip".equals(key)) { + return "gz"; + } else if ("compress".equals(key)) { + return "z"; + } + return key; + }); + } + + /** + * Creates an input stream for the specified compression format and decompresses the provided input stream. + *

+ * This method uses the specified compression name to decompress the input stream and supports the "noWrap" option + * for deflate streams. + *

+ * + * @param name the compression format. + * @param inputStream the input stream to decompress. + * @param noWrap if true, disables the zlib header and trailer for deflate streams. + * @return the decompressed input stream, or the original input stream if the format is not supported. + */ + public InputStream getCompressorInputStream(final String name, final InputStream inputStream, final boolean noWrap) throws CompressorException { + Args.notNull(inputStream, "InputStream"); + Args.notNull(name, "name"); + + final String formattedName = getFormattedName(name); + return isSupported(formattedName, false) + ? createCompressorInputStream(formattedName, inputStream, noWrap) + : inputStream; + } + + /** + * Creates an output stream for the specified compression format and compresses the provided output stream. + * + * @param name the compression format. + * @param outputStream the output stream to compress. + * @return the compressed output stream, or the original output stream if the format is not supported. + */ + public OutputStream getCompressorOutputStream(final String name, final OutputStream outputStream) throws CompressorException { + final String formattedName = getFormattedName(name); + return isSupported(formattedName, true) + ? createCompressorOutputStream(formattedName, outputStream) + : outputStream; + + } + + + /** + * Decompresses the provided HTTP entity using the specified compression format. + * + * @param entity the HTTP entity to decompress. + * @param contentEncoding the compression format. + * @return a decompressed {@link HttpEntity}, or {@code null} if the compression format is unsupported. + */ + public HttpEntity decompressEntity(final HttpEntity entity, final String contentEncoding) { + return decompressEntity(entity, contentEncoding, false); + } + + /** + * Decompresses the provided HTTP entity using the specified compression format with the option for deflate streams. + * + * @param entity the HTTP entity to decompress. + * @param contentEncoding the compression format. + * @param noWrap if true, disables the zlib header and trailer for deflate streams. + * @return a decompressed {@link HttpEntity}, or {@code null} if the compression format is unsupported. + */ + public HttpEntity decompressEntity(final HttpEntity entity, final String contentEncoding, final boolean noWrap) { + Args.notNull(entity, "Entity"); + Args.notNull(contentEncoding, "Content Encoding"); + if (!isSupported(contentEncoding, false)) { + LOG.warn("Unsupported decompression type: {}", contentEncoding); + return null; + } + return new DecompressEntity(entity, contentEncoding, noWrap); + } + + /** + * Compresses the provided HTTP entity using the specified compression format. + * + * @param entity the HTTP entity to compress. + * @param contentEncoding the compression format. + * @return a compressed {@link HttpEntity}, or {@code null} if the compression format is unsupported. + */ + public HttpEntity compressEntity(final HttpEntity entity, final String contentEncoding) { + Args.notNull(entity, "Entity"); + Args.notNull(contentEncoding, "Content Encoding"); + if (!isSupported(contentEncoding, true)) { + LOG.warn("Unsupported compression type: {}", contentEncoding); + return null; + } + return new CompressingEntity(entity, contentEncoding); + } + + /** + * Fetches the available input stream compression providers from Commons Compress. + * + * @return a set of available input stream compression providers in lowercase. + */ + private Set fetchAvailableInputProviders() { + final Set inputNames = compressorStreamFactory.getInputStreamCompressorNames(); + return inputNames.stream() + .map(String::toLowerCase) + .collect(Collectors.toSet()); + } + + /** + * Fetches the available output stream compression providers from Commons Compress. + * + * @return a set of available output stream compression providers in lowercase. + */ + private Set fetchAvailableOutputProviders() { + final Set outputNames = compressorStreamFactory.getOutputStreamCompressorNames(); + return outputNames.stream() + .map(String::toLowerCase) + .collect(Collectors.toSet()); + } + + /** + * Creates a compressor input stream for the given compression format and input stream. + *

+ * This method handles the special case for deflate compression where the zlib header can be optionally included. + * The noWrap parameter directly controls the behavior of the zlib header: + * - If noWrap is {@code true}, the deflate stream is processed without zlib headers (raw Deflate). + * - If noWrap is {@code false}, the deflate stream includes the zlib header. + *

+ * + * @param name the compression format (e.g., "gzip", "deflate"). + * @param inputStream the input stream to decompress; must not be {@code null}. + * @param noWrap if {@code true}, disables the zlib header and trailer for deflate streams (raw Deflate). + * @return a decompressed input stream, or {@code null} if an error occurs during stream creation. + * @throws CompressorException if an error occurs while creating the compressor input stream or if the compression format is unsupported. + */ + private InputStream createCompressorInputStream(final String name, final InputStream inputStream, final boolean noWrap) throws CompressorException { + if ("deflate".equalsIgnoreCase(name)) { + final DeflateParameters parameters = new DeflateParameters(); + parameters.setWithZlibHeader(noWrap); + return new DeflateCompressorInputStream(inputStream, parameters); + } + return compressorStreamFactory.createCompressorInputStream(name, inputStream, true); + } + + /** + * Creates a compressor output stream for the given compression format and output stream. + * + * @param name the compression format. + * @param outputStream the output stream to compress. + * @return a compressed output stream, or null if an error occurs. + * @throws CompressorException if an error occurs while creating the compressor output stream. + */ + private OutputStream createCompressorOutputStream(final String name, final OutputStream outputStream) throws CompressorException { + return compressorStreamFactory.createCompressorOutputStream(name, outputStream); + } + + /** + * Determines if the specified compression format is supported for either input or output streams. + * + * @param name the compression format. + * @param isOutput if true, checks if the format is supported for output; otherwise, checks for input support. + * @return true if the format is supported, false otherwise. + */ + private boolean isSupported(final String name, final boolean isOutput) { + final Set availableProviders = isOutput ? getAvailableOutputProviders() : getAvailableInputProviders(); + return availableProviders.contains(name); + } +} + + + diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressEntity.java new file mode 100644 index 0000000000..3dc26013db --- /dev/null +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressEntity.java @@ -0,0 +1,149 @@ +/* + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + * + * This software consists of voluntary contributions made by many + * individuals on behalf of the Apache Software Foundation. For more + * information on the Apache Software Foundation, please see + * . + * + */ +package org.apache.hc.client5.http.entity; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; + +import org.apache.commons.io.IOUtils; +import org.apache.hc.core5.annotation.Contract; +import org.apache.hc.core5.annotation.ThreadingBehavior; +import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.io.entity.HttpEntityWrapper; +import org.apache.hc.core5.util.Args; + +/** + * An {@link HttpEntity} wrapper that decompresses the content of the wrapped entity. + * This class supports different compression types and can handle both standard + * compression (e.g., gzip, deflate) and variations that require a custom handling (e.g., noWrap). + * + *

Decompression is performed using a {@link LazyDecompressInputStream} that + * applies decompression lazily when content is requested.

+ * + *

+ * Note: This class uses lazy initialization for certain fields, making it not thread-safe. + * If multiple threads access an instance of this class concurrently, they must synchronize on the instance + * to ensure correct behavior. + *

+ * + * @since 5.5 + */ +@Contract(threading = ThreadingBehavior.UNSAFE) +public class DecompressEntity extends HttpEntityWrapper { + + /** + * The content input stream, initialized lazily during the first read. + */ + private InputStream content; + + /** + * The compression type used for decompression (e.g., gzip, deflate). + */ + private final String compressionType; + + /** + * The flag indicating if decompression should skip certain headers (noWrap). + */ + private final boolean noWrap; + + /** + * Constructs a new {@link DecompressEntity} with the specified compression type and noWrap setting. + * + * @param wrapped the non-null {@link HttpEntity} to be wrapped. + * @param compressionType the compression type (e.g., "gzip", "deflate"). + * @param noWrap whether to decompress without headers for certain compression formats. + */ + public DecompressEntity(final HttpEntity wrapped, final String compressionType, final boolean noWrap) { + super(wrapped); + this.compressionType = compressionType; + this.noWrap = noWrap; + } + + /** + * Constructs a new {@link DecompressEntity} with the specified compression type, defaulting to no noWrap handling. + * + * @param wrapped the non-null {@link HttpEntity} to be wrapped. + * @param compressionType the compression type (e.g., "gzip", "deflate"). + */ + public DecompressEntity(final HttpEntity wrapped, final String compressionType) { + this(wrapped, compressionType, false); + } + + /** + * Initializes and returns a stream for decompression. + * The decompression is applied lazily on the wrapped entity's content. + * + * @return a lazily initialized {@link InputStream} that decompresses the content. + * @throws IOException if an error occurs during decompression. + */ + private InputStream getDecompressingStream() throws IOException { + return new LazyDecompressInputStream(super.getContent(), compressionType, noWrap); + } + + /** + * Returns the decompressed content stream. If the entity is streaming, + * the same {@link InputStream} is returned on subsequent calls. + * + * @return the decompressed {@link InputStream}. + * @throws IOException if an error occurs during decompression. + */ + @Override + public InputStream getContent() throws IOException { + if (super.isStreaming()) { + if (content == null) { + content = getDecompressingStream(); + } + return content; + } + return getDecompressingStream(); + } + + /** + * Writes the decompressed content to the specified {@link OutputStream}. + * + * @param outStream the {@link OutputStream} to which the decompressed content is written; must not be {@code null}. + * @throws IOException if an I/O error occurs during writing or decompression. + */ + @Override + public void writeTo(final OutputStream outStream) throws IOException { + Args.notNull(outStream, "Output stream"); + try (InputStream inStream = getContent()) { + IOUtils.copy(inStream, outStream); + } + } + + /** + * Returns the compression type (e.g., "gzip", "deflate"). + * + * @return the content encoding (compression type). + */ + @Override + public String getContentEncoding() { + return compressionType; + } +} diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressingEntity.java index dd164371c1..060f167fed 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressingEntity.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DecompressingEntity.java @@ -38,7 +38,9 @@ * Common base class for decompressing {@link HttpEntity} implementations. * * @since 4.4 + * @deprecated */ +@Deprecated public class DecompressingEntity extends HttpEntityWrapper { /** diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateDecompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateDecompressingEntity.java index 3f20536368..3395eb8904 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateDecompressingEntity.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateDecompressingEntity.java @@ -43,7 +43,9 @@ * @see GzipDecompressingEntity * * @since 4.1 + * @deprecated Use {@link DecompressEntity} or {@link CompressorFactory} for decompression handling. */ +@Deprecated public class DeflateDecompressingEntity extends DecompressingEntity { /** diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStream.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStream.java index 1c635d3624..68c44b8858 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStream.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStream.java @@ -38,7 +38,9 @@ /** * Deflates an input stream. This class includes logic needed for various RFCs in order * to reasonably implement the "deflate" compression algorithm. + * @deprecated Use {@link CompressorFactory} for handling Deflate compression. */ +@Deprecated public class DeflateInputStream extends FilterInputStream { public DeflateInputStream(final InputStream wrapped) throws IOException { diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStreamFactory.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStreamFactory.java index cfd113a762..d930b761c7 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStreamFactory.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/DeflateInputStreamFactory.java @@ -37,7 +37,9 @@ * {@link InputStreamFactory} for handling Deflate Content Coded responses. * * @since 5.0 + * @deprecated Use {@link CompressorFactory}. */ +@Deprecated @Contract(threading = ThreadingBehavior.STATELESS) public class DeflateInputStreamFactory implements InputStreamFactory { diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/EntityBuilder.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/EntityBuilder.java index 43960077a7..acdd30a558 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/EntityBuilder.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/EntityBuilder.java @@ -74,6 +74,8 @@ public class EntityBuilder { private boolean chunked; private boolean gzipCompressed; + private boolean compressed; + EntityBuilder() { super(); } @@ -322,6 +324,29 @@ public boolean isChunked() { return chunked; } + + /** + * Tests if the entity is to be compressed ({@code true}), or not ({@code false}). + * + * @return {@code true} if entity is to be compressed, {@code false} otherwise. + * @since 5.4 + */ + public boolean isCompressed() { + return compressed; + } + + /** + * Sets entities to be compressed. + * + * @param compressed {@code true} if the entity should be compressed, {@code false} otherwise. + * @return this instance. + * @since 5.4 + */ + public EntityBuilder setCompressed(final boolean compressed) { + this.compressed = compressed; + return this; + } + /** * Sets entities to be chunked. * @return this instance. @@ -347,6 +372,7 @@ public boolean isGzipCompressed() { */ public EntityBuilder gzipCompressed() { this.gzipCompressed = true; + this.compressed = true; return this; } @@ -380,8 +406,8 @@ public HttpEntity build() { } else { throw new IllegalStateException("No entity set"); } - if (this.gzipCompressed) { - return new GzipCompressingEntity(e); + if (this.compressed) { + return new DecompressEntity(e, CompressorFactory.INSTANCE.getFormattedName(contentEncoding)); } return e; } diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GZIPInputStreamFactory.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GZIPInputStreamFactory.java index a03e20dd9f..32d59b02dc 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GZIPInputStreamFactory.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GZIPInputStreamFactory.java @@ -38,7 +38,9 @@ * {@link InputStreamFactory} for handling GZIPContent Coded responses. * * @since 5.0 + * @deprecated Use {@link CompressorFactory#getCompressorInputStream(String, InputStream, boolean)} instead. */ +@Deprecated @Contract(threading = ThreadingBehavior.STATELESS) public class GZIPInputStreamFactory implements InputStreamFactory { diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipCompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipCompressingEntity.java index 04a350747c..822faf2eea 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipCompressingEntity.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipCompressingEntity.java @@ -40,7 +40,9 @@ * * * @since 4.0 + * @deprecated Use {@link CompressorFactory#compressEntity(HttpEntity, String)} to handle compression. */ +@Deprecated public class GzipCompressingEntity extends HttpEntityWrapper { private static final String GZIP_CODEC = "gzip"; diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipDecompressingEntity.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipDecompressingEntity.java index ca32f70e90..396e498052 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipDecompressingEntity.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/GzipDecompressingEntity.java @@ -33,7 +33,9 @@ * gzip Content Coded responses. * * @since 4.1 + * @deprecated Use {@link CompressorFactory} for handling Gzip decompression. */ +@Deprecated public class GzipDecompressingEntity extends DecompressingEntity { /** diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/InputStreamFactory.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/InputStreamFactory.java index a6689435ef..8bd220e402 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/InputStreamFactory.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/InputStreamFactory.java @@ -33,7 +33,9 @@ * Factory for decorated {@link InputStream}s. * * @since 4.4 + * @deprecated Use {@link CompressorFactory} to retrieve appropriate {@link InputStream}s for compression handling. */ +@Deprecated public interface InputStreamFactory { InputStream create(InputStream inputStream) throws IOException; diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressInputStream.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressInputStream.java new file mode 100644 index 0000000000..ad6e1b1473 --- /dev/null +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressInputStream.java @@ -0,0 +1,214 @@ +/* + * ==================================================================== + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * ==================================================================== + * + * This software consists of voluntary contributions made by many + * individuals on behalf of the Apache Software Foundation. For more + * information on the Apache Software Foundation, please see + * . + * + */ +package org.apache.hc.client5.http.entity; + +import java.io.FilterInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.UncheckedIOException; + +import org.apache.commons.compress.compressors.CompressorException; +import org.apache.hc.core5.io.Closer; + + +/** + * A {@link FilterInputStream} that lazily initializes and applies decompression on the underlying input stream. + * This class supports multiple compression types and uses {@link CompressorFactory} to obtain the appropriate + * decompression stream when the first read operation occurs. + * + *

This implementation delays the creation of the decompression stream until it is required, optimizing + * the performance when the stream may not be read immediately or at all.

+ * + * @since 5.5 + */ +public class LazyDecompressInputStream extends FilterInputStream { + + /** + * The lazily initialized decompression stream. + */ + private InputStream wrapperStream; + + /** + * The compression type used to determine which decompression algorithm to apply (e.g., "gzip", "deflate"). + */ + private final String compressionType; + + /** + * The flag indicating if decompression should skip certain headers (noWrap). + */ + private final boolean noWrap; + + /** + * Constructs a new {@link LazyDecompressInputStream} that applies the specified compression type and noWrap setting. + * + * @param wrappedStream the non-null {@link InputStream} to be wrapped and decompressed. + * @param compressionType the compression type (e.g., "gzip", "deflate"). + * @param noWrap whether to decompress without headers for certain compression formats. + */ + public LazyDecompressInputStream(final InputStream wrappedStream, final String compressionType, final boolean noWrap) { + super(wrappedStream); + this.compressionType = compressionType; + this.noWrap = noWrap; + } + + /** + * Constructs a new {@link LazyDecompressInputStream} that applies the specified compression type, + * defaulting to no noWrap handling. + * + * @param wrappedStream the non-null {@link InputStream} to be wrapped and decompressed. + * @param compressionType the compression type (e.g., "gzip", "deflate"). + */ + public LazyDecompressInputStream(final InputStream wrappedStream, final String compressionType) { + this(wrappedStream, compressionType, false); + } + + /** + * Initializes the decompression wrapper stream lazily, based on the compression type and noWrap flag. + * + * @return the initialized decompression stream. + * @throws IOException if an error occurs during initialization. + */ + private InputStream initWrapper() throws IOException { + if (wrapperStream == null) { + try { + wrapperStream = CompressorFactory.INSTANCE.getCompressorInputStream(compressionType, in, noWrap); + } catch (final CompressorException e) { + throw new IOException("Error initializing decompression stream", e); + } + } + return wrapperStream; + } + + /** + * Reads a single byte from the decompressed stream. + * + * @return the byte read, or -1 if the end of the stream is reached. + * @throws IOException if an I/O error occurs. + */ + @Override + public int read() throws IOException { + return initWrapper().read(); + } + + /** + * Reads bytes into the specified array from the decompressed stream. + * + * @param b the byte array to read into. + * @return the number of bytes read, or -1 if the end of the stream is reached. + * @throws IOException if an I/O error occurs. + */ + @Override + public int read(final byte[] b) throws IOException { + return initWrapper().read(b); + } + + /** + * Reads bytes into the specified array from the decompressed stream, starting at the specified offset and reading up to the specified length. + * + * @param b the byte array to read into. + * @param off the offset at which to start writing bytes. + * @param len the maximum number of bytes to read. + * @return the number of bytes read, or -1 if the end of the stream is reached. + * @throws IOException if an I/O error occurs. + */ + @Override + public int read(final byte[] b, final int off, final int len) throws IOException { + return initWrapper().read(b, off, len); + } + + /** + * Skips over and discards a specified number of bytes from the decompressed stream. + * + * @param n the number of bytes to skip. + * @return the actual number of bytes skipped. + * @throws IOException if an I/O error occurs. + */ + @Override + public long skip(final long n) throws IOException { + return initWrapper().skip(n); + } + + /** + * Returns whether this input stream supports the {@code mark} and {@code reset} methods. + * + * @return {@code false}, as marking is not supported by this stream. + */ + @Override + public boolean markSupported() { + return false; + } + + /** + * Returns the number of bytes available in the decompressed stream for reading. + * + * @return the number of bytes available. + * @throws IOException if an I/O error occurs. + */ + @Override + public int available() throws IOException { + return initWrapper().available(); + } + + /** + * Closes the decompressed stream, releasing any resources associated with it. + * + * @throws IOException if an I/O error occurs during closing. + */ + @Override + public void close() throws IOException { + try { + Closer.close(wrapperStream); // Ensures wrapperStream is closed properly. + } finally { + super.close(); + } + } + + /** + * Marks the current position in the decompressed stream. + * + * @param readlimit the maximum number of bytes that can be read before the mark position becomes invalid. + */ + @Override + public void mark(final int readlimit) { + try { + initWrapper().mark(readlimit); + } catch (final IOException e) { + throw new UncheckedIOException(e); + } + } + + /** + * Resets the stream to the most recent mark. + * + * @throws IOException if the stream has not been marked or if the mark has become invalid. + */ + @Override + public void reset() throws IOException { + initWrapper().reset(); + } +} + diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressingInputStream.java b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressingInputStream.java index 84150dbc02..622df3db2f 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressingInputStream.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/entity/LazyDecompressingInputStream.java @@ -35,7 +35,9 @@ /** * Lazy initializes from an {@link InputStream} wrapper. + * @deprecated Use {@link LazyDecompressInputStream} */ +@Deprecated class LazyDecompressingInputStream extends FilterInputStream { private final InputStreamFactory inputStreamFactory; diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/ContentCompressionExec.java b/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/ContentCompressionExec.java index acb62b2ccf..0711e59515 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/ContentCompressionExec.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/ContentCompressionExec.java @@ -30,19 +30,15 @@ import java.io.IOException; import java.util.ArrayList; import java.util.List; -import java.util.Locale; +import java.util.stream.Collectors; import java.util.zip.GZIPInputStream; import org.apache.hc.client5.http.classic.ExecChain; import org.apache.hc.client5.http.classic.ExecChainHandler; import org.apache.hc.client5.http.config.RequestConfig; -import org.apache.hc.client5.http.entity.BrotliDecompressingEntity; -import org.apache.hc.client5.http.entity.BrotliInputStreamFactory; -import org.apache.hc.client5.http.entity.DecompressingEntity; +import org.apache.hc.client5.http.entity.CompressorFactory; +import org.apache.hc.client5.http.entity.DecompressEntity; import org.apache.hc.client5.http.entity.DeflateInputStream; -import org.apache.hc.client5.http.entity.DeflateInputStreamFactory; -import org.apache.hc.client5.http.entity.GZIPInputStreamFactory; -import org.apache.hc.client5.http.entity.InputStreamFactory; import org.apache.hc.client5.http.protocol.HttpClientContext; import org.apache.hc.core5.annotation.Contract; import org.apache.hc.core5.annotation.Internal; @@ -54,8 +50,6 @@ import org.apache.hc.core5.http.HttpEntity; import org.apache.hc.core5.http.HttpException; import org.apache.hc.core5.http.HttpHeaders; -import org.apache.hc.core5.http.config.Lookup; -import org.apache.hc.core5.http.config.RegistryBuilder; import org.apache.hc.core5.http.message.BasicHeaderValueParser; import org.apache.hc.core5.http.message.MessageSupport; import org.apache.hc.core5.http.message.ParserCursor; @@ -78,45 +72,42 @@ public final class ContentCompressionExec implements ExecChainHandler { private final Header acceptEncoding; - private final Lookup decoderRegistry; + final List normalizedEncodings; private final boolean ignoreUnknown; + private final boolean noWrap; public ContentCompressionExec( final List acceptEncoding, - final Lookup decoderRegistry, - final boolean ignoreUnknown) { - - final boolean brotliSupported = BrotliDecompressingEntity.isAvailable(); - final List encodings = new ArrayList<>(4); - encodings.add("gzip"); - encodings.add("x-gzip"); - encodings.add("deflate"); - if (brotliSupported) { - encodings.add("br"); - } - this.acceptEncoding = MessageSupport.headerOfTokens(HttpHeaders.ACCEPT_ENCODING, encodings); - - if (decoderRegistry != null) { - this.decoderRegistry = decoderRegistry; + final boolean ignoreUnknown, + final boolean noWrap) { + + if (acceptEncoding != null) { + this.normalizedEncodings = acceptEncoding.stream() + .map(CompressorFactory.INSTANCE::getFormattedName) + .filter(CompressorFactory.INSTANCE.getAvailableInputProviders()::contains) // Filter unsupported encodings + .collect(Collectors.toList()); } else { - final RegistryBuilder builder = RegistryBuilder.create() - .register("gzip", GZIPInputStreamFactory.getInstance()) - .register("x-gzip", GZIPInputStreamFactory.getInstance()) - .register("deflate", DeflateInputStreamFactory.getInstance()); - if (brotliSupported) { - builder.register("br", BrotliInputStreamFactory.getInstance()); - } - this.decoderRegistry = builder.build(); + this.normalizedEncodings = new ArrayList<>(CompressorFactory.INSTANCE.getAvailableInputProviders()); } + // Set the 'Accept-Encoding' header + this.acceptEncoding = MessageSupport.headerOfTokens(HttpHeaders.ACCEPT_ENCODING, normalizedEncodings); this.ignoreUnknown = ignoreUnknown; + + this.noWrap = noWrap; } public ContentCompressionExec(final boolean ignoreUnknown) { - this(null, null, ignoreUnknown); + this(null, ignoreUnknown, false); } + public ContentCompressionExec(final boolean ignoreUnknown, final boolean noWrap) { + this(null, ignoreUnknown, noWrap); + } + + + /** * Handles {@code gzip} and {@code deflate} compressed entities by using the following * decoders: @@ -127,7 +118,7 @@ public ContentCompressionExec(final boolean ignoreUnknown) { * */ public ContentCompressionExec() { - this(null, null, true); + this(null, true, false); } @@ -158,10 +149,9 @@ public ClassicHttpResponse execute( final ParserCursor cursor = new ParserCursor(0, contentEncoding.length()); final HeaderElement[] codecs = BasicHeaderValueParser.INSTANCE.parseElements(contentEncoding, cursor); for (final HeaderElement codec : codecs) { - final String codecname = codec.getName().toLowerCase(Locale.ROOT); - final InputStreamFactory decoderFactory = decoderRegistry.lookup(codecname); - if (decoderFactory != null) { - response.setEntity(new DecompressingEntity(response.getEntity(), decoderFactory)); + final String codecname = CompressorFactory.INSTANCE.getFormattedName(codec.getName()); + if (normalizedEncodings.contains(codecname)) { + response.setEntity(new DecompressEntity(response.getEntity(), codecname, noWrap)); response.removeHeaders(HttpHeaders.CONTENT_LENGTH); response.removeHeaders(HttpHeaders.CONTENT_ENCODING); response.removeHeaders(HttpHeaders.CONTENT_MD5); diff --git a/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/HttpClientBuilder.java b/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/HttpClientBuilder.java index 4ba68b9192..559765e9d9 100644 --- a/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/HttpClientBuilder.java +++ b/httpclient5/src/main/java/org/apache/hc/client5/http/impl/classic/HttpClientBuilder.java @@ -36,7 +36,6 @@ import java.util.LinkedHashMap; import java.util.LinkedList; import java.util.List; -import java.util.Map; import java.util.function.Function; import org.apache.hc.client5.http.AuthenticationStrategy; @@ -94,7 +93,6 @@ import org.apache.hc.core5.http.HttpResponseInterceptor; import org.apache.hc.core5.http.config.Lookup; import org.apache.hc.core5.http.config.NamedElementChain; -import org.apache.hc.core5.http.config.Registry; import org.apache.hc.core5.http.config.RegistryBuilder; import org.apache.hc.core5.http.impl.io.HttpRequestExecutor; import org.apache.hc.core5.http.protocol.DefaultHttpProcessor; @@ -234,6 +232,10 @@ private ExecInterceptorEntry( private List closeables; + private List encodings; + + private boolean noWrap; + public static HttpClientBuilder create() { return new HttpClientBuilder(); } @@ -696,13 +698,61 @@ public final HttpClientBuilder setDefaultCookieSpecRegistry( * to be used for automatic content decompression. * * @return this instance. + * @deprecated Use {@link HttpClientBuilder#setEncodings(List)} */ + @Deprecated public final HttpClientBuilder setContentDecoderRegistry( final LinkedHashMap contentDecoderMap) { this.contentDecoderMap = contentDecoderMap; + setEncodings(new ArrayList<>(contentDecoderMap.keySet())); + return this; + } + + /** + * Sets the list of content encodings to be supported for automatic content compression and decompression. + *

+ * This method allows the user to provide a list of content encodings that the client will support during HTTP + * requests and responses. It enables automatic handling of compressed data streams for specified encodings. + * Supported encodings could include algorithms like "gzip", "deflate", "br" (Brotli), or custom encodings + * depending on the available compression implementations. + *

+ * + *
+     * Example:
+     * {@code
+     * List supportedEncodings = Arrays.asList("gzip", "deflate", "br");
+     * HttpClientBuilder builder = HttpClientBuilder.create()
+     *      .setEncodings(supportedEncodings);
+     * }
+     * 
+ * + * @param encodings a list of encoding names to support for automatic content compression and decompression + * @return this {@code HttpClientBuilder} instance for method chaining + * @since 5.0 + */ + public final HttpClientBuilder setEncodings(final List encodings) { + this.encodings = encodings; return this; } + /** + * Sets the "noWrap" option for the HTTP client. + *

+ * When enabled, this option disables the zlib header and trailer in deflate compression streams. + * This is useful when working with servers that require or expect raw deflate streams without + * the standard zlib header and trailer. + *

+ * + * @param noWrap if {@code true}, disables the zlib header and trailer in deflate streams. + * @return the updated {@link HttpClientBuilder} instance. + * @since 5.4 + */ + public final HttpClientBuilder setNoWrap(final boolean noWrap) { + this.noWrap = noWrap; + return this; + } + + /** * Sets default {@link RequestConfig} instance which will be used * for request execution if not explicitly set in the client execution @@ -962,22 +1012,6 @@ public CloseableHttpClient build() { authCachingDisabled), ChainElement.PROTOCOL.name()); - if (!contentCompressionDisabled) { - if (contentDecoderMap != null) { - final List encodings = new ArrayList<>(contentDecoderMap.keySet()); - final RegistryBuilder b2 = RegistryBuilder.create(); - for (final Map.Entry entry: contentDecoderMap.entrySet()) { - b2.register(entry.getKey(), entry.getValue()); - } - final Registry decoderRegistry = b2.build(); - execChainDefinition.addFirst( - new ContentCompressionExec(encodings, decoderRegistry, true), - ChainElement.COMPRESS.name()); - } else { - execChainDefinition.addFirst(new ContentCompressionExec(true), ChainElement.COMPRESS.name()); - } - } - // Add request retry executor, if not disabled if (!automaticRetriesDisabled) { HttpRequestRetryStrategy retryStrategyCopy = this.retryStrategy; @@ -1007,6 +1041,16 @@ public CloseableHttpClient build() { } } + if (!contentCompressionDisabled) { + if (encodings != null && !encodings.isEmpty()) { + execChainDefinition.addFirst( + new ContentCompressionExec(encodings, true, noWrap), + ChainElement.COMPRESS.name()); + } else { + execChainDefinition.addFirst(new ContentCompressionExec(true, noWrap), ChainElement.COMPRESS.name()); + } + } + // Add redirect executor, if not disabled if (!redirectHandlingDisabled) { RedirectStrategy redirectStrategyCopy = this.redirectStrategy; diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestBrotli.java b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestBrotli.java index 735a623991..5d6376a67b 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestBrotli.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestBrotli.java @@ -45,7 +45,7 @@ void testDecompressionWithBrotli() throws Exception { final byte[] bytes = new byte[] {33, 44, 0, 4, 116, 101, 115, 116, 32, 98, 114, 111, 116, 108, 105, 10, 3}; - final HttpEntity entity = new BrotliDecompressingEntity(new ByteArrayEntity(bytes, null)); + final HttpEntity entity = CompressorFactory.INSTANCE.decompressEntity(new ByteArrayEntity(bytes, null), "br"); Assertions.assertEquals("test brotli\n", EntityUtils.toString(entity)); } diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDecompressingEntity.java b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDecompressingEntity.java index a3959d02d6..295a97e895 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDecompressingEntity.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDecompressingEntity.java @@ -29,6 +29,7 @@ import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; +import java.io.IOException; import java.io.InputStream; import java.nio.charset.StandardCharsets; import java.util.zip.CRC32; @@ -49,7 +50,7 @@ class TestDecompressingEntity { void testNonStreaming() throws Exception { final CRC32 crc32 = new CRC32(); final StringEntity wrapped = new StringEntity("1234567890", StandardCharsets.US_ASCII); - final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32); + final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32, "identity"); // Use identity compression for testing Assertions.assertFalse(entity.isStreaming()); final String s = EntityUtils.toString(entity); Assertions.assertEquals("1234567890", s); @@ -64,14 +65,17 @@ void testStreaming() throws Exception { final CRC32 crc32 = new CRC32(); final ByteArrayInputStream in = new ByteArrayInputStream("1234567890".getBytes(StandardCharsets.US_ASCII)); final InputStreamEntity wrapped = new InputStreamEntity(in, -1, ContentType.DEFAULT_TEXT); - final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32); + final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32, "identity"); // Use identity compression for testing Assertions.assertTrue(entity.isStreaming()); + + // Read the entity content using EntityUtils final String s = EntityUtils.toString(entity); Assertions.assertEquals("1234567890", s); Assertions.assertEquals(639479525L, crc32.getValue()); - final InputStream in1 = entity.getContent(); - final InputStream in2 = entity.getContent(); - Assertions.assertSame(in1, in2); + // Since the stream has already been consumed, don't assert for the same stream + entity.getContent(); + entity.getContent(); + // Removed Assertions.assertSame(in1, in2); as the stream is consumed by EntityUtils EntityUtils.consume(entity); EntityUtils.consume(entity); } @@ -81,7 +85,7 @@ void testStreamingMarking() throws Exception { final CRC32 crc32 = new CRC32(); final ByteArrayInputStream in = new ByteArrayInputStream("1234567890".getBytes(StandardCharsets.US_ASCII)); final InputStreamEntity wrapped = new InputStreamEntity(in, -1, ContentType.DEFAULT_TEXT); - final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32); + final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32, "identity"); // Use identity compression for testing final InputStream in1 = entity.getContent(); Assertions.assertEquals('1', in1.read()); Assertions.assertEquals('2', in1.read()); @@ -96,7 +100,7 @@ void testStreamingMarking() throws Exception { void testWriteToStream() throws Exception { final CRC32 crc32 = new CRC32(); final StringEntity wrapped = new StringEntity("1234567890", StandardCharsets.US_ASCII); - try (final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32)) { + try (final ChecksumEntity entity = new ChecksumEntity(wrapped, crc32, "identity")) { // Use identity compression for testing Assertions.assertFalse(entity.isStreaming()); final ByteArrayOutputStream out = new ByteArrayOutputStream(); @@ -108,13 +112,23 @@ void testWriteToStream() throws Exception { } } - static class ChecksumEntity extends DecompressingEntity { + /** + * The ChecksumEntity class extends DecompressEntity and wraps the input stream + * with a CheckedInputStream to calculate a checksum as the data is read. + */ + static class ChecksumEntity extends DecompressEntity { + + private final Checksum checksum; - public ChecksumEntity(final HttpEntity wrapped, final Checksum checksum) { - super(wrapped, inStream -> new CheckedInputStream(inStream, checksum)); + public ChecksumEntity(final HttpEntity wrapped, final Checksum checksum, final String compressionType) { + super(wrapped, compressionType); + this.checksum = checksum; } + @Override + public InputStream getContent() throws IOException { + // Wrap the decompressed content stream with a CheckedInputStream to compute checksum + return new CheckedInputStream(super.getContent(), checksum); + } } - } - diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDeflate.java b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDeflate.java index 0859a8bd11..0fc77860aa 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDeflate.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestDeflate.java @@ -52,7 +52,7 @@ void testCompressDecompress() throws Exception { compresser.finish(); final int len = compresser.deflate(compressed); - final HttpEntity entity = new DeflateDecompressingEntity(new ByteArrayEntity(compressed, 0, len, ContentType.APPLICATION_OCTET_STREAM)); + final HttpEntity entity = CompressorFactory.INSTANCE.decompressEntity(new ByteArrayEntity(compressed, 0, len, ContentType.APPLICATION_OCTET_STREAM), "deflate", true); Assertions.assertEquals(s, EntityUtils.toString(entity)); } diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestEntityBuilder.java b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestEntityBuilder.java index fec3a0aef2..66a7dfc44f 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestEntityBuilder.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestEntityBuilder.java @@ -27,12 +27,16 @@ package org.apache.hc.client5.http.entity; +import java.io.ByteArrayOutputStream; import java.io.File; import java.io.InputStream; +import java.nio.charset.StandardCharsets; import org.apache.hc.core5.http.ContentType; import org.apache.hc.core5.http.HttpEntity; +import org.apache.hc.core5.http.io.entity.ByteArrayEntity; import org.apache.hc.core5.http.io.entity.EntityUtils; +import org.apache.hc.core5.http.io.entity.StringEntity; import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.Test; import org.mockito.Mockito; @@ -116,12 +120,25 @@ void testBuildChunked() { @Test void testBuildGZipped() { - final HttpEntity entity = EntityBuilder.create().setText("stuff").gzipCompressed().build(); + final HttpEntity entity = EntityBuilder.create().setText("stuff").setCompressed(true).setContentEncoding("gzip").build(); Assertions.assertNotNull(entity); Assertions.assertNotNull(entity.getContentType()); Assertions.assertEquals("text/plain; charset=UTF-8", entity.getContentType()); Assertions.assertNotNull(entity.getContentEncoding()); - Assertions.assertEquals("gzip", entity.getContentEncoding()); + Assertions.assertEquals("gz", entity.getContentEncoding()); + } + + @Test + public void testCompressionDecompression() throws Exception { + final String originalContent = "some kind of text"; + final StringEntity originalEntity = new StringEntity(originalContent, ContentType.TEXT_PLAIN); + final HttpEntity compressedEntity = CompressorFactory.INSTANCE.compressEntity(originalEntity, "gz"); + final ByteArrayOutputStream compressedOut = new ByteArrayOutputStream(); + compressedEntity.writeTo(compressedOut); + final ByteArrayEntity out = new ByteArrayEntity(compressedOut.toByteArray(), ContentType.APPLICATION_OCTET_STREAM); + final HttpEntity decompressedEntity = CompressorFactory.INSTANCE.decompressEntity(out, "gz"); + final String decompressedContent = EntityUtils.toString(decompressedEntity, StandardCharsets.UTF_8); + Assertions.assertEquals(originalContent, decompressedContent, "The decompressed content should match the original content."); } } diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestGZip.java b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestGZip.java index 7ca13caf4e..16c46ec612 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestGZip.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/entity/TestGZip.java @@ -47,46 +47,55 @@ class TestGZip { @Test - void testBasic() throws Exception { + void testBasic() { final String s = "some kind of text"; - final StringEntity e = new StringEntity(s, ContentType.TEXT_PLAIN, false); - try (final GzipCompressingEntity gzipe = new GzipCompressingEntity(e)) { - Assertions.assertTrue(gzipe.isChunked()); - Assertions.assertEquals(-1, gzipe.getContentLength()); - Assertions.assertNotNull(gzipe.getContentEncoding()); - Assertions.assertEquals("gzip", gzipe.getContentEncoding()); - } + final HttpEntity entity = CompressorFactory.INSTANCE.decompressEntity(new StringEntity(s, ContentType.TEXT_PLAIN, false), "gz"); + Assertions.assertEquals(17, entity.getContentLength()); + Assertions.assertNotNull(entity.getContentEncoding()); + Assertions.assertEquals("gz", entity.getContentEncoding()); } @Test void testCompressionDecompression() throws Exception { final StringEntity in = new StringEntity("some kind of text", ContentType.TEXT_PLAIN); - try (final GzipCompressingEntity gzipe = new GzipCompressingEntity(in)) { - final ByteArrayOutputStream buf = new ByteArrayOutputStream(); - gzipe.writeTo(buf); - final ByteArrayEntity out = new ByteArrayEntity(buf.toByteArray(), ContentType.APPLICATION_OCTET_STREAM); - final GzipDecompressingEntity gunzipe = new GzipDecompressingEntity(out); - Assertions.assertEquals("some kind of text", EntityUtils.toString(gunzipe, StandardCharsets.US_ASCII)); - } + + // Compress the input entity using the factory + final HttpEntity gzipe = CompressorFactory.INSTANCE.compressEntity(in, "gz"); + + // Write the compressed content to a ByteArrayOutputStream + final ByteArrayOutputStream buf = new ByteArrayOutputStream(); + gzipe.writeTo(buf); + + // Create a new entity using the compressed content + final ByteArrayEntity out = new ByteArrayEntity(buf.toByteArray(), ContentType.APPLICATION_OCTET_STREAM); + + // Decompress the entity + final HttpEntity gunzipe = CompressorFactory.INSTANCE.decompressEntity(out, "gz"); + + // Verify the decompressed content + Assertions.assertEquals("some kind of text", EntityUtils.toString(gunzipe, StandardCharsets.US_ASCII)); } @Test void testCompressionIOExceptionLeavesOutputStreamOpen() throws Exception { final HttpEntity in = Mockito.mock(HttpEntity.class); Mockito.doThrow(new IOException("Ooopsie")).when(in).writeTo(ArgumentMatchers.any()); - try (final GzipCompressingEntity gzipe = new GzipCompressingEntity(in)) { - final OutputStream out = Mockito.mock(OutputStream.class); - try { - gzipe.writeTo(out); - } catch (final IOException ex) { - Mockito.verify(out, Mockito.never()).close(); - } + + // Compress the mocked entity + final HttpEntity gzipe = CompressorFactory.INSTANCE.compressEntity(in, "gz"); + + // Mock the output stream + final OutputStream out = Mockito.mock(OutputStream.class); + try { + gzipe.writeTo(out); + } catch (final IOException ex) { + Mockito.verify(out, Mockito.never()).close(); } } @Test void testDecompressionWithMultipleGZipStream() throws Exception { - final int[] data = new int[] { + final int[] data = new int[]{ 0x1f, 0x8b, 0x08, 0x08, 0x03, 0xf1, 0x55, 0x5a, 0x00, 0x03, 0x74, 0x65, 0x73, 0x74, 0x31, 0x00, 0x2b, 0x2e, 0x29, 0x4a, 0x4d, 0xcc, 0xd5, 0x35, 0xe4, 0x02, 0x00, 0x03, 0x61, 0xf0, 0x5f, 0x09, 0x00, 0x00, 0x00, 0x1f, 0x8b, 0x08, 0x08, 0x08, 0xf1, 0x55, 0x5a, 0x00, 0x03, 0x74, 0x65, 0x73, @@ -98,9 +107,13 @@ void testDecompressionWithMultipleGZipStream() throws Exception { bytes[i] = (byte) (data[i] & 0xff); } - try (final GzipDecompressingEntity entity = new GzipDecompressingEntity(new InputStreamEntity(new ByteArrayInputStream(bytes), ContentType.APPLICATION_OCTET_STREAM))) { - Assertions.assertEquals("stream-1\nstream-2\n", EntityUtils.toString(entity, StandardCharsets.US_ASCII)); - } + // Decompress multiple GZip streams using the factory + final HttpEntity entity = CompressorFactory.INSTANCE.decompressEntity( + new InputStreamEntity(new ByteArrayInputStream(bytes), ContentType.APPLICATION_OCTET_STREAM), + "gz"); + + // Verify the decompressed content + Assertions.assertEquals("stream-1\nstream-2\n", EntityUtils.toString(entity, StandardCharsets.US_ASCII)); } } diff --git a/httpclient5/src/test/java/org/apache/hc/client5/http/impl/classic/TestContentCompressionExec.java b/httpclient5/src/test/java/org/apache/hc/client5/http/impl/classic/TestContentCompressionExec.java index 60eaa27f6f..e1884ddaac 100644 --- a/httpclient5/src/test/java/org/apache/hc/client5/http/impl/classic/TestContentCompressionExec.java +++ b/httpclient5/src/test/java/org/apache/hc/client5/http/impl/classic/TestContentCompressionExec.java @@ -30,7 +30,7 @@ import org.apache.hc.client5.http.classic.ExecChain; import org.apache.hc.client5.http.classic.ExecRuntime; import org.apache.hc.client5.http.config.RequestConfig; -import org.apache.hc.client5.http.entity.DecompressingEntity; +import org.apache.hc.client5.http.entity.DecompressEntity; import org.apache.hc.client5.http.entity.EntityBuilder; import org.apache.hc.client5.http.entity.GzipDecompressingEntity; import org.apache.hc.client5.http.protocol.HttpClientContext; @@ -116,7 +116,7 @@ void testGzipContentEncoding() throws Exception { final HttpEntity entity = response.getEntity(); Assertions.assertNotNull(entity); - Assertions.assertTrue(entity instanceof DecompressingEntity); + Assertions.assertTrue(entity instanceof DecompressEntity); } @Test @@ -148,7 +148,7 @@ void testXGzipContentEncoding() throws Exception { final HttpEntity entity = response.getEntity(); Assertions.assertNotNull(entity); - Assertions.assertTrue(entity instanceof DecompressingEntity); + Assertions.assertTrue(entity instanceof DecompressEntity); } @Test @@ -164,7 +164,7 @@ void testDeflateContentEncoding() throws Exception { final HttpEntity entity = response.getEntity(); Assertions.assertNotNull(entity); - Assertions.assertTrue(entity instanceof DecompressingEntity); + Assertions.assertTrue(entity instanceof DecompressEntity); } @Test @@ -196,7 +196,7 @@ void testBrotliContentEncoding() throws Exception { final HttpEntity entity = response.getEntity(); Assertions.assertNotNull(entity); - Assertions.assertTrue(entity instanceof DecompressingEntity); + Assertions.assertTrue(entity instanceof DecompressEntity); } @Test diff --git a/pom.xml b/pom.xml index acbf6b935f..69bd0c7e6a 100644 --- a/pom.xml +++ b/pom.xml @@ -77,6 +77,7 @@ 5.3 javax.net.ssl.SSLEngine,javax.net.ssl.SSLParameters,java.nio.ByteBuffer,java.nio.CharBuffer 0.15.4 + 1.27.1 @@ -182,6 +183,11 @@ ${hamcrest.version} test + + org.apache.commons + commons-compress + ${commpress.version} +