/** * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you 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.apache.cxf.jca.core.classloader; import java.io.IOException; import java.io.InputStream; import java.net.MalformedURLException; import java.net.URL; import java.security.ProtectionDomain; import java.security.SecureClassLoader; import java.util.ArrayList; import java.util.Enumeration; import java.util.Iterator; import java.util.List; import java.util.Properties; import java.util.logging.Logger; import org.apache.cxf.common.logging.LogUtils; public class PlugInClassLoader extends SecureClassLoader { private static final Logger LOG = LogUtils.getL7dLogger(PlugInClassLoader.class); private static final String FILE_COLON = "file:"; private static final String ZIP_COLON = "zip:"; private static final String URL_SCHEME_COLON = "classloader:"; private static final String JARS_PROPS_FILE = "jars.properties"; private static final String FILTERS_PROPS_FILE = "filters.properties"; private static final String NEFILTERS_PROPS_FILE = "negativefilters.properties"; private String jarUrls[] = new String[0]; private final ProtectionDomain protectionDomain; private final ClassLoader ploader; public PlugInClassLoader(final ClassLoader p) throws IOException { super(new FireWallClassLoader(p, getFilterList(p, FILTERS_PROPS_FILE), getFilterList(p, NEFILTERS_PROPS_FILE))); ploader = p; protectionDomain = getClass().getProtectionDomain(); jarUrls = loadUrls(p); processJarUrls(jarUrls); } private void processJarUrls(String urls[]) { for (int i = 0; i < urls.length; i++) { if (urls[i].startsWith(ZIP_COLON)) { urls[i] = FILE_COLON + urls[i].substring(ZIP_COLON.length()); } } } private static String[] getFilterList(ClassLoader parent, String propFile) throws IOException { Properties filtersProps = getProperties(parent, propFile); Iterator<Object> i = filtersProps.keySet().iterator(); while (i.hasNext()) { LOG.config("get Filter " + propFile + "::" + (String)i.next()); } return filtersProps.keySet().toArray(new String[filtersProps.keySet().size()]); } private static Properties getProperties(ClassLoader parent, String propsFileName) throws IOException { InputStream in = parent.getResourceAsStream(propsFileName); if (null == in) { in = PlugInClassLoader.class.getResourceAsStream(propsFileName); if (null == in) { String msg = "Internal rar classloader failed to locate configuration resource: " + propsFileName; IOException ioe = new IOException(msg); LOG.warning(ioe.toString()); throw ioe; } } Properties props = new Properties(); props.load(in); LOG.fine("Contents: " + propsFileName + props); return props; } private String[] loadUrls(ClassLoader parent) throws IOException { List<String> urlList = new ArrayList<>(); Properties props = getProperties(parent, JARS_PROPS_FILE); LOG.fine(props.toString()); Enumeration<Object> keys = props.keys(); while (keys.hasMoreElements()) { String key = (String)keys.nextElement(); URL url = parent.getResource(key); if (url != null) { LOG.config(url.toString()); urlList.add(url.toString()); } } return urlList.toArray(new String[urlList.size()]); } protected Class<?> findClass(String name) throws ClassNotFoundException { String path = name.replace('.', '/').concat(".class"); LOG.fine("findClass " + path); byte bytes[] = null; for (int i = 0; i < jarUrls.length; i++) { String fullpath = jarUrls[i] + "!/" + path; try { bytes = PlugInClassLoaderHelper.getResourceAsBytes(fullpath); if (bytes != null) { break; } } catch (IOException ex) { // we should find everything we look for but if we don't our // parent can when we throw cnf below LOG.fine("findClass: " + name + ": " + ex.toString()); } } if (bytes != null) { return defineClass(name, bytes, 0, bytes.length, protectionDomain); } else { LOG.config("can't find name " + name + " , try to using the ploader"); Class<?> result = ploader.loadClass(name); if (null == result) { throw new ClassNotFoundException(name); } else { return result; } } } protected URL findResource(String name) { LOG.fine("findResource: " + name); for (int i = 0; i < jarUrls.length; i++) { String fullpath = jarUrls[i] + "!/" + name; if (PlugInClassLoaderHelper.hasResource(fullpath)) { return genURL(fullpath); } } return null; } protected URL genURL(String path) { URL url = null; String urlString = URL_SCHEME_COLON + path; try { url = new URL(null, urlString, new Handler()); } catch (MalformedURLException mue) { LOG.warning(mue.toString()); } return url; } }