Support `@RequestPart` for `@HttpExchange` methods
Closes gh-29420
This commit is contained in:
parent
4b647a1801
commit
481389f761
|
|
@ -453,6 +453,11 @@ method parameters:
|
||||||
parameters are encoded in the request body. Otherwise, they are added as URL query
|
parameters are encoded in the request body. Otherwise, they are added as URL query
|
||||||
parameters.
|
parameters.
|
||||||
|
|
||||||
|
| `@RequestPart`
|
||||||
|
| Add a request part, which may be a String (form field), `Resource` (file part),
|
||||||
|
Object (entity to be encoded, e.g. as JSON), `HttpEntity` (part content and headers),
|
||||||
|
a Spring `Part`, or Reactive Streams `Publisher` of any of the above.
|
||||||
|
|
||||||
| `@CookieValue`
|
| `@CookieValue`
|
||||||
| Add a cookie or mutliple cookies. The argument may be a `Map<String, ?>` or
|
| Add a cookie or mutliple cookies. The argument may be a `Map<String, ?>` or
|
||||||
`MultiValueMap<String, ?>` with multiple cookies, a `Collection<?>` of values, or an
|
`MultiValueMap<String, ?>` with multiple cookies, a `Collection<?>` of values, or an
|
||||||
|
|
|
||||||
|
|
@ -24,6 +24,7 @@ import java.lang.annotation.Target;
|
||||||
|
|
||||||
import org.springframework.aot.hint.annotation.Reflective;
|
import org.springframework.aot.hint.annotation.Reflective;
|
||||||
import org.springframework.core.annotation.AliasFor;
|
import org.springframework.core.annotation.AliasFor;
|
||||||
|
import org.springframework.http.HttpEntity;
|
||||||
import org.springframework.web.bind.annotation.Mapping;
|
import org.springframework.web.bind.annotation.Mapping;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -92,6 +93,17 @@ import org.springframework.web.bind.annotation.Mapping;
|
||||||
* RequestParamArgumentResolver}</td>
|
* RequestParamArgumentResolver}</td>
|
||||||
* </tr>
|
* </tr>
|
||||||
* <tr>
|
* <tr>
|
||||||
|
* <td>{@link org.springframework.web.bind.annotation.RequestPart @RequestPart}</td>
|
||||||
|
* <td>Add a request part, which may be a String (form field),
|
||||||
|
* {@link org.springframework.core.io.Resource} (file part), Object (entity to be
|
||||||
|
* encoded, e.g. as JSON), {@link HttpEntity} (part content and headers), a
|
||||||
|
* {@link org.springframework.http.codec.multipart.Part}, or a
|
||||||
|
* {@link org.reactivestreams.Publisher} of any of the above.
|
||||||
|
* (</td>
|
||||||
|
* <td>{@link org.springframework.web.service.invoker.RequestPartArgumentResolver
|
||||||
|
* RequestPartArgumentResolver}</td>
|
||||||
|
* </tr>
|
||||||
|
* <tr>
|
||||||
* <td>{@link org.springframework.web.bind.annotation.CookieValue @CookieValue}</td>
|
* <td>{@link org.springframework.web.bind.annotation.CookieValue @CookieValue}</td>
|
||||||
* <td>Add a cookie</td>
|
* <td>Add a cookie</td>
|
||||||
* <td>{@link org.springframework.web.service.invoker.CookieValueArgumentResolver
|
* <td>{@link org.springframework.web.service.invoker.CookieValueArgumentResolver
|
||||||
|
|
|
||||||
|
|
@ -29,9 +29,11 @@ import java.util.function.Function;
|
||||||
import org.reactivestreams.Publisher;
|
import org.reactivestreams.Publisher;
|
||||||
|
|
||||||
import org.springframework.core.ParameterizedTypeReference;
|
import org.springframework.core.ParameterizedTypeReference;
|
||||||
|
import org.springframework.core.ResolvableType;
|
||||||
import org.springframework.http.HttpHeaders;
|
import org.springframework.http.HttpHeaders;
|
||||||
import org.springframework.http.HttpMethod;
|
import org.springframework.http.HttpMethod;
|
||||||
import org.springframework.http.MediaType;
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.http.client.MultipartBodyBuilder;
|
||||||
import org.springframework.http.codec.FormHttpMessageWriter;
|
import org.springframework.http.codec.FormHttpMessageWriter;
|
||||||
import org.springframework.lang.Nullable;
|
import org.springframework.lang.Nullable;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
|
|
@ -222,6 +224,9 @@ public final class HttpRequestValues {
|
||||||
@Nullable
|
@Nullable
|
||||||
private MultiValueMap<String, String> requestParams;
|
private MultiValueMap<String, String> requestParams;
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private MultipartBodyBuilder multipartBuilder;
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
private Map<String, Object> attributes;
|
private Map<String, Object> attributes;
|
||||||
|
|
||||||
|
|
@ -335,6 +340,26 @@ public final class HttpRequestValues {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a part to a multipart request. The part value may be as described
|
||||||
|
* in {@link MultipartBodyBuilder#part(String, Object)}.
|
||||||
|
*/
|
||||||
|
public Builder addRequestPart(String name, Object part) {
|
||||||
|
this.multipartBuilder = (this.multipartBuilder != null ? this.multipartBuilder : new MultipartBodyBuilder());
|
||||||
|
this.multipartBuilder.part(name, part);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Variant of {@link #addRequestPart(String, Object)} that allows the
|
||||||
|
* part value to be produced by a {@link Publisher}.
|
||||||
|
*/
|
||||||
|
public <T, P extends Publisher<T>> Builder addRequestPart(String name, P publisher, ResolvableType type) {
|
||||||
|
this.multipartBuilder = (this.multipartBuilder != null ? this.multipartBuilder : new MultipartBodyBuilder());
|
||||||
|
this.multipartBuilder.asyncPart(name, publisher, ParameterizedTypeReference.forType(type.getType()));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configure an attribute to associate with the request.
|
* Configure an attribute to associate with the request.
|
||||||
* @param name the attribute name
|
* @param name the attribute name
|
||||||
|
|
@ -399,6 +424,10 @@ public final class HttpRequestValues {
|
||||||
uriTemplate = appendQueryParams(uriTemplate, uriVars, this.requestParams);
|
uriTemplate = appendQueryParams(uriTemplate, uriVars, this.requestParams);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (this.multipartBuilder != null) {
|
||||||
|
Assert.isTrue(bodyValue == null && this.body == null, "Expected body or request parts, not both");
|
||||||
|
bodyValue = this.multipartBuilder.build();
|
||||||
|
}
|
||||||
|
|
||||||
HttpHeaders headers = HttpHeaders.EMPTY;
|
HttpHeaders headers = HttpHeaders.EMPTY;
|
||||||
if (this.headers != null) {
|
if (this.headers != null) {
|
||||||
|
|
|
||||||
|
|
@ -323,6 +323,7 @@ public final class HttpServiceProxyFactory implements InitializingBean, Embedded
|
||||||
resolvers.add(new RequestBodyArgumentResolver(this.reactiveAdapterRegistry));
|
resolvers.add(new RequestBodyArgumentResolver(this.reactiveAdapterRegistry));
|
||||||
resolvers.add(new PathVariableArgumentResolver(service));
|
resolvers.add(new PathVariableArgumentResolver(service));
|
||||||
resolvers.add(new RequestParamArgumentResolver(service));
|
resolvers.add(new RequestParamArgumentResolver(service));
|
||||||
|
resolvers.add(new RequestPartArgumentResolver(this.reactiveAdapterRegistry));
|
||||||
resolvers.add(new CookieValueArgumentResolver(service));
|
resolvers.add(new CookieValueArgumentResolver(service));
|
||||||
resolvers.add(new RequestAttributeArgumentResolver());
|
resolvers.add(new RequestAttributeArgumentResolver());
|
||||||
|
|
||||||
|
|
@ -497,6 +498,7 @@ public final class HttpServiceProxyFactory implements InitializingBean, Embedded
|
||||||
resolvers.add(new RequestBodyArgumentResolver(this.reactiveAdapterRegistry));
|
resolvers.add(new RequestBodyArgumentResolver(this.reactiveAdapterRegistry));
|
||||||
resolvers.add(new PathVariableArgumentResolver(conversionService));
|
resolvers.add(new PathVariableArgumentResolver(conversionService));
|
||||||
resolvers.add(new RequestParamArgumentResolver(conversionService));
|
resolvers.add(new RequestParamArgumentResolver(conversionService));
|
||||||
|
resolvers.add(new RequestPartArgumentResolver(this.reactiveAdapterRegistry));
|
||||||
resolvers.add(new CookieValueArgumentResolver(conversionService));
|
resolvers.add(new CookieValueArgumentResolver(conversionService));
|
||||||
resolvers.add(new RequestAttributeArgumentResolver());
|
resolvers.add(new RequestAttributeArgumentResolver());
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,81 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2022 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.springframework.web.service.invoker;
|
||||||
|
|
||||||
|
import org.reactivestreams.Publisher;
|
||||||
|
|
||||||
|
import org.springframework.core.MethodParameter;
|
||||||
|
import org.springframework.core.ReactiveAdapter;
|
||||||
|
import org.springframework.core.ReactiveAdapterRegistry;
|
||||||
|
import org.springframework.core.ResolvableType;
|
||||||
|
import org.springframework.http.HttpEntity;
|
||||||
|
import org.springframework.http.codec.multipart.Part;
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
import org.springframework.web.bind.annotation.RequestPart;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link HttpServiceArgumentResolver} for {@link RequestPart @RequestPart}
|
||||||
|
* annotated arguments.
|
||||||
|
*
|
||||||
|
* <p>The argument may be:
|
||||||
|
* <ul>
|
||||||
|
* <li>String -- form field
|
||||||
|
* <li>{@link org.springframework.core.io.Resource Resource} -- file part
|
||||||
|
* <li>Object -- content to be encoded (e.g. to JSON)
|
||||||
|
* <li>{@link HttpEntity} -- part content and headers although generally it's
|
||||||
|
* easier to add headers through the returned builder
|
||||||
|
* <li>{@link Part} -- a part from a server request
|
||||||
|
* <li>{@link Publisher} of any of the above
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @author Rossen Stoyanchev
|
||||||
|
* @since 6.0
|
||||||
|
*/
|
||||||
|
public class RequestPartArgumentResolver extends AbstractNamedValueArgumentResolver {
|
||||||
|
|
||||||
|
private final ReactiveAdapterRegistry reactiveAdapterRegistry;
|
||||||
|
|
||||||
|
|
||||||
|
public RequestPartArgumentResolver(ReactiveAdapterRegistry reactiveAdapterRegistry) {
|
||||||
|
this.reactiveAdapterRegistry = reactiveAdapterRegistry;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected NamedValueInfo createNamedValueInfo(MethodParameter parameter) {
|
||||||
|
RequestPart annot = parameter.getParameterAnnotation(RequestPart.class);
|
||||||
|
return (annot == null ? null :
|
||||||
|
new NamedValueInfo(annot.name(), annot.required(), null, "request part", true));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void addRequestValue(
|
||||||
|
String name, Object value, MethodParameter parameter, HttpRequestValues.Builder requestValues) {
|
||||||
|
|
||||||
|
Class<?> type = parameter.getParameterType();
|
||||||
|
ReactiveAdapter adapter = this.reactiveAdapterRegistry.getAdapter(type);
|
||||||
|
if (adapter != null) {
|
||||||
|
Assert.isTrue(!adapter.isNoValue(), "Expected publisher that produces a value");
|
||||||
|
Publisher<?> publisher = adapter.toPublisher(value);
|
||||||
|
requestValues.addRequestPart(name, publisher, ResolvableType.forMethodParameter(parameter.nested()));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
requestValues.addRequestPart(name, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,83 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2022 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.springframework.web.service.invoker;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
import org.springframework.http.HttpEntity;
|
||||||
|
import org.springframework.http.HttpHeaders;
|
||||||
|
import org.springframework.util.MultiValueMap;
|
||||||
|
import org.springframework.web.bind.annotation.RequestPart;
|
||||||
|
import org.springframework.web.service.annotation.PostExchange;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unit tests for {@link RequestPartArgumentResolver}.
|
||||||
|
*
|
||||||
|
* <p>Additional tests for this resolver:
|
||||||
|
* <ul>
|
||||||
|
* <li>Base class functionality in {@link NamedValueArgumentResolverTests}
|
||||||
|
* <li>Form data vs query params in {@link HttpRequestValuesTests}
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @author Rossen Stoyanchev
|
||||||
|
*/
|
||||||
|
public class RequestPartArgumentResolverTests {
|
||||||
|
|
||||||
|
private final TestHttpClientAdapter client = new TestHttpClientAdapter();
|
||||||
|
|
||||||
|
private Service service;
|
||||||
|
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setUp() throws Exception {
|
||||||
|
HttpServiceProxyFactory proxyFactory = HttpServiceProxyFactory.builder(this.client).build();
|
||||||
|
this.service = proxyFactory.createClient(Service.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Base class functionality should be tested in NamedValueArgumentResolverTests.
|
||||||
|
// Form data vs query params tested in HttpRequestValuesTests.
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void requestPart() {
|
||||||
|
HttpHeaders headers = new HttpHeaders();
|
||||||
|
headers.add("foo", "bar");
|
||||||
|
HttpEntity<String> part2 = new HttpEntity<>("part 2", headers);
|
||||||
|
this.service.postMultipart("part 1", part2, Mono.just("part 3"));
|
||||||
|
|
||||||
|
Object body = this.client.getRequestValues().getBodyValue();
|
||||||
|
assertThat(body).isNotNull().isInstanceOf(MultiValueMap.class);
|
||||||
|
MultiValueMap<String, HttpEntity<?>> map = (MultiValueMap<String, HttpEntity<?>>) body;
|
||||||
|
|
||||||
|
assertThat(map.getFirst("part1").getBody()).isEqualTo("part 1");
|
||||||
|
assertThat(map.getFirst("part2")).isEqualTo(part2);
|
||||||
|
assertThat(((Mono<?>) map.getFirst("part3").getBody()).block()).isEqualTo("part 3");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private interface Service {
|
||||||
|
|
||||||
|
@PostExchange
|
||||||
|
void postMultipart(@RequestPart String part1, @RequestPart HttpEntity<String> part2, @RequestPart Mono<String> part3);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue