1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34 package org.codehaus.groovy.bsf;
35
36 import groovy.lang.Binding;
37 import groovy.lang.GroovyClassLoader;
38 import groovy.lang.GroovyShell;
39 import groovy.lang.Script;
40 import org.apache.bsf.BSFDeclaredBean;
41 import org.apache.bsf.BSFException;
42 import org.apache.bsf.BSFManager;
43 import org.apache.bsf.util.BSFFunctions;
44 import org.codehaus.groovy.control.CompilerConfiguration;
45 import org.codehaus.groovy.runtime.InvokerHelper;
46
47 import java.io.ByteArrayInputStream;
48 import java.security.AccessController;
49 import java.security.PrivilegedAction;
50 import java.util.HashMap;
51 import java.util.Map;
52 import java.util.Vector;
53 import java.util.logging.Logger;
54 import java.util.logging.Level;
55
56 /***
57 * A Caching implementation of the GroovyEngine
58 *
59 * @author James Birchfield
60 */
61 public class CachingGroovyEngine extends GroovyEngine {
62 private static final Logger LOG = Logger.getLogger(CachingGroovyEngine.class.getName());
63 private static final Object[] EMPTY_ARGS = new Object[]{new String[]{}};
64
65 private Map evalScripts;
66 private Map execScripts;
67 private Binding context;
68 private GroovyClassLoader loader;
69
70 /***
71 * Evaluate an expression.
72 */
73 public Object eval(String source, int lineNo, int columnNo, Object script) throws BSFException {
74 try {
75 Class scriptClass = (Class) evalScripts.get(script);
76 if (scriptClass == null) {
77 scriptClass = loader.parseClass(new ByteArrayInputStream(script.toString().getBytes()), source);
78 evalScripts.put(script, scriptClass);
79 } else {
80 LOG.fine("eval() - Using cached script...");
81 }
82
83
84 Script s = InvokerHelper.createScript(scriptClass, context);
85 return s.run();
86 } catch (Exception e) {
87 throw new BSFException(BSFException.REASON_EXECUTION_ERROR, "exception from Groovy: " + e, e);
88 }
89 }
90
91 /***
92 * Execute a script.
93 */
94 public void exec(String source, int lineNo, int columnNo, Object script) throws BSFException {
95 try {
96
97
98 Class scriptClass = (Class) execScripts.get(script);
99 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
133
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 }