/environments/servlet/core/src/main/java/org/jboss/weld/environment/tomcat/WeldForwardingInstanceManager.java

http://github.com/weld/core · Java · 168 lines · 133 code · 24 blank · 11 comment · 0 complexity · 8621958750eaf15a7a3b74f9b9aaf7c6 MD5 · raw file

  1. package org.jboss.weld.environment.tomcat;
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.InvocationTargetException;
  4. import java.lang.reflect.Method;
  5. import javax.naming.NamingException;
  6. import jakarta.servlet.ServletContext;
  7. import org.apache.catalina.core.ApplicationContext;
  8. import org.apache.catalina.core.ApplicationContextFacade;
  9. import org.apache.catalina.core.StandardContext;
  10. import org.apache.tomcat.InstanceManager;
  11. import org.jboss.weld.environment.servlet.logging.TomcatLogger;
  12. import org.jboss.weld.environment.util.Reflections;
  13. import org.jboss.weld.manager.api.WeldManager;
  14. import org.jboss.weld.util.collections.Arrays2;
  15. /**
  16. * Forwards all calls in turn to two delegates: first to InstanceManager, then to WeldInstanceManager
  17. *
  18. * @author <a href="mailto:matija.mazi@gmail.com">Matija Mazi</a>
  19. */
  20. public class WeldForwardingInstanceManager extends ForwardingInstanceManager {
  21. private static final String CONTEXT_FIELD_NAME = "context";
  22. private static final String INSTANCE_MANAGER_SETTER_NAME = "setInstanceManager";
  23. private static final String INSTANCE_MANAGER_GETTER_NAME = "getInstanceManager";
  24. private static final String INSTANCE_MANAGER_FIELD_NAME = "instanceManager";
  25. private final InstanceManager firstProcessor;
  26. private final InstanceManager secondProcessor;
  27. public WeldForwardingInstanceManager(InstanceManager originalAnnotationProcessor, InstanceManager weldProcessor) {
  28. this.firstProcessor = originalAnnotationProcessor;
  29. this.secondProcessor = weldProcessor;
  30. }
  31. @Override
  32. protected InstanceManager delegate() {
  33. return firstProcessor;
  34. }
  35. @Override
  36. public void destroyInstance(Object o) throws IllegalAccessException, InvocationTargetException {
  37. super.destroyInstance(o);
  38. secondProcessor.destroyInstance(o);
  39. }
  40. @Override
  41. public void newInstance(Object o) throws IllegalAccessException, InvocationTargetException, NamingException {
  42. super.newInstance(o);
  43. secondProcessor.newInstance(o);
  44. }
  45. @Override
  46. public Object newInstance(String fqcn, ClassLoader classLoader) throws IllegalAccessException, InvocationTargetException, NamingException,
  47. InstantiationException, ClassNotFoundException, NoSuchMethodException {
  48. Object a = super.newInstance(fqcn, classLoader);
  49. secondProcessor.newInstance(a);
  50. return a;
  51. }
  52. @Override
  53. public Object newInstance(String fqcn) throws IllegalAccessException, InvocationTargetException, NamingException, InstantiationException,
  54. ClassNotFoundException, NoSuchMethodException {
  55. Object a = super.newInstance(fqcn);
  56. secondProcessor.newInstance(a);
  57. return a;
  58. }
  59. @Override
  60. public Object newInstance(Class<?> clazz) throws IllegalAccessException, InvocationTargetException, NamingException, InstantiationException, NoSuchMethodException {
  61. Object a = super.newInstance(clazz);
  62. secondProcessor.newInstance(a);
  63. return a;
  64. }
  65. public static void replaceInstanceManager(ServletContext context, WeldManager manager) {
  66. StandardContext stdContext = getStandardContext(context);
  67. setInstanceManager(stdContext, createInstance(manager, stdContext));
  68. }
  69. private static WeldForwardingInstanceManager createInstance(WeldManager manager, StandardContext stdContext) {
  70. try {
  71. InstanceManager weldProcessor = new WeldInstanceManager(manager);
  72. return new WeldForwardingInstanceManager(getInstanceManager(stdContext), weldProcessor);
  73. } catch (Exception e) {
  74. throw TomcatLogger.LOG.cannotCreatWeldForwardingAnnotationProcessor(e);
  75. }
  76. }
  77. private static StandardContext getStandardContext(ServletContext context) {
  78. try {
  79. // Hack into Tomcat to replace the InstanceManager using
  80. // reflection to access private fields
  81. ApplicationContext appContext = (ApplicationContext) getContextFieldValue((ApplicationContextFacade) context, ApplicationContextFacade.class);
  82. return (StandardContext) getContextFieldValue(appContext, ApplicationContext.class);
  83. } catch (Exception e) {
  84. throw TomcatLogger.LOG.cannotGetStandardContext(e);
  85. }
  86. }
  87. private static <E> Object getContextFieldValue(E obj, Class<E> clazz) throws NoSuchFieldException, IllegalAccessException {
  88. Field field = SecurityActions.lookupField(clazz, CONTEXT_FIELD_NAME);
  89. SecurityActions.ensureAccessible(field);
  90. return field.get(obj);
  91. }
  92. private static InstanceManager getInstanceManager(StandardContext stdContext) {
  93. try {
  94. Method method = SecurityActions.lookupMethod(stdContext.getClass(), INSTANCE_MANAGER_GETTER_NAME);
  95. SecurityActions.ensureAccessible(method);
  96. try {
  97. return Reflections.cast(method.invoke(stdContext));
  98. } catch (Exception e) {
  99. TomcatLogger.LOG.errorInvokingMethod(method.getName(), stdContext, Arrays2.EMPTY_ARRAY);
  100. }
  101. } catch (NoSuchMethodException e1) {
  102. // Getter/setter not found
  103. }
  104. try {
  105. Field field = SecurityActions.lookupField(stdContext.getClass(), INSTANCE_MANAGER_FIELD_NAME);
  106. SecurityActions.ensureAccessible(field);
  107. try {
  108. return Reflections.cast(field.get(stdContext));
  109. } catch (Exception e) {
  110. TomcatLogger.LOG.errorReadingField(field.getName(), stdContext);
  111. }
  112. } catch (NoSuchFieldException e1) {
  113. // Field not found
  114. }
  115. throw TomcatLogger.LOG.neitherFieldNorGetterSetterFound(stdContext.getClass());
  116. }
  117. private static void setInstanceManager(StandardContext stdContext, InstanceManager instanceManager) {
  118. try {
  119. Method method = SecurityActions.lookupMethod(stdContext.getClass(), INSTANCE_MANAGER_SETTER_NAME, InstanceManager.class);
  120. SecurityActions.ensureAccessible(method);
  121. try {
  122. method.invoke(stdContext, instanceManager);
  123. return;
  124. } catch (Exception e) {
  125. TomcatLogger.LOG.errorInvokingMethod(method.getName(), stdContext, instanceManager);
  126. }
  127. } catch (NoSuchMethodException e1) {
  128. // Getter/setter not found
  129. }
  130. try {
  131. Field field = SecurityActions.lookupField(stdContext.getClass(), INSTANCE_MANAGER_FIELD_NAME);
  132. SecurityActions.ensureAccessible(field);
  133. try {
  134. field.set(stdContext, instanceManager);
  135. return;
  136. } catch (Exception e) {
  137. TomcatLogger.LOG.errorWritingField(field.getName(), stdContext, instanceManager);
  138. }
  139. } catch (NoSuchFieldException e1) {
  140. // Field not found
  141. }
  142. throw TomcatLogger.LOG.neitherFieldNorGetterSetterFound(stdContext.getClass());
  143. }
  144. }