From 840a4d96ad7dfcbdaff11ae4214d8033972e2683 Mon Sep 17 00:00:00 2001 From: Krishnananthalingam Tharmigan <63336800+TharmiganK@users.noreply.github.com> Date: Fri, 6 Dec 2024 10:43:20 +0530 Subject: [PATCH] Fix rest path parameter generation with decoded value (#2245) --- .../service_dispatching_uri_template_matching.bal | 15 +++++++++++++++ changelog.md | 1 + .../http/api/service/signature/AllPathParams.java | 12 ++++++++---- 3 files changed, 24 insertions(+), 4 deletions(-) diff --git a/ballerina-tests/http-dispatching-tests/tests/service_dispatching_uri_template_matching.bal b/ballerina-tests/http-dispatching-tests/tests/service_dispatching_uri_template_matching.bal index bd119d06de..36be85d62e 100644 --- a/ballerina-tests/http-dispatching-tests/tests/service_dispatching_uri_template_matching.bal +++ b/ballerina-tests/http-dispatching-tests/tests/service_dispatching_uri_template_matching.bal @@ -383,6 +383,11 @@ service /encodedUri on utmTestEP { } service /restParam on utmTestEP { + + resource function 'default 'string/[string... aaa]() returns json { + return {aaa: aaa}; + } + resource function 'default 'int/[int... aaa](http:Caller caller) returns error? { http:Response res = new; json responseJson = {aaa: aaa}; @@ -1025,6 +1030,16 @@ function testEncodedPathParams() { } } +@test:Config {} +function testRestParamWithEncodedPathSegments() { +http:Response|error response = utmClient->get("/restParam/string/path%2Fseg/path%20seg/path%2Fseg+123"); + if response is http:Response { + common:assertJsonValue(response.getJsonPayload(), "aaa", ["path/seg", "path seg", "path/seg+123"]); + } else { + test:assertFail(msg = "Found unexpected output type: " + response.message()); + } +} + @test:Config {} function testMultipleIntTypedRestParams() { http:Response|error response = utmClient->get("/restParam/int/345/234/123"); diff --git a/changelog.md b/changelog.md index d262664c18..8d9fc3c94a 100644 --- a/changelog.md +++ b/changelog.md @@ -25,6 +25,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - [Fix duplicating `Content-Type` header via the `addHeader` method](https://github.com/ballerina-platform/ballerina-library/issues/7268) - [Update netty version](https://github.com/ballerina-platform/ballerina-library/issues/7358) - [Fix the issue of not being able to configure only server name in the secureSocket config](https://github.com/ballerina-platform/ballerina-library/issues/7443) +- [Fix rest path parameter generation with decoded path segments](https://github.com/ballerina-platform/ballerina-library/issues/7430) ## [2.12.0] - 2024-08-20 diff --git a/native/src/main/java/io/ballerina/stdlib/http/api/service/signature/AllPathParams.java b/native/src/main/java/io/ballerina/stdlib/http/api/service/signature/AllPathParams.java index afcaaa64ef..a6b92db8a0 100644 --- a/native/src/main/java/io/ballerina/stdlib/http/api/service/signature/AllPathParams.java +++ b/native/src/main/java/io/ballerina/stdlib/http/api/service/signature/AllPathParams.java @@ -33,6 +33,7 @@ import java.util.Collections; import java.util.List; import java.util.Map; +import java.util.stream.Stream; import static io.ballerina.stdlib.http.api.HttpConstants.EXTRA_PATH_INDEX; import static io.ballerina.stdlib.http.api.HttpConstants.PERCENTAGE; @@ -79,17 +80,16 @@ public void populateFeed(Object[] paramFeed, HttpCarbonMessage httpCarbonMessage if (argumentValue.endsWith(PERCENTAGE)) { argumentValue = argumentValue.replaceAll(PERCENTAGE, PERCENTAGE_ENCODED); } - argumentValue = URLDecoder.decode(argumentValue.replaceAll(PLUS_SIGN, PLUS_SIGN_ENCODED), - StandardCharsets.UTF_8); Object castedPathValue; try { Object parsedValue; if (pathParam.isArray()) { - String[] segments = argumentValue.substring(1).split(HttpConstants.SINGLE_SLASH); + String[] segments = Stream.of(argumentValue.substring(1).split(HttpConstants.SINGLE_SLASH)) + .map(AllPathParams::decodePathSegment).toArray(String[]::new); parsedValue = castParamArray(paramTypeTag, segments); } else { - parsedValue = castParam(paramTypeTag, argumentValue); + parsedValue = castParam(paramTypeTag, decodePathSegment(argumentValue)); } castedPathValue = ValueUtils.convert(parsedValue, paramType); } catch (Exception ex) { @@ -109,6 +109,10 @@ public void populateFeed(Object[] paramFeed, HttpCarbonMessage httpCarbonMessage } } + private static String decodePathSegment(String pathSegment) { + return URLDecoder.decode(pathSegment.replaceAll(PLUS_SIGN, PLUS_SIGN_ENCODED), StandardCharsets.UTF_8); + } + private static void updateWildcardToken(String wildcardToken, int wildCardIndex, Map> arguments) { if (wildcardToken == null) {