Browse Source
HandlerMapping, HandlerAdapter, HandlerResultHandler (+ HandlerResult) as the basic request handling contracts. DispatcherHandler to drive overall request handling. DispatcherApp provides minimal implementations of the above contracts enough to put together a running example that returns 200 text/plain "Hello world".pull/1111/head
6 changed files with 520 additions and 0 deletions
@ -0,0 +1,169 @@
@@ -0,0 +1,169 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
import java.util.ArrayList; |
||||
import java.util.List; |
||||
|
||||
import org.reactivestreams.Publisher; |
||||
import org.reactivestreams.Subscriber; |
||||
import org.reactivestreams.Subscription; |
||||
|
||||
import org.springframework.beans.factory.BeanFactoryUtils; |
||||
import org.springframework.context.ApplicationContext; |
||||
import org.springframework.http.HttpStatus; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public class DispatcherHttpHandler implements HttpHandler { |
||||
|
||||
private List<HandlerMapping> handlerMappings; |
||||
|
||||
private List<HandlerAdapter> handlerAdapters; |
||||
|
||||
private List<HandlerResultHandler> resultHandlers; |
||||
|
||||
|
||||
protected void initStrategies(ApplicationContext context) { |
||||
|
||||
this.handlerMappings = new ArrayList<>(BeanFactoryUtils.beansOfTypeIncludingAncestors( |
||||
context, HandlerMapping.class, true, false).values()); |
||||
|
||||
this.handlerAdapters = new ArrayList<>(BeanFactoryUtils.beansOfTypeIncludingAncestors( |
||||
context, HandlerAdapter.class, true, false).values()); |
||||
|
||||
this.resultHandlers = new ArrayList<>(BeanFactoryUtils.beansOfTypeIncludingAncestors( |
||||
context, HandlerResultHandler.class, true, false).values()); |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public Publisher<Void> handle(ServerHttpRequest request, ServerHttpResponse response) { |
||||
|
||||
Object handler = getHandler(request); |
||||
if (handler == null) { |
||||
// No exception handling mechanism yet
|
||||
response.setStatusCode(HttpStatus.NOT_FOUND); |
||||
return Publishers.complete(); |
||||
} |
||||
|
||||
HandlerAdapter handlerAdapter = getHandlerAdapter(handler); |
||||
final Publisher<HandlerResult> resultPublisher = handlerAdapter.handle(request, response, handler); |
||||
|
||||
return new Publisher<Void>() { |
||||
|
||||
@Override |
||||
public void subscribe(final Subscriber<? super Void> subscriber) { |
||||
|
||||
resultPublisher.subscribe(new Subscriber<HandlerResult>() { |
||||
|
||||
@Override |
||||
public void onSubscribe(Subscription subscription) { |
||||
subscription.request(Long.MAX_VALUE); |
||||
} |
||||
|
||||
@Override |
||||
public void onNext(HandlerResult result) { |
||||
for (HandlerResultHandler resultHandler : resultHandlers) { |
||||
if (resultHandler.supports(result)) { |
||||
Publisher<Void> publisher = resultHandler.handleResult(request, response, result); |
||||
publisher.subscribe(new Subscriber<Void>() { |
||||
@Override |
||||
public void onSubscribe(Subscription subscription) { |
||||
subscription.request(Long.MAX_VALUE); |
||||
} |
||||
|
||||
@Override |
||||
public void onNext(Void aVoid) { |
||||
// no op
|
||||
} |
||||
|
||||
@Override |
||||
public void onError(Throwable error) { |
||||
// Result handling error (no exception handling mechanism yet)
|
||||
subscriber.onError(error); |
||||
} |
||||
|
||||
@Override |
||||
public void onComplete() { |
||||
subscriber.onComplete(); |
||||
} |
||||
}); |
||||
} |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void onError(Throwable error) { |
||||
// Application handler error (no exception handling mechanism yet)
|
||||
subscriber.onError(error); |
||||
} |
||||
|
||||
@Override |
||||
public void onComplete() { |
||||
// do nothing
|
||||
} |
||||
}); |
||||
} |
||||
}; |
||||
} |
||||
|
||||
protected Object getHandler(ServerHttpRequest request) { |
||||
Object handler = null; |
||||
for (HandlerMapping handlerMapping : this.handlerMappings) { |
||||
handler = handlerMapping.getHandler(request); |
||||
if (handler != null) { |
||||
break; |
||||
} |
||||
} |
||||
return handler; |
||||
} |
||||
|
||||
protected HandlerAdapter getHandlerAdapter(Object handler) { |
||||
for (HandlerAdapter handlerAdapter : this.handlerAdapters) { |
||||
if (handlerAdapter.supports(handler)) { |
||||
return handlerAdapter; |
||||
} |
||||
} |
||||
// more specific exception
|
||||
throw new IllegalStateException("No HandlerAdapter for " + handler); |
||||
} |
||||
|
||||
|
||||
private static class Publishers { |
||||
|
||||
|
||||
public static Publisher<Void> complete() { |
||||
return subscriber -> { |
||||
subscriber.onSubscribe(new NoopSubscription()); |
||||
subscriber.onComplete(); |
||||
}; |
||||
} |
||||
} |
||||
|
||||
private static class NoopSubscription implements Subscription { |
||||
|
||||
@Override |
||||
public void request(long n) { |
||||
} |
||||
|
||||
@Override |
||||
public void cancel() { |
||||
} |
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,29 @@
@@ -0,0 +1,29 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
import org.reactivestreams.Publisher; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public interface HandlerAdapter { |
||||
|
||||
boolean supports(Object handler); |
||||
|
||||
Publisher<HandlerResult> handle(ServerHttpRequest request, ServerHttpResponse response, Object handler); |
||||
|
||||
} |
||||
@ -0,0 +1,25 @@
@@ -0,0 +1,25 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public interface HandlerMapping { |
||||
|
||||
Object getHandler(ServerHttpRequest request); |
||||
|
||||
} |
||||
@ -0,0 +1,35 @@
@@ -0,0 +1,35 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public class HandlerResult { |
||||
|
||||
private final Object returnValue; |
||||
|
||||
|
||||
public HandlerResult(Object returnValue) { |
||||
this.returnValue = returnValue; |
||||
} |
||||
|
||||
|
||||
public Object getReturnValue() { |
||||
return this.returnValue; |
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,29 @@
@@ -0,0 +1,29 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
import org.reactivestreams.Publisher; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public interface HandlerResultHandler { |
||||
|
||||
boolean supports(HandlerResult result); |
||||
|
||||
Publisher<Void> handleResult(ServerHttpRequest request, ServerHttpResponse response, HandlerResult result); |
||||
|
||||
} |
||||
@ -0,0 +1,233 @@
@@ -0,0 +1,233 @@
|
||||
/* |
||||
* 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; |
||||
|
||||
import java.nio.charset.Charset; |
||||
import java.util.HashMap; |
||||
import java.util.Map; |
||||
|
||||
import io.netty.buffer.ByteBuf; |
||||
import io.reactivex.netty.protocol.http.server.HttpServer; |
||||
import org.reactivestreams.Publisher; |
||||
import org.reactivestreams.Subscriber; |
||||
import org.reactivestreams.Subscription; |
||||
|
||||
import org.springframework.http.MediaType; |
||||
import org.springframework.reactive.web.rxnetty.RequestHandlerAdapter; |
||||
import org.springframework.web.context.support.StaticWebApplicationContext; |
||||
|
||||
/** |
||||
* @author Rossen Stoyanchev |
||||
*/ |
||||
public class DispatcherApp { |
||||
|
||||
public static void main(String[] args) { |
||||
|
||||
StaticWebApplicationContext wac = new StaticWebApplicationContext(); |
||||
wac.registerSingleton("handlerMapping", SimpleUrlHandlerMapping.class); |
||||
wac.registerSingleton("handlerAdapter", PlainTextHandlerAdapter.class); |
||||
wac.registerSingleton("resultHandler", PlainTextResultHandler.class); |
||||
wac.refresh(); |
||||
|
||||
SimpleUrlHandlerMapping handlerMapping = wac.getBean(SimpleUrlHandlerMapping.class); |
||||
handlerMapping.addHandler("/text", new HelloWorldTextHandler()); |
||||
|
||||
DispatcherHttpHandler dispatcherHandler = new DispatcherHttpHandler(); |
||||
dispatcherHandler.initStrategies(wac); |
||||
|
||||
RequestHandlerAdapter requestHandler = new RequestHandlerAdapter(dispatcherHandler); |
||||
HttpServer<ByteBuf, ByteBuf> server = HttpServer.newServer(8080); |
||||
server.start(requestHandler::handle); |
||||
server.awaitShutdown(); |
||||
} |
||||
|
||||
|
||||
private static class SimpleUrlHandlerMapping implements HandlerMapping { |
||||
|
||||
private final Map<String, Object> handlerMap = new HashMap<>(); |
||||
|
||||
|
||||
public void addHandler(String path, Object handler) { |
||||
this.handlerMap.put(path, handler); |
||||
} |
||||
|
||||
@Override |
||||
public Object getHandler(ServerHttpRequest request) { |
||||
return this.handlerMap.get(request.getURI().getPath()); |
||||
} |
||||
} |
||||
|
||||
private interface PlainTextHandler { |
||||
|
||||
Publisher<String> handle(ServerHttpRequest request, ServerHttpResponse response); |
||||
|
||||
} |
||||
|
||||
private static class HelloWorldTextHandler implements PlainTextHandler { |
||||
|
||||
@Override |
||||
public Publisher<String> handle(ServerHttpRequest request, ServerHttpResponse response) { |
||||
|
||||
return new Publisher<String>() { |
||||
|
||||
@Override |
||||
public void subscribe(Subscriber<? super String> subscriber) { |
||||
subscriber.onSubscribe(new AbstractSubscription<String>(subscriber) { |
||||
|
||||
@Override |
||||
protected void requestInternal(long n) { |
||||
invokeOnNext("Hello world."); |
||||
invokeOnComplete(); |
||||
} |
||||
}); |
||||
} |
||||
}; |
||||
} |
||||
|
||||
} |
||||
|
||||
private static class PlainTextHandlerAdapter implements HandlerAdapter { |
||||
|
||||
@Override |
||||
public boolean supports(Object handler) { |
||||
return PlainTextHandler.class.isAssignableFrom(handler.getClass()); |
||||
} |
||||
|
||||
@Override |
||||
public Publisher<HandlerResult> handle(ServerHttpRequest request, ServerHttpResponse response, |
||||
Object handler) { |
||||
|
||||
PlainTextHandler textHandler = (PlainTextHandler) handler; |
||||
final Publisher<String> resultPublisher = textHandler.handle(request, response); |
||||
|
||||
return new Publisher<HandlerResult>() { |
||||
|
||||
@Override |
||||
public void subscribe(Subscriber<? super HandlerResult> handlerResultSubscriber) { |
||||
handlerResultSubscriber.onSubscribe(new AbstractSubscription<HandlerResult>(handlerResultSubscriber) { |
||||
|
||||
@Override |
||||
protected void requestInternal(long n) { |
||||
resultPublisher.subscribe(new Subscriber<Object>() { |
||||
|
||||
@Override |
||||
public void onSubscribe(Subscription subscription) { |
||||
subscription.request(Long.MAX_VALUE); |
||||
} |
||||
|
||||
@Override |
||||
public void onNext(Object result) { |
||||
invokeOnNext(new HandlerResult(result)); |
||||
} |
||||
|
||||
@Override |
||||
public void onError(Throwable error) { |
||||
invokeOnError(error); |
||||
} |
||||
|
||||
@Override |
||||
public void onComplete() { |
||||
invokeOnComplete(); |
||||
} |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
}; |
||||
} |
||||
} |
||||
|
||||
private static class PlainTextResultHandler implements HandlerResultHandler { |
||||
|
||||
@Override |
||||
public boolean supports(HandlerResult result) { |
||||
Object value = result.getReturnValue(); |
||||
return (value != null && String.class.equals(value.getClass())); |
||||
} |
||||
|
||||
@Override |
||||
public Publisher<Void> handleResult(ServerHttpRequest request, ServerHttpResponse response, |
||||
HandlerResult result) { |
||||
|
||||
response.getHeaders().setContentType(MediaType.TEXT_PLAIN); |
||||
|
||||
return response.writeWith(new Publisher<byte[]>() { |
||||
|
||||
@Override |
||||
public void subscribe(Subscriber<? super byte[]> writeSubscriber) { |
||||
writeSubscriber.onSubscribe(new AbstractSubscription<byte[]>(writeSubscriber) { |
||||
|
||||
@Override |
||||
protected void requestInternal(long n) { |
||||
Charset charset = Charset.forName("UTF-8"); |
||||
invokeOnNext(((String) result.getReturnValue()).getBytes(charset)); |
||||
invokeOnComplete(); |
||||
} |
||||
}); |
||||
} |
||||
}); |
||||
} |
||||
} |
||||
|
||||
|
||||
private static abstract class AbstractSubscription<T> implements Subscription { |
||||
|
||||
private final Subscriber<? super T> subscriber; |
||||
|
||||
private volatile boolean terminated; |
||||
|
||||
|
||||
public AbstractSubscription(Subscriber<? super T> subscriber) { |
||||
this.subscriber = subscriber; |
||||
} |
||||
|
||||
protected boolean isTerminated() { |
||||
return this.terminated; |
||||
} |
||||
|
||||
@Override |
||||
public void request(long n) { |
||||
if (isTerminated()) { |
||||
return; |
||||
} |
||||
if (n > 0) { |
||||
requestInternal(n); |
||||
} |
||||
} |
||||
|
||||
protected abstract void requestInternal(long n); |
||||
|
||||
@Override |
||||
public void cancel() { |
||||
this.terminated = true; |
||||
} |
||||
|
||||
protected void invokeOnNext(T data) { |
||||
this.subscriber.onNext(data); |
||||
} |
||||
|
||||
protected void invokeOnError(Throwable error) { |
||||
this.terminated = true; |
||||
this.subscriber.onError(error); |
||||
} |
||||
|
||||
protected void invokeOnComplete() { |
||||
this.terminated = true; |
||||
this.subscriber.onComplete(); |
||||
} |
||||
} |
||||
|
||||
} |
||||
Loading…
Reference in new issue