Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Clarification about Response.readEntity(XXX) #1166

Closed
wants to merge 1 commit into from
Closed
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
99 changes: 49 additions & 50 deletions jaxrs-api/src/main/java/jakarta/ws/rs/core/Response.java
Original file line number Diff line number Diff line change
Expand Up @@ -74,16 +74,20 @@ protected Response() {
public abstract StatusType getStatusInfo();

/**
* Get the message entity Java instance. Returns {@code null} if the message does not contain an entity body.
* <p>
* If the entity is represented by an un-consumed {@link InputStream input stream} the method will return the input
* stream.
* </p>
* Get the message entity Java instance. Returns {@code null} if the message does not contain an entity body, otherwise
* behaves as follow:
* <ul>
* <li>If a Java instance was supplied by a call to {@link #accepted(Object)}, {@link #ok(Object)} or {@link ResponseBuilder#entity(Object)},
* then that instance is returned.
* <li>Else if {@link #bufferEntity()} was previously invoked successfully, a copy of the reset buffered entity input stream is returned.
* <li>Else if the message entity is represented by the original input stream not fully consumed already (i.e <code>inputStream.read() != -1</code>),
* then that input stream is returned.
* </ul>
*
* @return the message entity or {@code null} if message does not contain an entity body (i.e. when {@link #hasEntity()}
* returns {@code false}).
* @throws IllegalStateException if the entity was previously fully consumed as an {@link InputStream input stream}, or
* if the response has been {@link #close() closed}.
* @throws IllegalStateException if the message entity is represented by the original input stream fully consumed already
* (i.e <code>inputStream.read() == -1</code>), or if the response has been {@link #close() closed}.
*/
public abstract Object getEntity();

Expand All @@ -93,26 +97,25 @@ protected Response() {
* <p>
* Method throws an {@link ProcessingException} if the content of the message cannot be mapped to an entity of the
* requested type and {@link IllegalStateException} in case the entity is not backed by an input stream or if the
* original entity input stream has already been consumed without {@link #bufferEntity() buffering} the entity data
* prior consuming.
* original entity input stream has already been fully consumed (i.e <code>inputStream.read() == -1</code>) without
* being {@link #bufferEntity() buffered} prior consuming.
* </p>
* <p>
* A message instance returned from this method will be cached for subsequent retrievals via {@link #getEntity()}.
* Unless the supplied entity type is an {@link java.io.InputStream input stream}, this method automatically
* {@link #close() closes} the an unconsumed original response entity data stream if open. In case the entity data has
* {@link InputStream#close() closes} the original message entity input stream if open. In case the entity data has
* been buffered, the buffer will be reset prior consuming the buffered data to enable subsequent invocations of
* {@code readEntity(...)} methods on this response.
* </p>
*
* @param <T> entity instance Java type.
* @param entityType the type of entity.
* @return the message entity; for a zero-length response entities returns a corresponding Java object that represents
* @return the message entity; for a zero-length message entity returns a corresponding Java object that represents
* zero-length data. In case no zero-length representation is defined for the Java type, a {@link ProcessingException}
* wrapping the underlying {@link NoContentException} is thrown.
* @throws ProcessingException if the content of the message cannot be mapped to an entity of the requested type.
* @throws IllegalStateException if the entity is not backed by an input stream, the response has been {@link #close()
* closed} already, or if the entity input stream has been fully consumed already and has not been buffered prior
* consuming.
* closed} already, or if the entity input stream has been fully consumed already (i.e <code>inputStream.read() == -1</code>)
* and has not been buffered prior consuming.
* @see jakarta.ws.rs.ext.MessageBodyReader
* @since 2.0
*/
Expand All @@ -124,26 +127,25 @@ protected Response() {
* <p>
* Method throws an {@link ProcessingException} if the content of the message cannot be mapped to an entity of the
* requested type and {@link IllegalStateException} in case the entity is not backed by an input stream or if the
* original entity input stream has already been consumed without {@link #bufferEntity() buffering} the entity data
* prior consuming.
* original entity input stream has already been fully consumed (i.e <code>inputStream.read() == -1</code>) without
* being {@link #bufferEntity() buffered} prior consuming.
* </p>
* <p>
* A message instance returned from this method will be cached for subsequent retrievals via {@link #getEntity()}.
* Unless the supplied entity type is an {@link java.io.InputStream input stream}, this method automatically
* {@link #close() closes} the an unconsumed original response entity data stream if open. In case the entity data has
* {@link InputStream#close() closes} the original message entity input stream if open. In case the entity data has
* been buffered, the buffer will be reset prior consuming the buffered data to enable subsequent invocations of
* {@code readEntity(...)} methods on this response.
* </p>
*
* @param <T> entity instance Java type.
* @param entityType the type of entity; may be generic.
* @return the message entity; for a zero-length response entities returns a corresponding Java object that represents
* @return the message entity; for a zero-length response entity returns a corresponding Java object that represents
* zero-length data. In case no zero-length representation is defined for the Java type, a {@link ProcessingException}
* wrapping the underlying {@link NoContentException} is thrown.
* @throws ProcessingException if the content of the message cannot be mapped to an entity of the requested type.
* @throws IllegalStateException if the entity is not backed by an input stream, the response has been {@link #close()
* closed} already, or if the entity input stream has been fully consumed already and has not been buffered prior
* consuming.
* closed} already, or if the entity input stream has been fully consumed already (i.e <code>inputStream.read() == -1</code>)
* and has not been buffered prior consuming.
* @see jakarta.ws.rs.ext.MessageBodyReader
* @since 2.0
*/
Expand All @@ -155,27 +157,26 @@ protected Response() {
* <p>
* Method throws an {@link ProcessingException} if the content of the message cannot be mapped to an entity of the
* requested type and {@link IllegalStateException} in case the entity is not backed by an input stream or if the
* original entity input stream has already been consumed without {@link #bufferEntity() buffering} the entity data
* prior consuming.
* original entity input stream has already been fully consumed (i.e <code>inputStream.read() == -1</code>) without
* being {@link #bufferEntity() buffered} prior consuming.
* </p>
* <p>
* A message instance returned from this method will be cached for subsequent retrievals via {@link #getEntity()}.
* Unless the supplied entity type is an {@link java.io.InputStream input stream}, this method automatically
* {@link #close() closes} the an unconsumed original response entity data stream if open. In case the entity data has
* {@link InputStream#close() closes} the original message entity input stream if open. In case the entity data has
* been buffered, the buffer will be reset prior consuming the buffered data to enable subsequent invocations of
* {@code readEntity(...)} methods on this response.
* </p>
*
* @param <T> entity instance Java type.
* @param entityType the type of entity.
* @param annotations annotations that will be passed to the {@link MessageBodyReader}.
* @return the message entity; for a zero-length response entities returns a corresponding Java object that represents
* @return the message entity; for a zero-length response entity returns a corresponding Java object that represents
* zero-length data. In case no zero-length representation is defined for the Java type, a {@link ProcessingException}
* wrapping the underlying {@link NoContentException} is thrown.
* @throws ProcessingException if the content of the message cannot be mapped to an entity of the requested type.
* @throws IllegalStateException if the entity is not backed by an input stream, the response has been {@link #close()
* closed} already, or if the entity input stream has been fully consumed already and has not been buffered prior
* consuming.
* closed} already, or if the entity input stream has been fully consumed already (i.e <code>inputStream.read() == -1</code>)
* and has not been buffered prior consuming.
* @see jakarta.ws.rs.ext.MessageBodyReader
* @since 2.0
*/
Expand All @@ -187,27 +188,26 @@ protected Response() {
* <p>
* Method throws an {@link ProcessingException} if the content of the message cannot be mapped to an entity of the
* requested type and {@link IllegalStateException} in case the entity is not backed by an input stream or if the
* original entity input stream has already been consumed without {@link #bufferEntity() buffering} the entity data
* prior consuming.
* original entity input stream has already been fully consumed (i.e <code>inputStream.read() == -1</code>) without
* being {@link #bufferEntity() buffered} prior consuming.
* </p>
* <p>
* A message instance returned from this method will be cached for subsequent retrievals via {@link #getEntity()}.
* Unless the supplied entity type is an {@link java.io.InputStream input stream}, this method automatically
* {@link #close() closes} the an unconsumed original response entity data stream if open. In case the entity data has
* {@link InputStream#close() closes} the original message entity input stream if open. In case the entity data has
* been buffered, the buffer will be reset prior consuming the buffered data to enable subsequent invocations of
* {@code readEntity(...)} methods on this response.
* </p>
*
* @param <T> entity instance Java type.
* @param entityType the type of entity; may be generic.
* @param annotations annotations that will be passed to the {@link MessageBodyReader}.
* @return the message entity; for a zero-length response entities returns a corresponding Java object that represents
* @return the message entity; for a zero-length response entity returns a corresponding Java object that represents
* zero-length data. In case no zero-length representation is defined for the Java type, a {@link ProcessingException}
* wrapping the underlying {@link NoContentException} is thrown.
* @throws ProcessingException if the content of the message cannot be mapped to an entity of the requested type.
* @throws IllegalStateException if the entity is not backed by an input stream, the response has been {@link #close()
* closed} already, or if the entity input stream has been fully consumed already and has not been buffered prior
* consuming.
* closed} already, or if the entity input stream has been fully consumed already (i.e <code>inputStream.read() == -1</code>)
* and has not been buffered prior consuming.
* @see jakarta.ws.rs.ext.MessageBodyReader
* @since 2.0
*/
Expand All @@ -232,19 +232,18 @@ protected Response() {
public abstract boolean hasEntity();

/**
* Buffer the message entity data.
* Buffer the message entity input stream.
* <p>
* In case the message entity is backed by an unconsumed entity input stream, all the bytes of the original entity input
* stream are read and stored in a local buffer. The original entity input stream is consumed and automatically closed
* as part of the operation and the method returns {@code true}.
* In case the message entity is backed by an input stream not consumed at all, all the bytes of that input
* stream are read (i.e until <code>inputStream.read() == -1</code>) and stored in a local buffer. That input
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Overall, this looks like a step in the right direction. However, I think the phrase "not consumed at all" is a bit vague. If we mean, "not a single byte from the input stream has been read", we should be more explicit IMO. This phrase is used multiple times.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@NicoNes Could you review and respond to the comment?

Copy link
Contributor Author

@NicoNes NicoNes Dec 14, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@spericas I agree we can be more exeplicit.
What about saying " In case the message entity is backed by an input stream of which no byte has been read" ?

* stream is then automatically {@link InputStream#close() closed} as part of the operation and the method returns {@code true}.
* </p>
* <p>
* In case the response entity instance is not backed by an unconsumed input stream an invocation of
* {@code bufferEntity} method is ignored and the method returns {@code false}.
* Otherwise an invocation of {@code bufferEntity} method is ignored and the method returns {@code false}.
* </p>
* <p>
* This operation is idempotent, i.e. it can be invoked multiple times with the same effect which also means that
* calling the {@code bufferEntity()} method on an already buffered (and thus closed) message instance is legal and has
* calling the {@code bufferEntity()} method on an already buffered (and thus closed) input stream instance is legal and has
* no further effect. Also, the result returned by the {@code bufferEntity()} method is consistent across all
* invocations of the method on the same {@code Response} instance.
* </p>
Expand All @@ -268,13 +267,13 @@ protected Response() {
* associated with the response (e.g. {@link #bufferEntity() buffered message entity data}).
* <p>
* This operation is idempotent, i.e. it can be invoked multiple times with the same effect which also means that
* calling the {@code close()} method on an already closed message instance is legal and has no further effect.
* calling the {@code close()} method on an already closed message entity input stream is legal and has no further effect.
* </p>
* <p>
* The {@code close()} method should be invoked on all instances that contain an un-consumed entity input stream to
* ensure the resources associated with the instance are properly cleaned-up and prevent potential memory leaks. This is
* typical for client-side scenarios where application layer code processes only the response headers and ignores the
* response entity.
* The {@code close()} method should be invoked on all instances that contain an original entity input stream not fully
* consumed (i.e until inputStream.read() != -1) to ensure the resources associated with the instance are properly cleaned-up
* and prevent potential memory leaks. This is typical for client-side scenarios where application layer code processes only
* the response headers and ignores the response entity.
* </p>
* <p>
* Any attempts to manipulate (read, get, buffer) a message entity on a closed response will result in an
Expand Down Expand Up @@ -471,9 +470,9 @@ public MultivaluedMap<String, Object> getHeaders() {
* builder.
* </p>
* <p>
* Note that if the entity is backed by an un-consumed input stream, the reference to the stream is copied. In such case
* make sure to {@link #bufferEntity() buffer} the entity stream of the original response instance before passing it to
* this method.
* Note that if the entity is backed by the original input stream not consumed at all, the reference to the stream is copied.
* In such case make sure to {@link #bufferEntity() buffer} the entity stream of the original response instance before passing
* it to this method.
* </p>
*
* @param response a Response from which the status code, entity and {@link #getHeaders() response headers} will be
Expand Down