001 /*
002 * $Id: CachingGroovyEngine.java 4166 2006-10-25 07:07:33Z paulk $
003 *
004 * Copyright 2003 (C) James Strachan and Bob Mcwhirter. All Rights Reserved.
005 *
006 * Redistribution and use of this software and associated documentation
007 * ("Software"), with or without modification, are permitted provided that the
008 * following conditions are met: 1. Redistributions of source code must retain
009 * copyright statements and notices. Redistributions must also contain a copy
010 * of this document. 2. Redistributions in binary form must reproduce the above
011 * copyright notice, this list of conditions and the following disclaimer in
012 * the documentation and/or other materials provided with the distribution. 3.
013 * The name "groovy" must not be used to endorse or promote products derived
014 * from this Software without prior written permission of The Codehaus. For
015 * written permission, please contact info@codehaus.org. 4. Products derived
016 * from this Software may not be called "groovy" nor may "groovy" appear in
017 * their names without prior written permission of The Codehaus. "groovy" is a
018 * registered trademark of The Codehaus. 5. Due credit should be given to The
019 * Codehaus - http://groovy.codehaus.org/
020 *
021 * THIS SOFTWARE IS PROVIDED BY THE CODEHAUS AND CONTRIBUTORS ``AS IS'' AND ANY
022 * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
023 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
024 * DISCLAIMED. IN NO EVENT SHALL THE CODEHAUS OR ITS CONTRIBUTORS BE LIABLE FOR
025 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
026 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
027 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
028 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
029 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
030 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
031 * DAMAGE.
032 *
033 */
034 package org.codehaus.groovy.bsf;
035
036 import groovy.lang.Binding;
037 import groovy.lang.GroovyClassLoader;
038 import groovy.lang.GroovyShell;
039 import groovy.lang.Script;
040 import org.apache.bsf.BSFDeclaredBean;
041 import org.apache.bsf.BSFException;
042 import org.apache.bsf.BSFManager;
043 import org.apache.bsf.util.BSFFunctions;
044 import org.codehaus.groovy.control.CompilerConfiguration;
045 import org.codehaus.groovy.runtime.InvokerHelper;
046
047 import java.io.ByteArrayInputStream;
048 import java.security.AccessController;
049 import java.security.PrivilegedAction;
050 import java.util.HashMap;
051 import java.util.Map;
052 import java.util.Vector;
053 import java.util.logging.Logger;
054 import java.util.logging.Level;
055
056 /**
057 * A Caching implementation of the GroovyEngine
058 *
059 * @author James Birchfield
060 */
061 public class CachingGroovyEngine extends GroovyEngine {
062 private static final Logger LOG = Logger.getLogger(CachingGroovyEngine.class.getName());
063 private static final Object[] EMPTY_ARGS = new Object[]{new String[]{}};
064
065 private Map evalScripts;
066 private Map execScripts;
067 private Binding context;
068 private GroovyClassLoader loader;
069
070 /**
071 * Evaluate an expression.
072 */
073 public Object eval(String source, int lineNo, int columnNo, Object script) throws BSFException {
074 try {
075 Class scriptClass = (Class) evalScripts.get(script);
076 if (scriptClass == null) {
077 scriptClass = loader.parseClass(new ByteArrayInputStream(script.toString().getBytes()), source);
078 evalScripts.put(script, scriptClass);
079 } else {
080 LOG.fine("eval() - Using cached script...");
081 }
082 //can't cache the script because the context may be different.
083 //but don't bother loading parsing the class again
084 Script s = InvokerHelper.createScript(scriptClass, context);
085 return s.run();
086 } catch (Exception e) {
087 throw new BSFException(BSFException.REASON_EXECUTION_ERROR, "exception from Groovy: " + e, e);
088 }
089 }
090
091 /**
092 * Execute a script.
093 */
094 public void exec(String source, int lineNo, int columnNo, Object script) throws BSFException {
095 try {
096 // shell.run(script.toString(), source, EMPTY_ARGS);
097
098 Class scriptClass = (Class) execScripts.get(script);
099 if (scriptClass == null) {
100 scriptClass = loader.parseClass(new ByteArrayInputStream(script.toString().getBytes()), source);
101 execScripts.put(script, scriptClass);
102 } else {
103 LOG.fine("exec() - Using cached version of class...");
104 }
105 InvokerHelper.invokeMethod(scriptClass, "main", EMPTY_ARGS);
106 } catch (Exception e) {
107 LOG.log(Level.WARNING, "BSF trace", e);
108 throw new BSFException(BSFException.REASON_EXECUTION_ERROR, "exception from Groovy: " + e, e);
109 }
110 }
111
112 /**
113 * Initialize the engine.
114 */
115 public void initialize(final BSFManager mgr, String lang, Vector declaredBeans) throws BSFException {
116 super.initialize(mgr, lang, declaredBeans);
117 ClassLoader parent = mgr.getClassLoader();
118 if (parent == null)
119 parent = GroovyShell.class.getClassLoader();
120 final ClassLoader finalParent = parent;
121 this.loader =
122 (GroovyClassLoader) AccessController.doPrivileged(new PrivilegedAction() {
123 public Object run() {
124 CompilerConfiguration configuration = new CompilerConfiguration();
125 configuration.setClasspath(mgr.getClassPath());
126 return new GroovyClassLoader(finalParent, configuration);
127 }
128 });
129 execScripts = new HashMap();
130 evalScripts = new HashMap();
131 context = shell.getContext();
132 // create a shell
133 // register the mgr with object name "bsf"
134 context.setVariable("bsf", new BSFFunctions(mgr, this));
135 int size = declaredBeans.size();
136 for (int i = 0; i < size; i++) {
137 declareBean((BSFDeclaredBean) declaredBeans.elementAt(i));
138 }
139 }
140 }