3

我正在尝试使用 Grall.js 的实验性 ES 模块支持。我使用以下脚本: ES 模块“lib”

export const sqrt = Math.sqrt;
export function square(x) {
    return x * x;
}
export function diag(x, y) {
    return sqrt(square(x) + square(y));
}

和主脚本“脚本”

import { square, diag } from 'lib';
console.log('square(11)=' + square(11));
console.log('diag(4,3)=' + diag(4, 3));

我使用 graalvm-ce-19.2.1 并通过JSR 223在 JVM 中使用 Polyglot来运行主脚本。它不会尝试从磁盘上的某个位置加载 lib,而是抛出:

javax.script.ScriptException: org.graalvm.polyglot.PolyglotException: SyntaxError: script:1:0 Expected an operand but found import
import { square, diag } from 'lib';
^
script:1:30 Expected ; but found lib
import { square, diag } from 'lib';
                              ^
        at com.oracle.truffle.js.scriptengine.GraalJSScriptEngine.eval(GraalJSScriptEngine.java:348)
        at com.oracle.truffle.js.scriptengine.GraalJSScriptEngine.eval(GraalJSScriptEngine.java:323)

怎么了?

4

1 回答 1

3

文件名有一个约定,会触发将它们视为 ES 模块 - 文件必须以.mjs. 或者,可以application/javascript+moduleorg.graalvm.polyglot.Source对象上使用(非官方)mime 类型。没有这个import声明是不允许的1

这是使用 Polyglot Source/Context API 时的样子:

String script = "import {x} from 'lib'";

// these two support "import"
Source source1 =  Source.newBuilder("js", script, "script.mjs").build();
Source source2 =  Source.newBuilder("js", script, "script").mimeType("application/javascript+module").build();

// this one doesn't
//Source source3 =  Source.newBuilder("js", script, "script").build(); 

这适用于 JSR-223 API:

javax.script.ScriptEngine engine = factory.getEngineByName("graal.js");
engine.getContext().setAttribute(ScriptEngine.FILENAME, "script.mjs", ScriptContext.ENGINE_SCOPE);

似乎还有一个较旧的约定- 使用module:前缀,但这似乎不再起作用。

于 2019-12-03T16:29:38.180 回答