/core/src/main/java/com/ocpsoft/pretty/faces/config/annotation/AbstractClassFinder.java

http://github.com/ocpsoft/prettyfaces · Java · 235 lines · 85 code · 34 blank · 116 comment · 9 complexity · ea4723488d69906d7306273e28793300 MD5 · raw file

  1. /*
  2. * Copyright 2010 Lincoln Baxter, III
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.ocpsoft.pretty.faces.config.annotation;
  17. import java.io.IOException;
  18. import java.io.InputStream;
  19. import java.net.URL;
  20. import javax.servlet.ServletContext;
  21. import org.apache.commons.logging.Log;
  22. import org.apache.commons.logging.LogFactory;
  23. /**
  24. * Base class for implementations of the {@link ClassFinder} interface.
  25. *
  26. * @see ClassFinder
  27. * @author Christian Kaltepoth
  28. */
  29. public abstract class AbstractClassFinder implements ClassFinder
  30. {
  31. /**
  32. * Common logger for all implementations
  33. */
  34. protected final Log log = LogFactory.getLog(this.getClass());
  35. /**
  36. * The {@link ServletContext}
  37. */
  38. protected final ServletContext servletContext;
  39. /**
  40. * The {@link ClassLoader} to get classes from
  41. */
  42. protected final ClassLoader classLoader;
  43. /**
  44. * The filter for checking which classes to process
  45. */
  46. protected final PackageFilter packageFilter;
  47. /**
  48. * The filter to check bytecode for interesting annotations
  49. */
  50. private final ByteCodeAnnotationFilter byteCodeAnnotationFilter;
  51. /**
  52. * Initialization procedure
  53. *
  54. * @param servletContext
  55. * The {@link ServletContext} of the web application.
  56. * @param classLoader
  57. * The {@link ClassLoader} to use for loading classes
  58. * @param packageFilter
  59. * The {@link PackageFilter} used to check if a package has to be
  60. * scanned.
  61. */
  62. public AbstractClassFinder(ServletContext servletContext, ClassLoader classLoader, PackageFilter packageFilter)
  63. {
  64. this.servletContext = servletContext;
  65. this.classLoader = classLoader;
  66. this.packageFilter = packageFilter;
  67. this.byteCodeAnnotationFilter = new ByteCodeAnnotationFilter();
  68. }
  69. /**
  70. * Strip everything up to and including a given prefix from a string.
  71. *
  72. * @param str
  73. * The string to process
  74. * @param prefix
  75. * The prefix
  76. * @return the stripped string or <code>null</code> if the prefix has not
  77. * been found
  78. */
  79. protected String stripKnownPrefix(String str, String prefix)
  80. {
  81. int startIndex = str.lastIndexOf(prefix);
  82. if (startIndex != -1)
  83. {
  84. return str.substring(startIndex + prefix.length());
  85. }
  86. return null;
  87. }
  88. /**
  89. * <p>
  90. * Creates a FQCN from an {@link URL} representing a <code>.class</code>
  91. * file.
  92. * </p>
  93. *
  94. * @param url
  95. * The path of the class file
  96. * @return the FQCN of the class
  97. */
  98. protected static String getClassName(String filename)
  99. {
  100. // end index is just before ".class"
  101. int endIndex = filename.length() - ".class".length();
  102. // extract relevant part of the path
  103. String relativePath = filename.substring(0, endIndex);
  104. // replace / by . to create FQCN
  105. return relativePath.replace('/', '.');
  106. }
  107. /**
  108. * Checks if a supplied class has to be processed by checking the package
  109. * name against the {@link PackageFilter}.
  110. *
  111. * @param className
  112. * FQCN of the class
  113. * @return <code>true</code> for classes to process, <code>false</code> for
  114. * classes to ignore
  115. */
  116. protected boolean mustProcessClass(String className)
  117. {
  118. // the default package
  119. String packageName = "";
  120. // find last dot in class name to determine the package name
  121. int packageEndIndex = className.lastIndexOf(".");
  122. if (packageEndIndex != -1)
  123. {
  124. packageName = className.substring(0, packageEndIndex);
  125. }
  126. // check filter
  127. return packageFilter.isAllowedPackage(packageName);
  128. }
  129. /**
  130. * <p>
  131. * Handle a single class to process. This method should only be called if the
  132. * class name is accepted by the {@link PackageFilter}.
  133. * </p>
  134. * <p>
  135. * If <code>classFileStream</code> is not <code>null</code> the method will first
  136. * try to check whether the class files may contain annotations by scanning
  137. * it with the {@link ByteCodeAnnotationFilter}. If no {@link InputStream} is
  138. * supplied, this check will be skipped. After that the method will create an
  139. * instance of the class and then call
  140. * {@link PrettyAnnotationHandler#processClass(Class)}.
  141. * </p>
  142. * <p>
  143. * Please not the the called of this method is responsible to close the
  144. * supplied {@link InputStream}!
  145. * </p>
  146. *
  147. * @param className
  148. * The FQCN of the class
  149. * @param classFileStream
  150. * The Java class file of the class (may be <code>null</code>)
  151. * @param handler
  152. * the handler to notify
  153. */
  154. protected void processClass(String className, InputStream classFileStream, PrettyAnnotationHandler handler)
  155. {
  156. // bytecode check is only performed if the InputStream is available
  157. if (classFileStream != null)
  158. {
  159. // we must take care of IOExceptions thrown by ByteCodeAnnotationFilter
  160. try
  161. {
  162. // call bytecode filter
  163. boolean shouldScanClass = byteCodeAnnotationFilter.accept(classFileStream);
  164. // No annotations -> abort
  165. if (!shouldScanClass)
  166. {
  167. return;
  168. }
  169. // filter says we should scan the class
  170. if (log.isDebugEnabled())
  171. {
  172. log.debug("Bytecode filter recommends to scan class: " + className);
  173. }
  174. }
  175. catch (IOException e)
  176. {
  177. if (log.isDebugEnabled())
  178. {
  179. log.debug("Failed to parse class file: " + className, e);
  180. }
  181. }
  182. }
  183. try
  184. {
  185. // request this class from the ClassLoader
  186. Class<?> clazz = classLoader.loadClass(className);
  187. // call handler
  188. handler.processClass(clazz);
  189. }
  190. catch (NoClassDefFoundError e)
  191. {
  192. // reference to another class unknown to the classloader
  193. log.debug("Could not load class '" + className + "': " + e.toString());
  194. }
  195. catch (ClassNotFoundException e)
  196. {
  197. // should no happen, because we found the class on the classpath
  198. throw new IllegalStateException("Unable to load class: " + className, e);
  199. }
  200. }
  201. }