org.browsermob.proxy.jetty.http.ContextLoader.java Source code

Java tutorial

Introduction

Here is the source code for org.browsermob.proxy.jetty.http.ContextLoader.java

Source

// ========================================================================
// $Id: ContextLoader.java,v 1.37 2006/01/09 07:26:12 gregwilkins Exp $
// Copyright 1999-2004 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at 
// http://www.apache.org/licenses/LICENSE-2.0
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// ========================================================================

package org.browsermob.proxy.jetty.http;

import org.apache.commons.logging.Log;
import org.browsermob.proxy.jetty.log.LogFactory;
import org.browsermob.proxy.jetty.util.IO;
import org.browsermob.proxy.jetty.util.Resource;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.security.CodeSource;
import java.security.PermissionCollection;
import java.util.Arrays;
import java.util.StringTokenizer;

/* ------------------------------------------------------------ */
/** ClassLoader for HttpContext.
 * Specializes URLClassLoader with some utility and file mapping
 * methods.
 *
 * This loader defaults to the 2.3 servlet spec behaviour where non
 * system classes are loaded from the classpath in preference to the
 * parent loader.  Java2 compliant loading, where the parent loader
 * always has priority, can be selected with the setJava2Complient method.
 *
 * @version $Id: ContextLoader.java,v 1.37 2006/01/09 07:26:12 gregwilkins Exp $
 * @author Greg Wilkins (gregw)
 */
public class ContextLoader extends URLClassLoader {
    private static Log log = LogFactory.getLog(ContextLoader.class);

    private boolean _java2compliant = false;
    private ClassLoader _parent;
    private PermissionCollection _permissions;
    private String _urlClassPath;
    private HttpContext _context;

    /* ------------------------------------------------------------ */
    /** Constructor.
     * @param classPath Comma separated path of filenames or URLs
     * pointing to directories or jar files. Directories should end
     * with '/'.
     * @exception IOException
     */
    public ContextLoader(HttpContext context, String classPath, ClassLoader parent, PermissionCollection permisions)
            throws MalformedURLException, IOException {
        super(new URL[0], parent);
        _context = context;
        _permissions = permisions;
        _parent = parent;

        if (_parent == null)
            _parent = getSystemClassLoader();

        if (classPath == null) {
            _urlClassPath = "";
        } else {
            StringTokenizer tokenizer = new StringTokenizer(classPath, ",;");

            while (tokenizer.hasMoreTokens()) {
                Resource resource = Resource.newResource(tokenizer.nextToken());
                if (log.isDebugEnabled())
                    log.debug("Path resource=" + resource);

                // Resolve file path if possible
                File file = resource.getFile();

                if (file != null) {
                    URL url = resource.getURL();
                    addURL(url);
                    _urlClassPath = (_urlClassPath == null) ? url.toString()
                            : (_urlClassPath + "," + url.toString());
                } else {
                    // Add resource or expand jar/
                    if (!resource.isDirectory() && file == null) {
                        InputStream in = resource.getInputStream();
                        File lib = new File(context.getTempDirectory(), "lib");
                        if (!lib.exists()) {
                            lib.mkdir();
                            lib.deleteOnExit();
                        }
                        File jar = File.createTempFile("Jetty-", ".jar", lib);

                        jar.deleteOnExit();
                        if (log.isDebugEnabled())
                            log.debug("Extract " + resource + " to " + jar);
                        FileOutputStream out = null;
                        try {
                            out = new FileOutputStream(jar);
                            IO.copy(in, out);
                        } finally {
                            IO.close(out);
                        }

                        URL url = jar.toURL();
                        addURL(url);
                        _urlClassPath = (_urlClassPath == null) ? url.toString()
                                : (_urlClassPath + "," + url.toString());
                    } else {
                        URL url = resource.getURL();
                        addURL(url);
                        _urlClassPath = (_urlClassPath == null) ? url.toString()
                                : (_urlClassPath + "," + url.toString());
                    }
                }
            }
        }

        if (log.isDebugEnabled()) {
            if (log.isDebugEnabled())
                log.debug("ClassPath=" + _urlClassPath);
            if (log.isDebugEnabled())
                log.debug("Permissions=" + _permissions);
            if (log.isDebugEnabled())
                log.debug("URL=" + Arrays.asList(getURLs()));
        }
    }

    /* ------------------------------------------------------------ */
    /** Set Java2 compliant status.
     * @param compliant 
     */
    public void setJava2Compliant(boolean compliant) {
        _java2compliant = compliant;
    }

    /* ------------------------------------------------------------ */
    public boolean isJava2Compliant() {
        return _java2compliant;
    }

    /* ------------------------------------------------------------ */
    public String toString() {
        return "ContextLoader@" + hashCode() + "(" + _urlClassPath + ")\n  --parent--> " + _parent.toString();
    }

    /* ------------------------------------------------------------ */
    public PermissionCollection getPermissions(CodeSource cs) {
        PermissionCollection pc = (_permissions == null) ? super.getPermissions(cs) : _permissions;
        if (log.isDebugEnabled())
            log.debug("loader.getPermissions(" + cs + ")=" + pc);
        return pc;
    }

    /* ------------------------------------------------------------ */
    public Class loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
    }

    /* ------------------------------------------------------------ */
    protected Class loadClass(String name, boolean resolve) throws ClassNotFoundException {
        Class c = findLoadedClass(name);
        ClassNotFoundException ex = null;
        boolean tried_parent = false;
        if (c == null && (_java2compliant || isSystemPath(name)) && !isServerPath(name) && _parent != null) {
            if (log.isTraceEnabled())
                log.trace("try loadClass " + name + " from " + _parent);
            tried_parent = true;
            try {
                c = _parent.loadClass(name);
                if (log.isTraceEnabled())
                    log.trace("p0 loaded " + c);
            } catch (ClassNotFoundException e) {
                ex = e;
            }
        }

        if (c == null) {
            if (log.isTraceEnabled())
                log.trace("try findClass " + name + " from " + _urlClassPath);
            try {
                c = this.findClass(name);
                if (log.isTraceEnabled())
                    log.trace("cx loaded " + c);
            } catch (ClassNotFoundException e) {
                ex = e;
            }
        }

        if (c == null && !tried_parent && !isServerPath(name) && _parent != null) {
            if (log.isTraceEnabled())
                log.trace("try loadClass " + name + " from " + _parent);
            c = _parent.loadClass(name);
            if (log.isTraceEnabled())
                log.trace("p1 loaded " + c);
        }

        if (c == null)
            throw ex;

        if (resolve)
            resolveClass(c);

        return c;
    }

    /* ------------------------------------------------------------ */
    public URL getResource(String name) {
        URL url = null;
        boolean tried_parent = false;
        if (_parent != null && (_java2compliant || isSystemPath(name))) {
            if (log.isTraceEnabled())
                log.trace("try getResource " + name + " from " + _parent);
            tried_parent = true;
            url = _parent.getResource(name);
        }

        if (url == null) {
            if (log.isTraceEnabled())
                log.trace("try findResource " + name + " from " + _urlClassPath);
            url = this.findResource(name);

            if (url == null && name.startsWith("/")) {
                if (log.isDebugEnabled())
                    log.debug("HACK leading / off " + name);
                url = this.findResource(name.substring(1));
            }
        }

        if (_parent != null && url == null && !tried_parent) {
            if (log.isTraceEnabled())
                log.trace("try getResource " + name + " from " + _parent);
            url = _parent.getResource(name);
        }

        if (url != null)
            if (log.isTraceEnabled())
                log.trace("found " + url);

        return url;
    }

    /* ------------------------------------------------------------ */
    public boolean isServerPath(String name) {
        name = name.replace('/', '.');
        while (name.startsWith("."))
            name = name.substring(1);

        String[] server_classes = _context.getServerClasses();

        if (server_classes != null) {
            for (int i = 0; i < server_classes.length; i++) {
                boolean result = true;
                String c = server_classes[i];
                if (c.startsWith("-")) {
                    c = c.substring(1);
                    result = false;
                }

                if (c.endsWith(".")) {
                    if (name.startsWith(c))
                        return result;
                } else if (name.equals(c)) {
                    return result;
                }
            }
        }
        return false;
    }

    /* ------------------------------------------------------------ */
    public boolean isSystemPath(String name) {
        name = name.replace('/', '.');
        while (name.startsWith("."))
            name = name.substring(1);

        String[] system_classes = _context.getSystemClasses();
        if (system_classes != null) {
            for (int i = 0; i < system_classes.length; i++) {
                boolean result = true;
                String c = system_classes[i];
                if (c.startsWith("-")) {
                    c = c.substring(1);
                    result = false;
                }

                if (c.endsWith(".")) {
                    if (name.startsWith(c))
                        return result;
                } else if (name.equals(c))
                    return result;
            }
        }

        return false;
    }

    /* ------------------------------------------------------------ */
    public void destroy() {
        this._parent = null;
        this._permissions = null;
        this._urlClassPath = null;
    }

    /* ------------------------------------------------------------ */
    /**
     * @return Returns the serverClasses.
     */
    String[] getServerClasses() {
        return _context.getServerClasses();
    }

    /* ------------------------------------------------------------ */
    /**
     * @param serverClasses The serverClasses to set.
     */
    void setServerClasses(String[] serverClasses) {
        _context.setServerClasses(serverClasses);
    }

    /* ------------------------------------------------------------ */
    /**
     * @return Returns the systemClasses.
     */
    String[] getSystemClasses() {
        return _context.getSystemClasses();
    }

    /* ------------------------------------------------------------ */
    /**
     * @param systemClasses The systemClasses to set.
     */
    void setSystemClasses(String[] systemClasses) {
        _context.setSystemClasses(systemClasses);
    }
}