From 1418c493e1f090748ecea48f1d8913d78f6931ca Mon Sep 17 00:00:00 2001 From: Lucas Saldanha Date: Thu, 22 Nov 2018 10:14:29 +1300 Subject: [PATCH 1/2] NC-1950: Fixing WebSocket error response --- .../response/JsonRpcErrorResponse.java | 3 +++ .../websocket/WebSocketRequestHandler.java | 12 ++++++++---- .../WebSocketRequestHandlerTest.java | 19 +++++++++++++++---- 3 files changed, 26 insertions(+), 8 deletions(-) diff --git a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java index 5eddab85c8..ef08cfc02e 100644 --- a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java +++ b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java @@ -14,6 +14,8 @@ import com.fasterxml.jackson.annotation.JsonGetter; import com.fasterxml.jackson.annotation.JsonIgnore; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonPropertyOrder; import com.google.common.base.Objects; @@ -29,6 +31,7 @@ public JsonRpcErrorResponse(final Object id, final JsonRpcError error) { } @JsonGetter("id") + @JsonInclude(Include.NON_NULL) public Object getId() { return id; } diff --git a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandler.java b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandler.java index c84b9ef431..7ec02c054d 100644 --- a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandler.java +++ b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandler.java @@ -14,6 +14,7 @@ import tech.pegasys.pantheon.ethereum.jsonrpc.internal.methods.JsonRpcMethod; import tech.pegasys.pantheon.ethereum.jsonrpc.internal.response.JsonRpcError; +import tech.pegasys.pantheon.ethereum.jsonrpc.internal.response.JsonRpcErrorResponse; import tech.pegasys.pantheon.ethereum.jsonrpc.websocket.methods.WebSocketRpcRequest; import java.util.Map; @@ -45,12 +46,13 @@ public void handle(final String id, final Buffer buffer) { request = buffer.toJsonObject().mapTo(WebSocketRpcRequest.class); } catch (final IllegalArgumentException | DecodeException e) { LOG.debug("Error mapping json to WebSocketRpcRequest", e); - future.complete(JsonRpcError.INVALID_REQUEST); + future.complete(new JsonRpcErrorResponse(null, JsonRpcError.INVALID_REQUEST)); return; } if (!methods.containsKey(request.getMethod())) { - future.complete(JsonRpcError.METHOD_NOT_FOUND); + future.complete( + new JsonRpcErrorResponse(request.getId(), JsonRpcError.METHOD_NOT_FOUND)); LOG.debug("Can't find method {}", request.getMethod()); return; } @@ -61,14 +63,16 @@ public void handle(final String id, final Buffer buffer) { future.complete(method.response(request)); } catch (final Exception e) { LOG.error(JsonRpcError.INTERNAL_ERROR.getMessage(), e); - future.complete(JsonRpcError.INTERNAL_ERROR); + future.complete(new JsonRpcErrorResponse(request.getId(), JsonRpcError.INTERNAL_ERROR)); } }, result -> { if (result.succeeded()) { replyToClient(id, Json.encodeToBuffer(result.result())); } else { - replyToClient(id, Json.encodeToBuffer(JsonRpcError.INTERNAL_ERROR)); + replyToClient( + id, + Json.encodeToBuffer(new JsonRpcErrorResponse(null, JsonRpcError.INTERNAL_ERROR))); } }); } diff --git a/ethereum/jsonrpc/src/test/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandlerTest.java b/ethereum/jsonrpc/src/test/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandlerTest.java index 3e83505b97..2e9499f448 100644 --- a/ethereum/jsonrpc/src/test/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandlerTest.java +++ b/ethereum/jsonrpc/src/test/java/tech/pegasys/pantheon/ethereum/jsonrpc/websocket/WebSocketRequestHandlerTest.java @@ -20,6 +20,7 @@ import tech.pegasys.pantheon.ethereum.jsonrpc.internal.JsonRpcRequest; import tech.pegasys.pantheon.ethereum.jsonrpc.internal.methods.JsonRpcMethod; import tech.pegasys.pantheon.ethereum.jsonrpc.internal.response.JsonRpcError; +import tech.pegasys.pantheon.ethereum.jsonrpc.internal.response.JsonRpcErrorResponse; import tech.pegasys.pantheon.ethereum.jsonrpc.internal.response.JsonRpcSuccessResponse; import tech.pegasys.pantheon.ethereum.jsonrpc.websocket.methods.WebSocketRpcRequest; @@ -95,6 +96,9 @@ public void handlerDeliversResponseSuccessfully(final TestContext context) { public void jsonDecodeFailureShouldRespondInvalidRequest(final TestContext context) { final Async async = context.async(); + final JsonRpcErrorResponse expectedResponse = + new JsonRpcErrorResponse(null, JsonRpcError.INVALID_REQUEST); + final String websocketId = UUID.randomUUID().toString(); vertx @@ -102,7 +106,7 @@ public void jsonDecodeFailureShouldRespondInvalidRequest(final TestContext conte .consumer(websocketId) .handler( msg -> { - context.assertEquals(Json.encode(JsonRpcError.INVALID_REQUEST), msg.body()); + context.assertEquals(Json.encode(expectedResponse), msg.body()); verifyZeroInteractions(jsonRpcMethodMock); async.complete(); }) @@ -115,6 +119,9 @@ public void jsonDecodeFailureShouldRespondInvalidRequest(final TestContext conte public void objectMapperFailureShouldRespondInvalidRequest(final TestContext context) { final Async async = context.async(); + final JsonRpcErrorResponse expectedResponse = + new JsonRpcErrorResponse(null, JsonRpcError.INVALID_REQUEST); + final String websocketId = UUID.randomUUID().toString(); vertx @@ -122,7 +129,7 @@ public void objectMapperFailureShouldRespondInvalidRequest(final TestContext con .consumer(websocketId) .handler( msg -> { - context.assertEquals(Json.encode(JsonRpcError.INVALID_REQUEST), msg.body()); + context.assertEquals(Json.encode(expectedResponse), msg.body()); verifyZeroInteractions(jsonRpcMethodMock); async.complete(); }) @@ -137,6 +144,8 @@ public void absentMethodShouldRespondMethodNotFound(final TestContext context) { final JsonObject requestJson = new JsonObject().put("id", 1).put("method", "eth_nonexistentMethod"); + final JsonRpcErrorResponse expectedResponse = + new JsonRpcErrorResponse(1, JsonRpcError.METHOD_NOT_FOUND); final String websocketId = UUID.randomUUID().toString(); @@ -145,7 +154,7 @@ public void absentMethodShouldRespondMethodNotFound(final TestContext context) { .consumer(websocketId) .handler( msg -> { - context.assertEquals(Json.encode(JsonRpcError.METHOD_NOT_FOUND), msg.body()); + context.assertEquals(Json.encode(expectedResponse), msg.body()); async.complete(); }) .completionHandler(v -> handler.handle(websocketId, Buffer.buffer(requestJson.toString()))); @@ -160,6 +169,8 @@ public void onExceptionProcessingRequestShouldRespondInternalError(final TestCon final JsonObject requestJson = new JsonObject().put("id", 1).put("method", "eth_x"); final JsonRpcRequest expectedRequest = requestJson.mapTo(WebSocketRpcRequest.class); when(jsonRpcMethodMock.response(eq(expectedRequest))).thenThrow(new RuntimeException()); + final JsonRpcErrorResponse expectedResponse = + new JsonRpcErrorResponse(1, JsonRpcError.INTERNAL_ERROR); final String websocketId = UUID.randomUUID().toString(); @@ -168,7 +179,7 @@ public void onExceptionProcessingRequestShouldRespondInternalError(final TestCon .consumer(websocketId) .handler( msg -> { - context.assertEquals(Json.encode(JsonRpcError.INTERNAL_ERROR), msg.body()); + context.assertEquals(Json.encode(expectedResponse), msg.body()); async.complete(); }) .completionHandler(v -> handler.handle(websocketId, Buffer.buffer(requestJson.toString()))); From 4dfe119e55b84e42467614015813fae5cb96bb5f Mon Sep 17 00:00:00 2001 From: Lucas Saldanha Date: Thu, 22 Nov 2018 11:01:27 +1300 Subject: [PATCH 2/2] Adding null id in json rpc error response --- .../jsonrpc/internal/response/JsonRpcErrorResponse.java | 3 --- 1 file changed, 3 deletions(-) diff --git a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java index ef08cfc02e..5eddab85c8 100644 --- a/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java +++ b/ethereum/jsonrpc/src/main/java/tech/pegasys/pantheon/ethereum/jsonrpc/internal/response/JsonRpcErrorResponse.java @@ -14,8 +14,6 @@ import com.fasterxml.jackson.annotation.JsonGetter; import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonInclude.Include; import com.fasterxml.jackson.annotation.JsonPropertyOrder; import com.google.common.base.Objects; @@ -31,7 +29,6 @@ public JsonRpcErrorResponse(final Object id, final JsonRpcError error) { } @JsonGetter("id") - @JsonInclude(Include.NON_NULL) public Object getId() { return id; }