- GraalVM for JDK 23 (Latest)
- GraalVM for JDK 24 (Early Access)
- GraalVM for JDK 21
- GraalVM for JDK 17
- Archives
- Dev Build
- JavaScript and Node.js Reference
- Node.js Runtime
- Java Interoperability
- JavaScript Compatibility
- Using JavaScript Modules and Packages
- Multithreading
- Migration Guide from Nashorn to GraalVM JavaScript
- Migration Guide from Rhino to GraalVM JavaScript
- Differences Between Node.js and Java Embeddings
- Options
- Run GraalVM JavaScript on a Stock JDK
- ScriptEngine Implementation
- FAQ
Differences Between Node.js and Java Embeddings
GraalVM provides a fully-compliant ECMAScript 2023 JavaScript language runtime. As such, it can run JavaScript code in a variety of embedding scenarios, including Oracle Database, any Java-based application, and Node.js.
Depending on the GraalVM’s JavaScript embedding scenario, applications have access to different built-in capabilities.
For example, Node.js applications executed using GraalVM’s bin/node
executable have access to all of Node.js’ APIs, including built-in Node.js modules such as fs
, http
, etc.
Conversely, JavaScript code embedded in a Java application has access to limited capabilities, as specified through the Context API, and do not have access to Node.js built-in modules.
This guide describes the main differences between a Node.js application and a GraalVM JavaScript application embedded in Java.
Context Creation #
JavaScript code in GraalVM can be executed using a GraalVM execution Context.
In a Java application, a new context can be created using the Context
API.
New contexts can be configured in multiple ways, and configuration options include exposing access to Java classes, allowing access to IO, etc.
A list of context creation options can be found in the API documentation.
In this scenario, Java classes can be exposed to JavaScript by using GraalVM’s polyglot Bindings
.
In a Node.js application, the GraalVM Context
executing the application is pre-initialized by the Node.js runtime, and cannot be configured by the user application.
In this scenario, Java classes can be exposed to the Node.js application by using the --vm.cp=
command line option of the bin/node
command, as described below.
Java Interoperability #
JavaScript applications can interact with Java classes using the Java
built-in object.
This object is available by default in the js
and node
launchers, but accessing Java classes is only possible in the JVM standalones (that have -jvm
in the name).
When embedding GraalVM JavaScript using the Polyglot API, you have to explicitly enable host access in the Context.Builder
(allowHostAccess
, allowHostClassLookup
).
More details on the Java interoperability capabilities of GraalVM JavaScript are available in Java Interoperability.
Multithreading #
A GraalVM context running JavaScript enforces a “share-nothing” model of parallelism: no JavaScript values can be accessed by two concurrent Java threads at the same time. In order to leverage parallel execution, multiple contexts have to be created and executed from multiple threads:
- In Node.js mode, multiple contexts can be created using Node.js’ Worker threads API. The Worker threads API ensures that no sharing can happen between two parallel contexts.
- In Java, multiple contexts can be executed from multiple threads. As long as a context is not accessed by two threads at the same time, parallel execution happens safely.
More details on parallel execution in GraalVM JavaScript are available in this blog post.
Java Libraries #
Java libraries can be accessed from JavaScript in GraalVM through the Java
built-in object.
In order for a Java library to be accessible from a Context
, its jar
files need to be added to the GraalVM classpath. This can be done in the following way:
- In Node.js mode, the classpath can be modified using the
--vm.cp
option. - In Java, the default Java’s
-cp
option can be used.
More details on GraalVM command line options are available in Options.
JavaScript Packages and Modules #
Many popular JavaScript modules such as those available on the npm
package registry can be used from Node.js as well as from Java:
- In Node.js mode, JavaScript modules are handled by the Node.js runtime. Therefore, GraalVM JavaScript supports all modules supported by Node.js (including ES modules, CommonJS modules, and native modules).
- In Java mode, GraalVM JavaScript can execute any JavaScript module or package that does not depend on native Node.js built-in modules (such as
'fs'
,'http'
, etc.) Modules can be loaded using a package bundler, or using the available built-in mechanisms for ES modules. CommonJS modules are supported in Java mode under an experimental flag.
More details on JavaScript modules are available in Modules.