ratpack.handling.internal.DefaultContext Maven / Gradle / Ivy
/*
* Copyright 2013 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 ratpack.handling.internal;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.reflect.TypeToken;
import io.netty.channel.EventLoop;
import io.netty.handler.codec.http.HttpResponseStatus;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ratpack.error.ClientErrorHandler;
import ratpack.error.ServerErrorHandler;
import ratpack.event.internal.EventRegistry;
import ratpack.exec.ExecController;
import ratpack.exec.Execution;
import ratpack.exec.Promise;
import ratpack.file.FileSystemBinding;
import ratpack.func.Action;
import ratpack.func.Block;
import ratpack.func.Function;
import ratpack.handling.*;
import ratpack.handling.direct.DirectChannelAccess;
import ratpack.http.Request;
import ratpack.http.Response;
import ratpack.http.TypedData;
import ratpack.http.internal.DefaultRequest;
import ratpack.http.internal.HttpHeaderConstants;
import ratpack.parse.NoSuchParserException;
import ratpack.parse.Parse;
import ratpack.parse.Parser;
import ratpack.path.PathBinding;
import ratpack.path.PathTokens;
import ratpack.path.internal.DefaultPathTokens;
import ratpack.path.internal.RootPathBinding;
import ratpack.registry.NotInRegistryException;
import ratpack.registry.Registry;
import ratpack.registry.internal.DelegatingRegistry;
import ratpack.render.NoSuchRendererException;
import ratpack.render.internal.RenderController;
import ratpack.server.ServerConfig;
import ratpack.util.Exceptions;
import ratpack.util.Types;
import java.lang.reflect.UndeclaredThrowableException;
import java.nio.file.Path;
import java.util.*;
import static com.google.common.base.Throwables.getStackTraceAsString;
import static io.netty.handler.codec.http.HttpHeaderNames.IF_MODIFIED_SINCE;
import static io.netty.handler.codec.http.HttpResponseStatus.NOT_MODIFIED;
public class DefaultContext implements Context {
private static final TypeToken> PARSER_TYPE_TOKEN = new TypeToken>() {
};
private final static Logger LOGGER = LoggerFactory.getLogger(DefaultContext.class);
public static class ApplicationConstants {
private final RenderController renderController;
private final ExecController execController;
private final ServerConfig serverConfig;
private final Handler end;
public ApplicationConstants(Registry registry, RenderController renderController, ExecController execController, Handler end) {
this.renderController = renderController;
this.execController = execController;
this.serverConfig = registry.get(ServerConfig.class);
this.end = end;
}
}
public static class RequestConstants {
private final ApplicationConstants applicationConstants;
private final DefaultRequest request;
private final DirectChannelAccess directChannelAccess;
private final EventRegistry onCloseRegistry;
private Execution execution;
private final Deque indexes = Lists.newLinkedList();
public Response response;
public Context context;
public Handler handler;
public RequestConstants(
ApplicationConstants applicationConstants, DefaultRequest request,
DirectChannelAccess directChannelAccess, EventRegistry onCloseRegistry
) {
this.applicationConstants = applicationConstants;
this.request = request;
this.directChannelAccess = directChannelAccess;
this.onCloseRegistry = onCloseRegistry;
}
}
private static class ChainIndex implements Iterator {
final Handler[] handlers;
Registry registry;
final boolean first;
int i;
private ChainIndex(Handler[] handlers, Registry registry, boolean first) {
this.handlers = handlers;
this.registry = registry;
this.first = first;
}
public Handler next() {
return handlers[i++];
}
@Override
public boolean hasNext() {
return i < handlers.length;
}
}
private final RequestConstants requestConstants;
private final Registry joinedRegistry;
public static void start(EventLoop eventLoop, final RequestConstants requestConstants, Registry registry, Handler[] handlers, Action super Execution> onComplete) {
PathBinding initialPathBinding = new RootPathBinding(requestConstants.request.getPath());
Registry pathBindingRegistry = Registry.single(PathBinding.class, initialPathBinding);
ChainIndex index = new ChainIndex(handlers, registry.join(pathBindingRegistry), true);
requestConstants.indexes.push(index);
DefaultContext context = new DefaultContext(requestConstants);
requestConstants.context = context;
requestConstants.applicationConstants.execController.exec()
.onError(throwable -> requestConstants.context.error(throwable instanceof HandlerException ? throwable.getCause() : throwable))
.onComplete(onComplete)
.register(s -> s
.add(Context.class, context)
.add(Request.class, requestConstants.request)
.add(Response.class, requestConstants.response)
)
.eventLoop(eventLoop)
.onStart(e -> DefaultRequest.setDelegateRegistry(requestConstants.request, e))
.start(e -> {
requestConstants.execution = e;
context.next();
});
}
public DefaultContext(RequestConstants requestConstants) {
this.requestConstants = requestConstants;
this.joinedRegistry = ((DelegatingRegistry) DefaultContext.this::getContextRegistry).join(requestConstants.request);
}
private Registry getContextRegistry() {
return requestConstants.indexes.peek().registry;
}
@Override
public Context getContext() {
return this;
}
@Override
public Execution getExecution() {
return requestConstants.execution;
}
@Override
public ServerConfig getServerConfig() {
return requestConstants.applicationConstants.serverConfig;
}
public Request getRequest() {
return requestConstants.request;
}
public Response getResponse() {
return requestConstants.response;
}
public void next() {
Handler handler = null;
ChainIndex index = requestConstants.indexes.peek();
while (handler == null) {
if (index.hasNext()) {
handler = index.next();
if (handler.getClass().equals(ChainHandler.class)) {
requestConstants.indexes.push(new ChainIndex(((ChainHandler) handler).getHandlers(), getContextRegistry(), false));
index = requestConstants.indexes.peek();
handler = null;
}
} else if (index.first) {
handler = requestConstants.applicationConstants.end;
} else {
requestConstants.indexes.pop();
index = requestConstants.indexes.peek();
}
}
try {
requestConstants.handler = handler;
handler.handle(this);
} catch (Throwable e) {
if (e instanceof HandlerException) {
throw (HandlerException) e;
} else {
throw new HandlerException(e);
}
}
}
@Override
public void next(Registry registry) {
requestConstants.indexes.peek().registry = getContextRegistry().join(registry);
next();
}
public void insert(Handler... handlers) {
if (handlers.length == 0) {
throw new IllegalArgumentException("handlers is zero length");
}
requestConstants.indexes.push(new ChainIndex(handlers, getContextRegistry(), false));
next();
}
public void insert(final Registry registry, final Handler... handlers) {
if (handlers.length == 0) {
throw new IllegalArgumentException("handlers is zero length");
}
requestConstants.indexes.push(new ChainIndex(handlers, getContextRegistry().join(registry), false));
next();
}
public PathTokens getPathTokens() {
return maybeGet(PathBinding.class)
.map(PathBinding::getTokens)
.orElseGet(DefaultPathTokens::empty);
}
public PathTokens getAllPathTokens() {
return get(PathBinding.class).getAllTokens();
}
public Path file(String path) {
return get(FileSystemBinding.class).file(path);
}
public void render(Object object) throws NoSuchRendererException {
try {
requestConstants.applicationConstants.renderController.render(object, this);
} catch (NoSuchRendererException e) {
throw e;
} catch (Exception e) {
error(e);
}
}
@Override
public Promise parse(Parse parse) {
return getRequest().getBody().map(b -> parse(b, parse));
}
@Override
public T parse(TypedData body, Parse parse) {
String requestContentType = body.getContentType().getType();
if (requestContentType == null) {
requestContentType = "text/plain";
}
final String finalRequestContentType = requestContentType;
return Exceptions.uncheck(() -> {
return joinedRegistry
.first(PARSER_TYPE_TOKEN, parser -> {
if (parser.getContentType().equalsIgnoreCase(finalRequestContentType) && parser.getOptsType().isInstance(parse.getOpts())) {
Parser cast = Types.cast(parser);
return cast.parse(DefaultContext.this, body, parse);
}
return null;
});
}).orElseThrow(() -> new NoSuchParserException(parse.getType(), parse.getOpts(), finalRequestContentType));
}
@Override
public Promise parse(Class type) {
return parse(Parse.of(type));
}
@Override
public Promise parse(TypeToken type) {
return parse(Parse.of(type));
}
public Promise parse(Class type, O opts) {
return parse(Parse.of(type, opts));
}
public Promise parse(TypeToken type, O opts) {
return parse(Parse.of(type, opts));
}
@Override
public void onClose(Action super RequestOutcome> callback) {
requestConstants.onCloseRegistry.register(callback);
}
@Override
public DirectChannelAccess getDirectChannelAccess() {
return requestConstants.directChannelAccess;
}
public void redirect(String location) {
redirect(HttpResponseStatus.FOUND.code(), location);
}
public void redirect(int code, String location) {
Redirector redirector = joinedRegistry.get(Redirector.class);
redirector.redirect(this, location, code);
}
@Override
public void lastModified(Date date, Runnable runnable) {
Date ifModifiedSinceHeader = requestConstants.request.getHeaders().getDate(IF_MODIFIED_SINCE);
long lastModifiedSecs = date.getTime() / 1000;
if (ifModifiedSinceHeader != null) {
long time = ifModifiedSinceHeader.getTime();
long ifModifiedSinceSecs = time / 1000;
if (lastModifiedSecs == ifModifiedSinceSecs) {
requestConstants.response.status(NOT_MODIFIED.code()).send();
return;
}
}
requestConstants.response.getHeaders().setDate(HttpHeaderConstants.LAST_MODIFIED, date);
runnable.run();
}
public void error(Throwable throwable) {
ServerErrorHandler serverErrorHandler = get(ServerErrorHandler.class);
throwable = unpackThrowable(throwable);
ThrowableHolder throwableHolder = getRequest().maybeGet(ThrowableHolder.class).orElse(null);
if (throwableHolder == null) {
getRequest().add(ThrowableHolder.class, new ThrowableHolder(throwable));
try {
serverErrorHandler.error(this, throwable);
} catch (Throwable errorHandlerThrowable) {
onErrorHandlerError(serverErrorHandler, throwable, errorHandlerThrowable);
}
} else {
onErrorHandlerError(serverErrorHandler, throwableHolder.getThrowable(), throwable);
}
}
private void onErrorHandlerError(ServerErrorHandler serverErrorHandler, Throwable original, Throwable errorHandlerThrowable) {
String msg = "Throwable thrown by error handler " + serverErrorHandler + " while handling throwable\n"
+ "Original throwable: " + getStackTraceAsString(original) + "\n"
+ "Error handler throwable: " + getStackTraceAsString(errorHandlerThrowable);
LOGGER.error(msg);
Response response = requestConstants.response.status(500);
if (getServerConfig().isDevelopment()) {
response.send(msg);
} else {
response.send();
}
}
private Throwable unpackThrowable(Throwable throwable) {
if (throwable instanceof UndeclaredThrowableException) {
return throwable.getCause();
} else {
return throwable;
}
}
public void clientError(int statusCode) {
try {
get(ClientErrorHandler.class).error(this, statusCode);
} catch (Throwable e) {
error(Exceptions.toException(e));
}
}
public void byMethod(Action super ByMethodSpec> action) throws Exception {
Map blocks = Maps.newLinkedHashMap();
DefaultByMethodSpec spec = new DefaultByMethodSpec(blocks);
action.execute(spec);
new MultiMethodHandler(blocks).handle(this);
}
public void byContent(Action super ByContentSpec> action) throws Exception {
Map blocks = Maps.newLinkedHashMap();
DefaultByContentSpec spec = new DefaultByContentSpec(blocks);
action.execute(spec);
new ContentNegotiationHandler(blocks, spec.getNoMatchHandler()).handle(this);
}
@Override
public O get(TypeToken type) throws NotInRegistryException {
return joinedRegistry.get(type);
}
@Override
public Optional maybeGet(TypeToken type) {
return joinedRegistry.maybeGet(type);
}
@Override
public Iterable extends O> getAll(TypeToken type) {
return joinedRegistry.getAll(type);
}
@Override
public Optional first(TypeToken type, Function super T, ? extends O> function) throws Exception {
return joinedRegistry.first(type, function);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy