All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ratpack.guice.Guice 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.guice;

import com.google.inject.Injector;
import com.google.inject.Module;
import com.google.inject.Stage;
import ratpack.func.Function;
import ratpack.guice.internal.DefaultGuiceBackedHandlerFactory;
import ratpack.guice.internal.InjectorBackedRegistry;
import ratpack.guice.internal.JustInTimeInjectorRegistry;
import ratpack.handling.Chain;
import ratpack.handling.Handler;
import ratpack.launch.LaunchConfig;
import ratpack.registry.Registry;
import ratpack.func.Action;

import static com.google.inject.Guice.createInjector;
import static ratpack.handling.Handlers.chain;

/**
 * Static utility methods for creating Google Guice based Ratpack infrastructure.
 * 

* Any non trivial application will require supporting objects for the handler implementations, for persistence for example. * These supporting objects can be managed by Guice and made available to handlers (either by dependency injection or registry lookup) * for increased reusability, modularity and testability. *

*

* The Guice integration is not part of the Ratpack core library, but is available as a separate add on. * That said, Ratpack is designed to support something like Guice as an integral piece via it's {@link Registry} abstraction. * Guice is the “official” solution. *

*

Starting a Guice Ratpack app

*

* The user entry point for Ratpack applications is the {@link ratpack.launch.HandlerFactory} implementation * that is given to the {@link ratpack.launch.LaunchConfigBuilder#build(ratpack.launch.HandlerFactory)} method. * This implementation can use one of the {@code handler(...) } static methods. *

* Below is a complete example for bootstrapping a Guice based Ratpack application. *

*
 * import ratpack.handling.*;
 * import ratpack.registry.*;
 * import ratpack.guice.*;
 * import ratpack.launch.*;
 * import ratpack.func.*;
 * import com.google.inject.*;
 * import javax.inject.*;
 *
 * // Some service to be dependency injected
 * class SomeService {}
 *
 * // A Guice module that provides the service
 * class ServiceModule extends AbstractModule {
 *   protected void configure() {
 *     bind(SomeService.class);
 *   }
 * }
 *
 * // An action that registers the module with the registry, making it part of the application
 * class Bindings implements Action<BindingsSpec> {
 *   public void execute(BindingsSpec bindings) {
 *     bindings.add(new ServiceModule());
 *   }
 * }
 *
 * // A handler implementation that is dependency injected
 * {@literal @}Singleton
 * class InjectedHandler implements Handler {
 *   private final SomeService service;
 *
 *   {@literal @}Inject
 *   public InjectedHandler(SomeService service) {
 *     this.service = service;
 *   }
 *
 *   public void handle(Context exchange) {
 *     // …
 *   }
 * }
 *
 * // A chain configurer that adds a dependency injected handler to the chain
 * class HandlersBootstrap implements Action<Chain> {
 *   public void execute(Chain chain) {
 *
 *     // The registry in a Guice backed chain can be used to retrieve objects that were bound,
 *     // or to create objects that are bound “just-in-time”.
 *     Registry<Object> registry = chain.getRegistry();
 *     Handler injectedHandler = registry.get(InjectedHandler.class);
 *
 *     // Add the handler into the chain
 *     chain.get("some/path", injectedHandler);
 *   }
 * }
 *
 * // A HandlerFactory implementation used to bootstrap the application
 * class MyHandlerFactory implements HandlerFactory {
 *   public Handler create(LaunchConfig launchConfig) {
 *     return Guice.handler(launchConfig, new Bindings(), new HandlersBootstrap());
 *   }
 * }
 *
 * // Building a launch config with our handler factory
 * LaunchConfig launchConfig = LaunchConfigBuilder.baseDir(new File("appRoot")).build(new MyHandlerFactory());
 * 
*

Accessing Guice bound objects in Handlers

*

* There are two ways to use Guice bound objects in your handler implementations. *

*
Dependency Injected Handlers
*

* The {@code handler()} methods used to create a Guice backed application take an {@link Action} that operates on a {@link Chain} instance. * This chain instance given to this action provides a Guice backed {@link Registry} via its {@link Chain#getRegistry()} method. * This registry is able to retrieve objects that were explicitly bound (i.e. defined by a module), and bind objects “just in time”. * This means that it can be used to construct dependency injected {@link Handler} implementations. *

*

* Simply pass the class of the handler implementation to create a new dependency injected instance of to this method, * then add it to the chain. *

*

* See the code above for an example of this. *

*
Accessing dependencies via context registry lookup
*

* The {@link ratpack.handling.Context} object that is given to a handler's {@link Handler#handle(ratpack.handling.Context)} * method is also a registry implementation. In a Guice backed app, Guice bound objects can be retrieved via the {@link Registry#get(Class)} method of * the context. You can retrieve any bound object via its publicly bound type. *

*

* However, this will not create “just-in-time” bindings. Only objects that were explicitly bound can be retrieved this way. *

*

Guice modules as Ratpack “plugins”.

*

* Add on Ratpack functionality is typically provided via Guice modules. * For example, the Jackson integration for JSON serialisation * is provided by the {@code ratpack-jackson} add-on which ships a Guice module. * To use its functionality simply register the module it provides with the {@link BindingsSpec} used to bootstrap the application. *

*

Groovy Applications

*

* The Ratpack Groovy add-on provides application modes that automatically incorporate Guice (namely the “Ratpack Script” mode). * The module registration process is simpler and more convenient in this mode, and there are additional options for obtaining * Guice bound objects. *

*

* See the Groovy add-on's documentation for more details. *

*/ public abstract class Guice { private Guice() { } /** * Creates a handler that can be used as the entry point for a Guice backed Ratpack app. *

* Modules are processed eagerly. Before this method returns, the modules will be used to create a single {@link com.google.inject.Injector} instance. * This injector is available to the given handler via service lookup. * * @param launchConfig The launch config of the server * @param moduleConfigurer The configurer of the {@link BindingsSpec} to back the created handler * @param chainConfigurer The configurer that builds the handler chain * @return A handler that makes all Guice bound objects available to the handlers added to the {@link Chain} given to {@code chainConfigurer}. */ public static Handler handler(LaunchConfig launchConfig, Action moduleConfigurer, final Action chainConfigurer) throws Exception { return new DefaultGuiceBackedHandlerFactory(launchConfig).create(moduleConfigurer, newInjectorFactory(launchConfig), new InjectorHandlerTransformer(launchConfig, chainConfigurer)); } /** * Creates a handler that can be used as the entry point for a Guice backed Ratpack app. *

* This is a lower level version of {@link #handler(ratpack.launch.LaunchConfig, ratpack.func.Action, ratpack.func.Action)} * that supports a custom final {@link Handler} creation strategy. * * @param launchConfig The launch config of the server * @param moduleConfigurer The configurer of the {@link BindingsSpec} to back the created handler * @param injectorTransformer Takes the final {@link Injector} instance that was created from the modules and returns the {@link Handler} to use * @return A handler that makes all Guice bound objects available to the handlers added to the {@link Chain} given to {@code chainConfigurer}. */ public static Handler handler(LaunchConfig launchConfig, Action moduleConfigurer, Function injectorTransformer) throws Exception { return new DefaultGuiceBackedHandlerFactory(launchConfig).create(moduleConfigurer, newInjectorFactory(launchConfig), injectorTransformer); } /** * Creates a handler that can be used as the entry point for a Guice backed Ratpack app. *

* Similar to {@link #handler(ratpack.launch.LaunchConfig, ratpack.func.Action, ratpack.func.Action)}, * but provides the opportunity to use a parent injector when creating the injector to use for the application. * See Guice documentation for the semantics of parent/child injectors. *

* Typically only required when making a Ratpack application a component of a larger application. * * @param launchConfig The launch config of the server * @param parentInjector The injector to use as a parent of the injector to be created * @param moduleConfigurer The configurer of the {@link BindingsSpec} to back the created handler * @param chainConfigurer The configurer that builds the handler chain * @return A handler that makes all Guice bound objects available to the handlers added to the {@link Chain} given to {@code chainConfigurer}. */ public static Handler handler(LaunchConfig launchConfig, Injector parentInjector, Action moduleConfigurer, final Action chainConfigurer) throws Exception { return new DefaultGuiceBackedHandlerFactory(launchConfig).create(moduleConfigurer, childInjectorFactory(parentInjector), new InjectorHandlerTransformer(launchConfig, chainConfigurer)); } /** * Creates a handler that can be used as the entry point for a Guice backed Ratpack app. *

* This is a lower level version of {@link #handler(ratpack.launch.LaunchConfig, com.google.inject.Injector, ratpack.func.Action, ratpack.func.Action)} * that supports a custom final {@link Handler} creation strategy. * * @param launchConfig The launch config of the server * @param parentInjector The injector to use as a parent of the injector to be created * @param moduleConfigurer The configurer of the {@link BindingsSpec} to back the created handler * @param injectorTransformer Takes the final {@link Injector} instance that was created from the modules and returns the {@link Handler} to use * @return A handler that makes all Guice bound objects available to the handlers added to the {@link Chain} given to {@code chainConfigurer}. */ public static Handler handler(LaunchConfig launchConfig, Injector parentInjector, Action moduleConfigurer, Function injectorTransformer) throws Exception { return new DefaultGuiceBackedHandlerFactory(launchConfig).create(moduleConfigurer, childInjectorFactory(parentInjector), injectorTransformer); } /** * Creates a Ratpack {@link Registry} backed by the given {@link Injector} that will create objects via “just-in-time” binding. *

* The returned registry differs from the registry returned by {@link #registry(com.google.inject.Injector)} in that it may return objects that were not explicitly bound, * via Guice's “just-in-time” binding mechanism. * This only applies to the {@link Registry#get(Class)} and {@link Registry#maybeGet(Class)} (for all overloads). * All other methods do not perform any just in time bindings. * * @param injector The injector to back the registry * @return A registry that wraps the injector */ public static Registry justInTimeRegistry(Injector injector) { return new JustInTimeInjectorRegistry(injector); } /** * Creates a Ratpack {@link Registry} backed by the given {@link Injector} that will NOT create objects via “just-in-time” binding. * * @param injector The injector to back the registry * @return A registry that wraps the injector */ public static Registry registry(Injector injector) { return new InjectorBackedRegistry(injector); } /** * Creates a transformer that can build an injector from a module. *

* The module given to the {@code transform()} method may be {@code null}. * * @return a transformer that can build an injector from a module */ public static Function newInjectorFactory(final LaunchConfig launchConfig) { final Stage stage = launchConfig.isReloadable() ? Stage.DEVELOPMENT : Stage.PRODUCTION; return new Function() { @Override public Injector apply(Module from) { return from == null ? createInjector(stage) : createInjector(stage, from); } }; } /** * Creates a transformer that can build an injector from a module, as a child of the given parent. *

* The module given to the {@code transform()} method may be {@code null}. * * @return a transformer that can build an injector from a module, as a child of the given parent. */ public static Function childInjectorFactory(final Injector parent) { return new Function() { @Override public Injector apply(Module from) { return from == null ? parent.createChildInjector() : parent.createChildInjector(from); } }; } private static class InjectorHandlerTransformer implements Function { private final LaunchConfig launchConfig; private final Action action; public InjectorHandlerTransformer(LaunchConfig launchConfig, Action action) { this.launchConfig = launchConfig; this.action = action; } public Handler apply(Injector injector) throws Exception { return chain(launchConfig, justInTimeRegistry(injector), action); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy