Proxy Converter

Can I inject arbitrary Java objects and call all the API in JavaScript? Yes, JavetProxyConverter is designed for that. In general, the user experience is very much close to the one provided by GraalJS. As JavetProxyConverter opens almost the whole JVM to V8, it is very dangerous to allow end users to touch that V8 runtime, so JavetProxyConverter is not enabled by default. Here are the steps on how to enable that.

Usage

Preparation

// Step 1: Create an instance of JavetProxyConverter.
JavetProxyConverter javetProxyConverter = new JavetProxyConverter();
// Step 2: Set the V8Runtime converter to JavetProxyConverter.
v8Runtime.setConverter(javetProxyConverter);

Instance: File

File file = new File("/tmp/i-am-not-accessible");
v8Runtime.getGlobalObject().set("file", file);
assertEquals(file, v8Runtime.getGlobalObject().getObject("file"));
assertEquals(file.exists(), v8Runtime.getExecutor("file.exists()").executeBoolean());
assertEquals(file.isFile(), v8Runtime.getExecutor("file.isFile()").executeBoolean());
assertEquals(file.isDirectory(), v8Runtime.getExecutor("file.isDirectory()").executeBoolean());
assertEquals(file.canRead(), v8Runtime.getExecutor("file.canRead()").executeBoolean());
assertEquals(file.canWrite(), v8Runtime.getExecutor("file.canWrite()").executeBoolean());
assertEquals(file.canExecute(), v8Runtime.getExecutor("file.canExecute()").executeBoolean());
v8Runtime.getGlobalObject().delete("file");
v8Runtime.lowMemoryNotification();

Instance: List

javetProxyConverter.getConfig().setProxyListEnabled(true);
List<String> list = SimpleList.of("x", "y");
v8Runtime.getGlobalObject().set("list", list);
assertSame(list, v8Runtime.getGlobalObject().getObject("list"));
// contains
assertTrue(v8Runtime.getExecutor("list.contains('x')").executeBoolean());
assertTrue(v8Runtime.getExecutor("list.contains('y')").executeBoolean());
assertFalse(v8Runtime.getExecutor("list.contains('z')").executeBoolean());
// includes
assertTrue(v8Runtime.getExecutor("list.includes('x')").executeBoolean());
assertFalse(v8Runtime.getExecutor("list.includes('x', 1)").executeBoolean());
assertTrue(v8Runtime.getExecutor("list.includes('y', 1)").executeBoolean());
// push
assertEquals(4, v8Runtime.getExecutor("list.push('z', '1')").executeInteger());
assertTrue(v8Runtime.getExecutor("list.includes('z')").executeBoolean());
// pop
assertEquals("1", v8Runtime.getExecutor("list.pop()").executeString());
// toJSON
assertEquals(
        "[\"x\",\"y\",\"z\"]",
        v8Runtime.getExecutor("JSON.stringify(list);").executeString());
// Symbol.iterator
assertEquals(
        "[\"x\",\"y\",\"z\"]",
        v8Runtime.getExecutor("JSON.stringify([...list]);").executeString());
// unshift
assertEquals(5, v8Runtime.getExecutor("list.unshift('1', '2')").executeInteger());
// shift
assertEquals("1", v8Runtime.getExecutor("list.shift()").executeString());
assertEquals("2", v8Runtime.getExecutor("list.shift()").executeString());
// delete
assertTrue(v8Runtime.getExecutor("delete list[2]").executeBoolean());
assertEquals(2, v8Runtime.getExecutor("list.size()").executeInteger());
// length
assertEquals(2, v8Runtime.getExecutor("list.length").executeInteger());
v8Runtime.getGlobalObject().delete("list");
v8Runtime.lowMemoryNotification();
javetProxyConverter.getConfig().setProxyListEnabled(false);

Instance: Map

javetProxyConverter.getConfig().setProxyMapEnabled(true);
Map<String, Object> map = SimpleMap.of("x", 1, "y", "2");
v8Runtime.getGlobalObject().set("map", map);
assertTrue(map == v8Runtime.getGlobalObject().getObject("map"));
assertEquals(1, v8Runtime.getExecutor("map['x']").executeInteger());
assertEquals("2", v8Runtime.getExecutor("map['y']").executeString());
assertEquals(1, v8Runtime.getExecutor("map.x").executeInteger());
assertEquals("2", v8Runtime.getExecutor("map.y").executeString());
assertEquals("3", v8Runtime.getExecutor("map['z'] = '3'; map.z;").executeString());
assertEquals("3", map.get("z"));
assertEquals("4", v8Runtime.getExecutor("map.z = '4'; map.z;").executeString());
assertEquals("4", map.get("z"));
assertEquals(
        "[\"x\",\"y\",\"z\"]",
        v8Runtime.getExecutor("JSON.stringify(Object.getOwnPropertyNames(map));").executeString());
assertTrue(v8Runtime.getExecutor("delete map['x']").executeBoolean());
assertFalse(map.containsKey("x"));
assertTrue(v8Runtime.getExecutor("delete map['y']").executeBoolean());
assertFalse(map.containsKey("y"));
assertEquals(
        "{\"z\":\"z\"}",
        v8Runtime.getExecutor("JSON.stringify(map);").executeString());
v8Runtime.getGlobalObject().delete("map");
v8Runtime.lowMemoryNotification();
javetProxyConverter.getConfig().setProxyMapEnabled(false);

Instance: Path

Path path = new File("/tmp/i-am-not-accessible").toPath();
v8Runtime.getGlobalObject().set("path", path);
assertEquals(path, v8Runtime.getGlobalObject().getObject("path"));
assertEquals(path.toString(), v8Runtime.getExecutor("path.toString()").executeString());
Path newPath = v8Runtime.toObject(v8Runtime.getExecutor("path.resolve('abc')").execute(), true);
assertNotNull(newPath);
assertEquals(path.resolve("abc").toString(), newPath.toString());
assertEquals(path.resolve("abc").toString(), v8Runtime.getExecutor("path.resolve('abc').toString()").executeString());
v8Runtime.getGlobalObject().delete("path");
v8Runtime.lowMemoryNotification();

Instance: Set

javetProxyConverter.getConfig().setProxySetEnabled(true);
Set<String> set = SimpleSet.of("x", "y");
v8Runtime.getGlobalObject().set("set", set);
assertSame(set, v8Runtime.getGlobalObject().getObject("set"));
assertTrue(v8Runtime.getExecutor("set.contains('x')").executeBoolean());
assertTrue(v8Runtime.getExecutor("set.contains('y')").executeBoolean());
assertFalse(v8Runtime.getExecutor("set.contains('z')").executeBoolean());
assertFalse(v8Runtime.getExecutor("set.has('z')").executeBoolean());
assertTrue(v8Runtime.getExecutor("set.add('z')").executeBoolean());
assertTrue(v8Runtime.getExecutor("set.contains('z')").executeBoolean());
assertTrue(v8Runtime.getExecutor("set.has('z')").executeBoolean());
assertEquals(
        "[\"x\",\"y\",\"z\"]",
        v8Runtime.getExecutor("JSON.stringify(Object.getOwnPropertyNames(set));").executeString());
assertEquals(
        "[\"x\",\"y\",\"z\"]",
        v8Runtime.getExecutor("const keys = []; for (let key of set.keys()) { keys.push(key); } JSON.stringify(keys);").executeString());
assertTrue(v8Runtime.getExecutor("set.delete('z')").executeBoolean());
assertFalse(v8Runtime.getExecutor("set.delete('z')").executeBoolean());
assertFalse(v8Runtime.getExecutor("set.has('z')").executeBoolean());
v8Runtime.getGlobalObject().delete("set");
v8Runtime.getGlobalObject().delete("set");
v8Runtime.lowMemoryNotification();
javetProxyConverter.getConfig().setProxySetEnabled(false);

Static: StringBuilder

v8Runtime.getGlobalObject().set("StringBuilder", StringBuilder.class);
assertEquals("abc def", v8Runtime.getExecutor(
        "function main() {\n" +
                "  return new StringBuilder().append('abc').append(' ').append('def').toString();\n" +
                "}\n" +
                "main();").executeString());
v8Runtime.getGlobalObject().delete("StringBuilder");
v8Runtime.lowMemoryNotification();

Static: Pattern

v8Runtime.getGlobalObject().set("Pattern", Pattern.class);
assertTrue(v8Runtime.getExecutor("let p = Pattern.compile('^\\\\d+$'); p;").executeObject() instanceof Pattern);
assertTrue(v8Runtime.getExecutor("p.matcher('123').matches();").executeBoolean());
assertFalse(v8Runtime.getExecutor("p.matcher('a123').matches();").executeBoolean());
v8Runtime.getGlobalObject().delete("Pattern");
v8Runtime.getExecutor("p = undefined;").executeVoid();
v8Runtime.lowMemoryNotification();

Static: Enum

Static class usually does not have an instance. The dynamic proxy based converter is smart enough to handle that.

v8Runtime.getGlobalObject().set("JavetErrorType", JavetErrorType.class);
assertEquals(JavetErrorType.Converter, v8Runtime.getExecutor("JavetErrorType.Converter").executeObject());
assertThrows(
        JavetExecutionException.class,
        () -> v8Runtime.getExecutor("JavetErrorType.Converter = 1;").executeVoid(),
        "Public final field should not be writable.");
v8Runtime.getGlobalObject().delete("JavetErrorType");
v8Runtime.getGlobalObject().set("Converter", JavetErrorType.Converter);
assertEquals(JavetErrorType.Converter, v8Runtime.getGlobalObject().getObject("Converter"));
v8Runtime.getGlobalObject().delete("Converter");
v8Runtime.lowMemoryNotification();

Static: Interface

Sometimes an interface or annotation class can be injected for enabling Java reflection in V8.

v8Runtime.getGlobalObject().set("AutoCloseable", AutoCloseable.class);
v8Runtime.getGlobalObject().set("IJavetClosable", IJavetClosable.class);
assertTrue(AutoCloseable.class.isAssignableFrom(IJavetClosable.class));
assertTrue(v8Runtime.getExecutor("AutoCloseable.isAssignableFrom(IJavetClosable);").executeBoolean());
assertEquals(AutoCloseable.class, v8Runtime.getExecutor("AutoCloseable").executeObject());
assertEquals(IJavetClosable.class, v8Runtime.getExecutor("IJavetClosable").executeObject());
v8Runtime.getGlobalObject().delete("AutoCloseable");
v8Runtime.getGlobalObject().delete("IJavetClosable");
v8Runtime.lowMemoryNotification();

Dynamic: Anonymous Function

This feature is quite special as it allows implementing Java interfaces in JavaScript via anonymous functions, also known as lambda expressions.

  1. Define a simple interface IStringJoiner for joining two strings.

interface IStringJoiner extends AutoCloseable {
    String join(String a, String b);
}
  1. Define a simple class StringJoiner which holds the interface IStringJoiner.

public class StringJoiner implements AutoCloseable {
    private IStringJoiner joiner;

    public StringJoiner() {
        joiner = null;
    }

    @Override
    public void close() throws Exception {
        if (joiner != null) {
            joiner.close();
            joiner = null;
        }
    }

    public IStringJoiner getJoiner() {
        return joiner;
    }

    public void setJoiner(IStringJoiner joiner) {
        this.joiner = joiner;
    }
}
  1. Inject the implementation from JavaScript.

try (StringJoiner stringJoiner = new StringJoiner()) {
    v8Runtime.getGlobalObject().set("stringJoiner", stringJoiner);
    v8Runtime.getExecutor("stringJoiner.setJoiner((a, b) => a + ',' + b);").executeVoid();
    IStringJoiner joiner = stringJoiner.getJoiner();
    assertEquals("a,b", joiner.join("a", "b"));
    assertEquals("a,b,c", joiner.join(joiner.join("a", "b"), "c"));
    v8Runtime.getGlobalObject().delete("stringJoiner");
}
v8Runtime.lowMemoryNotification();

Voilà! It works.

Note

The JavaScript implementation is backed up by V8ValueFunction which is an orphan object. After its internal V8Runtime is closed, it will no longer callable. It's recommended to have the interface implement AutoCloseable as the sample shows so that the orphan V8ValueFunction can be recycled explicitly. If you don't own the interface, Javet will force the recycle of the orphan V8ValueFunction when the V8Runtime is being closed. Be careful, if you keep the application running for long while without recycling them in time, OutOfMemoryError may occur.

Dynamic: Anonymous Object for Interface

This feature is similar to the dynamic anonymous function, but is an enhanced version because it allows implementing all methods exposed by a Java interface.

  1. Define a simple interface IStringUtils for joining two strings.

interface IStringUtils extends AutoCloseable {
    String hello();
    String join(String separator, String... strings);
    List<String> split(String separator, String string);
}
  1. Define a simple class StringUtils which holds the interface IStringUtils.

public class StringUtils implements AutoCloseable {
    private IStringUtils utils;

    public StringUtils() {
        utils = null;
    }

    @Override
    public void close() throws Exception {
        if (utils != null) {
            utils.close();
            utils = null;
        }
    }

    public IStringUtils getUtils() {
        return utils;
    }

    public void setUtils(IStringUtils utils) {
        this.utils = utils;
    }
}
  1. Inject the implementation from JavaScript.

try (StringUtils stringUtils = new StringUtils()) {
    v8Runtime.getGlobalObject().set("stringUtils", stringUtils);
    v8Runtime.getExecutor(
            "stringUtils.setUtils({\n" +
            "  hello: () => 'hello',\n" +
            "  join: (separator, ...strings) => [...strings].join(separator),\n" +
            "  split: (separator, str) => str.split(separator),\n" +
            "});"
    ).executeVoid();
    IStringUtils utils = stringUtils.getUtils();
    assertEquals("hello", utils.hello());
    assertEquals("a,b,c", utils.join(",", "a", "b", "c"));
    assertArrayEquals(
            new String[]{"a", "b", "c"},
            utils.split(",", "a,b,c").toArray(new String[0]));
    v8Runtime.getGlobalObject().delete("stringUtils");
}
v8Runtime.lowMemoryNotification();

Voilà aussi! It works again.

Dynamic: Anonymous Object for Class

This feature is similar to the dynamic anonymous object for interface, but it allows implementing all methods exposed by a non-final Java class.

  1. Add ByteBuddy and JavetBuddy to the dependency. Please refer to JavetBuddy for detail.

  2. Define a simple class DynamicClass for adding two integers.

public class DynamicClass {
    public int add(int a, int b) {
        return 0;
    }
}
  1. Create an instance of a class which takes an instance of the DynamicClass.

IJavetAnonymous anonymous = new IJavetAnonymous() {
    @V8Function
    public void test(DynamicClass dynamicClass) throws Exception {
        assertEquals(3, dynamicClass.add(1, 2), "Add should work.");
        ((AutoCloseable) dynamicClass).close();
    }
};
  1. Inject the implementation from JavaScript. Please note that dynamic object support is disabled by default and JavetReflectionObjectFactory needs to be set to the converter config for JavetProxyConverter to enable this feature.

try {
    javetProxyConverter.getConfig().setReflectionObjectFactory(JavetReflectionObjectFactory.getInstance());
    v8Runtime.getGlobalObject().set("a", anonymous);
    String codeString = "a.test({\n" +
            "  add: (a, b) => a + b,\n" +
            "});";
    v8Runtime.getExecutor(codeString).executeVoid();
    v8Runtime.getGlobalObject().delete("a");
} finally {
    javetProxyConverter.getConfig().setReflectionObjectFactory(null);
    v8Runtime.lowMemoryNotification();
}

Voilà aussi! It works again.

Note

The JavaScript implementation is backed up by V8ValueObject which is an orphan object. After its internal V8Runtime is closed, it will no longer be callable. It's recommended to have the interface or the object implement AutoCloseable as the sample shows so that the orphan V8ValueObject can be recycled explicitly.

If you don't own the interface or the object, there are 2 ways of recycling it to avoid memory leak.

  1. Manually calling System.gc(); System.runFinalization(); will recycle the orphan V8ValueObject via the Java garbage collector.

  2. Javet will force the recycle of the orphan V8ValueObject when the V8Runtime is being closed. Be careful, if you keep the application running for long time without recycling them in time, OutOfMemoryError may occur. Of course, that is less likely going to happen because the Java garbage collector runs periodically.

Features

  • Any Java objects generated inside V8 are automatically handled by the converter.

  • Getters and setters (get, is, set and put) are smartly handled.

  • Overloaded methods and varargs methods are identified well.

  • Primitive types, Set, Map, List, Array are not handled. Map is special because it can be enabled.

  • Java interfaces can be implemented by anonymous functions in JavaScript.

  • Annotations can be applied to classes or methods to alter the default behaviors.

Annotation

Type

Description

@V8Convert

Class

It tells the converter which mode to be applied to the annotated class.

@V8Allow

Constructor / Field / Method

It tells the converter to bind the constructor / field / method.

@V8Block

Constructor / Field / Method

It tells the converter to ignore the constructor / field / method.

@V8Property

Field

It tells the converter to bind the field.

@V8Function

Method

It tells the converter to bind the method.

@V8Getter

Method

It tells the converter to bind the method as getter.

@V8Setter

Method

It tells the converter to bind the method as setter.

@V8Convert::mode

It tells the converter how to treat the annotated class.

  • Transparent - Transparent mode maps the Java objects directly to V8 and ignores any annotations. It is the default mode.

  • AllowOnly - AllowOnly mode only maps the API with @V8Allow.

  • BlockOnly - BlockOnly mode only ignores the API with @V8Block.

@V8Property::name

It tells the converter to bind the property to an alias name.

@V8Function::name

It tells the converter to bind the function to an alias name.

How does JavetProxyConverter Work?

JavetProxyConverter creates a JavaScript proxy per Java object. For now, the proxy intercepts get, has and set to achieve the complete virtualization of Java objects in JavaScript runtime.

How to Customize JavetProxyConverter?

It is recommended to subclass JavetProxyConverter and override few internal API to achieve complete customization.