This documentation is for the unreleased GraalVM version.Download Early Access Builds from GitHub.

Migration Guide from Nashorn to GraalJS

This guide serves as a migration guide for code previously targeted to the Nashorn engine. See the Java Interoperability guide for an overview of supported Java interoperability features.

The Nashorn engine has been deprecated in JDK 11 as part of JEP 335 and has been removed from JDK15 as part of JEP 372.

GraalJS can step in as a replacement for JavaScript code previously executed on the Nashorn engine. GraalJS provides all the features for JavaScript previously provided by Nashorn. Many are available by default, some are behind options, and others require minor modifications to your source code.

Both Nashorn and GraalJS support a similar set of syntax and semantics for Java interoperability. One notable difference is that GraalJS takes a secure by default approach, meaning some features need to be explicitly enabled that were available by default on Nashorn. The most important differences relevant for migration are listed here.

Nashorn features available by default (dependent on security settings):

  • Java.type, Java.typeName
  • Java.from, Java.to
  • Java.extend, Java.super
  • Java package globals: Packages, java, javafx, javax, com, org, edu

Nashorn Compatibility Mode #

GraalJS provides a Nashorn compatibility mode. Some of the functionality necessary for Nashorn compatibility is only available when the js.nashorn-compat option is enabled. This is the case for Nashorn-specific extensions that GraalJS does not want to expose by default.

Note that you have to unlock experimental features to use this option. Further note that setting this option defeats the secure by default approach of GraalJS in some cases, for example, when operating on a legacy ScriptEngine.

When you use the Nashorn compatibility mode, by default, ECMAScript 5 is set as compatibility level. You can specify a different ECMAScript version using the js.ecmascript-version option. Note that this might conflict with full Nashorn compatibility. A code example how to set the option is given near the end of this section.

The js.nashorn-compat option can be set:

  • By using a command line option:
      js --experimental-options --js.nashorn-compat=true
    
  • By using the Polyglot API:
      import org.graalvm.polyglot.Context;
    
      try (Context context = Context.newBuilder().allowExperimentalOptions(true).option("js.nashorn-compat", "true").build()) {
          context.eval("js", "print(__LINE__)");
      }
    
  • By using a system property when starting a Java application (remember to enable allowExperimentalOptions on the Context.Builder in your application as well):
      java -Dpolyglot.js.nashorn-compat=true MyApplication
    

Functionality only available under the nashorn-compat option includes:

  • Java.isJavaFunction, Java.isJavaMethod, Java.isScriptObject, Java.isScriptFunction
  • new Interface|AbstractClass(fn|obj)
  • JavaImporter
  • JSAdapter
  • java.lang.String methods on string values
  • load("nashorn:parser.js"), load("nashorn:mozilla_compat.js")
  • exit, quit

The js.ecmascript-version option can be set in similar fashion. As this is a supported option, there is no need to provide the experimental-options option just for setting the ecmascript-version:

js --js.ecmascript-version=2020

Nashorn Syntax Extensions #

Nashorn syntax extensions can be enabled using the js.syntax-extensions experimental option. They are also enabled by default in the Nashorn compatibility mode (js.nashorn-compat).

GraalJS vs Nashorn #

GraalJS differs from Nashorn in some aspects that were intentional design decisions.

Secure by Default #

GraalJS takes a secure by default approach. Unless explicitly permitted by the embedder, JavaScript code cannot access Java classes or access the file system, among other restrictions. Several features of GraalJS, including Nashorn compatibility features, are only available when the relevant security settings are permissive enough.

Make sure you understand the security implications of any change that lifts the secure default limits to your application and the host system.

For a full list of available settings, see Context.Builder. Those options can be defined when building the context with the Polyglot API.

Options frequently required to enable features of GraalJS are:

  • allowHostAccess(): configure which public constructors, methods or fields of public classes are accessible by a guest application. Use HostAccess.EXPLICIT or a custom HostAccess policy to selectively enable access. Set to HostAccess.ALL to allow unrestricted access.
  • allowHostClassLookup(): set a filter that specifies the Java host classes that can be looked up by a guest application. Set to the Predicate className -> true to allow lookup of all classes.
  • allowIO(): allow a guest language to perform unrestricted IO operations on the host system, required, for example, to load() from the file system. Set to true to enable IO.

If you run code on the legacy ScriptEngine, see Setting Options via Bindings regarding how to set them there.

Finally, note that the nashorn-compat mode enables the relevant options when executing code on the ScriptEngine (but not on Context), to provide better compatibility with Nashorn in that setup.

Launcher Name js #

GraalJS comes with a binary launcher named js. Note that, depending on the build environment, GraalJS might still ship Nashorn and its jjs launcher.

ScriptEngine Name graal.js #

GraalJS is shipped with support for ScriptEngine. It registers under several names, including “graal.js”, “JavaScript”, and “js”. Be sure to activate the Nashorn compatibility mode as described above if you need full Nashorn compatibility. Depending on the build setup, GraalJS might still ship Nashorn and provide it via ScriptEngine. For more details, see ScriptEngine Implementation.

ClassFilter #

GraalJS provides a class filter when starting with a polyglot Context. See Context.Builder.hostClassFilter.

Fully Qualified Names #

GraalJS requires the use of Java.type(typename). It does not support accessing classes just by their fully qualified class name by default. Java.type brings more clarity and avoids the accidental use of Java classes in JavaScript code. For example, look at this pattern:

var bd = new java.math.BigDecimal('10');

It should be expressed as:

var BigDecimal = Java.type('java.math.BigDecimal');
var bd = new BigDecimal('10');

Lossy Conversion #

GraalJS does not allow lossy conversions of arguments when calling Java methods. This could lead to bugs with numeric values that are hard to detect.

GraalJS always selects the overloaded method with the narrowest possible argument types that can be converted to without loss. If no such overloaded method is available, GraalJS throws a TypeError instead of lossy conversion. In general, this affects which overloaded method is executed.

Custom targetTypeMappings can be used to customize behavior. See HostAccess.Builder#targetTypeMapping.

ScriptObjectMirror Objects #

GraalJS does not provide objects of the class ScriptObjectMirror. Instead, JavaScript objects are exposed to Java code as objects implementing Java’s Map interface.

Code referencing ScriptObjectMirror instances can be rewritten by changing the type to either an interface (Map or List) or the polyglot Value class which provides similar capabilities.

Multithreading #

Running JavaScript on GraalVM supports multithreading by creating several Context objects from Java code. Contexts can be shared between threads, but each context must be accessed by a single thread at a time. Multiple JavaScript engines can be created from a Java application, and can be safely executed in parallel on multiple threads:

Context polyglot = Context.create();
Value array = polyglot.eval("js", "[1,2,42,4]");

GraalJS does not allow the creation of threads from JavaScript with access to the current Context. Moreover, GraalJS does not allow concurrent threads to access the same Context at the same time. This could lead to unmanageable synchronization problems like data races in a language that is not prepared for multithreading. For example:

new Thread(function() {
    print('printed from another thread'); // throws Exception due to potential synchronization problems
}).start();

JavaScript code can create and start threads with Runnables implemented in Java. The child thread may not access the Context of the parent thread or of any other polyglot thread. In case of violations, an IllegalStateException will be thrown. A child thread may create a new Context instance, though.

new Thread(aJavaRunnable).start(); // allowed on GraalJS

With proper synchronization in place, multiple contexts can be shared between different threads. The example Java applications using JavaScript Contexts from multiple threads can be found here.

Extensions Only Available in Nashorn Compatibility Mode #

The following JavaScript extensions available in Nashorn are deactivated in GraalJS by default. They are provided in the Nashorn compatibility mode. It is highly recommended not to implement new applications based on those features, but only to use it as a means to migrate existing applications to GraalVM.

String length Property #

GraalJS does not treat the length property of a String specially. The canonical way of accessing the String length is reading the length property:

myJavaString.length;

Nashorn enables users to access length as both a property and a function. Existing function calls length() should be expressed as property access. Nashorn behavior is mimicked in the Nashorn compatibility mode.

Java Packages in the JavaScript Global Object #

GraalJS requires the use of Java.type instead of fully qualified names. In the Nashorn compatibility mode, the following Java packages are added to the JavaScript global object: java, javafx, javax, com, org, and edu.

JavaImporter #

The JavaImporter feature is available only in the Nashorn compatibility mode.

JSAdapter #

The use of the non-standard JSAdapter feature is discouraged and should be replaced with the equivalent standard Proxy feature. For compatibility, JSAdapter is still available in the Nashorn compatibility mode.

Java.* Methods #

Several methods provided by Nashorn on the Java global object are available only in the Nashorn compatibility mode, or currently not supported by GraalJS. Available in the Nashorn compatibility mode are: Java.isJavaFunction, Java.isJavaMethod, Java.isScriptObject, and Java.isScriptFunction. Java.asJSONCompatible is currently not supported.

Accessors #

In the Nashorn compatibility mode, GraalJS allows users to access getters and setters just by using the names as properties, while omitting get, set, or is:

var Date = Java.type('java.util.Date');
var date = new Date();

var myYear = date.year; // calls date.getYear()
date.year = myYear + 1; // calls date.setYear(myYear + 1);

GraalJS mimics the behavior of Nashorn regarding the ordering of the access:

  • In case of a read operation, GraalJS will first try to call a getter with the name get and the property name in camel case. If that is not available, a getter with the name is and the property name in camel case is called. In the second case, unlike Nashorn, the resulting value is returned even if it is not of type boolean. Only if both methods are not available, the property itself will be read.
  • In case of a write operation, GraalJS will try to call a setter with the name set and the property name in camel case, providing the value as argument to that function. If the setter is not available, the property itself will be written.

Note that Nashorn (and thus, GraalJS) makes a clear distinction between property read/writes and function calls. When the Java class has both a field and a method of the same name publicly available, obj.property will always read the field (or the getter as discussed above), while obj.property() will always call the respective method.

Additional Aspects to Consider #

Features of GraalJS #

GraalJS supports features of the newest ECMAScript specification and some extensions to it. See JavaScript Compatibility. Note that this example adds objects to the global scope that might interfere with existing source code unaware of those extensions.

Console Output #

GraalJS provides a print builtin function compatible with Nashorn.

Note that GraalJS also provides a console.log function. This is an alias for print in pure JavaScript mode, but uses an implementation provided by Node.js when running in Node mode. The behavior around Java objects differs for console.log in Node mode as Node.js does not implement special treatment for such objects.

Connect with us