View Javadoc

1   package liquibase.servicelocator;
2   
3   import java.io.File;
4   import java.io.FileInputStream;
5   import java.io.IOException;
6   import java.io.InputStream;
7   import java.net.URI;
8   import java.net.URISyntaxException;
9   import java.net.URL;
10  import java.net.URLConnection;
11  import java.net.URLDecoder;
12  import java.util.Arrays;
13  import java.util.Collections;
14  import java.util.Enumeration;
15  import java.util.HashMap;
16  import java.util.HashSet;
17  import java.util.LinkedHashSet;
18  import java.util.Map;
19  import java.util.Set;
20  import java.util.jar.JarEntry;
21  import java.util.jar.JarInputStream;
22  
23  import liquibase.logging.Logger;
24  import liquibase.logging.core.DefaultLogger;
25  
26  /**
27   * Default implement of {@link PackageScanClassResolver}
28   */
29  public class DefaultPackageScanClassResolver implements PackageScanClassResolver {
30  
31      private static Map<String, Set<String>> classesByJarUrl = new HashMap<String, Set<String>>();
32  
33      protected final transient Logger log = new DefaultLogger();
34      private Set<ClassLoader> classLoaders;
35      private Set<PackageScanFilter> scanFilters;
36  
37      @Override
38      public void addClassLoader(ClassLoader classLoader) {
39          try {
40              getClassLoaders().add(classLoader);
41          } catch (UnsupportedOperationException ex) {
42              // Ignore this exception as the PackageScanClassResolver
43              // don't want use any other classloader
44          }
45      }
46  
47      @Override
48      public void addFilter(PackageScanFilter filter) {
49          if (scanFilters == null) {
50              scanFilters = new LinkedHashSet<PackageScanFilter>();
51          }
52          scanFilters.add(filter);
53      }
54  
55      @Override
56      public void removeFilter(PackageScanFilter filter) {
57          if (scanFilters != null) {
58              scanFilters.remove(filter);
59          }
60      }
61  
62      @Override
63      public Set<ClassLoader> getClassLoaders() {
64          if (classLoaders == null) {
65              classLoaders = new HashSet<ClassLoader>();
66              ClassLoader ccl = Thread.currentThread().getContextClassLoader();
67              if (ccl != null) {
68                  log.debug("The thread context class loader: " + ccl + "  is used to load the class");
69                  classLoaders.add(ccl);
70              }
71              classLoaders.add(DefaultPackageScanClassResolver.class.getClassLoader());
72          }
73          return classLoaders;
74      }
75  
76      @Override
77      public void setClassLoaders(Set<ClassLoader> classLoaders) {
78          this.classLoaders = classLoaders;
79      }
80  
81      @Override
82      @SuppressWarnings("unchecked")
83      public Set<Class<?>> findImplementations(Class parent, String... packageNames) {
84          if (packageNames == null) {
85              return Collections.EMPTY_SET;
86          }
87  
88          log.debug("Searching for implementations of " + parent.getName() + " in packages: "
89                  + Arrays.asList(packageNames));
90  
91          PackageScanFilter test = getCompositeFilter(new AssignableToPackageScanFilter(parent));
92          Set<Class<?>> classes = new LinkedHashSet<Class<?>>();
93          for (String pkg : packageNames) {
94              find(test, pkg, classes);
95          }
96  
97          log.debug("Found: " + classes);
98  
99          return classes;
100     }
101 
102     @Override
103     @SuppressWarnings("unchecked")
104     public Set<Class<?>> findByFilter(PackageScanFilter filter, String... packageNames) {
105         if (packageNames == null) {
106             return Collections.EMPTY_SET;
107         }
108 
109         Set<Class<?>> classes = new LinkedHashSet<Class<?>>();
110         for (String pkg : packageNames) {
111             find(filter, pkg, classes);
112         }
113 
114         log.debug("Found: " + classes);
115 
116         return classes;
117     }
118 
119     protected void find(PackageScanFilter test, String packageName, Set<Class<?>> classes) {
120         packageName = packageName.replace('.', '/');
121 
122         Set<ClassLoader> set = getClassLoaders();
123 
124         for (ClassLoader classLoader : set) {
125             find(test, packageName, classLoader, classes);
126         }
127     }
128 
129     protected void find(PackageScanFilter test, String packageName, ClassLoader loader, Set<Class<?>> classes) {
130         log.debug("Searching for: " + test + " in package: " + packageName + " using classloader: "
131                 + loader.getClass().getName());
132 
133         Enumeration<URL> urls;
134         try {
135             urls = getResources(loader, packageName);
136             if (!urls.hasMoreElements()) {
137                 log.debug("No URLs returned by classloader");
138             }
139         } catch (IOException ioe) {
140             log.warning("Cannot read package: " + packageName, ioe);
141             return;
142         }
143 
144         while (urls.hasMoreElements()) {
145             URL url = null;
146             try {
147                 url = urls.nextElement();
148                 log.debug("URL from classloader: " + url);
149 
150                 url = customResourceLocator(url);
151 
152                 String urlPath = url.getFile();
153                 String host = null;
154                 urlPath = URLDecoder.decode(urlPath, "UTF-8");
155 
156                 if (url.getProtocol().equals("vfs") && !urlPath.startsWith("vfs")) {
157                     urlPath = "vfs:" + urlPath;
158                 }
159                 if (url.getProtocol().equals("vfszip") && !urlPath.startsWith("vfszip")) {
160                     urlPath = "vfszip:" + urlPath;
161                 }
162 
163                 log.debug("Decoded urlPath: " + urlPath + " with protocol: " + url.getProtocol());
164 
165                 // If it's a file in a directory, trim the stupid file: spec
166                 if (urlPath.startsWith("file:")) {
167                     // file path can be temporary folder which uses characters that the URLDecoder decodes wrong
168                     // for example + being decoded to something else (+ can be used in temp folders on Mac OS)
169                     // to remedy this then create new path without using the URLDecoder
170                     try {
171                         URI uri = new URI(url.getFile());
172                         host = uri.getHost();
173                         urlPath = uri.getPath();
174                     } catch (URISyntaxException e) {
175                         // fallback to use as it was given from the URLDecoder
176                         // this allows us to work on Windows if users have spaces in paths
177                     }
178 
179                     if (urlPath.startsWith("file:")) {
180                         urlPath = urlPath.substring(5);
181                     }
182                 }
183 
184                 // osgi bundles should be skipped
185                 if (url.toString().startsWith("bundle:") || urlPath.startsWith("bundle:")) {
186                     log.debug("It's a virtual osgi bundle, skipping");
187                     continue;
188                 }
189 
190                 // Else it's in a JAR, grab the path to the jar
191                 if (urlPath.contains(".jar/")) {
192                     urlPath = urlPath.replace(".jar/", ".jar!/");
193                 }
194 
195                 if (urlPath.indexOf('!') > 0) {
196                     urlPath = urlPath.substring(0, urlPath.indexOf('!'));
197                 }
198 
199                 // If a host component was given prepend it to the decoded path.
200                 // This still has its problems as we silently skip user and password
201                 // information etc. but it fixes UNC urls on windows.
202                 if (host != null) {
203                     if (urlPath.startsWith("/")) {
204                         urlPath = "//" + host + urlPath;
205                     } else {
206                         urlPath = "//" + host + "/" + urlPath;
207                     }
208                 }
209 
210                 log.debug("Scanning for classes in [" + urlPath + "] matching criteria: " + test);
211 
212                 File file = new File(urlPath);
213                 if (file.isDirectory()) {
214                     log.debug("Loading from directory using file: " + file);
215                     loadImplementationsInDirectory(test, packageName, file, classes);
216                 } else {
217                     InputStream stream;
218                     if (urlPath.startsWith("http:") || urlPath.startsWith("https:") || urlPath.startsWith("sonicfs:")
219                             || urlPath.startsWith("vfs:") || urlPath.startsWith("vfszip:")) {
220                         // load resources using http/https
221                         // sonic ESB requires to be loaded using a regular URLConnection
222                         URL urlStream = new URL(urlPath);
223                         log.debug("Loading from jar using " + urlStream.getProtocol() + ": " + urlPath);
224                         URLConnection con = urlStream.openConnection();
225                         // disable cache mainly to avoid jar file locking on Windows
226                         con.setUseCaches(false);
227                         stream = con.getInputStream();
228                     } else {
229                         log.debug("Loading from jar using file: " + file);
230                         stream = new FileInputStream(file);
231                     }
232 
233                     loadImplementationsInJar(test, packageName, stream, urlPath, classes);
234                 }
235             } catch (IOException e) {
236                 // use debug logging to avoid being to noisy in logs
237                 log.debug("Cannot read entries in url: " + url, e);
238             }
239         }
240     }
241 
242     // We can override this method to support the custom ResourceLocator
243 
244     protected URL customResourceLocator(URL url) throws IOException {
245         // Do nothing here
246         return url;
247     }
248 
249     /**
250      * Strategy to get the resources by the given classloader.
251      * <p/>
252      * Notice that in WebSphere platforms there is a {@link WebSpherePackageScanClassResolver} to take care of
253      * WebSphere's odditiy of resource loading.
254      * 
255      * @param loader
256      *            the classloader
257      * @param packageName
258      *            the packagename for the package to load
259      * @return URL's for the given package
260      * @throws IOException
261      *             is thrown by the classloader
262      */
263     protected Enumeration<URL> getResources(ClassLoader loader, String packageName) throws IOException {
264         log.debug("Getting resource URL for package: " + packageName + " with classloader: " + loader);
265 
266         // If the URL is a jar, the URLClassloader.getResources() seems to require a trailing slash. The
267         // trailing slash is harmless for other URLs
268         if (!packageName.endsWith("/")) {
269             packageName = packageName + "/";
270         }
271         return loader.getResources(packageName);
272     }
273 
274     private PackageScanFilter getCompositeFilter(PackageScanFilter filter) {
275         if (scanFilters != null) {
276             CompositePackageScanFilter composite = new CompositePackageScanFilter(scanFilters);
277             composite.addFilter(filter);
278             return composite;
279         }
280         return filter;
281     }
282 
283     /**
284      * Finds matches in a physical directory on a filesystem. Examines all files within a directory - if the File object
285      * is not a directory, and ends with <i>.class</i> the file is loaded and tested to see if it is acceptable
286      * according to the Test. Operates recursively to find classes within a folder structure matching the package
287      * structure.
288      * 
289      * @param test
290      *            a Test used to filter the classes that are discovered
291      * @param parent
292      *            the package name up to this directory in the package hierarchy. E.g. if /classes is in the classpath
293      *            and we wish to examine files in /classes/org/apache then the values of <i>parent</i> would be
294      *            <i>org/apache</i>
295      * @param location
296      *            a File object representing a directory
297      */
298     private void loadImplementationsInDirectory(PackageScanFilter test, String parent, File location,
299             Set<Class<?>> classes) {
300         File[] files = location.listFiles();
301         StringBuilder builder = null;
302 
303         for (File file : files) {
304             builder = new StringBuilder(100);
305             String name = file.getName();
306             if (name != null) {
307                 name = name.trim();
308                 builder.append(parent).append("/").append(name);
309                 String packageOrClass = parent == null ? name : builder.toString();
310 
311                 if (file.isDirectory()) {
312                     loadImplementationsInDirectory(test, packageOrClass, file, classes);
313                 } else if (name.endsWith(".class")) {
314                     addIfMatching(test, packageOrClass, classes);
315                 }
316             }
317         }
318     }
319 
320     /**
321      * Finds matching classes within a jar files that contains a folder structure matching the package structure. If the
322      * File is not a JarFile or does not exist a warning will be logged, but no error will be raised.
323      * 
324      * @param test
325      *            a Test used to filter the classes that are discovered
326      * @param parent
327      *            the parent package under which classes must be in order to be considered
328      * @param stream
329      *            the inputstream of the jar file to be examined for classes
330      * @param urlPath
331      *            the url of the jar file to be examined for classes
332      */
333     protected void loadImplementationsInJar(PackageScanFilter test, String parent, InputStream stream, String urlPath,
334             Set<Class<?>> classes) {
335         JarInputStream jarStream = null;
336         try {
337 
338             if (!classesByJarUrl.containsKey(urlPath)) {
339                 Set<String> names = new HashSet<String>();
340 
341                 if (stream instanceof JarInputStream) {
342                     jarStream = (JarInputStream) stream;
343                 } else {
344                     jarStream = new JarInputStream(stream);
345                 }
346 
347                 JarEntry entry;
348                 while ((entry = jarStream.getNextJarEntry()) != null) {
349                     String name = entry.getName();
350                     if (name != null) {
351                         name = name.trim();
352                         if (!entry.isDirectory() && name.endsWith(".class")) {
353                             names.add(name);
354                         }
355                     }
356                 }
357 
358                 classesByJarUrl.put(urlPath, names);
359             }
360 
361             for (String name : classesByJarUrl.get(urlPath)) {
362                 if (name.startsWith(parent)) {
363                     addIfMatching(test, name, classes);
364                 }
365             }
366         } catch (IOException ioe) {
367             log.warning("Cannot search jar file '" + urlPath + "' for classes matching criteria: " + test
368                     + " due to an IOException: " + ioe.getMessage(), ioe);
369         } finally {
370             try {
371                 if (jarStream != null) {
372                     jarStream.close();
373                 }
374             } catch (IOException ignore) {
375             }
376         }
377     }
378 
379     /**
380      * Add the class designated by the fully qualified class name provided to the set of resolved classes if and only if
381      * it is approved by the Test supplied.
382      * 
383      * @param test
384      *            the test used to determine if the class matches
385      * @param fqn
386      *            the fully qualified name of a class
387      */
388     protected void addIfMatching(PackageScanFilter test, String fqn, Set<Class<?>> classes) {
389         try {
390             String externalName = fqn.substring(0, fqn.indexOf('.')).replace('/', '.');
391             Set<ClassLoader> set = getClassLoaders();
392             boolean found = false;
393             for (ClassLoader classLoader : set) {
394                 log.debug("Testing that class " + externalName + " matches criteria [" + test + "] using classloader:"
395                         + classLoader);
396                 try {
397                     Class<?> type = classLoader.loadClass(externalName);
398                     log.debug("Loaded the class: " + type + " in classloader: " + classLoader);
399                     if (test.matches(type)) {
400                         log.debug("Found class: " + type + " which matches the filter in classloader: " + classLoader);
401                         classes.add(type);
402                     }
403                     found = true;
404                     break;
405                 } catch (ClassNotFoundException e) {
406                     log.debug("Cannot find class '" + fqn + "' in classloader: " + classLoader + ". Reason: " + e, e);
407                 } catch (NoClassDefFoundError e) {
408                     log.debug("Cannot find the class definition '" + fqn + "' in classloader: " + classLoader
409                             + ". Reason: " + e, e);
410                 } catch (Throwable e) {
411                     log.severe("Cannot load class '" + fqn + "' in classloader: " + classLoader + ".  Reason: " + e, e);
412                 }
413             }
414             if (!found) {
415                 // use debug to avoid being noisy in logs
416                 log.debug("Cannot find class '" + fqn + "' in any classloaders: " + set);
417             }
418         } catch (Exception e) {
419             log.warning(
420                     "Cannot examine class '" + fqn + "' due to a " + e.getClass().getName() + " with message: "
421                             + e.getMessage(), e);
422         }
423     }
424 
425 }