package org.jruby.embed.jsr223;
import java.io.PrintStream;
import java.io.Writer;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import javax.script.Bindings;
import javax.script.ScriptContext;
import javax.script.ScriptEngine;
import org.jruby.Ruby;
import org.jruby.RubyGlobal.OutputGlobalVariable;
import org.jruby.RubyIO;
import org.jruby.embed.AttributeName;
import org.jruby.embed.LocalVariableBehavior;
import org.jruby.embed.ScriptingContainer;
import org.jruby.embed.io.WriterOutputStream;
import org.jruby.embed.variable.TransientLocalVariable;
import org.jruby.embed.variable.VariableInterceptor;
import org.jruby.internal.runtime.GlobalVariable;
public class Utils {
static int getLineNumber(ScriptContext context) {
Object obj = context.getAttribute(AttributeName.LINENUMBER.toString(), ScriptContext.ENGINE_SCOPE);
if (obj instanceof Integer) {
return (Integer) obj;
}
return 0;
}
static Object getReceiver(ScriptContext context) {
return context.getAttribute(AttributeName.RECEIVER.toString(), ScriptContext.ENGINE_SCOPE);
}
static String getFilename(ScriptContext context) {
Object filename = context.getAttribute(ScriptEngine.FILENAME);
return filename != null ? (String)filename : "<script>";
}
static boolean isTerminationOn(ScriptContext context) {
Object obj = context.getAttribute(AttributeName.TERMINATION.toString());
if (obj != null && obj instanceof Boolean && ((Boolean) obj) == true) {
return true;
}
return false;
}
static boolean isClearVariablesOn(ScriptContext context) {
Object obj = context.getAttribute(AttributeName.CLEAR_VARAIBLES.toString());
if (obj != null && obj instanceof Boolean && ((Boolean) obj) == true) {
return true;
}
return false;
}
static void preEval(ScriptingContainer container, ScriptContext context) {
Object receiver = Utils.getReceiverObject(context);
Bindings bindings = context.getBindings(ScriptContext.ENGINE_SCOPE);
for (Map.Entry<String, Object> entry : bindings.entrySet()) {
put(container, receiver, entry.getKey(), entry.getValue(), context);
}
setStdOut(container, context.getWriter());
setStdErr(container, context.getErrorWriter());
bindings = context.getBindings(ScriptContext.GLOBAL_SCOPE);
if (bindings == null) return;
for (Map.Entry<String, Object> entry : bindings.entrySet()) {
if (container.getVarMap().containsKey(entry.getKey())) continue;
put(container, receiver, entry.getKey(), entry.getValue(), context);
}
}
private static Object getReceiverObject(ScriptContext context) {
if (context == null) return null;
return context.getAttribute(AttributeName.RECEIVER.toString(), ScriptContext.ENGINE_SCOPE);
}
private static void setStdOut(ScriptingContainer container, Writer writer) {
if (writer == null) {
return;
}
Map map = container.getAttributeMap();
if (map.containsKey(AttributeName.WRITER)) {
Writer old = (Writer) map.get(AttributeName.WRITER);
if (old == writer) {
return;
}
}
map.put(AttributeName.WRITER, writer);
Ruby runtime = container.getProvider().getRuntime();
RubyIO io = getRubyIO(runtime, writer);
runtime.defineVariable(new OutputGlobalVariable(runtime, "$stdout", io), GlobalVariable.Scope.GLOBAL);
runtime.getObject().storeConstant("STDOUT", io);
runtime.getGlobalVariables().alias("$>", "$stdout");
runtime.getGlobalVariables().alias("$defout", "$stdout");
}
private static void setStdErr(ScriptingContainer container, Writer writer) {
if (writer == null) {
return;
}
Map map = container.getAttributeMap();
if (map.containsKey(AttributeName.ERROR_WRITER)) {
Writer old = (Writer) map.get(AttributeName.ERROR_WRITER);
if (old == writer) {
return;
}
}
map.put(AttributeName.ERROR_WRITER, writer);
Ruby runtime = container.getProvider().getRuntime();
RubyIO io = getRubyIO(runtime, writer);
runtime.defineVariable(new OutputGlobalVariable(runtime, "$stderr", io), GlobalVariable.Scope.GLOBAL);
runtime.getObject().storeConstant("STDERR", io);
runtime.getGlobalVariables().alias("$deferr", "$stderr");
}
private static RubyIO getRubyIO(Ruby runtime, Writer writer) {
PrintStream stream = new PrintStream(new WriterOutputStream(writer, runtime.getDefaultCharset().name()), true);
RubyIO io = new RubyIO(runtime, stream, false);
boolean locked = io.getOpenFile().lock();
try {
io.getOpenFile().setSync(true);
io.getOpenFile().io_fflush(runtime.getCurrentContext());
return io;
} finally {
if (locked) io.getOpenFile().unlock();
}
}
static void postEval(ScriptingContainer container, ScriptContext context) {
if (context == null) return;
Object receiver = Utils.getReceiverObject(context);
Bindings engineMap = context.getBindings(ScriptContext.ENGINE_SCOPE);
Iterator<Map.Entry<String, Object>> iter = engineMap.entrySet().iterator();
for (;iter.hasNext();) {
Map.Entry<String, Object> entry = iter.next();
if (Utils.shouldLVarBeDeleted(container, entry.getKey())) {
iter.remove();
}
}
Set<String> keys = container.getVarMap().keySet();
if (keys != null && keys.size() > 0) {
for (String key : keys) {
Object value = container.getVarMap().get(key);
engineMap.put(Utils.adjustKey(key), value);
}
}
Bindings globalMap = context.getBindings(ScriptContext.GLOBAL_SCOPE);
if (globalMap == null) return;
keys = globalMap.keySet();
if (keys != null && keys.size() > 0) {
for (String key : keys) {
if (engineMap.containsKey(key)) continue;
Object value = container.getVarMap().get(receiver, key);
globalMap.put(key, value);
}
}
}
private static Object put(ScriptingContainer container, Object receiver, String key, Object value, ScriptContext context) {
Object oldValue = null;
String adjustedKey = Utils.adjustKey(key);
if (Utils.isRubyVariable(container, adjustedKey)) {
boolean sharing_variables = true;
Object obj = context.getAttribute(AttributeName.SHARING_VARIABLES.toString(), ScriptContext.ENGINE_SCOPE);
if (obj != null && obj instanceof Boolean && ((Boolean) obj) == false) {
sharing_variables = false;
}
if (sharing_variables || "ARGV".equals(adjustedKey)) {
oldValue = container.put(receiver, adjustedKey, value);
}
} else {
if (adjustedKey.equals(AttributeName.SHARING_VARIABLES.toString())) {
oldValue = container.setAttribute(AttributeName.SHARING_VARIABLES, value);
} else {
oldValue = container.setAttribute(adjustedKey, value);
}
}
return oldValue;
}
static boolean isRubyVariable(ScriptingContainer container, String name) {
return VariableInterceptor.isKindOfRubyVariable(container.getProvider().getLocalVariableBehavior(), name);
}
private static String adjustKey(String key) {
if (key.equals(ScriptEngine.ARGV)) {
return "ARGV";
} if ("ARGV".equals(key)) {
return ScriptEngine.ARGV;
} else {
return key;
}
}
private static boolean shouldLVarBeDeleted(ScriptingContainer container, String key) {
LocalVariableBehavior behavior = container.getProvider().getLocalVariableBehavior();
if (behavior != LocalVariableBehavior.TRANSIENT) return false;
return TransientLocalVariable.isValidName(key);
}
}