Skip to content

Commit 6415f12

Browse files
committed
Prevent duplicate encoding issue in HTTP request parameter rewrite filter
Signed-off-by: raccoonback <kosb15@naver.com>
1 parent ff2c7e7 commit 6415f12

File tree

2 files changed

+33
-6
lines changed

2 files changed

+33
-6
lines changed

spring-cloud-gateway-server/src/main/java/org/springframework/cloud/gateway/filter/factory/RewriteRequestParameterGatewayFilterFactory.java

Lines changed: 20 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -26,10 +26,14 @@
2626
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
2727
import org.springframework.http.server.reactive.ServerHttpRequest;
2828
import org.springframework.util.Assert;
29+
import org.springframework.util.LinkedMultiValueMap;
30+
import org.springframework.util.MultiValueMap;
2931
import org.springframework.web.server.ServerWebExchange;
3032
import org.springframework.web.util.UriComponentsBuilder;
33+
import org.springframework.web.util.UriUtils;
3134

3235
import static org.springframework.cloud.gateway.support.GatewayToStringStyler.filterToStringCreator;
36+
import static org.springframework.util.CollectionUtils.unmodifiableMultiValueMap;
3337

3438
/**
3539
* @author Fredrich Ombico
@@ -59,14 +63,25 @@ public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
5963
ServerHttpRequest req = exchange.getRequest();
6064

6165
UriComponentsBuilder uriComponentsBuilder = UriComponentsBuilder.fromUri(req.getURI());
62-
if (req.getQueryParams().containsKey(config.getName())) {
63-
uriComponentsBuilder.replaceQueryParam(config.getName(), config.getReplacement());
66+
67+
MultiValueMap<String, String> queryParams = new LinkedMultiValueMap<>(req.getQueryParams());
68+
if (queryParams.containsKey(config.getName())) {
69+
queryParams.remove(config.getName());
70+
queryParams.add(config.getName(), config.getReplacement());
6471
}
6572

66-
URI uri = uriComponentsBuilder.build().toUri();
67-
ServerHttpRequest request = req.mutate().uri(uri).build();
73+
try {
74+
MultiValueMap<String, String> encodedQueryParams = UriUtils.encodeQueryParams(queryParams);
75+
URI uri = uriComponentsBuilder.replaceQueryParams(unmodifiableMultiValueMap(encodedQueryParams))
76+
.build(true)
77+
.toUri();
6878

69-
return chain.filter(exchange.mutate().request(request).build());
79+
ServerHttpRequest request = req.mutate().uri(uri).build();
80+
return chain.filter(exchange.mutate().request(request).build());
81+
}
82+
catch (IllegalArgumentException ex) {
83+
throw new IllegalStateException("Invalid URI query: \"" + queryParams + "\"");
84+
}
7085
}
7186

7287
@Override

spring-cloud-gateway-server/src/test/java/org/springframework/cloud/gateway/filter/factory/RewriteRequestParameterGatewayFilterFactoryTests.java

Lines changed: 13 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -71,11 +71,23 @@ void rewriteRequestParameterFilterDoesNotAddParamIfNameNotFound() {
7171
}
7272

7373
@Test
74-
void rewriteRequestParameterFilterWorksWithSpecialCharacters() {
74+
void rewriteRequestParameterFilterWithSpecialCharactersInParameterValue() {
7575
testRewriteRequestParameterFilter("campaign", "black friday~(1.A-B_C!)", "campaign=old&color=green",
7676
Map.of("campaign", List.of("black friday~(1.A-B_C!)"), "color", List.of("green")));
7777
}
7878

79+
@Test
80+
void rewriteRequestParameterFilterWithSpecialCharactersInParameterName() {
81+
testRewriteRequestParameterFilter("campaign[]", "red", "campaign%5B%5D=blue&color=green",
82+
Map.of("campaign[]", List.of("red"), "color", List.of("green")));
83+
}
84+
85+
@Test
86+
void rewriteRequestParameterFilterKeepsOtherParamsEncoded() {
87+
testRewriteRequestParameterFilter("color", "white", "campaign%5B%5D=blue&color=green",
88+
Map.of("campaign[]", List.of("blue"), "color", List.of("white")));
89+
}
90+
7991
private void testRewriteRequestParameterFilter(String name, String replacement, String query,
8092
Map<String, List<String>> expectedQueryParams) {
8193
GatewayFilter filter = new RewriteRequestParameterGatewayFilterFactory()

0 commit comments

Comments
 (0)