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

Allow for external customization and i8n of the "detail" of an RFC 7807 response #28814

Closed
1 task done
Tracked by #27052
rstoyanchev opened this issue Jul 13, 2022 · 5 comments
Closed
1 task done
Tracked by #27052
Assignees
Labels
in: web Issues in web modules (web, webmvc, webflux, websocket) type: enhancement A general enhancement
Milestone

Comments

@rstoyanchev
Copy link
Contributor

rstoyanchev commented Jul 13, 2022

After #27052, each Spring MVC / WebFlux exception implements ErrorResponse and exposes ProblemDetail for the response body. By default, the "detail" field is set to a minimal message (and not than the actual Exception message), to avoid leaking implementation details. An application can extend ResponseEntityExceptionHandler and override protected methods customize the details for each exception.

We should consider a way to allow customization of the error "detail" for each Spring MVC and WebFlux exception, as well as for any ErrorResponseException, for example by performing a lookup in a MessageSource with a message code derived from the exception name. That would allow an application to customize and internationalize exception messages through property files. This is already supported for @ResponseStatus exceptions, and we could support the same for ErrorResponse exceptions.

One specific challenge will be the ability to parameterize messages with information that is specific to each exception.

@rstoyanchev rstoyanchev added in: web Issues in web modules (web, webmvc, webflux, websocket) type: enhancement A general enhancement labels Jul 13, 2022
@rstoyanchev rstoyanchev added this to the 6.0.0-M6 milestone Jul 13, 2022
@rstoyanchev rstoyanchev self-assigned this Jul 13, 2022
@rstoyanchev rstoyanchev modified the milestones: 6.0.0-M6, 6.0.0-RC1 Sep 14, 2022
rstoyanchev added a commit that referenced this issue Oct 5, 2022
rstoyanchev added a commit that referenced this issue Oct 5, 2022
Expand the documentation for error responses and add details on
using a MessageSource to customize and internationalize error details.

See gh-28814
@rstoyanchev rstoyanchev changed the title Provide options for external control over ProblemDetail properties for Spring web exceptions Allow for external customization and i8n of the "detail" of an RFC 7807 response Oct 5, 2022
@rstoyanchev
Copy link
Contributor Author

This is now largely done. Check the updated documentation section on Error Responses section and the sub-section on customization and internationalization.

@biergit
Copy link

biergit commented Oct 25, 2022

Hi @rstoyanchev
I was wondering how I as a user of Spring should go about providing i18n messages for my custom exceptions.
It seems you are proposing to override the ResponseEntityExceptionHandler but I don't see as much support for reusing the default message source lookup mechanism as used for the Spring exceptions as I hoped for.
Ideally I would like to provide my problemDetail.com.example.MyException message in my message.properties and write something like

@ControllerAdvice
class MyExceptionHandler extends ResponseEntityExceptionHandler {

  @ExceptionHandler
  public ResponseEntity<Object> handleMyException(MyException ex, NativeWebRequest request) {
    return handleExceptionInternal(ex, BAD_REQUEST, request, "My default message", new Object[] {ex.getParam1(), ex.getParam2()});
  }
}

Instead what I thought I had to do was something like this

@ControllerAdvice
public class MyExceptionHandler extends ResponseEntityExceptionHandler {
    @ExceptionHandler
    ResponseEntity<Object> handleMyException(MyException myException, NativeWebRequest nativeWebRequest) {
        ProblemDetail body = ProblemDetail.forStatusAndDetail(HttpStatus.BAD_REQUEST, myException.getMessage());
        ErrorResponseException mappedException = new ErrorResponseException(HttpStatus.BAD_REQUEST, body, myException,
            ErrorResponse.getDefaultDetailMessageCode(myOtherException.getClass(), null), null);
        return handleExceptionInternal(mappedException, null, null, HttpStatus.BAD_REQUEST,
            nativeWebRequest);
    }
}

This feels quite awkward and in finally getting to that solution I stumbled across an NPE here when I did ErrorResponseException mappedException = new ErrorResponseException(HttpStatus.BAD_REQUEST, !!null!!, myOtherException, ErrorResponse.getDefaultDetailMessageCode(myOtherException.getClass(), null), null); instead.

Does ProblemDetail have to be so weird to use? Why not go for a full builder?

Also the current implementation essentially only supports customizing the message detail via the message source for ErrorResponses. So problemDetail.org.springframework.web.servlet.NoHandlerFoundException=No handler works while problemDetail.org.springframework.beans.TypeMismatchException=Ooops does not. Would you agree that this is not ideal and also not clearly described in the documentation here?

@rstoyanchev
Copy link
Contributor Author

rstoyanchev commented Oct 26, 2022

@biergit, thanks for taking a look and for the feedback.

Generally, you don't need to extend ResponseEntityExceptionHandler. It can be any @ControllerAdvice, and it's easy for such a class to perform a lookup via MessageSource. That said, you're right that ResponseEntityExceptionHandler could make it easier to map custom exceptions.

I've created #29384 to make an improvement, and there is a specific proposal. Please, take a look and let's continue the discussion there.

As for TypeMismatchException, that should work, see here, but it's not actually passing the messageCode, so that's an oversight. I'll fix this that as part of #29384 since the handling for TypeMismatchException will take advantage of the same improvement.

@NaitYoussef
Copy link

NaitYoussef commented Oct 30, 2023

This is now largely done. Check the updated documentation section on Error Responses section and the sub-section on customization and internationalization.

The link above is broken

@bclozel
Copy link
Member

bclozel commented Oct 30, 2023

See https://docs.spring.io/spring-framework/reference/web/webmvc/mvc-ann-rest-exceptions.html#mvc-ann-rest-exceptions-i18n

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in: web Issues in web modules (web, webmvc, webflux, websocket) type: enhancement A general enhancement
Projects
None yet
Development

No branches or pull requests

4 participants