Scripting in Java, by Java, for Java
JavaBox tries to answer the question, "Where is the thing that will let me run scripts written in Java within my Java application?"
JavaBox is a simple container ("sandbox") for executing scripts written in Java. JavaBox does not provide a secure sandbox; it's not safe for untrusted code.
Instead, it provides a basic sandbox that allows you to impose simple controls like time limits, instruction count limits, and restrictions on accessible classes. This allows, for example, an application to use Java as a runtime configuration language while having the ability to restrict unwanted functionality like network I/O, System.exit()
, etc.
Each JavaBox instance relies on an underlying JShell instance configured for local execution mode. That means JavaBox scripts are really JShell scripts that happen to be executing on the same JVM instance. Unlike normal JShell scripts, which can only return strings, JavaBox scripts can return arbitrary Java objects, and those objects can then be used outside of the scripting environment.
JavaBox supports imposing restrictions on scripts using Controls. Controls are allowed to intercept a script's class loading step and each time it executes.
JavaBox supports interrupting scripts in the middle of execution if necessary, and it also has suspend and resume functions, allowing an application thread to block scripts on certain operations to regain control.
Here's a "Hello, World" example:
Object returnValue;
try (JavaBox box = new JavaBox(Config.builder().build())) {
box.initialize();
box.setVariable("target", "World");
String script = "String.format(\"Hello, %s!\", target);";
returnValue = box.execute(script).returnValue();
}
System.out.println(returnValue); // prints "Hello, World!"
Here's an example that shows how to use a TimeLimitControl
to kill a script that takes too long:
// Set up control
Config config = Config.builder()
.withControl(new TimeLimitControl(Duration.ofSeconds(5)))
.build();
// Execute script
ScriptResult result;
try (JavaBox box = new JavaBox(config)) {
box.initialize();
result = box.execute("""
while (true) {
Thread.yield();
}
""");
}
// Check result
switch (result.snippetOutcomes().get(0)) {
case SnippetOutcome.ExceptionThrown e when e.exception() instanceof TimeLimitExceededException
-> System.out.println("script was taking too long");
}
- The
controls
module requires Java 24+ because it uses the newClassFile
API; everything else requires JDK 17+.
JavaBox is available from Maven Central.