ratpack.test.handling.RequestFixture Maven / Gradle / Ivy
Show all versions of ratpack-test Show documentation
/*
* 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.test.handling;
import com.google.common.net.HostAndPort;
import ratpack.func.Action;
import ratpack.handling.Chain;
import ratpack.handling.Handler;
import ratpack.registry.RegistrySpec;
import ratpack.server.ServerConfig;
import ratpack.test.handling.internal.DefaultRequestFixture;
import java.nio.file.Path;
import java.util.Map;
/**
* A contrived request environment, suitable for unit testing {@link Handler} implementations.
*
* A request fixture emulates a request, and the effective state of the request handling in the handler pipeline.
*
* A request fixture can be obtained by the {@link ratpack.test.handling.RequestFixture#requestFixture()} method.
* However it is often more convenient to use the alternative {@link ratpack.test.handling.RequestFixture#handle(ratpack.handling.Handler, ratpack.func.Action)} method.
*
* @see #handle(ratpack.handling.Handler)
*/
public interface RequestFixture {
/**
* Unit test a single {@link Handler}.
*
*
{@code
* import ratpack.handling.Context;
* import ratpack.handling.Handler;
* import ratpack.test.handling.RequestFixture;
* import ratpack.test.handling.HandlingResult;
*
* import static org.junit.Assert.assertEquals;
*
* public class Example {
*
* public static class MyHandler implements Handler {
* public void handle(Context ctx) throws Exception {
* String outputHeaderValue = ctx.getRequest().getHeaders().get("input-value") + ":bar";
* ctx.getResponse().getHeaders().set("output-value", outputHeaderValue);
* ctx.render("received: " + ctx.getRequest().getPath());
* }
* }
*
* public static void main(String[] args) throws Exception {
* HandlingResult result = RequestFixture.handle(new MyHandler(), fixture ->
* fixture.header("input-value", "foo").uri("some/path")
* );
*
* assertEquals("received: some/path", result.rendered(String.class));
* assertEquals("foo:bar", result.getHeaders().get("output-value"));
* }
* }
* }
*
* @param handler The handler to invoke
* @param action The configuration of the context for the handler
* @return A result object indicating what happened
* @throws ratpack.test.handling.HandlerTimeoutException if the handler takes more than {@link ratpack.test.handling.RequestFixture#timeout(int)} seconds to send a response or call {@code next()} on the context
* @throws Exception any thrown by {@code action}
* @see #handle(Action, Action)
*/
@SuppressWarnings("overloads")
public static HandlingResult handle(Handler handler, Action super RequestFixture> action) throws Exception {
RequestFixture requestFixture = requestFixture();
action.execute(requestFixture);
return requestFixture.handle(handler);
}
/**
* Unit test a {@link Handler} chain.
*
* {@code
* import ratpack.func.Action;
* import ratpack.handling.Chain;
* import ratpack.test.handling.RequestFixture;
* import ratpack.test.handling.HandlingResult;
*
* import static org.junit.Assert.assertEquals;
*
* public class Example {
*
* public static class MyHandlers implements Action {
* public void execute(Chain chain) throws Exception {
* chain.handler(ctx -> {
* String outputHeaderValue = ctx.getRequest().getHeaders().get("input-value") + ":bar";
* ctx.getResponse().getHeaders().set("output-value", outputHeaderValue);
* ctx.next();
* });
* chain.handler(ctx -> ctx.render("received: " + ctx.getRequest().getPath()) );
* }
* }
*
* public static void main(String[] args) throws Exception {
* HandlingResult result = RequestFixture.handle(new MyHandlers(), fixture ->
* fixture.header("input-value", "foo").uri("some/path")
* );
*
* assertEquals("received: some/path", result.rendered(String.class));
* assertEquals("foo:bar", result.getHeaders().get("output-value"));
* }
* }
* }
*
* @param chainAction the definition of a handler chain to test
* @param requestFixtureAction the configuration of the request fixture
* @return a result object indicating what happened
* @throws ratpack.test.handling.HandlerTimeoutException if the handler takes more than {@link ratpack.test.handling.RequestFixture#timeout(int)} seconds to send a response or call {@code next()} on the context
* @throws Exception any thrown by {@code chainAction} or {@code requestFixtureAction}
* @see #handle(Handler, Action)
*/
@SuppressWarnings("overloads")
public static HandlingResult handle(Action super Chain> chainAction, Action super RequestFixture> requestFixtureAction) throws Exception {
RequestFixture requestFixture = requestFixture();
requestFixtureAction.execute(requestFixture);
return requestFixture.handleChain(chainAction);
}
/**
* Create a request fixture, for unit testing of {@link Handler handlers}.
*
* @see #handle(ratpack.handling.Handler, ratpack.func.Action)
* @see #handle(ratpack.func.Action, ratpack.func.Action)
* @return a request fixture
*/
public static RequestFixture requestFixture() {
return new DefaultRequestFixture();
}
/**
* Sets the request body to be the given bytes, and adds a {@code Content-Type} request header of the given value.
*
* By default the body is empty.
*
* @param bytes the request body in bytes
* @param contentType the content type of the request body
* @return this
*/
RequestFixture body(byte[] bytes, String contentType);
/**
* Sets the request body to be the given string in utf8 bytes, and adds a {@code Content-Type} request header of the given value.
*
* By default the body is empty.
*
* @param text the request body as a string
* @param contentType the content type of the request body
* @return this
*/
RequestFixture body(String text, String contentType);
/**
* A specification of the context registry.
*
* Can be used to make objects (e.g. support services) available via context registry lookup.
*
* By default, only a {@link ratpack.error.ServerErrorHandler} and {@link ratpack.error.ClientErrorHandler} are in the context registry.
*
* @return a specification of the context registry
*/
RegistrySpec getRegistry();
/**
* Invokes the given handler with a newly created {@link ratpack.handling.Context} based on the state of this fixture.
*
* The return value can be used to examine the effective result of the handler.
*
* A result may be one of the following:
*
* - The sending of a response via one of the {@link ratpack.http.Response#send} methods
* - Rendering to the response via the {@link ratpack.handling.Context#render(Object)}
* - Raising of a client error via {@link ratpack.handling.Context#clientError(int)}
* - Raising of a server error via {@link ratpack.handling.Context#error(Throwable)}
* - Raising of a server error by the throwing of an exception
* - Delegating to the next handler by invoking one of the {@link ratpack.handling.Context#next} methods
*
* Note that any handlers {@link ratpack.handling.Context#insert inserted} by the handler under test will be invoked.
* If the last inserted handler delegates to the next handler, the handling will terminate with a result indicating that the effective result was delegating to the next handler.
*
* This method blocks until a result is achieved, even if the handler performs an asynchronous operation (such as performing {@link ratpack.handling.Context#blocking(java.util.concurrent.Callable) blocking IO}).
* As such, a time limit on the execution is imposed which by default is 5 seconds.
* The time limit can be changed via the {@link #timeout(int)} method.
* If the time limit is reached, a {@link HandlerTimeoutException} is thrown.
*
* @param handler the handler to test
* @return the effective result of the handling
* @throws HandlerTimeoutException if the handler does not produce a result in the time limit defined by this fixture
*/
HandlingResult handle(Handler handler) throws HandlerTimeoutException;
/**
* Similar to {@link #handle(ratpack.handling.Handler)}, but for testing a handler chain.
*
* @param chainAction the handler chain to test
* @return the effective result of the handling
* @throws HandlerTimeoutException if the handler does not produce a result in the time limit defined by this fixture
* @throws Exception any thrown by {@code chainAction}
*/
HandlingResult handleChain(Action super Chain> chainAction) throws Exception;
/**
* Set a request header value.
*
* By default there are no request headers.
*
* @param name the header name
* @param value the header value
* @return this
*/
RequestFixture header(String name, String value);
/**
* Configures the server config to have the given base dir and given configuration.
*
* By default the server config is equivalent to {@link ratpack.server.ServerConfig#noBaseDir() ServerConfigBuilder.noBaseDir()}.{@link ratpack.server.ServerConfig.Builder#build() build()}.
*
* @param baseDir the server config base dir
* @param action configuration of the server config
* @return this
* @throws Exception any thrown by {@code action}
*/
RequestFixture serverConfig(Path baseDir, Action super ServerConfig.Builder> action) throws Exception;
/**
* Configures the server config to have no base dir and given configuration.
*
* By default the server config is equivalent to {@link ratpack.server.ServerConfig#noBaseDir() ServerConfigBuilder.noBaseDir()}.{@link ratpack.server.ServerConfig.Builder#build() build()}.
*
* @param action configuration of the server config
* @return this
* @throws Exception any thrown by {@code action}
*/
RequestFixture serverConfig(Action super ServerConfig.Builder> action) throws Exception;
/**
* Set the request method (case insensitive).
*
* The default method is {@code "GET"}.
*
* @param method the request method
* @return this
*/
RequestFixture method(String method);
/**
* Adds a path binding, with the given path tokens.
*
* By default, there are no path tokens and no path binding.
*
* @param pathTokens the path tokens to make available to the handler(s) under test
* @return this
*/
RequestFixture pathBinding(Map pathTokens);
/**
* Adds a path binding, with the given path tokens and parts.
*
* By default, there are no path tokens and no path binding.
*
* @param boundTo the part of the request path that the binding bound to
* @param pastBinding the part of the request path past {@code boundTo}
* @param pathTokens the path tokens and binding to make available to the handler(s) under test
* @return this
*/
RequestFixture pathBinding(String boundTo, String pastBinding, Map pathTokens);
/**
* Configures the context registry.
*
* @param action a registry specification action
* @return this
* @throws Exception any thrown by {@code action}
*/
RequestFixture registry(Action super RegistrySpec> action) throws Exception;
/**
* Set a response header value.
*
* Can be used to simulate the setting of a response header by an upstream handler.
*
* By default there are no request headers.
*
* @param name the header name
* @param value the header value
* @return this
*/
RequestFixture responseHeader(String name, String value);
/**
* Sets the maximum time to allow the handler under test to produce a result.
*
* As handlers may execute asynchronously, a maximum time limit must be used to guard against never ending handlers.
*
* @param timeoutSeconds the maximum number of seconds to allow the handler(s) under test to produce a result
* @return this
*/
RequestFixture timeout(int timeoutSeconds);
/**
* The URI of the request.
*
* No encoding is performed on the given value.
* It is expected to be a well formed URI path string (potentially including query and fragment strings)
*
* @param uri the URI of the request
* @return this
*/
RequestFixture uri(String uri);
/**
* Set the remote address from which the request is made.
*
* Effectively the return value of {@link ratpack.http.Request#getRemoteAddress()}.
* @param remote the remote host and port address
* @return this
*/
RequestFixture remoteAddress(HostAndPort remote);
/**
* Set the local address to which this request is made.
*
* Effectively the return value of {@link ratpack.http.Request#getLocalAddress()}.
*
* @param local the local host and port address
* @return this
*/
RequestFixture localAddress(HostAndPort local);
}