Initial commit 🎉
fbshipit-source-id: b6fc29740c6875d2e78953b8a7123890a67930f2 Co-authored-by: Sebastian McKenzie <sebmck@fb.com> Co-authored-by: John Knox <jknox@fb.com> Co-authored-by: Emil Sjölander <emilsj@fb.com> Co-authored-by: Pritesh Nandgaonkar <prit91@fb.com>
This commit is contained in:
169
android/plugins/console/JavascriptSession.java
Normal file
169
android/plugins/console/JavascriptSession.java
Normal file
@@ -0,0 +1,169 @@
|
||||
/*
|
||||
* Copyright (c) 2018-present, Facebook, Inc.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the LICENSE
|
||||
* file in the root directory of this source tree.
|
||||
*
|
||||
*/
|
||||
package com.facebook.sonar.plugins.console;
|
||||
|
||||
import com.facebook.sonar.plugins.console.iface.ScriptingSession;
|
||||
import java.io.Closeable;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.atomic.AtomicInteger;
|
||||
import org.json.JSONException;
|
||||
import org.json.JSONObject;
|
||||
import org.json.JSONTokener;
|
||||
import org.mozilla.javascript.Context;
|
||||
import org.mozilla.javascript.ContextFactory;
|
||||
import org.mozilla.javascript.Function;
|
||||
import org.mozilla.javascript.NativeJSON;
|
||||
import org.mozilla.javascript.NativeJavaMethod;
|
||||
import org.mozilla.javascript.NativeJavaObject;
|
||||
import org.mozilla.javascript.Scriptable;
|
||||
import org.mozilla.javascript.ScriptableObject;
|
||||
import org.mozilla.javascript.Undefined;
|
||||
|
||||
public class JavascriptSession implements Closeable, ScriptingSession {
|
||||
|
||||
private static final String TYPE = "type";
|
||||
private static final String VALUE = "value";
|
||||
public static final String JSON = "json";
|
||||
private final Context mContext;
|
||||
private final ContextFactory mContextFactory;
|
||||
private final Scriptable mScope;
|
||||
private final AtomicInteger lineNumber = new AtomicInteger(0);
|
||||
|
||||
JavascriptSession(ContextFactory contextFactory, Map<String, Object> globals) {
|
||||
mContextFactory = contextFactory;
|
||||
mContext = contextFactory.enterContext();
|
||||
|
||||
// Interpreted mode, or it will produce Dalvik incompatible bytecode.
|
||||
mContext.setOptimizationLevel(-1);
|
||||
mScope = mContext.initStandardObjects();
|
||||
|
||||
for (Map.Entry<String, Object> entry : globals.entrySet()) {
|
||||
final Object value = entry.getValue();
|
||||
|
||||
if (value instanceof Number || value instanceof String) {
|
||||
ScriptableObject.putConstProperty(mScope, entry.getKey(), entry.getValue());
|
||||
} else {
|
||||
// Calling java methods in the VM produces objects wrapped in NativeJava*.
|
||||
// So passing in wrapped objects keeps them consistent.
|
||||
ScriptableObject.putConstProperty(
|
||||
mScope,
|
||||
entry.getKey(),
|
||||
new NativeJavaObject(mScope, entry.getValue(), entry.getValue().getClass()));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public JSONObject evaluateCommand(String userScript) throws JSONException {
|
||||
return evaluateCommand(userScript, mScope);
|
||||
}
|
||||
|
||||
@Override
|
||||
public JSONObject evaluateCommand(String userScript, Object context) throws JSONException {
|
||||
Scriptable scope = new NativeJavaObject(mScope, context, context.getClass());
|
||||
return evaluateCommand(userScript, scope);
|
||||
}
|
||||
|
||||
private JSONObject evaluateCommand(String command, Scriptable scope) throws JSONException {
|
||||
try {
|
||||
// This may be called by any thread, and contexts have to be entered in the current thread
|
||||
// before being used, so enter/exit every time.
|
||||
mContextFactory.enterContext();
|
||||
return toJson(
|
||||
mContext.evaluateString(
|
||||
scope, command, "sonar-console", lineNumber.incrementAndGet(), null));
|
||||
} finally {
|
||||
Context.exit();
|
||||
}
|
||||
}
|
||||
|
||||
private JSONObject toJson(Object result) throws JSONException {
|
||||
|
||||
if (result instanceof String) {
|
||||
return new JSONObject().put(TYPE, JSON).put(VALUE, result);
|
||||
}
|
||||
|
||||
if (result instanceof Class) {
|
||||
return new JSONObject().put(TYPE, "class").put(VALUE, ((Class) result).getName());
|
||||
}
|
||||
|
||||
if (result instanceof NativeJavaObject
|
||||
&& ((NativeJavaObject) result).unwrap() instanceof String) {
|
||||
return new JSONObject().put(TYPE, JSON).put(VALUE, ((NativeJavaObject) result).unwrap());
|
||||
}
|
||||
|
||||
if (result instanceof NativeJavaObject
|
||||
&& ((NativeJavaObject) result).unwrap() instanceof Class) {
|
||||
return new JSONObject()
|
||||
.put(TYPE, "class")
|
||||
.put(VALUE, ((NativeJavaObject) result).unwrap().toString());
|
||||
}
|
||||
|
||||
if (result instanceof NativeJavaObject) {
|
||||
final JSONObject o = new JSONObject();
|
||||
o.put("toString", ((NativeJavaObject) result).unwrap().toString());
|
||||
for (Object id : ((NativeJavaObject) result).getIds()) {
|
||||
if (id instanceof String) {
|
||||
final String name = (String) id;
|
||||
final Object value = ((NativeJavaObject) result).get(name, (NativeJavaObject) result);
|
||||
if (value != null && value instanceof NativeJavaMethod) {
|
||||
continue;
|
||||
}
|
||||
final String valueString = value == null ? null : safeUnwrap(value).toString();
|
||||
o.put(name, valueString);
|
||||
}
|
||||
}
|
||||
return new JSONObject().put(TYPE, "javaObject").put(VALUE, o);
|
||||
}
|
||||
|
||||
if (result instanceof NativeJavaMethod) {
|
||||
final JSONObject o = new JSONObject();
|
||||
o.put(TYPE, "method");
|
||||
o.put("name", ((NativeJavaMethod) result).getFunctionName());
|
||||
return o;
|
||||
}
|
||||
|
||||
if (result == null || result instanceof Undefined) {
|
||||
return new JSONObject().put(TYPE, "null");
|
||||
}
|
||||
|
||||
if (result instanceof Function) {
|
||||
final JSONObject o = new JSONObject();
|
||||
o.put(TYPE, "function");
|
||||
o.put(VALUE, Context.toString(result));
|
||||
return o;
|
||||
}
|
||||
|
||||
if (result instanceof ScriptableObject) {
|
||||
return new JSONObject()
|
||||
.put(TYPE, JSON)
|
||||
.put(
|
||||
VALUE,
|
||||
new JSONTokener(NativeJSON.stringify(mContext, mScope, result, null, null).toString())
|
||||
.nextValue());
|
||||
}
|
||||
|
||||
if (result instanceof Number) {
|
||||
return new JSONObject().put(TYPE, JSON).put(VALUE, result);
|
||||
}
|
||||
|
||||
return new JSONObject().put(TYPE, "unknown").put(VALUE, result.toString());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void close() {
|
||||
Context.exit();
|
||||
}
|
||||
|
||||
private static Object safeUnwrap(Object o) {
|
||||
if (o instanceof NativeJavaObject) {
|
||||
return ((NativeJavaObject) o).unwrap();
|
||||
}
|
||||
return o;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user