View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements. See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache license, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License. You may obtain a copy of the License at
8    *
9    *      http://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the license for the specific language governing permissions and
15   * limitations under the license.
16   */
17  package org.apache.logging.log4j.core.config.plugins.util;
18  
19  import java.io.File;
20  import java.io.FileInputStream;
21  import java.io.IOException;
22  import java.io.InputStream;
23  import java.io.UnsupportedEncodingException;
24  import java.net.JarURLConnection;
25  import java.net.URI;
26  import java.net.URISyntaxException;
27  import java.net.URL;
28  import java.net.URLDecoder;
29  import java.nio.charset.StandardCharsets;
30  import java.util.Arrays;
31  import java.util.Collection;
32  import java.util.Enumeration;
33  import java.util.HashSet;
34  import java.util.List;
35  import java.util.Set;
36  import java.util.jar.JarEntry;
37  import java.util.jar.JarFile;
38  import java.util.jar.JarInputStream;
39  
40  import org.apache.logging.log4j.Logger;
41  import org.apache.logging.log4j.core.util.Loader;
42  import org.apache.logging.log4j.status.StatusLogger;
43  import org.osgi.framework.FrameworkUtil;
44  import org.osgi.framework.wiring.BundleWiring;
45  
46  /**
47   * <p>
48   * ResolverUtil is used to locate classes that are available in the/a class path and meet arbitrary conditions. The two
49   * most common conditions are that a class implements/extends another class, or that is it annotated with a specific
50   * annotation. However, through the use of the {@link Test} class it is possible to search using arbitrary conditions.
51   * </p>
52   *
53   * <p>
54   * A ClassLoader is used to locate all locations (directories and jar files) in the class path that contain classes
55   * within certain packages, and then to load those classes and check them. By default the ClassLoader returned by
56   * {@code Thread.currentThread().getContextClassLoader()} is used, but this can be overridden by calling
57   * {@link #setClassLoader(ClassLoader)} prior to invoking any of the {@code find()} methods.
58   * </p>
59   *
60   * <p>
61   * General searches are initiated by calling the {@link #find(ResolverUtil.Test, String...)} method and supplying a
62   * package name and a Test instance. This will cause the named package <b>and all sub-packages</b> to be scanned for
63   * classes that meet the test. There are also utility methods for the common use cases of scanning multiple packages for
64   * extensions of particular classes, or classes annotated with a specific annotation.
65   * </p>
66   *
67   * <p>
68   * The standard usage pattern for the ResolverUtil class is as follows:
69   * </p>
70   *
71   * <pre>
72   * ResolverUtil resolver = new ResolverUtil();
73   * resolver.findInPackage(new CustomTest(), pkg1);
74   * resolver.find(new CustomTest(), pkg1);
75   * resolver.find(new CustomTest(), pkg1, pkg2);
76   * Set&lt;Class&lt;?&gt;&gt; beans = resolver.getClasses();
77   * </pre>
78   *
79   * <p>
80   * This class was copied and modified from Stripes - http://stripes.mc4j.org/confluence/display/stripes/Home
81   * </p>
82   */
83  public class ResolverUtil {
84      /** An instance of Log to use for logging in this class. */
85      private static final Logger LOGGER = StatusLogger.getLogger();
86  
87      private static final String VFSZIP = "vfszip";
88  
89      private static final String VFS = "vfs";
90  
91      private static final String JAR = "jar";
92  
93      private static final String BUNDLE_RESOURCE = "bundleresource";
94  
95      /** The set of matches being accumulated. */
96      private final Set<Class<?>> classMatches = new HashSet<>();
97  
98      /** The set of matches being accumulated. */
99      private final Set<URI> resourceMatches = new HashSet<>();
100 
101     /**
102      * The ClassLoader to use when looking for classes. If null then the ClassLoader returned by
103      * Thread.currentThread().getContextClassLoader() will be used.
104      */
105     private ClassLoader classloader;
106 
107     /**
108      * Provides access to the classes discovered so far. If no calls have been made to any of the {@code find()}
109      * methods, this set will be empty.
110      *
111      * @return the set of classes that have been discovered.
112      */
113     public Set<Class<?>> getClasses() {
114         return classMatches;
115     }
116 
117     /**
118      * Returns the matching resources.
119      *
120      * @return A Set of URIs that match the criteria.
121      */
122     public Set<URI> getResources() {
123         return resourceMatches;
124     }
125 
126     /**
127      * Returns the ClassLoader that will be used for scanning for classes. If no explicit ClassLoader has been set by
128      * the calling, the context class loader will be used.
129      *
130      * @return the ClassLoader that will be used to scan for classes
131      */
132     public ClassLoader getClassLoader() {
133         return classloader != null ? classloader : (classloader = Loader.getClassLoader(ResolverUtil.class, null));
134     }
135 
136     /**
137      * Sets an explicit ClassLoader that should be used when scanning for classes. If none is set then the context
138      * ClassLoader will be used.
139      *
140      * @param aClassloader
141      *        a ClassLoader to use when scanning for classes
142      */
143     public void setClassLoader(final ClassLoader aClassloader) {
144         this.classloader = aClassloader;
145     }
146 
147     /**
148      * Attempts to discover classes that pass the test. Accumulated classes can be accessed by calling
149      * {@link #getClasses()}.
150      *
151      * @param test
152      *        the test to determine matching classes
153      * @param packageNames
154      *        one or more package names to scan (including subpackages) for classes
155      */
156     public void find(final Test test, final String... packageNames) {
157         if (packageNames == null) {
158             return;
159         }
160 
161         for (final String pkg : packageNames) {
162             findInPackage(test, pkg);
163         }
164     }
165 
166     /**
167      * Scans for classes starting at the package provided and descending into subpackages. Each class is offered up to
168      * the Test as it is discovered, and if the Test returns true the class is retained. Accumulated classes can be
169      * fetched by calling {@link #getClasses()}.
170      *
171      * @param test
172      *        an instance of {@link Test} that will be used to filter classes
173      * @param packageName
174      *        the name of the package from which to start scanning for classes, e.g. {@code net.sourceforge.stripes}
175      */
176     public void findInPackage(final Test test, String packageName) {
177         packageName = packageName.replace('.', '/');
178         final ClassLoader loader = getClassLoader();
179         Enumeration<URL> urls;
180 
181         try {
182             urls = loader.getResources(packageName);
183         } catch (final IOException ioe) {
184             LOGGER.warn("Could not read package: {}", packageName, ioe);
185             return;
186         }
187 
188         while (urls.hasMoreElements()) {
189             try {
190                 final URL url = urls.nextElement();
191                 final String urlPath = extractPath(url);
192 
193                 LOGGER.info("Scanning for classes in '{}' matching criteria {}", urlPath , test);
194                 // Check for a jar in a war in JBoss
195                 if (VFSZIP.equals(url.getProtocol())) {
196                     final String path = urlPath.substring(0, urlPath.length() - packageName.length() - 2);
197                     final URL newURL = new URL(url.getProtocol(), url.getHost(), path);
198                     @SuppressWarnings("resource")
199                     final JarInputStream stream = new JarInputStream(newURL.openStream());
200                     try {
201                         loadImplementationsInJar(test, packageName, path, stream);
202                     } finally {
203                         close(stream, newURL);
204                     }
205                 } else if (VFS.equals(url.getProtocol())) {
206                     final String containerPath = urlPath.substring(1, urlPath.length() - packageName.length() - 2);
207                     final File containerFile = new File(containerPath);
208                     if (containerFile.exists()) {
209                         if (containerFile.isDirectory()) {
210                             loadImplementationsInDirectory(test, packageName, new File(containerFile, packageName));
211                         } else {
212                             loadImplementationsInJar(test, packageName, containerFile);
213                         }
214                     } else {
215                         // fallback code for Jboss/Wildfly, if the file couldn't be found
216                         // by loading the path as a file, try to read the jar as a stream
217                         final String path = urlPath.substring(0, urlPath.length() - packageName.length() - 2);
218                         final URL newURL = new URL(url.getProtocol(), url.getHost(), path);
219 
220                         try (final InputStream is = newURL.openStream()) {
221                             final JarInputStream jarStream;
222                             if (is instanceof JarInputStream) {
223                                 jarStream = (JarInputStream) is;
224                             } else {
225                                 jarStream = new JarInputStream(is);
226                             }
227                             loadImplementationsInJar(test, packageName, path, jarStream);
228                         }
229                     }
230                 } else if (BUNDLE_RESOURCE.equals(url.getProtocol())) {
231                     loadImplementationsInBundle(test, packageName);
232                 } else if (JAR.equals(url.getProtocol())) {
233                     loadImplementationsInJar(test, packageName, url);
234                 } else {
235                     final File file = new File(urlPath);
236                     if (file.isDirectory()) {
237                         loadImplementationsInDirectory(test, packageName, file);
238                     } else {
239                         loadImplementationsInJar(test, packageName, file);
240                     }
241                 }
242             } catch (final IOException | URISyntaxException ioe) {
243                 LOGGER.warn("Could not read entries", ioe);
244             }
245         }
246     }
247 
248     String extractPath(final URL url) throws UnsupportedEncodingException, URISyntaxException {
249         String urlPath = url.getPath(); // same as getFile but without the Query portion
250         // System.out.println(url.getProtocol() + "->" + urlPath);
251 
252         // I would be surprised if URL.getPath() ever starts with "jar:" but no harm in checking
253         if (urlPath.startsWith("jar:")) {
254             urlPath = urlPath.substring(4);
255         }
256         // For jar: URLs, the path part starts with "file:"
257         if (urlPath.startsWith("file:")) {
258             urlPath = urlPath.substring(5);
259         }
260         // If it was in a JAR, grab the path to the jar
261         final int bangIndex = urlPath.indexOf('!');
262         if (bangIndex > 0) {
263             urlPath = urlPath.substring(0, bangIndex);
264         }
265 
266         // LOG4J2-445
267         // Finally, decide whether to URL-decode the file name or not...
268         final String protocol = url.getProtocol();
269         final List<String> neverDecode = Arrays.asList(VFS, VFSZIP, BUNDLE_RESOURCE);
270         if (neverDecode.contains(protocol)) {
271             return urlPath;
272         }
273         final String cleanPath = new URI(urlPath).getPath();
274         if (new File(cleanPath).exists()) {
275             // if URL-encoded file exists, don't decode it
276             return cleanPath;
277         }
278         return URLDecoder.decode(urlPath, StandardCharsets.UTF_8.name());
279     }
280 
281     private void loadImplementationsInBundle(final Test test, final String packageName) {
282         final BundleWiring wiring = FrameworkUtil.getBundle(ResolverUtil.class).adapt(BundleWiring.class);
283         final Collection<String> list = wiring.listResources(packageName, "*.class",
284                 BundleWiring.LISTRESOURCES_RECURSE);
285         for (final String name : list) {
286             addIfMatching(test, name);
287         }
288     }
289 
290     /**
291      * Finds matches in a physical directory on a file system. Examines all files within a directory - if the File object
292      * is not a directory, and ends with <i>.class</i> the file is loaded and tested to see if it is acceptable
293      * according to the Test. Operates recursively to find classes within a folder structure matching the package
294      * structure.
295      *
296      * @param test
297      *        a Test used to filter the classes that are discovered
298      * @param parent
299      *        the package name up to this directory in the package hierarchy. E.g. if /classes is in the classpath and
300      *        we wish to examine files in /classes/org/apache then the values of <i>parent</i> would be
301      *        <i>org/apache</i>
302      * @param location
303      *        a File object representing a directory
304      */
305     private void loadImplementationsInDirectory(final Test test, final String parent, final File location) {
306         final File[] files = location.listFiles();
307         if (files == null) {
308             return;
309         }
310 
311         StringBuilder builder;
312         for (final File file : files) {
313             builder = new StringBuilder();
314             builder.append(parent).append('/').append(file.getName());
315             final String packageOrClass = parent == null ? file.getName() : builder.toString();
316 
317             if (file.isDirectory()) {
318                 loadImplementationsInDirectory(test, packageOrClass, file);
319             } else if (isTestApplicable(test, file.getName())) {
320                 addIfMatching(test, packageOrClass);
321             }
322         }
323     }
324 
325     private boolean isTestApplicable(final Test test, final String path) {
326         return test.doesMatchResource() || path.endsWith(".class") && test.doesMatchClass();
327     }
328 
329     /**
330      * Finds matching classes within a jar files that contains a folder structure matching the package structure. If the
331      * File is not a JarFile or does not exist a warning will be logged, but no error will be raised.
332      *
333      * @param test
334      *        a Test used to filter the classes that are discovered
335      * @param parent
336      *        the parent package under which classes must be in order to be considered
337      * @param url
338      *        the url that identifies the jar containing the resource.
339      */
340     private void loadImplementationsInJar(final Test test, final String parent, final URL url) {
341         JarURLConnection connection = null;
342         try {
343             connection = (JarURLConnection) url.openConnection();
344             if (connection != null) {
345                 try (JarFile jarFile = connection.getJarFile()) {
346                     Enumeration<JarEntry> entries = jarFile.entries();
347                     while (entries.hasMoreElements()) {
348                         JarEntry entry = entries.nextElement();
349                         final String name = entry.getName();
350                         if (!entry.isDirectory() && name.startsWith(parent) && isTestApplicable(test, name)) {
351                             addIfMatching(test, name);
352                         }
353                     }
354                 }
355             } else {
356                 LOGGER.error("Could not establish connection to {}", url.toString());
357             }
358         } catch (final IOException ex) {
359             LOGGER.error("Could not search JAR file '{}' for classes matching criteria {}, file not found",
360                 url.toString(), test, ex);
361         }
362     }
363 
364     /**
365      * Finds matching classes within a jar files that contains a folder structure matching the package structure. If the
366      * File is not a JarFile or does not exist a warning will be logged, but no error will be raised.
367      *
368      * @param test
369      *        a Test used to filter the classes that are discovered
370      * @param parent
371      *        the parent package under which classes must be in order to be considered
372      * @param jarFile
373      *        the jar file to be examined for classes
374      */
375     private void loadImplementationsInJar(final Test test, final String parent, final File jarFile) {
376         JarInputStream jarStream = null;
377         try {
378             jarStream = new JarInputStream(new FileInputStream(jarFile));
379             loadImplementationsInJar(test, parent, jarFile.getPath(), jarStream);
380         } catch (final IOException ex) {
381             LOGGER.error("Could not search JAR file '{}' for classes matching criteria {}, file not found", jarFile,
382                     test, ex);
383         } finally {
384             close(jarStream, jarFile);
385         }
386     }
387 
388     /**
389      * @param jarStream
390      * @param source
391      */
392     private void close(final JarInputStream jarStream, final Object source) {
393         if (jarStream != null) {
394             try {
395                 jarStream.close();
396             } catch (final IOException e) {
397                 LOGGER.error("Error closing JAR file stream for {}", source, e);
398             }
399         }
400     }
401 
402     /**
403      * Finds matching classes within a jar files that contains a folder structure matching the package structure. If the
404      * File is not a JarFile or does not exist a warning will be logged, but no error will be raised.
405      *
406      * @param test
407      *        a Test used to filter the classes that are discovered
408      * @param parent
409      *        the parent package under which classes must be in order to be considered
410      * @param stream
411      *        The jar InputStream
412      */
413     private void loadImplementationsInJar(final Test test, final String parent, final String path,
414             final JarInputStream stream) {
415 
416         try {
417             JarEntry entry;
418 
419             while ((entry = stream.getNextJarEntry()) != null) {
420                 final String name = entry.getName();
421                 if (!entry.isDirectory() && name.startsWith(parent) && isTestApplicable(test, name)) {
422                     addIfMatching(test, name);
423                 }
424             }
425         } catch (final IOException ioe) {
426             LOGGER.error("Could not search JAR file '{}' for classes matching criteria {} due to an IOException", path,
427                     test, ioe);
428         }
429     }
430 
431     /**
432      * Add the class designated by the fully qualified class name provided to the set of resolved classes if and only if
433      * it is approved by the Test supplied.
434      *
435      * @param test
436      *        the test used to determine if the class matches
437      * @param fqn
438      *        the fully qualified name of a class
439      */
440     protected void addIfMatching(final Test test, final String fqn) {
441         try {
442             final ClassLoader loader = getClassLoader();
443             if (test.doesMatchClass()) {
444                 final String externalName = fqn.substring(0, fqn.indexOf('.')).replace('/', '.');
445                 if (LOGGER.isDebugEnabled()) {
446                     LOGGER.debug("Checking to see if class {} matches criteria {}", externalName, test);
447                 }
448 
449                 final Class<?> type = loader.loadClass(externalName);
450                 if (test.matches(type)) {
451                     classMatches.add(type);
452                 }
453             }
454             if (test.doesMatchResource()) {
455                 URL url = loader.getResource(fqn);
456                 if (url == null) {
457                     url = loader.getResource(fqn.substring(1));
458                 }
459                 if (url != null && test.matches(url.toURI())) {
460                     resourceMatches.add(url.toURI());
461                 }
462             }
463         } catch (final Throwable t) {
464             LOGGER.warn("Could not examine class {}", fqn, t);
465         }
466     }
467 
468     /**
469      * A simple interface that specifies how to test classes to determine if they are to be included in the results
470      * produced by the ResolverUtil.
471      */
472     public interface Test {
473         /**
474          * Will be called repeatedly with candidate classes. Must return True if a class is to be included in the
475          * results, false otherwise.
476          *
477          * @param type
478          *        The Class to match against.
479          * @return true if the Class matches.
480          */
481         boolean matches(Class<?> type);
482 
483         /**
484          * Test for a resource.
485          *
486          * @param resource
487          *        The URI to the resource.
488          * @return true if the resource matches.
489          */
490         boolean matches(URI resource);
491 
492         boolean doesMatchClass();
493 
494         boolean doesMatchResource();
495     }
496 
497 }