◀Table of Contents
Java Interoperability
GraalVM provides a JavaScript language execution runtime and allows interoperability with Java code. This document describes the features and usage of this JavaScript-to-Java interoperability feature.
For a reference of GraalVM public API, see JavaScript Compatibility. See the Embedding Reference on how to interact with a guest language like JavaScript from a Java host application. The Polyglot Programming guide can be of additional help in that area. Specific migration guides for Rhino and Nashorn are also available.
Enabling Java Interoperability
In GraalVM, the js
and node
launchers are started in an ahead-of-time compiled native mode by default. In that mode, Java interoperability is not available.
To enable Java interoperability, the --jvm
option has to be provided to the native launcher.
This way, GraalVM JavaScript is executed on a traditional JVM and allows full Java interoperability.
Classpath
In order to load Java classes you need to have them on the Java classpath.
You can specify the classpath with the --vm.classpath=<classpath>
option (or short: --vm.cp=<classpath>
):
node --jvm --vm.cp=/my/class/path
js --jvm --vm.cp=/my/class/path
The method Java.addToClasspath()
can be used to programmatically add to the classpath at runtime.
Polyglot Context
The preferred method of launching GraalVM JavaScript with Java interop support is via polyglot Context
.
For that, a new org.graalvm.polyglot.Context
is built with the hostAccess
option allowing access and a hostClassLookup
predicate defining the Java classes you allow access to:
Context context = Context.newBuilder("js")
.allowHostAccess(HostAccess.ALL)
//allows access to all Java classes
.allowHostClassLookup(className -> true)
.build();
context.eval("js", jsSourceCode);
See the Embedding Reference on how to interact with a guest language like JavaScript from a Java host application. The Polyglot Programming guide can also be of help in that area.
ScriptEngine (JSR 223)
The org.graalvm.polyglot.Context
is the preferred execution method for interoperability with GraalVM’s languages and tools.
In addition, JavaScript running on GraalVM is fully compatible with JSR 223 and supports the ScriptEngine API
.
Internally, the GraalVM’s JavaScript ScriptEngine wraps a polyglot context instance:
ScriptEngine eng = new ScriptEngineManager()
.getEngineByName("graal.js");
Object fn = eng.eval("(function() { return this; })");
Invocable inv = (Invocable) eng;
Object result = inv.invokeMethod(fn, "call", fn);
See the ScriptEngine guide for more details on how to use it from GraalVM JavaScript.
Access Java from JavaScript
GraalVM provides a set of features to allow interoperability from JavaScript
to Java
.
While Rhino, Nashorn, and GraalVM JavaScript have a mostly comparable overall feature set, they differ in exact syntax, and, partly, semantics.
Class Access
To access a Java class, GraalVM JavaScript supports the Java.type(typeName)
function:
var FileClass = Java.type('java.io.File');
By default, Java classes are not automatically mapped to global variables, e.g., there is no java
global property in GraalVM JavaScript.
Existing code accessing, e.g., java.io.File
, should be rewritten to use the Java.type(name)
function:
//GraalVM JavaScript compliant syntax
var FileClass = Java.type("java.io.File");
//backwards-compatible syntax
var FileClass = java.io.File;
GraalVM JavaScript provides Packages
, java
, and similar global properties for compatibility.
However, explicitly accessing the required class with Java.type
is preferred whenever possible for two reasons:
- It allows resolving the class in one step rather than trying to resolve each property as a class.
Java.type
immediately throws aTypeError
if the class cannot be found or is not accessible, rather than silently treating an unresolved name as a package.
The js.java-package-globals
flag can be used to deactivate the global fields of Java packages (set false
to avoid creation of the fields; default is true
).
Constructing Java Objects
Java objects can be constructed with JavaScript’s new
keyword:
var FileClass = Java.type('java.io.File');
var file = new FileClass("myFile.md");
Field and Method Access
Static fields of a Java class, or fields of a Java object, can be accessed like JavaScript properties:
var JavaPI = Java.type('java.lang.Math').PI;
Java methods can be called like JavaScript functions:
var file = new (Java.type('java.io.File'))("test.md");
var fileName = file.getName();
Conversion of Method Arguments
JavaScript is defined to operate on the double
number type.
GraalVM JavaScript might internally use additional Java data types for performance reasons (e.g., the int
type).
When calling Java methods, a value conversion might be required.
This happens when the Java method expects a long
parameter, and an int
is provided from GraalVM JavaScript (type widening
).
If this conversion causes a lossy conversion, a TypeError
is thrown:
//Java
void longArg (long arg1);
void doubleArg (double arg2);
void intArg (int arg3);
//JavaScript
javaObject.longArg(1); //widening, OK
javaObject.doubleArg(1); //widening, OK
javaObject.intArg(1); //match, OK
javaObject.longArg(1.1); //lossy conversion, TypeError!
javaObject.doubleArg(1.1); //match, OK
javaObject.intArg(1.1); //lossy conversion, TypeError!
Note how the argument values have to fit into the parameter types. You can override this behavior using custom target type mappings.
Selection of Method
Java allows overloading of methods by argument types. When calling from JavaScript to Java, the method with the narrowest available type that the actual argument can be converted to without loss is selected:
//Java
void foo(int arg);
void foo(short arg);
void foo(double arg);
void foo(long arg);
//JavaScript
javaObject.foo(1); // will call foo(short);
javaObject.foo(Math.pow(2,16)); // will call foo(int);
javaObject.foo(1.1); // will call foo(double);
javaObject.foo(Math.pow(2,32)); // will call foo(long);
To override this behavior, an explicit method overload can be selected using the javaObject['methodName(paramTypes)']
syntax.
Parameter types need to be comma-separated without spaces, and Object types need to be fully qualified (e.g., 'get(java.lang.String,java.lang.String[])'
).
Note that this is different from Nashorn which allows extra spaces and simple names.
In the example above, one might always want to call, e.g., foo(long)
, even when foo(short)
can be reached with lossless conversion (foo(1)
):
javaObject['foo(int)'](1);
javaObject['foo(long)'](1);
javaObject['foo(double)'](1);
Note that the argument values still have to fit into the parameter types. You can override this behavior using custom target type mappings.
An explicit method selection can also be useful when the method overloads are ambiguous and cannot be automatically resolved as well as when you want to override the default choice:
//Java
void sort(List<Object> array, Comparator<Object> callback);
void sort(List<Integer> array, IntBinaryOperator callback);
void consumeArray(List<Object> array);
void consumeArray(Object[] array);
//JavaScript
var array = [3, 13, 3, 7];
var compare = (x, y) => (x < y) ? -1 : ((x == y) ? 0 : 1);
// throws TypeError: Multiple applicable overloads found
javaObject.sort(array, compare);
// explicitly select sort(List, Comparator)
javaObject['sort(java.util.List,java.util.Comparator)'](array, compare);
// will call consumeArray(List)
javaObject.consumeArray(array);
// explicitly select consumeArray(Object[])
javaObject['consumeArray(java.lang.Object[])'](array);
Note that there is currently no way to explicitly select constructor overloads. Future versions of GraalVM JavaScript might lift that restriction.
Package Access
GraalVM JavaScript provides a Packages
global property:
> Packages.java.io.File
JavaClass[java.io.File]
Array Access
GraalVM JavaScript supports the creation of Java arrays from JavaScript code. Both the patterns suggested by Rhino and Nashorn are supported:
//Rhino pattern
var JArray = Java.type('java.lang.reflect.Array');
var JString = Java.type('java.lang.String');
var sarr = JArray.newInstance(JString, 5);
//Nashorn pattern
var IntArray = Java.type("int[]");
var iarr = new IntArray(5);
The arrays created are Java types, but can be used in JavaScript code:
iarr[0] = iarr[iarr.length] * 2;
Map Access
In GraalVM JavaScript you can create and access Java Maps, e.g., java.util.HashMap
:
var HashMap = Java.type('java.util.HashMap');
var map = new HashMap();
map.put(1, "a");
map.get(1);
GraalVM JavaScript supports iterating over such maps similar to Nashorn:
for (var key in map) {
print(key);
print(map.get(key));
}
List Access
In GraalVM JavaScript you can create and access Java Lists, e.g., java.util.ArrayList
:
var ArrayList = Java.type('java.util.ArrayList');
var list = new ArrayList();
list.add(42);
list.add("23");
list.add({});
for (var idx in list) {
print(idx);
print(list.get(idx));
}
String Access
GraalVM JavaScript can create Java strings with Java interoperability.
The length of the string can be queried with the length
property (note that length
is a value property and cannot be called as a function):
var javaString = new (Java.type('java.lang.String'))("Java");
javaString.length === 4;
Note that GraalVM JavaScript uses Java strings internally to represent JavaScript strings, so the above code and the JavaScript string literal "Java"
are actually not distinguishable.
Iterating Properties
Properties (fields and methods) of Java classes and Java objects can be iterated with a JavaScript for..in
loop:
var m = Java.type('java.lang.Math')
for (var i in m) { print(i); }
> E
> PI
> abs
> sin
> ...
Access to JavaScript Objects from Java
JavaScript objects are exposed to Java code as instances of com.oracle.truffle.api.interop.java.TruffleMap
.
This class implements Java’s Map
interface.
JavaImporter
The JavaImporter
feature is available only in Nashorn compatibility mode (js.nashorn-compat
option).
Console Output of Java Classes and Java Objects
GraalVM JavaScript provides both print
and console.log
.
GraalVM JavaScript provides a print
built-in function compatible with Nashorn.
The console.log
is provided by Node.js directly.
It does not provide special treatment of interop objects.
Note that the default implementation of console.log
on GraalVM JavaScript is just an alias for print
, and Node’s implementation is only available when running on Node.js.
Exceptions
Exceptions thrown in Java code can be caught in JavaScript code. They are represented as Java objects:
try {
Java.type('java.lang.Class')
.forName("nonexistent");
} catch (e) {
print(e.getMessage());
}
Promises
GraalVM JavaScript provides support for interoperability between JavaScript Promise
objects and Java.
Java objects can be exposed to JavaScript code as thenable objects, allowing JavaScript code to await
Java objects.
Moreover, JavaScript Promise
objects are regular JavaScript objects, and can be accessed from Java using the mechanisms described in this document.
This allows Java code to be called back from JavaScript when a JavaScript promise is resolved or rejected.
Creating JavaScript Promise
Objects That Can Be Resolved from Java
JavaScript applications can create Promise
objects delegating to Java the resolution of the Promise
instance.
This can be achieved from JavaScript by using a Java object as the “executor” function of the JavaScript Promise
.
For example, Java objects implementing the following functional interface can be used to create new Promise
objects:
@FunctionalInterface
public interface PromiseExecutor {
void onPromiseCreation(Value onResolve, Value onReject);
}
Any Java object implementing PromiseExecutor
can be used to create a JavaScript Promise
:
// `javaExecutable` is a Java object implementing the `PromiseExecutor` interface
var myPromise = new Promise(javaExecutable).then(...);
JavaScript Promise
objects can be created not only using functional interfaces, but also using any other Java object that can be executed by the GraalVM JavaScript engine (for example, any Java object implementing the Polyglot ProxyExecutable interface).
More detailed example usages are available in the GraalVM JavaScript unit tests.
Using await
with Java Objects
JavaScript applications can use the await
expression with Java objects.
This can be useful when Java and JavaScript have to interact with asynchronous events.
To expose a Java object to GraalVM JavaScript as a thenable object, the Java object should implement a method called then()
having the following signature:
void then(Value onResolve, Value onReject);
When await
is used with a Java object implementing then()
, the GraalVM JavaScript runtime will treat the object as a JavaScript Promise
.
The onResolve
and onReject
arguments are executable Value
objects that should be used by the Java code to resume or abort the JavaScript await
expression associated with the corresponding Java object.
More detailed example usages are available in the GraalVM JavaScript unit tests.
Using JavaScript Promises from Java
Promise
objects created in JavaScript can be exposed to Java code like any other JavaScript object.
Java code can access such objects like normal Value
objects, with the possibility to register new promise resolution functions using the Promise
’s default then()
and catch()
functions.
As an example, the following Java code registers a Java callback to be executed when a JavaScript promise resolves:
Value jsPromise = context.eval(ID, "Promise.resolve(42);");
Consumer<Object> javaThen = (value)
-> System.out.println("Resolved from JavaScript: " + value);
jsPromise.invokeMember("then", javaThen);
More detailed example usages are available in the GraalVM JavaScript unit tests.
Multithreading
GraalVM JavaScript supports multithreading when used in combination with Java. More details about the GraalVM JavaScript multithreading model can be found in the Multithreading documentation.
Extending Java classes
In the JVM mode (--jvm
), GraalVM JavaScript provides support for extending Java classes and interfaces using the Java.extend
function.
Note that host access has to be enabled in the polyglot context for this feature to be available.
Java.extend
Java.extend(types...)
returns a generated adapter Java class object that extends the specified Java class and/or interfaces.
For example:
var Ext = Java.extend(Java.type("some.AbstractClass"),
Java.type("some.Interface1"),
Java.type("some.Interface2"));
var impl = new Ext({
superclassMethod: function() {/*...*/},
interface1Method: function() {/*...*/},
interface2Method: function() {/*...*/},
toString() {return "MyClass";}
});
impl.superclassMethod();
Super methods can be called via Java.super(adapterInstance)
.
See a combined example:
var sw = new (Java.type("java.io.StringWriter"));
var FilterWriterAdapter = Java.extend(Java.type("java.io.FilterWriter"));
var fw = new FilterWriterAdapter(sw, {
write: function(s, off, len) {
s = s.toUpperCase();
if (off === undefined) {
fw_super.write(s, 0, s.length)
} else {
fw_super.write(s, off, len)
}
}
});
var fw_super = Java.super(fw);
fw.write("abcdefg");
fw.write("h".charAt(0));
fw.write("**ijk**", 2, 3);
fw.write("***lmno**", 3, 4);
print(sw); // ABCDEFGHIJKLMNO
Note that in the nashorn-compat
mode, you can also extend interfaces and abstract classes using a new operator on a type object of an interface or an abstract class:
// --experimental-options --js.nashorn-compat
var JFunction = Java.type('java.util.function.Function');
var sqFn = new JFunction({
apply: function(x) { return x * x; }
});
sqFn.apply(6); // 36