001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package gololang; 012 013import org.eclipse.golo.compiler.GoloClassLoader; 014import org.eclipse.golo.compiler.GoloCompilationException; 015 016import java.io.StringReader; 017import java.io.IOException; 018import java.io.Reader; 019import java.nio.charset.StandardCharsets; 020import java.lang.reflect.InvocationTargetException; 021import java.util.*; 022 023/** 024 * An evaluation environment offers facilities for dynamic code compilation, loading and execution from Golo code as 025 * strings. 026 * <p> 027 * An evaluation environment is reusable across several executions. The only exception is when using {@code asModule()}, 028 * as attempts to load a module with the same name as an already loaded one fails. 029 * <p> 030 * Each instance of this class uses a dedicated {@link GoloClassLoader}, hence usual rules about classloader delegation 031 * and isolation apply to evaluation environments. 032 * <p> 033 * While dynamic code evaluation is useful, it shall still be used with care and parsimony. It is especially important 034 * not to abuse {@code run()}, as each invocation triggers the generation of a one-shot class. 035 * <p> 036 * Here is an example usage of this API: 037 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 038 * let env = EvaluationEnvironment() 039 * let code = 040 * """ 041 * function a = -> "a." 042 * function b = -> "b." 043 * """ 044 * let mod = env: anonymousModule(code) 045 * let a = fun("a", mod) 046 * let b = fun("b", mod) 047 * println(a()) 048 * println(b()) 049 * </code></pre> 050 * <p> 051 * While this class is expected to be used from Golo code, it can also be used as a convenient way to embed Golo into 052 * polyglot JVM applications. 053 */ 054public class EvaluationEnvironment { 055 056 private final GoloClassLoader goloClassLoader; 057 private final List<String> imports = new LinkedList<>(); 058 059 private static String anonymousFilename() { 060 return "$Anonymous$_" + System.nanoTime() + ".golo"; 061 } 062 063 private static String anonymousModuleName() { 064 return "module anonymous" + System.nanoTime(); 065 } 066 067 /** 068 * Creates an evaluation environment using the current thread context classloader. 069 */ 070 public EvaluationEnvironment() { 071 this(Thread.currentThread().getContextClassLoader()); 072 } 073 074 /** 075 * Creates an evaluation environment using a parent classloader. 076 * 077 * @param parentClassLoader the parent classloader. 078 */ 079 public EvaluationEnvironment(ClassLoader parentClassLoader) { 080 goloClassLoader = new GoloClassLoader(parentClassLoader); 081 } 082 083 /** 084 * Imports symbols. 085 * <p> 086 * Each symbol generates an equivalent {@code import} statement in the corresponding Golo code. Calling 087 * {@code imports("foo.Bar", "bar.Baz")} means that the subsequent code evaluations have {@code import foo.Bar} and 088 * {@code import bar.Baz} statements. 089 * <p> 090 * Note that this has no effect for {@link #asModule(String)}. Also, calling this method several times accumulates 091 * the imports, in order. 092 * 093 * @param head the first imported symbol. 094 * @param tail the next imported symbols. 095 * @return this evaluation environment. 096 */ 097 public EvaluationEnvironment imports(String head, String... tail) { 098 imports.add(head); 099 Collections.addAll(imports, tail); 100 return this; 101 } 102 103 /** 104 * Clears all import symbols for the next code evaluation requests. 105 * 106 * @return this evaluation environment. 107 */ 108 public EvaluationEnvironment clearImports() { 109 imports.clear(); 110 return this; 111 } 112 113 /** 114 * Evaluates a complete module string. 115 *<p> 116 * For instance: 117 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 118 * let code = 119 * """ 120 * module foo 121 * 122 * function a = -> "a!" 123 * function b = -> "b!" 124 * """ 125 * let mod = env: asModule(code) 126 * let a = fun("a", mod) 127 * let b = fun("b", mod) 128 * println(a()) 129 * println(b()) 130 * </code></pre> 131 * 132 * @param source the module Golo source code as a string. 133 * @return the corresponding module, as a {@link Class}. 134 * @see gololang.Predefined#fun(Class, Object, Object) 135 */ 136 public Object asModule(String source) { 137 try (Reader in = new StringReader(source)) { 138 return goloClassLoader.load(anonymousFilename(), in); 139 } catch (IOException e) { 140 throw new RuntimeException(e); 141 } catch (GoloCompilationException e) { 142 e.setSourceCode(source); 143 throw e; 144 } 145 } 146 147 /** 148 * Loads an anonymous module. This is the same as {@link #asModule(String)}, except that the code does not contain 149 * a {@code module} declaration. 150 * 151 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 152 * let code = 153 * """ 154 * function a = -> "a!" 155 * function b = -> "b!" 156 * """ 157 * let mod = env: anonymousModule(code) 158 * let a = fun("a", mod) 159 * let b = fun("b", mod) 160 * println(a()) 161 * println(b()) 162 * </code></pre> 163 * 164 * @param source the module Golo source code as a string. 165 * @return the corresponding module, as a {@link Class}. 166 * @see gololang.Predefined#fun(Class, Object, Object) 167 */ 168 public Object anonymousModule(String source) { 169 return asModule(anonymousModuleName() + "\n\n" + source); 170 } 171 172 /** 173 * Defines a function, and returns it. 174 * 175 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 176 * let code = "|a, b| -> (a + b) * 2" 177 * let f = env: def(code) 178 * println(f(10, 20)) 179 * </code></pre> 180 * 181 * @param source the function code. 182 * @return the function as a {@link gololang.FunctionReference} instance. 183 */ 184 public Object def(String source) { 185 return loadAndRun("return " + source, "$_code"); 186 } 187 188 /** 189 * Evaluates some code as the body of a function and returns it. 190 * 191 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 192 * let code = "return (a + b) * 2" 193 * let f = env: asFunction(code, "a", "b") 194 * println(f(10, 20)) 195 * </code></pre> 196 * 197 * @param source the function body source code. 198 * @param argumentNames the argument names. 199 * @return the function as a {@link gololang.FunctionReference} instance. 200 */ 201 public Object asFunction(String source, String... argumentNames) { 202 return loadAndRun(source, "$_code_ref", argumentNames); 203 } 204 205 /** 206 * Runs some code as the body of a function and returns the value. The code shall use {@code return} statements 207 * to provide return values, if any. 208 * 209 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 210 * let code = """println(">>> run") 211 * foreach (i in range(0, 3)) { 212 * println("w00t") 213 * } 214 * return 666""" 215 * env: run(code) 216 * 217 * </code></pre> 218 * 219 * @param source the source to run. 220 * @return the return value, or {@code null} if no {@code return} statement is used. 221 */ 222 public Object run(String source) { 223 return loadAndRun(source, "$_code"); 224 } 225 226 /** 227 * Runs some code as the body of a function and returns the value. This is the same as {@link #run(String)}, but it 228 * takes a set of reference bindings in a map. Each reference is equivalent to a {@code let} statement. 229 * 230 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 231 * let code = """println(">>> run_map") 232 * println(a) 233 * println(b) 234 * """ 235 * let values = java.util.TreeMap(): add("a", 1): add("b", 2) 236 * env: run(code, values) 237 * </code></pre> 238 * 239 * @param source the source to run. 240 * @param context a map of bindings from name to values. 241 * @return the return value, or {@code null} if no {@code return} statement is used. 242 */ 243 public Object run(String source, Map<String, Object> context) { 244 StringBuilder builder = new StringBuilder(); 245 for (String param : context.keySet()) { 246 builder 247 .append("let ") 248 .append(param) 249 .append(" = $_env: get(\"") 250 .append(param) 251 .append("\")\n"); 252 } 253 builder.append(source); 254 return loadAndRun(builder.toString(), "$_code", new String[]{"$_env"}, new Object[]{context}); 255 } 256 257 private Class<?> wrapAndLoad(String source, String... argumentNames) { 258 StringBuilder builder = new StringBuilder() 259 .append(anonymousModuleName()) 260 .append("\n"); 261 for (String importSymbol : imports) { 262 builder.append("import ").append(importSymbol).append("\n"); 263 } 264 builder.append("\nfunction $_code = "); 265 if (argumentNames.length > 0) { 266 builder.append("| "); 267 final int lastIndex = argumentNames.length - 1; 268 for (int i = 0; i < argumentNames.length; i++) { 269 builder.append(argumentNames[i]); 270 if (i < lastIndex) { 271 builder.append(", "); 272 } 273 } 274 builder.append(" |"); 275 } 276 builder 277 .append(" {\n") 278 .append(source) 279 .append("\n}\n\n") 280 .append("function $_code_ref = -> ^$_code\n\n"); 281 return (Class<?>) asModule(builder.toString()); 282 } 283 284 private Object loadAndRun(String source, String target, String... argumentNames) { 285 try { 286 Class<?> module = wrapAndLoad(source, argumentNames); 287 return module.getMethod(target).invoke(null); 288 } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { 289 throw new RuntimeException(e); 290 } 291 } 292 293 private Object loadAndRun(String source, String target, String[] argumentNames, Object[] arguments) { 294 try { 295 Class<?> module = wrapAndLoad(source, argumentNames); 296 Class<?>[] type = new Class<?>[argumentNames.length]; 297 Arrays.fill(type, Object.class); 298 return module.getMethod(target, type).invoke(null, arguments); 299 } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { 300 throw new RuntimeException(e); 301 } 302 } 303}