All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.jetty.webapp.ClasspathPattern Maven / Gradle / Ivy

The newest version!
//
//  ========================================================================
//  Copyright (c) 1995-2019 Mort Bay Consulting Pty. Ltd.
//  ------------------------------------------------------------------------
//  All rights reserved. This program and the accompanying materials
//  are made available under the terms of the Eclipse Public License v1.0
//  and Apache License v2.0 which accompanies this distribution.
//
//      The Eclipse Public License is available at
//      http://www.eclipse.org/legal/epl-v10.html
//
//      The Apache License v2.0 is available at
//      http://www.opensource.org/licenses/apache2.0.php
//
//  You may elect to redistribute this code under either of these licenses.
//  ========================================================================
//


package org.eclipse.jetty.webapp;

import java.io.File;
import java.io.IOException;
import java.net.URI;
import java.net.URL;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Predicate;

import org.eclipse.jetty.io.RuntimeIOException;
import org.eclipse.jetty.util.ArrayTernaryTrie;
import org.eclipse.jetty.util.IncludeExcludeSet;
import org.eclipse.jetty.util.TypeUtil;
import org.eclipse.jetty.util.URIUtil;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.resource.Resource;

/**
 * Classpath classes list performs pattern matching of a class name 
 * against an internal array of classpath pattern entries.
 * A class pattern is a string of one of the forms:
    *
  • 'org.package.SomeClass' will match a specific class *
  • 'org.package.' will match a specific package hierarchy *
  • 'org.package.SomeClass$NestedClass ' will match a nested class exactly otherwise. * Nested classes are matched by their containing class. (eg. org.example.MyClass * matches org.example.MyClass$AnyNestedClass) *
  • 'file:///some/location/' - A file system directory from which * the class was loaded *
  • 'file:///some/location.jar' - The URI of a jar file from which * the class was loaded *
  • 'jrt:/modulename' - A Java9 module name
  • *
  • Any of the above patterns preceded by '-' will exclude rather than include the match. *
* When class is initialized from a classpath pattern string, entries * in this string should be separated by ':' (semicolon) or ',' (comma). */ public class ClasspathPattern extends AbstractSet { private static final Logger LOG = Log.getLogger(ClasspathPattern.class); private static class Entry { private final String _pattern; private final String _name; private final boolean _inclusive; protected Entry(String name, boolean inclusive) { _name = name; _inclusive = inclusive; _pattern = inclusive ? _name : ("-"+_name); } public String getPattern() { return _pattern; } public String getName() { return _name; } @Override public String toString() { return _pattern; } @Override public int hashCode() { return _pattern.hashCode(); } @Override public boolean equals(Object o) { return (o instanceof Entry) && _pattern.equals(((Entry)o)._pattern); } public boolean isInclusive() { return _inclusive; } } private static class PackageEntry extends Entry { protected PackageEntry(String name, boolean inclusive) { super(name, inclusive); } } private static class ClassEntry extends Entry { protected ClassEntry(String name, boolean inclusive) { super(name, inclusive); } } private static class LocationEntry extends Entry { private final File _file; protected LocationEntry(String name, boolean inclusive) { super(name, inclusive); if (!getName().startsWith("file:")) throw new IllegalArgumentException(name); try { _file = Resource.newResource(getName()).getFile(); } catch(IOException e) { throw new RuntimeIOException(e); } } public File getFile() { return _file; } } private static class ModuleEntry extends Entry { private final String _module; protected ModuleEntry(String name, boolean inclusive) { super(name, inclusive); if (!getName().startsWith("jrt:")) throw new IllegalArgumentException(name); _module = getName().split("/")[1]; } public String getModule() { return _module; } } public static class ByPackage extends AbstractSet implements Predicate { private final ArrayTernaryTrie.Growing _entries = new ArrayTernaryTrie.Growing<>(false,512,512); @Override public boolean test(String name) { return _entries.getBest(name)!=null; } @Override public Iterator iterator() { return _entries.keySet().stream().map(_entries::get).iterator(); } @Override public int size() { return _entries.size(); } @Override public boolean isEmpty() { return _entries.isEmpty(); } @Override public boolean add(Entry entry) { String name = entry.getName(); if (entry instanceof ClassEntry) name+="$"; else if (!(entry instanceof PackageEntry)) throw new IllegalArgumentException(entry.toString()); else if (".".equals(name)) name=""; if (_entries.get(name)!=null) return false; return _entries.put(name,entry); } @Override public boolean remove(Object entry) { if (!(entry instanceof Entry)) return false; return _entries.remove(((Entry)entry).getName())!=null; } @Override public void clear() { _entries.clear(); } } @SuppressWarnings("serial") public static class ByClass extends HashSet implements Predicate { private final Map _entries = new HashMap<>(); @Override public boolean test(String name) { return _entries.containsKey(name); } @Override public Iterator iterator() { return _entries.values().iterator(); } @Override public int size() { return _entries.size(); } @Override public boolean add(Entry entry) { if (!(entry instanceof ClassEntry)) throw new IllegalArgumentException(entry.toString()); return _entries.put(entry.getName(),entry)==null; } @Override public boolean remove(Object entry) { if (!(entry instanceof Entry)) return false; return _entries.remove(((Entry)entry).getName())!=null; } } public static class ByPackageOrName extends AbstractSet implements Predicate { private final ByClass _byClass = new ByClass(); private final ByPackage _byPackage = new ByPackage(); @Override public boolean test(String name) { return _byPackage.test(name) || _byClass.test(name) ; } @Override public Iterator iterator() { // by package contains all entries (classes are also $ packages). return _byPackage.iterator(); } @Override public int size() { return _byPackage.size(); } @Override public boolean add(Entry entry) { if (entry instanceof PackageEntry) return _byPackage.add(entry); if (entry instanceof ClassEntry) { // Add class name to packages also as classes act // as packages for nested classes. boolean added = _byPackage.add(entry); added = _byClass.add(entry) || added; return added; } throw new IllegalArgumentException(); } @Override public boolean remove(Object o) { if (!(o instanceof Entry)) return false; boolean removedPackage = _byPackage.remove(o); boolean removedClass = _byClass.remove(o); return removedPackage || removedClass; } @Override public void clear() { _byPackage.clear(); _byClass.clear(); } } @SuppressWarnings("serial") public static class ByLocation extends HashSet implements Predicate { @Override public boolean test(URI uri) { if (!uri.getScheme().equals("file")) return false; Path path = Paths.get(uri); for (Entry entry : this) { if (!(entry instanceof LocationEntry)) throw new IllegalStateException(); File file = ((LocationEntry)entry).getFile(); if (file.isDirectory()) { if (path.startsWith(file.toPath())) { return true; } } else { if (path.equals(file.toPath())) { return true; } } } return false; } } @SuppressWarnings("serial") public static class ByModule extends HashSet implements Predicate { private final ArrayTernaryTrie.Growing _entries = new ArrayTernaryTrie.Growing<>(false,512,512); @Override public boolean test(URI uri) { if (!uri.getScheme().equalsIgnoreCase("jrt")) return false; String module = uri.getPath(); int end = module.indexOf('/',1); if (end<1) end = module.length(); return _entries.get(module,1,end-1)!=null; } @Override public Iterator iterator() { return _entries.keySet().stream().map(_entries::get).iterator(); } @Override public int size() { return _entries.size(); } @Override public boolean add(Entry entry) { if (!(entry instanceof ModuleEntry)) throw new IllegalArgumentException(entry.toString()); String module = ((ModuleEntry)entry).getModule(); if (_entries.get(module)!=null) return false; _entries.put(module,entry); return true; } @Override public boolean remove(Object entry) { if (!(entry instanceof Entry)) return false; return _entries.remove(((Entry)entry).getName())!=null; } } public static class ByLocationOrModule extends AbstractSet implements Predicate { private final ByLocation _byLocation = new ByLocation(); private final ByModule _byModule = new ByModule(); @Override public boolean test(URI name) { return _byLocation.test(name) || _byModule.test(name); } @Override public Iterator iterator() { Set entries = new HashSet<>(); entries.addAll(_byLocation); entries.addAll(_byModule); return entries.iterator(); } @Override public int size() { return _byLocation.size()+_byModule.size(); } @Override public boolean add(Entry entry) { if (entry instanceof LocationEntry) return _byLocation.add(entry); if (entry instanceof ModuleEntry) return _byModule.add(entry); throw new IllegalArgumentException(entry.toString()); } @Override public boolean remove(Object o) { if (o instanceof LocationEntry) return _byLocation.remove(o); if (o instanceof ModuleEntry) return _byModule.remove(o); return false; } @Override public void clear() { _byLocation.clear(); _byModule.clear(); } } Map _entries = new HashMap<>(); IncludeExcludeSet _patterns = new IncludeExcludeSet<>(ByPackageOrName.class); IncludeExcludeSet _locations = new IncludeExcludeSet<>(ByLocationOrModule.class); public ClasspathPattern() { } public ClasspathPattern(String[] patterns) { setAll(patterns); } public ClasspathPattern(String pattern) { add(pattern); } public boolean include(String name) { if (name==null) return false; return add(newEntry(name,true)); } public boolean include(String... name) { boolean added = false; for (String n:name) if (n!=null) added = add(newEntry(n,true)) || added; return added; } public boolean exclude(String name) { if (name==null) return false; return add(newEntry(name,false)); } public boolean exclude(String... name) { boolean added = false; for (String n:name) if (n!=null) added = add(newEntry(n,false)) || added; return added; } @Override public boolean add(String pattern) { if (pattern==null) return false; return add(newEntry(pattern)); } public boolean add(String... pattern) { boolean added = false; for (String p:pattern) if (p!=null) added = add(newEntry(p)) || added; return added; } protected Entry newEntry(String pattern) { if (pattern.startsWith("-")) return newEntry(pattern.substring(1),false); return newEntry(pattern,true); } protected Entry newEntry(String name, boolean inclusive) { if (name.startsWith("-")) throw new IllegalStateException(name); if (name.startsWith("file:")) return new LocationEntry(name, inclusive); if (name.startsWith("jrt:")) return new ModuleEntry(name, inclusive); if (name.endsWith(".")) return new PackageEntry(name, inclusive); return new ClassEntry(name,inclusive); } protected boolean add(Entry entry) { if (_entries.containsKey(entry.getPattern())) return false; _entries.put(entry.getPattern(),entry); if (entry instanceof LocationEntry || entry instanceof ModuleEntry) { if (entry.isInclusive()) _locations.include(entry); else _locations.exclude(entry); } else { if (entry.isInclusive()) _patterns.include(entry); else _patterns.exclude(entry); } return true; } @Override public boolean remove(Object o) { if (!(o instanceof String)) return false; String pattern = (String)o; Entry entry = _entries.remove(pattern); if (entry==null) return false; List saved = new ArrayList<>(_entries.values()); clear(); for (Entry e:saved) add(e); return true; } @Override public void clear() { _entries.clear(); _patterns.clear(); _locations.clear(); } @Override public Iterator iterator() { return _entries.keySet().iterator(); } @Override public int size() { return _entries.size(); } /** * Initialize the matcher by parsing each classpath pattern in an array * * @param classes array of classpath patterns */ private void setAll(String[] classes) { _entries.clear(); addAll(classes); } /** * @param classes array of classpath patterns */ private void addAll(String[] classes) { if (classes!=null) addAll(Arrays.asList(classes)); } /** * @return array of classpath patterns */ public String[] getPatterns() { return toArray(new String[_entries.size()]); } /** * Match the class name against the pattern * * @param name name of the class to match * @return true if class matches the pattern */ public boolean match(String name) { return _patterns.test(name); } /** * Match the class name against the pattern * * @param clazz A class to try to match * @return true if class matches the pattern */ public boolean match(Class clazz) { try { Boolean byName = _patterns.isIncludedAndNotExcluded(clazz.getName()); if (Boolean.FALSE.equals(byName)) return byName; // Already excluded so no need to check location. URI location = TypeUtil.getLocationOfClass(clazz); Boolean byLocation = location == null ? null : _locations.isIncludedAndNotExcluded(location); if (LOG.isDebugEnabled()) LOG.debug("match {} from {} byName={} byLocation={} in {}",clazz,location,byName,byLocation,this); // Combine the tri-state match of both IncludeExclude Sets boolean included = Boolean.TRUE.equals(byName) || Boolean.TRUE.equals(byLocation) || (byName==null && !_patterns.hasIncludes() && byLocation==null && !_locations.hasIncludes()); boolean excluded = Boolean.FALSE.equals(byName) || Boolean.FALSE.equals(byLocation); return included && !excluded; } catch (Exception e) { LOG.warn(e); } return false; } public boolean match(String name, URL url) { // Strip class suffix for name matching if (name.endsWith(".class")) name=name.substring(0,name.length()-6); // Treat path elements as packages for name matching name=name.replace("/","."); Boolean byName = _patterns.isIncludedAndNotExcluded(name); if (Boolean.FALSE.equals(byName)) return byName; // Already excluded so no need to check location. // Try to find a file path for location matching Boolean byLocation = null; try { URI jarUri = URIUtil.getJarSource(url.toURI()); if ("file".equalsIgnoreCase(jarUri.getScheme())) { byLocation = _locations.isIncludedAndNotExcluded(jarUri); } } catch(Exception e) { LOG.ignore(e); } // Combine the tri-state match of both IncludeExclude Sets boolean included = Boolean.TRUE.equals(byName) || Boolean.TRUE.equals(byLocation) || (byName==null && !_patterns.hasIncludes() && byLocation==null && !_locations.hasIncludes()); boolean excluded = Boolean.FALSE.equals(byName) || Boolean.FALSE.equals(byLocation); return included && !excluded; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy