Addressing bug 303460:

1. Now shell can take compile script. This based on the initial patch from
Patrick Beard <pcbeard@mac.com>.

2. Using common code to read source and compiled scripts and for script
execution.
This commit is contained in:
igor%mir2.org 2005-08-14 14:36:17 +00:00
parent 1263f7496d
commit 12199852d5
2 changed files with 217 additions and 88 deletions

View file

@ -36,6 +36,7 @@
package org.mozilla.javascript; package org.mozilla.javascript;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream;
import java.io.Reader; import java.io.Reader;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.util.Hashtable; import java.util.Hashtable;
@ -437,6 +438,33 @@ public class Kit
return new String(buffer, 0, cursor); return new String(buffer, 0, cursor);
} }
public static byte[] readStream(InputStream is, int initialBufferCapacity)
throws IOException
{
if (initialBufferCapacity <= 0) {
throw new IllegalArgumentException(
"Bad initialBufferCapacity: "+initialBufferCapacity);
}
byte[] buffer = new byte[initialBufferCapacity];
int cursor = 0;
for (;;) {
int n = is.read(buffer, cursor, buffer.length - cursor);
if (n < 0) { break; }
cursor += n;
if (cursor == buffer.length) {
byte[] tmp = new byte[buffer.length * 2];
System.arraycopy(buffer, 0, tmp, 0, cursor);
buffer = tmp;
}
}
if (cursor != buffer.length) {
byte[] tmp = new byte[cursor];
System.arraycopy(buffer, 0, tmp, 0, cursor);
buffer = tmp;
}
return buffer;
}
/** /**
* Throws RuntimeException to indicate failed assertion. * Throws RuntimeException to indicate failed assertion.
* The function never returns and its return type is RuntimeException * The function never returns and its return type is RuntimeException

View file

@ -41,6 +41,7 @@ package org.mozilla.javascript.tools.shell;
import java.io.*; import java.io.*;
import java.net.URL; import java.net.URL;
import java.net.URLConnection;
import java.net.MalformedURLException; import java.net.MalformedURLException;
import java.util.*; import java.util.*;
import java.lang.reflect.*; import java.lang.reflect.*;
@ -82,8 +83,11 @@ public class Main
if (type == PROCESS_FILES) { if (type == PROCESS_FILES) {
processFiles(cx, args); processFiles(cx, args);
} else if (type == EVAL_INLINE_SCRIPT) { } else if (type == EVAL_INLINE_SCRIPT) {
evaluateScript(cx, getGlobal(), null, scriptText, Script script = loadScriptFromSource(cx, scriptText,
"<command>", 1, null); "<command>", 1, null);
if (script != null) {
evaluateScript(script, cx, getGlobal());
}
} else { } else {
throw Kit.codeBug(); throw Kit.codeBug();
} }
@ -338,8 +342,10 @@ public class Main
if (cx.stringIsCompilableUnit(source)) if (cx.stringIsCompilableUnit(source))
break; break;
} }
Object result = evaluateScript(cx, global, null, source, Script script = loadScriptFromSource(cx, source, "<stdin>",
"<stdin>", startline, null); lineno, null);
if (script != null) {
Object result = evaluateScript(script, cx, global);
if (result != cx.getUndefinedValue()) { if (result != cx.getUndefinedValue()) {
try { try {
global.getErr().println(cx.toString(result)); global.getErr().println(cx.toString(result));
@ -350,6 +356,7 @@ public class Main
NativeArray h = global.history; NativeArray h = global.history;
h.put((int)h.getLength(), h, source); h.put((int)h.getLength(), h, source);
} }
}
global.getErr().println(); global.getErr().println();
} else { } else {
processFile(cx, global, filename); processFile(cx, global, filename);
@ -368,91 +375,44 @@ public class Main
} }
static void processFileSecure(Context cx, Scriptable scope, static void processFileSecure(Context cx, Scriptable scope,
String filename, Object securityDomain) String path, Object securityDomain)
{ {
Reader in = null; Script script;
// Try filename first as URL if (path.endsWith(".class")) {
try { script = loadCompiledScript(cx, path, securityDomain);
URL url = new URL(filename); } else {
InputStream is = url.openStream(); String source = (String)readFileOrUrl(path, true);
in = new BufferedReader(new InputStreamReader(is)); if (source == null) {
} catch (MalformedURLException mfex) {
// fall through to try it as a file
in = null;
} catch (IOException ioex) {
Context.reportError(ToolErrorReporter.getMessage(
"msg.couldnt.open.url", filename, ioex.toString()));
exitCode = EXITCODE_FILE_NOT_FOUND; exitCode = EXITCODE_FILE_NOT_FOUND;
return; return;
} }
if (in == null) {
// Try filename as file
try {
in = new PushbackReader(new FileReader(filename));
int c = in.read();
// Support the executable script #! syntax: If // Support the executable script #! syntax: If
// the first line begins with a '#', treat the whole // the first line begins with a '#', treat the whole
// line as a comment. // line as a comment.
if (c == '#') { if (source.length() > 0 && source.charAt(0) == '#') {
while ((c = in.read()) != -1) { for (int i = 1; i != source.length(); ++i) {
if (c == '\n' || c == '\r') int c = source.charAt(i);
if (c == '\n' || c == '\r') {
source = source.substring(i);
break; break;
} }
((PushbackReader) in).unread(c);
} else {
// No '#' line, just reopen the file and forget it
// ever happened. OPT closing and reopening
// undoubtedly carries some cost. Is this faster
// or slower than leaving the PushbackReader
// around?
in.close();
in = new FileReader(filename);
}
filename = new java.io.File(filename).getCanonicalPath();
}
catch (FileNotFoundException ex) {
Context.reportError(ToolErrorReporter.getMessage(
"msg.couldnt.open",
filename));
exitCode = EXITCODE_FILE_NOT_FOUND;
return;
} catch (IOException ioe) {
global.getErr().println(ioe.toString());
} }
} }
// Here we evalute the entire contents of the file as script = loadScriptFromSource(cx, source, path, 1, securityDomain);
// a script. Text is printed only if the print() function }
// is called. if (script != null) {
evaluateScript(cx, scope, in, null, filename, 1, securityDomain); evaluateScript(script, cx, scope);
}
} }
public static Object evaluateScript(Context cx, Scriptable scope, public static Script loadScriptFromSource(Context cx, String scriptSource,
Reader in, String script, String path, int lineno,
String sourceName, Object securityDomain)
int lineno, Object securityDomain)
{ {
if (!global.initialized) {
global.init(cx);
}
Object result = cx.getUndefinedValue();
try { try {
if (in != null) { return cx.compileString(scriptSource, path, lineno,
try {
try {
result = cx.evaluateReader(scope, in,
sourceName, lineno,
securityDomain); securityDomain);
} finally {
in.close();
}
} catch (IOException ioe) {
global.getErr().println(ioe.toString());
}
} else {
result = cx.evaluateString(scope, script, sourceName, lineno,
securityDomain);
}
} catch (WrappedException we) { } catch (WrappedException we) {
global.getErr().println(we.getWrappedException().toString()); global.getErr().println(we.getWrappedException().toString());
we.printStackTrace(); we.printStackTrace();
@ -470,7 +430,76 @@ public class Main
exitCode = EXITCODE_RUNTIME_ERROR; exitCode = EXITCODE_RUNTIME_ERROR;
Context.reportError(msg); Context.reportError(msg);
} }
return result; return null;
}
private static Script loadCompiledScript(Context cx, String path,
Object securityDomain)
{
byte[] data = (byte[])readFileOrUrl(path, false);
if (data == null) {
exitCode = EXITCODE_FILE_NOT_FOUND;
return null;
}
// XXX: For now extract class name of compiled Script from path
// instead of parsing class bytes
int nameStart = path.lastIndexOf('/');
if (nameStart < 0) {
nameStart = 0;
} else {
++nameStart;
}
int nameEnd = path.lastIndexOf('.');
if (nameEnd < nameStart) {
// '.' does not exist in path (nameEnd < 0)
// or it comes before nameStart
nameEnd = path.length();
}
String name = path.substring(nameStart, nameEnd);
try {
GeneratedClassLoader loader = SecurityController.createLoader(cx.getApplicationClassLoader(), securityDomain);
Class clazz = loader.defineClass(name, data);
loader.linkClass(clazz);
if (!Script.class.isAssignableFrom(clazz)) {
throw Context.reportRuntimeError("msg.must.implement.Script");
}
return (Script) clazz.newInstance();
} catch (RhinoException rex) {
errorReporter.reportException(rex);
exitCode = EXITCODE_RUNTIME_ERROR;
} catch (IllegalAccessException iaex) {
exitCode = EXITCODE_RUNTIME_ERROR;
Context.reportError(iaex.toString());
} catch (InstantiationException inex) {
exitCode = EXITCODE_RUNTIME_ERROR;
Context.reportError(inex.toString());
}
return null;
}
public static Object evaluateScript(Script script, Context cx,
Scriptable scope)
{
if (!global.initialized) {
global.init(cx);
}
try {
return script.exec(cx, scope);
} catch (WrappedException we) {
global.getErr().println(we.getWrappedException().toString());
we.printStackTrace();
} catch (RhinoException rex) {
errorReporter.reportException(rex);
exitCode = EXITCODE_RUNTIME_ERROR;
} catch (VirtualMachineError ex) {
// Treat StackOverflow and OutOfMemory as runtime errors
ex.printStackTrace();
String msg = ToolErrorReporter.getMessage(
"msg.uncaughtJSException", ex.toString());
exitCode = EXITCODE_RUNTIME_ERROR;
Context.reportError(msg);
}
return cx.getUndefinedValue();
} }
private static void p(String s) { private static void p(String s) {
@ -508,6 +537,78 @@ public class Main
Global.getInstance(getGlobal()).setErr(err); Global.getInstance(getGlobal()).setErr(err);
} }
/**
* Read file or url specified by <tt>path</tt>.
* @return file or url content as <tt>byte[]</tt> or as <tt>String</tt> if
* <tt>convertToString</tt> is true.
*/
private static Object readFileOrUrl(String path, boolean convertToString)
{
URL url = null;
// Assume path is URL if it contains dot and there are at least
// 2 characters in the protocol part. The later allows under Windows
// to interpret paths with driver letter as file, not URL.
if (path.indexOf(':') >= 2) {
try {
url = new URL(path);
} catch (MalformedURLException ex) {
}
}
InputStream is = null;
int capacityHint = 0;
if (url == null) {
File file = new File(path);
capacityHint = (int)file.length();
try {
is = new FileInputStream(file);
} catch (IOException ex) {
Context.reportError(ToolErrorReporter.getMessage(
"msg.couldnt.open", path));
return null;
}
} else {
try {
URLConnection uc = url.openConnection();
is = uc.getInputStream();
capacityHint = uc.getContentLength();
// Ignore insane values for Content-Length
if (capacityHint > (1 << 20)) {
capacityHint = -1;
}
} catch (IOException ex) {
Context.reportError(ToolErrorReporter.getMessage(
"msg.couldnt.open.url", url.toString(), ex.toString()));
return null;
}
}
if (capacityHint <= 0) {
capacityHint = 4096;
}
byte[] data;
try {
try {
data = Kit.readStream(is, capacityHint);
} finally {
is.close();
}
} catch (IOException ex) {
Context.reportError(ex.toString());
return null;
}
Object result;
if (!convertToString) {
result = data;
} else {
// Convert to String using the default encoding
// XXX: Use 'charset=' argument of Content-Type if URL?
result = new String(data);
}
return result;
}
static protected final Global global = new Global(); static protected final Global global = new Global();
static protected ToolErrorReporter errorReporter; static protected ToolErrorReporter errorReporter;
static protected int exitCode = 0; static protected int exitCode = 0;