
org.jruby.embed.internal.EmbedRubyRuntimeAdapterImpl Maven / Gradle / Ivy
/*
* **** BEGIN LICENSE BLOCK *****
* Version: EPL 2.0/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Eclipse Public
* License Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.eclipse.org/legal/epl-v20.html
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* Copyright (C) 2009-2010 Yoko Harada
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the EPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the EPL, the GPL or the LGPL.
* **** END LICENSE BLOCK *****
*/
package org.jruby.embed.internal;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.StringReader;
import java.net.URL;
import java.util.Objects;
import org.jruby.Ruby;
import org.jruby.RubyInstanceConfig.CompileMode;
import org.jruby.RubyRuntimeAdapter;
import org.jruby.RubyString;
import org.jruby.ast.Node;
import org.jruby.ast.executable.Script;
import org.jruby.embed.AttributeName;
import org.jruby.embed.EmbedEvalUnit;
import org.jruby.embed.EmbedRubyRuntimeAdapter;
import org.jruby.embed.ParseFailedException;
import org.jruby.embed.PathType;
import org.jruby.embed.ScriptingContainer;
import org.jruby.embed.io.ReaderInputStream;
import org.jruby.embed.util.SystemPropertyCatcher;
import org.jruby.exceptions.RaiseException;
import org.jruby.internal.runtime.GlobalVariable;
import org.jruby.internal.runtime.ValueAccessor;
import org.jruby.javasupport.JavaEmbedUtils;
import org.jruby.javasupport.JavaEmbedUtils.EvalUnit;
import org.jruby.parser.StaticScope;
import org.jruby.parser.StaticScopeFactory;
import org.jruby.runtime.DynamicScope;
import org.jruby.runtime.Helpers;
import org.jruby.runtime.IAccessor;
import org.jruby.runtime.builtin.IRubyObject;
import org.jruby.runtime.load.LoadService;
import org.jruby.runtime.scope.ManyVarsDynamicScope;
import static org.jruby.embed.internal.EmbedEvalUnitImpl.isSharingVariables;
/**
*
* @author Yoko Harada
*/
public class EmbedRubyRuntimeAdapterImpl implements EmbedRubyRuntimeAdapter {
private final RubyRuntimeAdapter adapter = JavaEmbedUtils.newRuntimeAdapter();
private final ScriptingContainer container;
private final boolean wrapExceptions;
public EmbedRubyRuntimeAdapterImpl(ScriptingContainer container) {
this(container, false);
}
public EmbedRubyRuntimeAdapterImpl(ScriptingContainer container, boolean wrapExceptions) {
this.container = container;
this.wrapExceptions = wrapExceptions;
}
public EmbedEvalUnit parse(String script, int... lines) {
if (script == null) {
return null;
}
boolean unicode_escape = false;
Object obj = container.getAttribute(AttributeName.UNICODE_ESCAPE);
if (obj instanceof Boolean) {
unicode_escape = (Boolean)obj;
}
if (unicode_escape) {
InputStream istream = new ReaderInputStream(new StringReader(script));
return runParser(istream, null, lines);
} else {
return runParser(script, null, lines);
}
}
public EmbedEvalUnit parse(Reader reader, String filename, int... lines) {
Objects.requireNonNull(reader, "reader");
InputStream istream = new ReaderInputStream(reader);
return runParser(istream, filename, lines);
}
public EmbedEvalUnit parse(PathType type, String filename, int... lines) {
Objects.requireNonNull(filename, "filename");
if (type == null) type = PathType.ABSOLUTE;
InputStream istream = null;
try {
switch (type) {
case ABSOLUTE:
istream = new FileInputStream(filename);
break;
case RELATIVE:
String basedir = (String) container.getAttribute(AttributeName.BASE_DIR);
if (basedir == null) {
basedir = SystemPropertyCatcher.getBaseDir();
}
String absolutePath = basedir + File.separator + filename;
istream = new FileInputStream(absolutePath);
break;
case CLASSPATH:
URL loc = container.getProvider().getRuntime().getJRubyClassLoader().getResource(filename);
if (loc != null) {
filename = LoadService.classpathFilenameFromURL(filename, loc, true);
try {
istream = loc.openStream();
} catch (IOException ioe) {
istream = null; // as in getClassLoader.getResourceAsStream
}
}
break;
}
return parse(istream, filename, lines);
} catch (FileNotFoundException e) {
if (wrapExceptions) throw new ParseFailedException(e);
// NOTE: we do not declare throws IOException due source
Helpers.throwException(e); return null;
} finally {
if (istream != null) {
try { istream.close(); } catch (IOException ioe) {}
}
}
}
public EmbedEvalUnit parse(InputStream istream, String filename, int... lines) {
if (istream != null) {
return runParser(istream, filename, lines);
} else {
return null;
}
}
private EmbedEvalUnit runParser(Object input, String filename, int... lines) {
if (filename == null || filename.length() == 0) {
filename = container.getScriptFilename();
}
final Ruby runtime = container.getProvider().getRuntime();
IAccessor $0 = new ValueAccessor(RubyString.newString(runtime, filename));
runtime.getGlobalVariables().define("$PROGRAM_NAME", $0, GlobalVariable.Scope.GLOBAL);
runtime.getGlobalVariables().define("$0", $0, GlobalVariable.Scope.GLOBAL);
int line = 0;
if (lines != null && lines.length > 0) {
line = lines[0];
}
DynamicScope scope = null;
if (isSharingVariables(container)) {
scope = createLocalVarScope(runtime, container.getVarMap().getLocalVarNames());
}
try {
final Node node;
if (input instanceof String) {
node = runtime.parseEval((String) input, filename, scope, line);
} else {
node = runtime.parseFile((InputStream) input, filename, scope, line);
}
CompileMode compileMode = runtime.getInstanceConfig().getCompileMode();
if (compileMode == CompileMode.FORCE) {
// CON FIXME: We may need to force heap variables here so the compiled script uses our provided scope
Script script = runtime.tryCompile(node);
return new EmbedEvalUnitImpl(container, node, scope, script, wrapExceptions);
}
return new EmbedEvalUnitImpl(container, node, scope, null, wrapExceptions);
} catch (RaiseException e) {
if (wrapExceptions) throw new ParseFailedException(e.getMessage(), e);
throw e;
} catch (RuntimeException e) {
if (wrapExceptions) throw new ParseFailedException(e);
throw e;
}
}
public IRubyObject eval(Ruby runtime, String script) {
return adapter.eval(runtime, script);
}
public EvalUnit parse(Ruby runtime, String script, String filename, int lineNumber) {
return adapter.parse(runtime, script, filename, lineNumber);
}
public EvalUnit parse(Ruby runtime, InputStream istream, String filename, int lineNumber) {
return adapter.parse(runtime, istream, filename, lineNumber);
}
static DynamicScope createLocalVarScope(Ruby runtime, final String[] varNames) {
ManyVarsDynamicScope scope;
StaticScopeFactory scopeFactory = runtime.getStaticScopeFactory();
// root our parsing scope with a dummy scope
StaticScope topStaticScope = scopeFactory.newLocalScope(null);
topStaticScope.setModule(runtime.getObject());
DynamicScope currentScope = new ManyVarsDynamicScope(topStaticScope, null);
StaticScope evalScope = scopeFactory.newEvalScope(currentScope.getStaticScope(), varNames);
scope = new ManyVarsDynamicScope(evalScope, currentScope);
// JRUBY-5501: ensure we've set up a cref for the scope too
scope.getStaticScope().determineModule();
return scope;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy