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

org.eclipse.jetty.ee10.webapp.WebAppClassLoader Maven / Gradle / Ivy

//
// ========================================================================
// Copyright (c) 1995 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//

package org.eclipse.jetty.ee10.webapp;

import java.io.IOException;
import java.io.InputStream;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Path;
import java.security.CodeSource;
import java.security.PermissionCollection;
import java.security.PrivilegedExceptionAction;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.concurrent.CopyOnWriteArrayList;

import org.eclipse.jetty.util.ClassVisibilityChecker;
import org.eclipse.jetty.util.FileID;
import org.eclipse.jetty.util.IO;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.TypeUtil;
import org.eclipse.jetty.util.resource.Resource;
import org.eclipse.jetty.util.resource.ResourceCollators;
import org.eclipse.jetty.util.resource.ResourceFactory;
import org.eclipse.jetty.util.resource.Resources;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * ClassLoader for HttpContext.
 * 

* Specializes URLClassLoader with some utility and file mapping * methods. *

* This loader implements the Servlet specification behavior that may invert the normal Java classloader parent priority * behaviour. The {@link ClassVisibilityChecker} API of the {@link WebAppClassLoader.Context} implementation is * used to determine which classes from the parent classloader are hidden from the context, and which are protected * from being overridden by the context. *

* Java compliant loading, where the parent loader always has priority, can be selected with the * {@link WebAppContext#setParentLoaderPriority(boolean)} method. */ public class WebAppClassLoader extends URLClassLoader implements ClassVisibilityChecker { static { registerAsParallelCapable(); } private static final Logger LOG = LoggerFactory.getLogger(WebAppClassLoader.class); private static final ThreadLocal __loadServerClasses = new ThreadLocal<>(); private final Context _context; private final ClassLoader _parent; private final Set _extensions = new HashSet<>(); private String _name = String.valueOf(hashCode()); private final List _transformers = new CopyOnWriteArrayList<>(); private final ResourceFactory.Closeable _resourceFactory = ResourceFactory.closeable(); /** * The Context in which the classloader operates. */ public interface Context extends ClassVisibilityChecker { /** * Convert a URL or path to a Resource. * The default implementation * is a wrapper for {@link ResourceFactory#newResource(String)}. * * @param urlOrPath The URL or path to convert * @return The Resource for the URL/path * @throws IOException The Resource could not be created. */ Resource newResource(String urlOrPath) throws IOException; /** * @return Returns the permissions. */ PermissionCollection getPermissions(); /** * @return True if the classloader should delegate first to the parent * classloader (standard java behaviour) or false if the classloader * should first try to load from WEB-INF/lib or WEB-INF/classes (servlet * spec recommendation). */ boolean isParentLoaderPriority(); List getExtraClasspath(); /** * @deprecated use {@link #isHiddenResource(String, URL)} */ @Deprecated(since = "12.0.8", forRemoval = true) default boolean isServerResource(String name, URL parentUrl) { return isHiddenResource(name, parentUrl); } /** * @deprecated use {@link #isProtectedResource(String, URL)} */ @Deprecated(since = "12.0.8", forRemoval = true) default boolean isSystemResource(String name, URL webappUrl) { return isProtectedResource(name, webappUrl); } default boolean isHiddenResource(String name, URL parentUrl) { return false; } default boolean isProtectedResource(String name, URL webappUrl) { return false; } } /** * Run an action with access to ServerClasses *

Run the passed {@link PrivilegedExceptionAction} with the classloader * configured so as to allow server classes to be visible

* * @param action The action to run * @param the type of PrivilegedExceptionAction and the type returned by the action * @return The return from the action * @throws Exception if thrown by the action */ public static T runWithServerClassAccess(PrivilegedExceptionAction action) throws Exception { Boolean lsc = __loadServerClasses.get(); try { __loadServerClasses.set(true); return action.run(); } finally { if (lsc == null) __loadServerClasses.remove(); else __loadServerClasses.set(lsc); } } /** * Constructor. *

* The {@link Thread#currentThread()} {@link Thread#getContextClassLoader()} will be used as the parent loader, * unless {@code null}, in which case the classloader that loaded this class is used as the parent. * * @param context the context for this classloader */ public WebAppClassLoader(Context context) { this(null, context); } /** * Constructor. * * @param parent the parent classloader; if {@code null} then the {@link Thread#currentThread()} * {@link Thread#getContextClassLoader()} will be used as the parent loader, * unless also {@code null}, in which case the classloader that loaded this class is used as the parent. * @param context the context for this classloader */ public WebAppClassLoader(ClassLoader parent, Context context) { super(new URL[]{}, parent != null ? parent : (Thread.currentThread().getContextClassLoader() != null ? Thread.currentThread().getContextClassLoader() : (WebAppClassLoader.class.getClassLoader() != null ? WebAppClassLoader.class.getClassLoader() : ClassLoader.getSystemClassLoader()))); _parent = getParent(); _context = context; if (_parent == null) throw new IllegalArgumentException("no parent classloader!"); _extensions.add("jar"); _extensions.add("zip"); // TODO remove this system property String extensions = System.getProperty(WebAppClassLoader.class.getName() + ".extensions"); if (extensions != null) { StringTokenizer tokenizer = new StringTokenizer(extensions, StringUtil.DEFAULT_DELIMS); while (tokenizer.hasMoreTokens()) { _extensions.add(tokenizer.nextToken().trim()); } } for (Resource extra : context.getExtraClasspath()) addClassPath(extra); } /** * Get the name of the classloader. * @return the name of the classloader */ public String getName() { return _name; } /** * Set the name of the classloader. * @param name the name of the classloader */ public void setName(String name) { _name = name; } public Context getContext() { return _context; } /** * @param resource The resources to add to the classpath */ public void addClassPath(Resource resource) { for (Resource r : resource) { if (resource.exists()) { try { addURL(r.getURI().toURL()); } catch (MalformedURLException e) { throw new IllegalArgumentException("File not resolvable or incompatible with URLClassloader: " + resource); } } else { if (LOG.isDebugEnabled()) LOG.debug("Check resource exists and is not a nested jar: {}", resource); throw new IllegalArgumentException("File not resolvable or incompatible with URLClassloader: " + resource); } } } /** * @param classPathList Comma or semicolon separated path of filenames or URLs * pointing to directories or jar files. Directories should end * with '/'. * @throws IOException if unable to add classpath */ public void addClassPath(String classPathList) throws IOException { if (classPathList == null) return; _resourceFactory.split(classPathList).forEach(this::addClassPath); } /** * @param file Checks if this file type can be added to the classpath. */ private boolean isFileSupported(String file) { String ext = FileID.getExtension(file); return ext != null && _extensions.contains(ext); } private boolean isFileSupported(Path path) { return isFileSupported(path.getFileName().toString()); } /** * Add elements to the class path for the context from the jar and zip files found * in the specified resource. * * @param libs the directory resource that contains the jar and/or zip files. */ public void addJars(Resource libs) { if (!Resources.isReadableDirectory(libs)) return; libs.list().stream().filter(r -> isFileSupported(r.getName())).sorted(ResourceCollators.byName(true)).forEach(this::addClassPath); } @Override public PermissionCollection getPermissions(CodeSource cs) { PermissionCollection permissions = _context.getPermissions(); return (permissions == null) ? super.getPermissions(cs) : permissions; } @Override public Enumeration getResources(String name) throws IOException { List fromParent = new ArrayList<>(); List fromWebapp = new ArrayList<>(); Enumeration urls = _parent.getResources(name); while (urls != null && urls.hasMoreElements()) { URL url = urls.nextElement(); if (Boolean.TRUE.equals(__loadServerClasses.get()) || !_context.isHiddenResource(name, url)) fromParent.add(url); } urls = this.findResources(name); while (urls != null && urls.hasMoreElements()) { URL url = urls.nextElement(); if (!_context.isProtectedResource(name, url) || fromParent.isEmpty()) fromWebapp.add(url); } List resources; if (_context.isParentLoaderPriority()) { fromParent.addAll(fromWebapp); resources = fromParent; } else { fromWebapp.addAll(fromParent); resources = fromWebapp; } if (LOG.isDebugEnabled()) LOG.debug("getResources {} {}", name, resources); return Collections.enumeration(resources); } /** * Get a resource from the classloader *

* NOTE: this method provides a convenience of hacking off a leading / * should one be present. This is non-standard and it is recommended * to not rely on this behavior */ @Override public URL getResource(String name) { URL resource = null; if (_context.isParentLoaderPriority()) { URL parentUrl = _parent.getResource(name); // return if we have a url the webapp is allowed to see if (parentUrl != null && (Boolean.TRUE.equals(__loadServerClasses.get()) || !_context.isHiddenResource(name, parentUrl))) resource = parentUrl; else { URL webappUrl = this.findResource(name); // If found here then OK to use regardless of system or server classes // If it is a system resource, we've already tried to load from parent, so // would have returned it. // If it is a server resource, doesn't matter as we have loaded it from the // webapp if (webappUrl != null) resource = webappUrl; } } else { URL webappUrl = this.findResource(name); if (webappUrl != null && !_context.isProtectedResource(name, webappUrl)) resource = webappUrl; else { // Couldn't find or see a webapp resource, so try a parent URL parentUrl = _parent.getResource(name); if (parentUrl != null && (Boolean.TRUE.equals(__loadServerClasses.get()) || !_context.isHiddenResource(name, parentUrl))) resource = parentUrl; // We couldn't find a parent resource, so OK to return a webapp one if it exists // and we just couldn't see it before else if (webappUrl != null) resource = webappUrl; } } // Perhaps this failed due to leading / if (resource == null && name.startsWith("/")) resource = getResource(name.substring(1)); if (LOG.isDebugEnabled()) LOG.debug("getResource {} {}", name, resource); return resource; } @Override protected Class loadClass(String name, boolean resolve) throws ClassNotFoundException { synchronized (getClassLoadingLock(name)) { ClassNotFoundException ex = null; Class parentClass; Class webappClass; // Has this loader loaded the class already? webappClass = findLoadedClass(name); if (webappClass != null) { return webappClass; } // Should we try the parent loader first? if (_context.isParentLoaderPriority()) { // Try the parent loader try { parentClass = _parent.loadClass(name); if (parentClass == null) throw new ClassNotFoundException("Bad ClassLoader: returned null for loadClass(" + name + ")"); // If the webapp is allowed to see this class if (Boolean.TRUE.equals(__loadServerClasses.get()) || !_context.isHiddenClass(parentClass)) { return parentClass; } } catch (ClassNotFoundException e) { // Save it for later ex = e; } // Try the webapp loader try { // If found here then OK to use regardless of system or server classes // If it is a system class, we've already tried to load from parent, so // would have returned it. // If it is a server class, doesn't matter as we have loaded it from the // webapp webappClass = this.findClass(name); if (resolve) resolveClass(webappClass); return webappClass; } catch (ClassNotFoundException e) { if (ex == null) ex = e; else if (e != ex) ex.addSuppressed(e); } throw ex; } else { // Not parent loader priority, so... webappClass = loadAsResource(name, true); if (webappClass != null) { return webappClass; } // Try the parent loader try { parentClass = _parent.loadClass(name); // If the webapp is allowed to see this class if (Boolean.TRUE.equals(__loadServerClasses.get()) || !_context.isHiddenClass(parentClass)) { return parentClass; } } catch (ClassNotFoundException e) { ex = e; } // We couldn't find a parent class, so OK to return a webapp one if it exists // and we just couldn't see it before webappClass = loadAsResource(name, false); if (webappClass != null) { return webappClass; } throw ex == null ? new ClassNotFoundException(name) : ex; } } } public void addTransformer(ClassFileTransformer transformer) { _transformers.add(transformer); } public boolean removeTransformer(ClassFileTransformer transformer) { return _transformers.remove(transformer); } /** * Look for the classname as a resource to avoid loading a class that is * potentially a system resource. * * @param name the name of the class to load * @param checkSystemResource if true and the class isn't a system class we return it * @return the loaded class * @throws ClassNotFoundException if the class cannot be found */ protected Class loadAsResource(final String name, boolean checkSystemResource) throws ClassNotFoundException { // Try the webapp classloader first // Look in the webapp classloader as a resource, to avoid // loading a system class. Class webappClass = null; String path = TypeUtil.toClassReference(name); URL webappUrl = findResource(path); if (webappUrl != null && (!checkSystemResource || !_context.isProtectedResource(name, webappUrl))) { webappClass = this.foundClass(name, webappUrl); resolveClass(webappClass); if (LOG.isDebugEnabled()) LOG.debug("WAP webapp loaded {}", webappClass); } return webappClass; } @Override protected Class findClass(final String name) throws ClassNotFoundException { if (_transformers.isEmpty()) { return super.findClass(name); } String path = TypeUtil.toClassReference(name); URL url = findResource(path); if (url == null) throw new ClassNotFoundException(name); return foundClass(name, url); } protected Class foundClass(final String name, URL url) throws ClassNotFoundException { if (_transformers.isEmpty()) return super.findClass(name); InputStream content = null; try { content = url.openStream(); byte[] bytes = IO.readBytes(content); for (ClassFileTransformer transformer : _transformers) { byte[] tmp = transformer.transform(this, name, null, null, bytes); if (tmp != null) bytes = tmp; } return defineClass(name, bytes, 0, bytes.length); } catch (IOException | IllegalClassFormatException e) { throw new ClassNotFoundException(name, e); } finally { IO.close(content); } } @Override public void close() throws IOException { super.close(); IO.close(_resourceFactory); } @Override public String toString() { return String.format("%s{%s}@%x", this.getClass().getSimpleName(), _name, hashCode()); } @Override public boolean isProtectedClass(Class clazz) { return _context.isProtectedClass(clazz); } @Override public boolean isHiddenClass(Class clazz) { return _context.isHiddenClass(clazz); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy