Browse Source

Polishing ProblemDetail support

See gh-28665
pull/28715/head
rstoyanchev 2 years ago
parent
commit
45ee7913bf
  1. 13
      spring-web/src/main/java/org/springframework/http/ProblemDetail.java
  2. 9
      spring-web/src/main/java/org/springframework/http/ResponseEntity.java
  3. 2
      spring-web/src/main/java/org/springframework/web/ErrorResponseException.java
  4. 2
      spring-webflux/src/main/java/org/springframework/web/reactive/result/method/annotation/AbstractMessageWriterResultHandler.java
  5. 2
      spring-webflux/src/main/java/org/springframework/web/reactive/result/method/annotation/ResponseEntityResultHandler.java
  6. 2
      spring-webmvc/src/main/java/org/springframework/web/servlet/mvc/method/annotation/AbstractMessageConverterMethodProcessor.java
  7. 2
      spring-webmvc/src/main/java/org/springframework/web/servlet/mvc/method/annotation/HttpEntityMethodProcessor.java

13
spring-web/src/main/java/org/springframework/http/ProblemDetail.java

@ -23,8 +23,11 @@ import org.springframework.util.Assert; @@ -23,8 +23,11 @@ import org.springframework.util.Assert;
/**
* Representation of an RFC 7807 problem detail, including all RFC-defined
* fields. For an extended response with more fields, create a subclass that
* exposes the additional fields.
* properties.
*
* <p>For an extended response, create a subclass with additional properties.
* A subclass can use the {@link ProblemDetail#ProblemDetail(ProblemDetail)}
* copy constructor to extend an existing {@code ProblemDetail} instance.
*
* @author Rossen Stoyanchev
* @since 6.0
@ -63,8 +66,8 @@ public class ProblemDetail { @@ -63,8 +66,8 @@ public class ProblemDetail {
}
/**
* Copy constructor that could be used from a subclass to re-create a
* {@code ProblemDetail} in order to extend it with more fields.
* Copy constructor that a subclass can use to re-create and extend a
* {@code ProblemDetail} with additional properties.
*/
protected ProblemDetail(ProblemDetail other) {
this.type = other.type;
@ -75,7 +78,7 @@ public class ProblemDetail { @@ -75,7 +78,7 @@ public class ProblemDetail {
}
/**
* For deserialization.
* No-arg constructor, for deserialization.
*/
protected ProblemDetail() {
}

9
spring-web/src/main/java/org/springframework/http/ResponseEntity.java

@ -264,10 +264,11 @@ public class ResponseEntity<T> extends HttpEntity<T> { @@ -264,10 +264,11 @@ public class ResponseEntity<T> extends HttpEntity<T> {
/**
* Create a builder for a {@code ResponseEntity} with the given
* {@link ProblemDetail} as the body, also matching to its
* {@link ProblemDetail#getStatus() status}. An {@code @ExceptionHandler}
* method can use to add response headers, or otherwise it can return
* {@code ProblemDetail}.
* {@link ProblemDetail} as the body, and its
* {@link ProblemDetail#getStatus() status} as the status.
* <p>Note that {@code ProblemDetail} is supported as a return value from
* controller methods and from {@code @ExceptionHandler} methods. The method
* here is convenient to also add response headers.
* @param body the details for an HTTP error response
* @return the created builder
* @since 6.0

2
spring-web/src/main/java/org/springframework/web/ErrorResponseException.java

@ -32,7 +32,7 @@ import org.springframework.lang.Nullable; @@ -32,7 +32,7 @@ import org.springframework.lang.Nullable;
* <p>The exception can be used as is, or it can be extended as a more specific
* exception that populates the {@link ProblemDetail#setType(URI) type} or
* {@link ProblemDetail#setDetail(String) detail} fields, or potentially adds
* other non-standard fields.
* other non-standard properties.
*
* @author Rossen Stoyanchev
* @since 6.0

2
spring-webflux/src/main/java/org/springframework/web/reactive/result/method/annotation/AbstractMessageWriterResultHandler.java

@ -167,7 +167,7 @@ public abstract class AbstractMessageWriterResultHandler extends HandlerResultHa @@ -167,7 +167,7 @@ public abstract class AbstractMessageWriterResultHandler extends HandlerResultHa
throw ex;
}
// Fall back on RFC 7807 format for ProblemDetail
// For ProblemDetail, fall back on RFC 7807 format
if (bestMediaType == null && elementType.toClass().equals(ProblemDetail.class)) {
bestMediaType = selectMediaType(exchange, () -> getMediaTypesFor(elementType), this.problemMediaTypes);
}

2
spring-webflux/src/main/java/org/springframework/web/reactive/result/method/annotation/ResponseEntityResultHandler.java

@ -144,7 +144,7 @@ public class ResponseEntityResultHandler extends AbstractMessageWriterResultHand @@ -144,7 +144,7 @@ public class ResponseEntityResultHandler extends AbstractMessageWriterResultHand
httpEntity = new ResponseEntity<>(response.getBody(), response.getHeaders(), response.getStatusCode());
}
else if (returnValue instanceof ProblemDetail detail) {
httpEntity = new ResponseEntity<>(returnValue, HttpHeaders.EMPTY, detail.getStatus());
httpEntity = ResponseEntity.of(detail).build();
}
else if (returnValue instanceof HttpHeaders) {
httpEntity = new ResponseEntity<>((HttpHeaders) returnValue, HttpStatus.OK);

2
spring-webmvc/src/main/java/org/springframework/web/servlet/mvc/method/annotation/AbstractMessageConverterMethodProcessor.java

@ -243,7 +243,7 @@ public abstract class AbstractMessageConverterMethodProcessor extends AbstractMe @@ -243,7 +243,7 @@ public abstract class AbstractMessageConverterMethodProcessor extends AbstractMe
List<MediaType> compatibleMediaTypes = new ArrayList<>();
determineCompatibleMediaTypes(acceptableTypes, producibleTypes, compatibleMediaTypes);
// Fall back on RFC 7807 format for ProblemDetail
// For ProblemDetail, fall back on RFC 7807 format
if (compatibleMediaTypes.isEmpty() && ProblemDetail.class.isAssignableFrom(valueType)) {
determineCompatibleMediaTypes(this.problemMediaTypes, producibleTypes, compatibleMediaTypes);
}

2
spring-webmvc/src/main/java/org/springframework/web/servlet/mvc/method/annotation/HttpEntityMethodProcessor.java

@ -185,7 +185,7 @@ public class HttpEntityMethodProcessor extends AbstractMessageConverterMethodPro @@ -185,7 +185,7 @@ public class HttpEntityMethodProcessor extends AbstractMessageConverterMethodPro
httpEntity = new ResponseEntity<>(response.getBody(), response.getHeaders(), response.getStatusCode());
}
else if (returnValue instanceof ProblemDetail detail) {
httpEntity = new ResponseEntity<>(returnValue, HttpHeaders.EMPTY, detail.getStatus());
httpEntity = ResponseEntity.of(detail).build();
}
else {
Assert.isInstanceOf(HttpEntity.class, returnValue);

Loading…
Cancel
Save