Return HandlerResult in HandlerAdapter#handle()
This commit updates HandlerAdapter#handle() to return HandlerResult instead of Publisher<HandlerResult>. A new SimpleHandlerResultHandler class has been introduced for handlers returning Publisher<Void>.
This commit is contained in:
parent
49fc32e214
commit
9516c9992f
|
@ -91,9 +91,9 @@ public class DispatcherHandler implements HttpHandler, ApplicationContextAware {
|
||||||
}
|
}
|
||||||
|
|
||||||
HandlerAdapter handlerAdapter = getHandlerAdapter(handler);
|
HandlerAdapter handlerAdapter = getHandlerAdapter(handler);
|
||||||
Publisher<HandlerResult> resultPublisher = handlerAdapter.handle(request, response, handler);
|
|
||||||
|
|
||||||
return Streams.wrap(resultPublisher).concatMap((HandlerResult result) -> {
|
try {
|
||||||
|
HandlerResult result = handlerAdapter.handle(request, response, handler);
|
||||||
for (HandlerResultHandler resultHandler : resultHandlers) {
|
for (HandlerResultHandler resultHandler : resultHandlers) {
|
||||||
if (resultHandler.supports(result)) {
|
if (resultHandler.supports(result)) {
|
||||||
return resultHandler.handleResult(request, response, result);
|
return resultHandler.handleResult(request, response, result);
|
||||||
|
@ -101,7 +101,11 @@ public class DispatcherHandler implements HttpHandler, ApplicationContextAware {
|
||||||
}
|
}
|
||||||
return Streams.fail(new IllegalStateException(
|
return Streams.fail(new IllegalStateException(
|
||||||
"No HandlerResultHandler for " + result.getValue()));
|
"No HandlerResultHandler for " + result.getValue()));
|
||||||
});
|
}
|
||||||
|
catch(Exception ex) {
|
||||||
|
return Streams.fail(ex);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Object getHandler(ServerHttpRequest request) {
|
protected Object getHandler(ServerHttpRequest request) {
|
||||||
|
|
|
@ -15,8 +15,6 @@
|
||||||
*/
|
*/
|
||||||
package org.springframework.reactive.web.dispatch;
|
package org.springframework.reactive.web.dispatch;
|
||||||
|
|
||||||
import org.reactivestreams.Publisher;
|
|
||||||
|
|
||||||
import org.springframework.reactive.web.http.ServerHttpRequest;
|
import org.springframework.reactive.web.http.ServerHttpRequest;
|
||||||
import org.springframework.reactive.web.http.ServerHttpResponse;
|
import org.springframework.reactive.web.http.ServerHttpResponse;
|
||||||
|
|
||||||
|
@ -27,6 +25,6 @@ public interface HandlerAdapter {
|
||||||
|
|
||||||
boolean supports(Object handler);
|
boolean supports(Object handler);
|
||||||
|
|
||||||
Publisher<HandlerResult> handle(ServerHttpRequest request, ServerHttpResponse response, Object handler);
|
HandlerResult handle(ServerHttpRequest request, ServerHttpResponse response, Object handler) throws Exception;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2015 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.reactive.web.dispatch;
|
||||||
|
|
||||||
|
import org.reactivestreams.Publisher;
|
||||||
|
|
||||||
|
import org.springframework.core.Ordered;
|
||||||
|
import org.springframework.reactive.web.http.ServerHttpRequest;
|
||||||
|
import org.springframework.reactive.web.http.ServerHttpResponse;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Supports {@link HandlerResult} with a {@code Publisher<Void>} value.
|
||||||
|
*
|
||||||
|
* @author Sebastien Deleuze
|
||||||
|
*/
|
||||||
|
public class SimpleHandlerResultHandler implements Ordered, HandlerResultHandler {
|
||||||
|
|
||||||
|
private int order = Ordered.LOWEST_PRECEDENCE;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getOrder() {
|
||||||
|
return this.order;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supports(HandlerResult result) {
|
||||||
|
Object value = result.getValue();
|
||||||
|
return value != null && Publisher.class.isAssignableFrom(value.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Publisher<Void> handleResult(ServerHttpRequest request, ServerHttpResponse response, HandlerResult result) {
|
||||||
|
return (Publisher<Void>)result.getValue();
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,7 +16,6 @@
|
||||||
package org.springframework.reactive.web.dispatch.handler;
|
package org.springframework.reactive.web.dispatch.handler;
|
||||||
|
|
||||||
import org.reactivestreams.Publisher;
|
import org.reactivestreams.Publisher;
|
||||||
import reactor.rx.Streams;
|
|
||||||
|
|
||||||
import org.springframework.reactive.web.dispatch.HandlerAdapter;
|
import org.springframework.reactive.web.dispatch.HandlerAdapter;
|
||||||
import org.springframework.reactive.web.dispatch.HandlerResult;
|
import org.springframework.reactive.web.dispatch.HandlerResult;
|
||||||
|
@ -34,6 +33,7 @@ import org.springframework.reactive.web.http.ServerHttpResponse;
|
||||||
* handler mappings.
|
* handler mappings.
|
||||||
*
|
*
|
||||||
* @author Rossen Stoyanchev
|
* @author Rossen Stoyanchev
|
||||||
|
* @author Sebastien Deleuze
|
||||||
*/
|
*/
|
||||||
public class HttpHandlerAdapter implements HandlerAdapter {
|
public class HttpHandlerAdapter implements HandlerAdapter {
|
||||||
|
|
||||||
|
@ -44,10 +44,10 @@ public class HttpHandlerAdapter implements HandlerAdapter {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Publisher<HandlerResult> handle(ServerHttpRequest request, ServerHttpResponse response, Object handler) {
|
public HandlerResult handle(ServerHttpRequest request, ServerHttpResponse response, Object handler) {
|
||||||
HttpHandler httpHandler = (HttpHandler) handler;
|
HttpHandler httpHandler = (HttpHandler)handler;
|
||||||
Publisher<Void> publisher = httpHandler.handle(request, response);
|
Publisher<Void> completion = httpHandler.handle(request, response);
|
||||||
return Streams.wrap(publisher).map(aVoid -> null);
|
return new HandlerResult(httpHandler, completion);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,9 +19,6 @@ import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.reactivestreams.Publisher;
|
|
||||||
import reactor.rx.Streams;
|
|
||||||
|
|
||||||
import org.springframework.beans.factory.InitializingBean;
|
import org.springframework.beans.factory.InitializingBean;
|
||||||
import org.springframework.reactive.codec.decoder.JacksonJsonDecoder;
|
import org.springframework.reactive.codec.decoder.JacksonJsonDecoder;
|
||||||
import org.springframework.reactive.codec.decoder.JsonObjectDecoder;
|
import org.springframework.reactive.codec.decoder.JsonObjectDecoder;
|
||||||
|
@ -64,21 +61,15 @@ public class RequestMappingHandlerAdapter implements HandlerAdapter, Initializin
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Publisher<HandlerResult> handle(ServerHttpRequest request, ServerHttpResponse response,
|
public HandlerResult handle(ServerHttpRequest request, ServerHttpResponse response,
|
||||||
Object handler) {
|
Object handler) throws Exception {
|
||||||
|
|
||||||
final InvocableHandlerMethod invocable = new InvocableHandlerMethod((HandlerMethod) handler);
|
final InvocableHandlerMethod invocable = new InvocableHandlerMethod((HandlerMethod) handler);
|
||||||
invocable.setHandlerMethodArgumentResolvers(this.argumentResolvers);
|
invocable.setHandlerMethodArgumentResolvers(this.argumentResolvers);
|
||||||
|
|
||||||
Object result;
|
Object result = invocable.invokeForRequest(request);
|
||||||
try {
|
|
||||||
result = invocable.invokeForRequest(request);
|
|
||||||
}
|
|
||||||
catch (Exception ex) {
|
|
||||||
return Streams.fail(ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Streams.just(new HandlerResult(invocable, result));
|
return new HandlerResult(invocable, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
package org.springframework.reactive.web.dispatch.method.annotation;
|
package org.springframework.reactive.web.dispatch.method.annotation;
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.charset.Charset;
|
import java.nio.charset.Charset;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
@ -25,7 +26,6 @@ import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
import org.reactivestreams.Publisher;
|
import org.reactivestreams.Publisher;
|
||||||
import reactor.rx.Promise;
|
import reactor.rx.Promise;
|
||||||
import reactor.rx.Stream;
|
|
||||||
import reactor.rx.Streams;
|
import reactor.rx.Streams;
|
||||||
import rx.Observable;
|
import rx.Observable;
|
||||||
import rx.RxReactiveStreams;
|
import rx.RxReactiveStreams;
|
||||||
|
@ -33,6 +33,7 @@ import rx.Single;
|
||||||
|
|
||||||
import org.springframework.core.MethodParameter;
|
import org.springframework.core.MethodParameter;
|
||||||
import org.springframework.core.Ordered;
|
import org.springframework.core.Ordered;
|
||||||
|
import org.springframework.core.ParameterizedTypeReference;
|
||||||
import org.springframework.core.ResolvableType;
|
import org.springframework.core.ResolvableType;
|
||||||
import org.springframework.core.annotation.AnnotatedElementUtils;
|
import org.springframework.core.annotation.AnnotatedElementUtils;
|
||||||
import org.springframework.http.HttpHeaders;
|
import org.springframework.http.HttpHeaders;
|
||||||
|
@ -60,7 +61,7 @@ public class ResponseBodyResultHandler implements HandlerResultHandler, Ordered
|
||||||
private final List<MessageToByteEncoder<?>> serializers;
|
private final List<MessageToByteEncoder<?>> serializers;
|
||||||
private final List<MessageToByteEncoder<ByteBuffer>> postProcessors;
|
private final List<MessageToByteEncoder<ByteBuffer>> postProcessors;
|
||||||
|
|
||||||
private int order = Ordered.LOWEST_PRECEDENCE;
|
private int order = 0;
|
||||||
|
|
||||||
|
|
||||||
public ResponseBodyResultHandler(List<MessageToByteEncoder<?>> serializers) {
|
public ResponseBodyResultHandler(List<MessageToByteEncoder<?>> serializers) {
|
||||||
|
@ -86,8 +87,10 @@ public class ResponseBodyResultHandler implements HandlerResultHandler, Ordered
|
||||||
public boolean supports(HandlerResult result) {
|
public boolean supports(HandlerResult result) {
|
||||||
Object handler = result.getHandler();
|
Object handler = result.getHandler();
|
||||||
if (handler instanceof HandlerMethod) {
|
if (handler instanceof HandlerMethod) {
|
||||||
Method method = ((HandlerMethod) handler).getMethod();
|
HandlerMethod handlerMethod = (HandlerMethod) handler;
|
||||||
return AnnotatedElementUtils.isAnnotated(method, ResponseBody.class.getName());
|
Type publisherVoidType = new ParameterizedTypeReference<Publisher<Void>>(){}.getType();
|
||||||
|
return AnnotatedElementUtils.isAnnotated(handlerMethod.getMethod(), ResponseBody.class.getName()) &&
|
||||||
|
!handlerMethod.getReturnType().getGenericParameterType().equals(publisherVoidType);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,7 @@ import reactor.rx.Streams;
|
||||||
import org.springframework.http.RequestEntity;
|
import org.springframework.http.RequestEntity;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.reactive.web.dispatch.DispatcherHandler;
|
import org.springframework.reactive.web.dispatch.DispatcherHandler;
|
||||||
|
import org.springframework.reactive.web.dispatch.SimpleHandlerResultHandler;
|
||||||
import org.springframework.reactive.web.http.AbstractHttpHandlerIntegrationTests;
|
import org.springframework.reactive.web.http.AbstractHttpHandlerIntegrationTests;
|
||||||
import org.springframework.reactive.web.http.HttpHandler;
|
import org.springframework.reactive.web.http.HttpHandler;
|
||||||
import org.springframework.reactive.web.http.ServerHttpRequest;
|
import org.springframework.reactive.web.http.ServerHttpRequest;
|
||||||
|
@ -52,6 +53,7 @@ public class SimpleUrlHandlerMappingIntegrationTests extends AbstractHttpHandler
|
||||||
StaticWebApplicationContext wac = new StaticWebApplicationContext();
|
StaticWebApplicationContext wac = new StaticWebApplicationContext();
|
||||||
wac.registerSingleton("hm", TestHandlerMapping.class);
|
wac.registerSingleton("hm", TestHandlerMapping.class);
|
||||||
wac.registerSingleton("ha", HttpHandlerAdapter.class);
|
wac.registerSingleton("ha", HttpHandlerAdapter.class);
|
||||||
|
wac.registerSingleton("hhrh", SimpleHandlerResultHandler.class);
|
||||||
wac.refresh();
|
wac.refresh();
|
||||||
|
|
||||||
DispatcherHandler dispatcherHandler = new DispatcherHandler();
|
DispatcherHandler dispatcherHandler = new DispatcherHandler();
|
||||||
|
|
|
@ -0,0 +1,68 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2015 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.reactive.web.dispatch.method.annotation;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.reactivestreams.Publisher;
|
||||||
|
|
||||||
|
import org.springframework.reactive.web.dispatch.HandlerResult;
|
||||||
|
import org.springframework.web.bind.annotation.ResponseBody;
|
||||||
|
import org.springframework.web.method.HandlerMethod;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Sebastien Deleuze
|
||||||
|
*/
|
||||||
|
public class ResponseBodyResultHandlerTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void supports() throws NoSuchMethodException {
|
||||||
|
ResponseBodyResultHandler resultHandler = new ResponseBodyResultHandler(Collections.emptyList());
|
||||||
|
TestController controller = new TestController();
|
||||||
|
|
||||||
|
HandlerMethod notAnnotatedMethod = new HandlerMethod(controller, TestController.class.getMethod("notAnnotated"));
|
||||||
|
assertFalse(resultHandler.supports(new HandlerResult(notAnnotatedMethod, null)));
|
||||||
|
|
||||||
|
HandlerMethod publisherStringMethod = new HandlerMethod(controller, TestController.class.getMethod("publisherString"));
|
||||||
|
assertTrue(resultHandler.supports(new HandlerResult(publisherStringMethod, null)));
|
||||||
|
|
||||||
|
HandlerMethod publisherVoidMethod = new HandlerMethod(controller, TestController.class.getMethod("publisherVoid"));
|
||||||
|
assertFalse(resultHandler.supports(new HandlerResult(publisherVoidMethod, null)));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class TestController {
|
||||||
|
|
||||||
|
public Publisher<String> notAnnotated() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@ResponseBody
|
||||||
|
public Publisher<String> publisherString() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@ResponseBody
|
||||||
|
public Publisher<Void> publisherVoid() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue