com.oracle.bmc.http.client.jersey3.WrappedResponseInputStream Maven / Gradle / Ivy
/**
* Copyright (c) 2016, 2024, Oracle and/or its affiliates. All rights reserved.
* This software is dual-licensed to you under the Universal Permissive License (UPL) 1.0 as shown at https://oss.oracle.com/licenses/upl or Apache License 2.0 as shown at http://www.apache.org/licenses/LICENSE-2.0. You may choose either license.
*/
package com.oracle.bmc.http.client.jersey3;
import jakarta.annotation.Nonnull;
import jakarta.ws.rs.ProcessingException;
import jakarta.ws.rs.core.Response;
import java.io.IOException;
import java.io.InputStream;
/**
* A wrapper over an {@link java.io.InputStream} that ensures that the delegate {@link
* jakarta.ws.rs.core.Response} is closed when the delegate input stream is closed.
*/
public final class WrappedResponseInputStream extends InputStream {
private static final org.slf4j.Logger LOG =
org.slf4j.LoggerFactory.getLogger(WrappedResponseInputStream.class);
protected final InputStream delegate;
protected final Response delegateResponse;
public WrappedResponseInputStream(
@Nonnull InputStream delegate, @Nonnull Response delegateResponse) {
if (delegate == null) {
throw new java.lang.NullPointerException("delegate is marked non-null but is null");
}
if (delegateResponse == null) {
throw new java.lang.NullPointerException(
"delegateResponse is marked non-null but is null");
}
this.delegate = delegate;
this.delegateResponse = delegateResponse;
}
/**
* Closes a given {@link Response} and ignores any thrown exceptions. Closing a response is
* idempotent, so there is no harm in closing a {@link Response} once the response has been
* fully processed to ensure that the client connection is not left in use.
*
* @param response the response to close
*/
private static void closeResponseSilently(final Response response) {
synchronized (response) {
try {
response.close();
} catch (Throwable t) {
LOG.debug("Exception while closing response", t);
}
}
}
@Override
public int read() throws IOException {
return delegate.read();
}
@Override
public int read(final byte[] b) throws IOException {
return delegate.read(b);
}
@Override
public int read(final byte[] b, final int off, final int len) throws IOException {
return delegate.read(b, off, len);
}
@Override
public long skip(final long n) throws IOException {
return delegate.skip(n);
}
@Override
public synchronized void mark(int readlimit) {
delegate.mark(readlimit);
}
@Override
public synchronized void reset() throws IOException {
delegate.reset();
}
@Override
public int available() throws IOException {
return delegate.available();
}
@Override
public boolean markSupported() {
return delegate.markSupported();
}
@Override
public void close() throws IOException {
try {
delegate.close();
} catch (ProcessingException e) {
LOG.debug("Exception caused while trying to close the stream", e);
} finally {
// When the input stream has been consumed, try also closing the wrapped response entity
// to free up
// resources for the Jersey Client (e.g., a connection via a connection pool).
closeResponseSilently(delegateResponse);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy