Uploaded image for project: 'RESTEasy'
  1. RESTEasy
  2. RESTEASY-2505

Client Proxy methods returning void eagerly close body even in error conditions

XMLWordPrintable

    • Icon: Bug Bug
    • Resolution: Not a Bug
    • Icon: Minor Minor
    • None
    • 4.4.2.Final
    • jaxrs
    • None

      We write a JAX-RS client method, like

      @Produces(JSON)
      @PUT
      void storeSomething(Something value);
      

      Due to an error, the server returns 400 Bad Request unconditionally. This correctly causes the client to throw BadRequestException.

      We have a post-processor that attempts to inspect WebApplicationExceptions (including this BadRequestException) and attach diagnostic info for the operator:

          public static <X extends WebApplicationException> X summarize(final X wae) {
              wae.addSuppressed(new BodySummary(wae.getResponse()));
              throw wae;
          }
      
          @SuppressWarnings("PMD.DoNotExtendJavaLangThrowable")
          static class BodySummary extends Throwable {
              private static final long serialVersionUID = 1L;
      
              BodySummary(final Response response) {
                  super("Failing body is: " + abbreviate(response));
              }
      
              private static String abbreviate(final Response response) {
                  if (response instanceof BuiltResponse && ((BuiltResponse) response).isClosed()) {
                      return "<closed>";
                  }
                  return StringUtils.abbreviate(response.readEntity(String.class), 5120);
              }
          }
      

      This normally works just fine. We noticed it does not work at all on methods returning void, however. This seems to be due to DefaultEntityExtractorFactory:

         public static final EntityExtractor createVoidExtractor()
         {
            return new EntityExtractor()
            {
               public Object extractEntity(ClientContext context, Object... args)
               {
                  ClientResponse response = context.getClientResponse();
                  int status = response.getStatus();
                  if (status >= 400)
                  {
                     response.bufferEntity();
                     response.close();
                     ClientInvocation.handleErrorStatus(response);
                  }
                  response.close();
                  return null;
               }
            };
         }
      

      The extractor takes care to ensure the response is closed before returning it. This is fine in the normal case – obviously void methods don't have data returned – but in case of an exception, there may in fact be a body present describing the error.

      In this case it is extremely tricky to inspect the returned body via WebApplicationException.getResponse(). While it is buffered explicitly, since the response is already closed, any attempt to check e.g. hasEntity throws. And in fact the spec says:

      Buffering the message entity data allows for multiple invocations of readEntity(...) methods on the response instance. Note however, that once the response instance itself is closed, the implementations are expected to release the buffered message entity data too. Therefore any subsequent attempts to read a message entity stream on such closed response will result in an IllegalStateException being thrown. 
      

      So, if you have a void-returning method in a proxied interface, it seems to be impossible to read an error body in a spec-compliant manner.

      Please let me know if I can clarify further or if I'm "holding it wrong". Thanks!

        1. 2CB7248D-64DE-4E56-BDDE-51CEDDCEE724.png
          281 kB
          Weinan Li
        2. 31CCCE15-2A90-4F88-BB1F-13ADA9680D3B.png
          300 kB
          Weinan Li
        3. 35DE50AF-C45C-4115-A8A1-C9009A2A9984.png
          337 kB
          Weinan Li
        4. 4231E51D-A51D-4E4E-B8CD-B386BD7A6E22.png
          159 kB
          Weinan Li
        5. 44111429-F8E4-4510-A553-91EDBCD524E6.png
          412 kB
          Weinan Li
        6. 5BD6E24A-C788-4972-80A9-CCCB173D709E.png
          701 kB
          Weinan Li
        7. A0B82097-B52C-4058-8DAD-23B7C7453B26.png
          1.42 MB
          Weinan Li
        8. A0B82097-B52C-4058-8DAD-23B7C7453B26.png
          1.42 MB
          Weinan Li
        9. AD7AFB1A-DF47-4FDD-A1BC-3D806167A01B.png
          295 kB
          Weinan Li
        10. BDFC5E87-EF49-4514-A3AE-7AA359383CBA.png
          913 kB
          Weinan Li
        11. C4677079-59D5-4529-AD2D-F7438FD9990B.png
          515 kB
          Weinan Li
        12. E16DAFBB-FA36-4F02-9842-BC2410653DFB.png
          682 kB
          Weinan Li
        13. image-2020-06-16-20-19-07-184.png
          153 kB
          Weinan Li
        14. image-2020-06-16-20-19-32-539.png
          191 kB
          Weinan Li
        15. image-2020-06-22-19-38-15-210.png
          15 kB
          Weinan Li
        16. image-2020-06-22-19-38-29-708.png
          127 kB
          Weinan Li
        17. image-2020-06-22-19-38-50-200.png
          222 kB
          Weinan Li

              weli@redhat.com Weinan Li
              stevenschlansker Steven Schlansker (Inactive)
              Votes:
              0 Vote for this issue
              Watchers:
              4 Start watching this issue

                Created:
                Updated:
                Resolved: