Add multipart support to ServerWebExchange
Issue: SPR-14546
This commit is contained in:
parent
8e272bc5b0
commit
4bfd04b3c5
|
|
@ -24,6 +24,7 @@ import java.util.function.Consumer;
|
|||
|
||||
import reactor.core.publisher.Mono;
|
||||
|
||||
import org.springframework.http.codec.multipart.Part;
|
||||
import org.springframework.http.server.reactive.ServerHttpRequest;
|
||||
import org.springframework.http.server.reactive.ServerHttpResponse;
|
||||
import org.springframework.util.MultiValueMap;
|
||||
|
|
@ -82,6 +83,12 @@ public interface ServerWebExchange {
|
|||
*/
|
||||
Mono<MultiValueMap<String, String>> getFormData();
|
||||
|
||||
/**
|
||||
* Return the form parts from the body of the request or an empty {@code Mono}
|
||||
* if the Content-Type is not "multipart/form-data".
|
||||
*/
|
||||
Mono<MultiValueMap<String, Part>> getMultipartData();
|
||||
|
||||
/**
|
||||
* Return a combined map that represents both
|
||||
* {@link ServerHttpRequest#getQueryParams()} and {@link #getFormData()}
|
||||
|
|
|
|||
|
|
@ -22,6 +22,7 @@ import java.util.Optional;
|
|||
|
||||
import reactor.core.publisher.Mono;
|
||||
|
||||
import org.springframework.http.codec.multipart.Part;
|
||||
import org.springframework.http.server.reactive.ServerHttpRequest;
|
||||
import org.springframework.http.server.reactive.ServerHttpResponse;
|
||||
import org.springframework.util.Assert;
|
||||
|
|
@ -93,6 +94,11 @@ public class ServerWebExchangeDecorator implements ServerWebExchange {
|
|||
return getDelegate().getFormData();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Mono<MultiValueMap<String, Part>> getMultipartData() {
|
||||
return getDelegate().getMultipartData();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Mono<MultiValueMap<String, String>> getRequestParams() {
|
||||
return getDelegate().getRequestParams();
|
||||
|
|
|
|||
|
|
@ -36,6 +36,7 @@ import org.springframework.http.InvalidMediaTypeException;
|
|||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.codec.HttpMessageReader;
|
||||
import org.springframework.http.codec.ServerCodecConfigurer;
|
||||
import org.springframework.http.codec.multipart.Part;
|
||||
import org.springframework.http.server.reactive.ServerHttpRequest;
|
||||
import org.springframework.http.server.reactive.ServerHttpResponse;
|
||||
import org.springframework.util.Assert;
|
||||
|
|
@ -48,6 +49,7 @@ import org.springframework.web.server.WebSession;
|
|||
import org.springframework.web.server.session.WebSessionManager;
|
||||
|
||||
import static org.springframework.http.MediaType.*;
|
||||
import static org.springframework.http.codec.multipart.MultipartHttpMessageReader.*;
|
||||
|
||||
/**
|
||||
* Default implementation of {@link ServerWebExchange}.
|
||||
|
|
@ -66,6 +68,10 @@ public class DefaultServerWebExchange implements ServerWebExchange {
|
|||
Mono.just(CollectionUtils.unmodifiableMultiValueMap(new LinkedMultiValueMap<String, String>(0)))
|
||||
.cache();
|
||||
|
||||
private static final Mono<MultiValueMap<String, Part>> EMPTY_MULTIPART_DATA =
|
||||
Mono.just(CollectionUtils.unmodifiableMultiValueMap(new LinkedMultiValueMap<String, Part>(0)))
|
||||
.cache();
|
||||
|
||||
|
||||
private final ServerHttpRequest request;
|
||||
|
||||
|
|
@ -77,6 +83,8 @@ public class DefaultServerWebExchange implements ServerWebExchange {
|
|||
|
||||
private final Mono<MultiValueMap<String, String>> formDataMono;
|
||||
|
||||
private final Mono<MultiValueMap<String, Part>> multipartDataMono;
|
||||
|
||||
private final Mono<MultiValueMap<String, String>> requestParamsMono;
|
||||
|
||||
private volatile boolean notModified;
|
||||
|
|
@ -97,6 +105,7 @@ public class DefaultServerWebExchange implements ServerWebExchange {
|
|||
this.response = response;
|
||||
this.sessionMono = sessionManager.getSession(this).cache();
|
||||
this.formDataMono = initFormData(request, codecConfigurer);
|
||||
this.multipartDataMono = initMultipartData(request, codecConfigurer);
|
||||
this.requestParamsMono = initRequestParams(request, this.formDataMono);
|
||||
|
||||
}
|
||||
|
|
@ -126,6 +135,31 @@ public class DefaultServerWebExchange implements ServerWebExchange {
|
|||
return EMPTY_FORM_DATA;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
private static Mono<MultiValueMap<String, Part>> initMultipartData(
|
||||
ServerHttpRequest request, ServerCodecConfigurer codecConfigurer) {
|
||||
|
||||
MediaType contentType;
|
||||
try {
|
||||
contentType = request.getHeaders().getContentType();
|
||||
if (MULTIPART_FORM_DATA.isCompatibleWith(contentType)) {
|
||||
return ((HttpMessageReader<MultiValueMap<String, Part>>)codecConfigurer
|
||||
.getReaders()
|
||||
.stream()
|
||||
.filter(messageReader -> messageReader.canRead(MULTIPART_VALUE_TYPE, MULTIPART_FORM_DATA))
|
||||
.findFirst()
|
||||
.orElseThrow(() -> new IllegalStateException("Could not find HttpMessageReader that supports " + MULTIPART_FORM_DATA)))
|
||||
.readMono(FORM_DATA_VALUE_TYPE, request, Collections.emptyMap())
|
||||
.switchIfEmpty(EMPTY_MULTIPART_DATA)
|
||||
.cache();
|
||||
}
|
||||
}
|
||||
catch (InvalidMediaTypeException ex) {
|
||||
// Ignore
|
||||
}
|
||||
return EMPTY_MULTIPART_DATA;
|
||||
}
|
||||
|
||||
private static Mono<MultiValueMap<String, String>> initRequestParams(
|
||||
ServerHttpRequest request, Mono<MultiValueMap<String, String>> formDataMono) {
|
||||
|
||||
|
|
@ -184,6 +218,11 @@ public class DefaultServerWebExchange implements ServerWebExchange {
|
|||
return this.formDataMono;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Mono<MultiValueMap<String, Part>> getMultipartData() {
|
||||
return this.multipartDataMono;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Mono<MultiValueMap<String, String>> getRequestParams() {
|
||||
return this.requestParamsMono;
|
||||
|
|
|
|||
|
|
@ -0,0 +1,120 @@
|
|||
/*
|
||||
* Copyright 2002-2017 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
|
||||
*
|
||||
* http://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.http.server.reactive;
|
||||
|
||||
import java.net.URI;
|
||||
import java.util.Optional;
|
||||
|
||||
import static org.junit.Assert.*;
|
||||
import org.junit.Test;
|
||||
import reactor.core.publisher.Mono;
|
||||
|
||||
import org.springframework.core.io.ClassPathResource;
|
||||
import org.springframework.core.io.buffer.DataBuffer;
|
||||
import org.springframework.http.HttpEntity;
|
||||
import org.springframework.http.HttpHeaders;
|
||||
import org.springframework.http.HttpStatus;
|
||||
import org.springframework.http.MediaType;
|
||||
import org.springframework.http.RequestEntity;
|
||||
import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.util.LinkedMultiValueMap;
|
||||
import org.springframework.util.MultiValueMap;
|
||||
import org.springframework.web.client.RestTemplate;
|
||||
import org.springframework.http.codec.multipart.Part;
|
||||
import org.springframework.web.server.ServerWebExchange;
|
||||
import org.springframework.web.server.WebHandler;
|
||||
import org.springframework.web.server.adapter.HttpWebHandlerAdapter;
|
||||
|
||||
public class MultipartIntegrationTests extends AbstractHttpHandlerIntegrationTests {
|
||||
|
||||
@Override
|
||||
protected HttpHandler createHttpHandler() {
|
||||
HttpWebHandlerAdapter handler = new HttpWebHandlerAdapter(new CheckRequestHandler());
|
||||
return handler;
|
||||
}
|
||||
|
||||
@Test
|
||||
public void getFormParts() throws Exception {
|
||||
RestTemplate restTemplate = new RestTemplate();
|
||||
RequestEntity<MultiValueMap<String, Object>> request = RequestEntity
|
||||
.post(new URI("http://localhost:" + port + "/form-parts"))
|
||||
.contentType(MediaType.MULTIPART_FORM_DATA)
|
||||
.body(generateBody());
|
||||
ResponseEntity<Void> response = restTemplate.exchange(request, Void.class);
|
||||
assertEquals(HttpStatus.OK, response.getStatusCode());
|
||||
}
|
||||
|
||||
private MultiValueMap<String, Object> generateBody() {
|
||||
HttpHeaders fooHeaders = new HttpHeaders();
|
||||
fooHeaders.setContentType(MediaType.TEXT_PLAIN);
|
||||
ClassPathResource fooResource = new ClassPathResource("org/springframework/http/codec/multipart/foo.txt");
|
||||
HttpEntity<ClassPathResource> fooPart = new HttpEntity<>(fooResource, fooHeaders);
|
||||
HttpEntity<String> barPart = new HttpEntity<>("bar");
|
||||
MultiValueMap<String, Object> parts = new LinkedMultiValueMap<>();
|
||||
parts.add("fooPart", fooPart);
|
||||
parts.add("barPart", barPart);
|
||||
return parts;
|
||||
}
|
||||
|
||||
public static class CheckRequestHandler implements WebHandler {
|
||||
|
||||
@Override
|
||||
public Mono<Void> handle(ServerWebExchange exchange) {
|
||||
|
||||
if (exchange.getRequest().getURI().getPath().equals("/form-parts")) {
|
||||
return assertGetFormParts(exchange);
|
||||
}
|
||||
return Mono.error(new AssertionError());
|
||||
}
|
||||
|
||||
private Mono<Void> assertGetFormParts(ServerWebExchange exchange) {
|
||||
return exchange
|
||||
.getMultipartData()
|
||||
.doOnNext(parts -> {
|
||||
assertEquals(2, parts.size());
|
||||
assertTrue(parts.containsKey("fooPart"));
|
||||
assertFooPart(parts.getFirst("fooPart"));
|
||||
assertTrue(parts.containsKey("barPart"));
|
||||
assertBarPart(parts.getFirst("barPart"));
|
||||
})
|
||||
.then();
|
||||
}
|
||||
|
||||
private void assertFooPart(Part part) {
|
||||
assertEquals("fooPart", part.getName());
|
||||
Optional<String> filename = part.getFilename();
|
||||
assertTrue(filename.isPresent());
|
||||
assertEquals("foo.txt", filename.get());
|
||||
DataBuffer buffer = part
|
||||
.getContent()
|
||||
.reduce((s1, s2) -> s1.write(s2))
|
||||
.block();
|
||||
assertEquals(12, buffer.readableByteCount());
|
||||
byte[] byteContent = new byte[12];
|
||||
buffer.read(byteContent);
|
||||
assertEquals("Lorem\nIpsum\n", new String(byteContent));
|
||||
}
|
||||
|
||||
private void assertBarPart(Part part) {
|
||||
assertEquals("barPart", part.getName());
|
||||
Optional<String> filename = part.getFilename();
|
||||
assertFalse(filename.isPresent());
|
||||
assertEquals("bar", part.getContentAsString().block());
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Reference in New Issue