JSR-223:如何将可变参数宿主函数绑定到ScriptEngine

时间:2019-03-26 15:26:44

标签: javascript java jsr223 scriptengine graalvm

我正在尝试在Java VM下通过JSR-223 ScriptEngine运行Javascript来调用用户提供的代码,并将其传递给“承诺式”回调:即,一个不接受任何参数的函数(成功完成,没有价值);一个参数(null,表示成功完成,没有值或错误对象);或两个参数(成功完成的null,一个值)。

我正在为此使用GaalVM,并将回调添加到Javascript绑定对象中。像这样:

var engine = new ScriptEngineManager().getEngineByName("graal.js");
var binding = engine.getBindings(ScriptContext.ENGINE_SCOPE);
binding.put("exports", engine.eval("new Object()"));
engine.eval(userCode, binding);
binding.put("data", data);
binding.put("callback", callback);
engine.eval("exports.handler(data, callback)", binding);

问题是我无法弄清楚"callback"绑定中的内容:

我尝试过简单的方法:

BiFunction<Object, Object, Object> callback = (err, value) -> { /* ... */ };

在这种情况下,使用两个参数调用回调可以很好地工作,但是使用单个参数(或没有参数)时会出现错误:

org.graalvm.polyglot.PolyglotException: TypeError: EXECUTE on 
  JavaObject[my.package.JavascriptRun$$Lambda$771/0x0000000840844040@4cf04c6a 
  (my.package.JavascriptRun$$Lambda$771/0x0000000840844040)] failed due to: 
  Arity error - expected: 2 actual: 1

我已经尝试过像这样获取Object数组:

Function<Object[], Object> callback = (args) -> { /* ... */ };

这导致GraalVM向我扔了这个东西:

org.graalvm.polyglot.PolyglotException: TypeError: EXECUTE on
   JavaObject[my.package.JavascriptRun$$Lambda$771/0x0000000840844040@855ef90 
  (my.package.JavascriptRun$$Lambda$771/0x0000000840844040)] failed due to:
  java.lang.ClassCastException: class com.oracle.truffle.polyglot.PolyglotMap
  cannot be cast to class [Ljava.lang.Object; 
  (com.oracle.truffle.polyglot.PolyglotMap is in unnamed module of loader 'app';
   [Ljava.lang.Object; is in module java.base of loader 'bootstrap')

我目前正在使用Javascript进行手动多次调度-基本上定义了多个回调宿主函数,然后在Javascript中检查参数数量并相应地进行调度:

engine.eval("exports.handler(data, function() { "+
  "switch(arguments.length) { "+
  "case 0: return callback0(); "+
  "case 1: return callback1(arguments[0]); "+
  "default: return callback2(arguments[0],arguments[1]); "+
  "} })");

但这对我来说似乎是个坏主意。

1 个答案:

答案 0 :(得分:1)

MyCallback callback = (arguments) -> { /*...*/ };

其中

@FunctionalInterface
public static interface MyCallback {
    Object call(Object... arguments);
}

我想,它将为您服务。