001    /**
002     * Copyright 2005-2011 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.kew.plugin;
017    
018    import org.apache.commons.lang.StringUtils;
019    import org.apache.log4j.Logger;
020    import org.kuali.rice.core.api.CoreConstants;
021    import org.kuali.rice.core.api.config.CoreConfigHelper;
022    import org.kuali.rice.core.api.config.property.Config;
023    import org.kuali.rice.core.api.config.property.ConfigContext;
024    import org.kuali.rice.core.api.util.ClassLoaderUtils;
025    import org.kuali.rice.core.api.util.ContextClassLoaderBinder;
026    import org.kuali.rice.core.api.util.xml.XmlException;
027    
028    import javax.xml.namespace.QName;
029    import java.io.File;
030    import java.io.FileNotFoundException;
031    import java.io.IOException;
032    import java.net.MalformedURLException;
033    import java.net.URL;
034    
035    
036    /**
037     * Abstract base PluginLoader implementation.
038     * Delegates to template methods to obtain plugin ClassLoader and plugin config file URL,
039     * then load the config under the plugin ClassLoader, and constructs a Plugin object.
040     *
041     * @author Kuali Rice Team (rice.collab@kuali.org)
042     */
043    public abstract class BasePluginLoader implements PluginLoader {
044        private static final Logger LOG = Logger.getLogger(BasePluginLoader.class);
045    
046        private static final String META_INF_PATH = "META-INF";
047        private static final String PLUGIN_CONFIG_PATH = META_INF_PATH + "/workflow.xml";
048    
049        protected final String simplePluginName;
050        protected String logPrefix;
051    
052        protected final ClassLoader parentClassLoader;
053        protected final Config parentConfig;
054        protected final File sharedPluginDirectory;
055        protected String pluginConfigPath = PLUGIN_CONFIG_PATH;
056    
057        public BasePluginLoader(String simplePluginName, File sharedPluginDirectory, ClassLoader parentClassLoader, Config parentConfig) {
058            this.sharedPluginDirectory = sharedPluginDirectory;
059            if (parentClassLoader == null) {
060                parentClassLoader = ClassLoaderUtils.getDefaultClassLoader();
061            }
062            this.parentClassLoader = parentClassLoader;
063            this.parentConfig = parentConfig;
064            this.simplePluginName = simplePluginName;
065            this.logPrefix = simplePluginName;
066        }
067    
068        protected String getLogPrefix() {
069            return logPrefix;
070        }
071        
072        public String getPluginName() {
073            return simplePluginName;
074        }
075    
076        public void setPluginConfigPath(String pluginConfigPath) {
077            this.pluginConfigPath = pluginConfigPath;
078        }
079    
080        protected String getSimplePluginName() {
081            return simplePluginName;
082        }
083    
084        /**
085         * Template method that subclasses should implement to supply an appropriate
086         * plugin ClassLoader
087         * @return an appropriate PluginClassLoader
088         * @throws IOException if anything goes awry
089         */
090        protected abstract PluginClassLoader createPluginClassLoader() throws IOException;
091        /**
092         * Template method that subclasses should implement to supply an appropriate
093         * URL to the plugin's configuration
094         * @return an appropriate URL to the plugin's configuration
095         * @throws IOException if anything goes awry
096         */
097        protected abstract URL getPluginConfigURL() throws PluginException, IOException;
098    
099        /**
100         * Loads and creates the Plugin.
101         */
102        public Plugin load() throws Exception {
103            PluginClassLoader classLoader = createPluginClassLoader();
104            LOG.info("Created plugin ClassLoader: " + classLoader);
105            ContextClassLoaderBinder.bind(classLoader);
106            try {
107                return loadWithinContextClassLoader(classLoader);
108            } finally {
109                ContextClassLoaderBinder.unbind();
110            }
111        }
112    
113        public boolean isRemoved() {
114            return false;
115        }
116    
117        /**
118         * Executes loading of the plugin within the current context classloader set to the Plugin's classloader.
119         */
120        protected Plugin loadWithinContextClassLoader(PluginClassLoader classLoader) throws PluginException, IOException {
121            URL url = getPluginConfigURL();
122            PluginConfig pluginConfig = loadPluginConfig(url);
123            QName qPluginName = getPluginName(pluginConfig);
124            classLoader.setConfig(pluginConfig);
125            ConfigContext.init(classLoader, pluginConfig);
126            configureExtraClasspath(classLoader, pluginConfig);
127            this.logPrefix = PluginUtils.getLogPrefix(qPluginName).toString();
128            LOG.info("Constructing plugin '" + simplePluginName + "' with classloader: " + classLoader);
129            Plugin plugin = new Plugin(qPluginName, pluginConfig, classLoader);
130            installResourceLoader(plugin);
131            installPluginListeners(plugin);
132            return plugin;
133        }
134    
135        protected void installResourceLoader(Plugin plugin) {
136            PluginUtils.installResourceLoader(plugin);
137        }
138    
139        protected void installPluginListeners(Plugin plugin) {
140            PluginUtils.installPluginListeners(plugin);
141        }
142    
143        protected void configureExtraClasspath(PluginClassLoader classLoader, PluginConfig config) throws MalformedURLException {
144                    String extraClassesDirs = config.getProperty(Config.EXTRA_CLASSES_DIR);
145                    if (!org.apache.commons.lang.StringUtils.isEmpty(extraClassesDirs)) {
146                            String[] extraClasses = extraClassesDirs.split(",");
147                            for (int index = 0; index < extraClasses.length; index++) {
148                                    File extraClassesDir = new File(extraClasses[index]);
149                                    if (extraClassesDir.exists()) {
150                                            classLoader.addClassesDirectory(extraClassesDir);
151                                    }
152                            }
153                    }
154                    String extraLibDirs = config.getProperty(Config.EXTRA_LIB_DIR);
155                    if (!org.apache.commons.lang.StringUtils.isEmpty(extraLibDirs)) {
156                            String[] extraLibs = extraLibDirs.split(",");
157                            for (int index = 0; index < extraLibs.length; index++) {
158                                    File extraLibDir = new File(extraLibs[index]);
159                                    if (extraLibDir.exists()) {
160                                            classLoader.addLibDirectory(extraLibDir);
161                                    }
162                            }
163                    }
164            }
165    
166    
167        protected QName getPluginName(PluginConfig pluginConfig) {
168            String applicationId = pluginConfig.getProperty(CoreConstants.Config.APPLICATION_ID);
169            QName qPluginName = null;
170            if (StringUtils.isBlank(applicationId)) {
171                    qPluginName = new QName(CoreConfigHelper.getApplicationId(), simplePluginName);
172            } else {
173                    qPluginName = new QName(applicationId, simplePluginName);
174            }
175            return qPluginName;
176        }
177    
178        protected PluginConfig loadPluginConfig(URL url) {
179            PluginConfigParser parser = new PluginConfigParser();
180            try {
181                PluginConfig pluginConfig  = parser.parse(url, parentConfig);
182                pluginConfig.parseConfig();
183                return pluginConfig;
184            } catch (FileNotFoundException e) {
185                throw new PluginException(getLogPrefix() + " Could not locate the plugin config file at path " + url, e);
186            } catch (IOException ioe) {
187                throw new PluginException(getLogPrefix() + " Could not read the plugin config file", ioe);
188            } catch (XmlException ixe) {
189                throw new PluginException(getLogPrefix() + " Could not parse the plugin config file", ixe);
190            }
191        }
192    }