// serverClassLoader.java 
// -----------------------
// (C) by Michael Peter Christen; mc@anomic.de
// first published on http://www.anomic.de
// Frankfurt, Germany, 2004
// last major change: 11.07.2004
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 2 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
//
// Using this software in any meaning (reading, learning, copying, compiling,
// running) means that you agree that the Author(s) is (are) not responsible
// for cost, loss of data or any harm that may be caused directly or indirectly
// by usage of this softare or this documentation. The usage of this software
// is on your own risk. The installation and usage (starting/running) of this
// software may allow other people or application to access your computer and
// any attached devices and is highly dependent on the configuration of the
// software which must be done by the user of the software; the author(s) is
// (are) also not responsible for proper configuration and usage of the
// software, even if provoked by documentation provided together with
// the software.
//
// Any changes to this file according to the GPL as documented in the file
// gpl.txt aside this file in the shipment you received can be done to the
// lines that follows this copyright notice here, but changes must not be
// done inside the copyright notive above. A re-distribution must contain
// the intact and unchanged copyright notice.
// Contributions and changes to the program code must be marked as such.


package de.anomic.server;

import java.io.File;
import java.io.IOException;
import java.util.HashMap;

public final class serverClassLoader extends ClassLoader {

    private final HashMap classes;

    public serverClassLoader() {
        //super(ClassLoader.getSystemClassLoader());
    	super(Thread.currentThread().getContextClassLoader());
        this.classes = new HashMap();
    }

    public serverClassLoader(ClassLoader parent) {
        super(parent);
        classes = new HashMap();
    }

    public Package[] packages() {
        return super.getPackages();
    }

    public Class loadClass(File classfile) throws ClassNotFoundException {
        // we consider that the classkey can either be only the name of a class, or a partial or
        // complete path to a class file
        
        // normalize classkey: strip off '.class'
        //if (classkey.endsWith(".class")) classkey = classkey.substring(0, classkey.length() - 6);
        
        String classFileName = null;
        try {
            classFileName = classfile.getCanonicalPath();
        } catch (IOException e) {
            throw new ClassNotFoundException("Unable to resolve the classfile path");
        }
        
        // try to load the class
        synchronized(classFileName.intern()) {
            // first try: take the class out of the cache, denoted by the classname    
            Class c = (Class) this.classes.get(classfile);
            if (c != null) return c;
            
            // consider classkey as a file and extract the file name
            //File classfile = new File(classkey);
            // this file cannot exist for real, since we stripped off the .class
            // we constructed the classfile for the only purpose to strip off the name:
            
            // get the class name out of the classfile
            String classname = classfile.getName();
            int p = classname.indexOf(".");
            classname = classname.substring(0, p);
            
            // now that we have the name, we can create the real class file
            //classfile = new File(classkey + ".class");
            

// This code doesn't work properly if there are multiple classes with the same name
// This is because we havn't definded package names in our servlets
//                
//          try {
//          c = findLoadedClass(classname);
//          if (c == null) {
//          // second try: ask the system
//          c = findSystemClass(classname);
//          }
//          if (c == null) {
//          // third try
//          throw new ClassNotFoundException("internal trigger");
//          }
//          } catch (ClassNotFoundException e) {
            //System.out.println("INTERNAL ERROR1 in cachedClassLoader: " + e.getMessage());
            
            // third try: load the file from the file system
            byte[] b;
            try {
                b = serverFileUtils.read(classfile);
                // now make a class out of the stream
                //  System.out.println("loading class " + classname + " from file " + classfile.toString());
                c = this.defineClass(null, b, 0, b.length);
                resolveClass(c);
                this.classes.put(classfile, c);
            } catch (LinkageError ee) {
                c = findLoadedClass(classname);
                if (c!=null) return c;                
            } catch (IOException ee) {
                //System.out.println("INTERNAL ERROR2 in cachedClassLoader: " + ee.getMessage());
                throw new ClassNotFoundException(classfile.toString());
            }
//          }
            return c;
        }
    }

}