From 7a5fce1acd4887eb167ea7d74de743089d44f8c0 Mon Sep 17 00:00:00 2001 From: Harold Hotelling Date: Fri, 5 Nov 2021 22:20:57 -0700 Subject: [PATCH] Throw an exception when an entity-level action would be invoked without an id. Entity-level actions are performed on a specific resource, and so the id value in the request is mandatory. Today, if the client forgets to call .id(...) to supply a value the request can be built and sent to the server where it will cause an error with a mysterious error message that implies that there is no action of that name, i.e. "POST operation named myAction not supported on resource '...' URI: '...'" Here we fix that so that a) The problem is caught on the client side during the RequestBuilder.build() method call so that the request cannot be sent to the server. b) A sensible error message is given where the stack trace's line number will lead the developer straight to the root of the problem: "Missing or null id value; we cannot invoke this entity-level action." --- CHANGELOG.md | 1 + .../http2/Http2ProtocolUpgradeHandler.java | 2 +- .../restli/client/ActionRequestBuilder.java | 5 ++++ .../base/EntityActionRequestBuilderBase.java | 27 +++++++++++++++++++ .../restli/examples/TestGreetingsClient.java | 8 ++++++ .../JavaRequestBuilderGenerator.java | 13 ++++++--- 6 files changed, 52 insertions(+), 4 deletions(-) create mode 100644 restli-client/src/main/java/com/linkedin/restli/client/base/EntityActionRequestBuilderBase.java diff --git a/CHANGELOG.md b/CHANGELOG.md index 758688defa..0e2d9b828c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,6 +13,7 @@ When updating the changelog, remember to be very clear about what behavior has c and what APIs have changed, if applicable. ## [Unreleased] +- For entity level action request builders, issue a sensible error if the id is null. ## [29.22.10] - 2021-10-20 - SmoothRateLimiter - do not double-count execution delays on setRate diff --git a/r2-netty/src/main/java/com/linkedin/r2/netty/handler/http2/Http2ProtocolUpgradeHandler.java b/r2-netty/src/main/java/com/linkedin/r2/netty/handler/http2/Http2ProtocolUpgradeHandler.java index 2d78497590..8048671c6d 100644 --- a/r2-netty/src/main/java/com/linkedin/r2/netty/handler/http2/Http2ProtocolUpgradeHandler.java +++ b/r2-netty/src/main/java/com/linkedin/r2/netty/handler/http2/Http2ProtocolUpgradeHandler.java @@ -78,7 +78,7 @@ public void channelActive(ChannelHandlerContext ctx) public static void processChannelActive(ChannelHandlerContext ctx, Logger log, ChannelPromise upgradePromise) { - // For an upgrade request, clients should use an OPTIONS request for path “*” or a HEAD request for “/”. + // For an upgrade request, clients should use an OPTIONS request for path "*" or a HEAD request for "/". // RFC: https://tools.ietf.org/html/rfc7540#section-3.2 // Implementation detail: https://http2.github.io/faq/#can-i-implement-http2-without-implementing-http11 final DefaultFullHttpRequest request = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.OPTIONS, "*"); diff --git a/restli-client/src/main/java/com/linkedin/restli/client/ActionRequestBuilder.java b/restli-client/src/main/java/com/linkedin/restli/client/ActionRequestBuilder.java index 32770e80ee..301c2873cd 100644 --- a/restli-client/src/main/java/com/linkedin/restli/client/ActionRequestBuilder.java +++ b/restli-client/src/main/java/com/linkedin/restli/client/ActionRequestBuilder.java @@ -241,6 +241,11 @@ public ActionRequest build() } + protected boolean hasId() + { + return _id != null; + } + private Map, Object> buildReadOnlyActionParameters() { return buildReadOnlyActionParameters(_actionParams); diff --git a/restli-client/src/main/java/com/linkedin/restli/client/base/EntityActionRequestBuilderBase.java b/restli-client/src/main/java/com/linkedin/restli/client/base/EntityActionRequestBuilderBase.java new file mode 100644 index 0000000000..349ec817ae --- /dev/null +++ b/restli-client/src/main/java/com/linkedin/restli/client/base/EntityActionRequestBuilderBase.java @@ -0,0 +1,27 @@ +package com.linkedin.restli.client.base; + +import com.linkedin.restli.client.ActionRequest; +import com.linkedin.restli.client.RestliRequestOptions; +import com.linkedin.restli.common.ResourceSpec; + +/** + * The abstract base class for all generated request builders classes for entity-level actions. + */ +public abstract class EntityActionRequestBuilderBase> + extends ActionRequestBuilderBase +{ + protected EntityActionRequestBuilderBase(String baseUriTemplate, Class elementClass, ResourceSpec resourceSpec, + RestliRequestOptions requestOptions) + { + super(baseUriTemplate, elementClass, resourceSpec, requestOptions); + } + + @Override + public ActionRequest build() { + if (!hasId()) + { + throw new IllegalStateException("Entity-level action request is missing required id value."); + } + return super.build(); + } +} diff --git a/restli-int-test/src/test/java/com/linkedin/restli/examples/TestGreetingsClient.java b/restli-int-test/src/test/java/com/linkedin/restli/examples/TestGreetingsClient.java index cd63a25b47..86ca692ceb 100644 --- a/restli-int-test/src/test/java/com/linkedin/restli/examples/TestGreetingsClient.java +++ b/restli-int-test/src/test/java/com/linkedin/restli/examples/TestGreetingsClient.java @@ -121,6 +121,14 @@ public void testActionRequestOptionsPropagation(RootBuilderWrapper builders) + { + builders.action("SomeAction").build(); + } + @Test(dataProvider = com.linkedin.restli.internal.common.TestConstants.RESTLI_PROTOCOL_1_2_PREFIX + "requestBuilderDataProvider") public void testGetRequestOptionsPropagation(RootBuilderWrapper builders) { diff --git a/restli-tools/src/main/java/com/linkedin/restli/tools/clientgen/JavaRequestBuilderGenerator.java b/restli-tools/src/main/java/com/linkedin/restli/tools/clientgen/JavaRequestBuilderGenerator.java index cd557fd8a1..791c4478ab 100644 --- a/restli-tools/src/main/java/com/linkedin/restli/tools/clientgen/JavaRequestBuilderGenerator.java +++ b/restli-tools/src/main/java/com/linkedin/restli/tools/clientgen/JavaRequestBuilderGenerator.java @@ -42,6 +42,7 @@ import com.linkedin.pegasus.generator.JavaDataTemplateGenerator; import com.linkedin.pegasus.generator.TemplateSpecGenerator; import com.linkedin.pegasus.generator.spec.ClassTemplateSpec; +import com.linkedin.restli.client.ActionRequestBuilder; import com.linkedin.restli.client.OptionsRequestBuilder; import com.linkedin.restli.client.RestliRequestOptions; import com.linkedin.restli.client.base.ActionRequestBuilderBase; @@ -57,6 +58,7 @@ import com.linkedin.restli.client.base.CreateIdEntityRequestBuilderBase; import com.linkedin.restli.client.base.CreateIdRequestBuilderBase; import com.linkedin.restli.client.base.DeleteRequestBuilderBase; +import com.linkedin.restli.client.base.EntityActionRequestBuilderBase; import com.linkedin.restli.client.base.FindRequestBuilderBase; import com.linkedin.restli.client.base.GetAllRequestBuilderBase; import com.linkedin.restli.client.base.GetRequestBuilderBase; @@ -1378,7 +1380,9 @@ private void generateActions(JDefinedClass facadeClass, { for (ActionSchema action : resourceActions) { - generateActionMethod(facadeClass, baseUriExpr, _voidClass, action, resourceSpecField, resourceName, pathKeys, pathKeyTypes, assocKeyTypes, pathToAssocKeys, requestOptionsExpr, rootPath); + generateActionMethod(facadeClass, baseUriExpr, _voidClass, ActionRequestBuilderBase.class, action, + resourceSpecField, resourceName, pathKeys, pathKeyTypes, assocKeyTypes, pathToAssocKeys, requestOptionsExpr, + rootPath); } } @@ -1386,7 +1390,9 @@ private void generateActions(JDefinedClass facadeClass, { for (ActionSchema action : entityActions) { - generateActionMethod(facadeClass, baseUriExpr, keyClass, action, resourceSpecField, resourceName, pathKeys, pathKeyTypes, assocKeyTypes, pathToAssocKeys, requestOptionsExpr, rootPath); + generateActionMethod(facadeClass, baseUriExpr, keyClass, EntityActionRequestBuilderBase.class, action, + resourceSpecField, resourceName, pathKeys, pathKeyTypes, assocKeyTypes, pathToAssocKeys, requestOptionsExpr, + rootPath); } } } @@ -1394,6 +1400,7 @@ private void generateActions(JDefinedClass facadeClass, private void generateActionMethod(JDefinedClass facadeClass, JExpression baseUriExpr, JClass keyClass, + Class builderParentClass, ActionSchema action, JVar resourceSpecField, String resourceName, @@ -1406,7 +1413,7 @@ private void generateActionMethod(JDefinedClass facadeClass, throws JClassAlreadyExistsException { final JClass returnType = getActionReturnType(facadeClass, action.getReturns()); - final JClass vanillaActionBuilderClass = getCodeModel().ref(ActionRequestBuilderBase.class).narrow(keyClass, returnType); + final JClass vanillaActionBuilderClass = getCodeModel().ref(builderParentClass).narrow(keyClass, returnType); final String actionName = action.getName(); final String actionBuilderClassName = CodeUtil.capitalize(resourceName) + "Do" + CodeUtil.capitalize(actionName) + METHOD_BUILDER_SUFFIX.get(_version);