/gwt-test-utils/src/main/java/com/octo/gwt/test/gin/GInjectorCreateHandler.java

http://gwt-test-utils.googlecode.com/ · Java · 112 lines · 74 code · 25 blank · 13 comment · 12 complexity · af39eee5f6185f298352688049d2ace7 MD5 · raw file

  1. package com.octo.gwt.test.gin;
  2. import java.lang.reflect.Proxy;
  3. import java.util.HashMap;
  4. import java.util.HashSet;
  5. import java.util.Map;
  6. import java.util.Set;
  7. import org.slf4j.Logger;
  8. import org.slf4j.LoggerFactory;
  9. import com.google.gwt.core.client.GWT;
  10. import com.google.gwt.inject.client.GinModule;
  11. import com.google.gwt.inject.client.GinModules;
  12. import com.google.gwt.inject.client.Ginjector;
  13. import com.google.gwt.inject.rebind.adapter.GinModuleAdapter;
  14. import com.google.inject.Guice;
  15. import com.google.inject.Injector;
  16. import com.google.inject.Module;
  17. import com.octo.gwt.test.GwtCreateHandler;
  18. /**
  19. * Handle {@link GWT#create(Class)} for {@link Ginjector}.
  20. *
  21. * @author Alex Dobjanschi
  22. * @author Gael Lazzari
  23. *
  24. */
  25. public class GInjectorCreateHandler implements GwtCreateHandler {
  26. private static final Logger LOGGER = LoggerFactory.getLogger(GInjectorCreateHandler.class);
  27. // map used as cache to store bindings between gin and guice proxy injectors
  28. private Map<Class<? extends Ginjector>, Object> injectors;
  29. public Object create(Class<?> classLiteral) throws Exception {
  30. // Make sure this is a Ginjector
  31. if (!Ginjector.class.isAssignableFrom(classLiteral)) {
  32. return null;
  33. }
  34. @SuppressWarnings("unchecked")
  35. Class<? extends Ginjector> ginInjectorClass = (Class<? extends Ginjector>) classLiteral;
  36. if (injectors == null) {
  37. injectors = new HashMap<Class<? extends Ginjector>, Object>();
  38. }
  39. Object guiceInjectorProxy = injectors.get(classLiteral);
  40. if (guiceInjectorProxy != null) {
  41. LOGGER.debug("Proxy for class '" + ginInjectorClass.getName()
  42. + "'has been found in cache. It is returned");
  43. return guiceInjectorProxy;
  44. }
  45. Class<? extends GinModule>[] ginModules = readGinModules(ginInjectorClass);
  46. // try to instantiate an injector, based on the modules read above.
  47. Module[] guiceModules = readGuiceModules(ginInjectorClass, ginModules);
  48. Injector injector = Guice.createInjector(guiceModules);
  49. LOGGER.debug("creating new Proxy for class '" + ginInjectorClass.getName()
  50. + "'");
  51. guiceInjectorProxy = Proxy.newProxyInstance(
  52. this.getClass().getClassLoader(), new Class[]{ginInjectorClass},
  53. new GinInjectorInvocationHandler(injector));
  54. injectors.put(ginInjectorClass, guiceInjectorProxy);
  55. return guiceInjectorProxy;
  56. }
  57. private Class<? extends GinModule>[] readGinModules(
  58. Class<? extends Ginjector> classLiteral) {
  59. LOGGER.debug("inspecting classLiteral " + classLiteral);
  60. GinModules annotation = classLiteral.getAnnotation(GinModules.class);
  61. if (annotation == null) {
  62. // Throw an exception if we don't find this specific annotation.
  63. throw new IllegalArgumentException(classLiteral.getName()
  64. + " doesn't have any @GinModules annotation present");
  65. }
  66. Class<? extends GinModule>[] ginModules = annotation.value();
  67. if (ginModules == null || ginModules.length == 0) {
  68. // there are no GinModules present in the Ginjector.
  69. throw new IllegalArgumentException(classLiteral.getName()
  70. + " doesn't have any GinModules. " + "Runtime should not work.");
  71. }
  72. LOGGER.debug("discovered modules " + annotation);
  73. return ginModules;
  74. }
  75. private Module[] readGuiceModules(Class<? extends Ginjector> ginjectorClass,
  76. Class<? extends GinModule>[] classLiterals) throws Exception {
  77. Set<Module> modules = new HashSet<Module>();
  78. for (Class<? extends GinModule> literal : classLiterals) {
  79. LOGGER.debug("wrapping GinModule literal " + literal);
  80. modules.add(new GinModuleAdapter(literal.newInstance()));
  81. }
  82. // Use Guice SPI to solve deferred binding dependencies.
  83. modules.add(new DeferredBindingModule(ginjectorClass,
  84. modules.toArray(new Module[modules.size()])));
  85. return modules.toArray(new Module[modules.size()]);
  86. }
  87. }