Skip to content

Latest commit

 

History

History
189 lines (137 loc) · 4.54 KB

File metadata and controls

189 lines (137 loc) · 4.54 KB

lua2java

Lua to Java deserializer for Data2Java.

Depends on:

implementation "org.msuo:lua2java:<version>"

Supported Lua versions:

  • Lua 5.2 and above (LuaJ provides Lua 5.2 semantics via luaj-jse:3.0.1)

Leaf values

class DataModel {
    public String name;
    public Integer port;
}

String lua = "return { name = 'svc', port = 8080 }";
DataModel data = new LuaDeserializer().deserialize(lua, DataModel.class);

assertEquals("svc", data.name);
assertEquals(Integer.valueOf(8080), data.port);

Missing keys keep defaults (unknown keys fail)

Missing keys preserve defaults. Unknown keys are reported as UnknownField errors.

class DataModel {
    public String name = "default-name";
    public Integer port;
}

DataModel data = new LuaDeserializer().deserialize("return { port = 8080 }", DataModel.class);
assertEquals("default-name", data.name);

Nested objects

class DataModel {
    public Db db;
    static class Db {
        public String host;
        public Integer port;
    }
}

String lua = "return { db = { host = 'db', port = 15432 } }";
DataModel data = new LuaDeserializer().deserialize(lua, DataModel.class);

assertEquals("db", data.db.host);
assertEquals(Integer.valueOf(15432), data.db.port);

Optionals

import java.util.Optional;

class DataModel {
    public Optional<String> user = Optional.of("default-user");
}

DataModel present = new LuaDeserializer().deserialize("return { user = 'alice' }", DataModel.class);
DataModel missing = new LuaDeserializer().deserialize("return {}", DataModel.class);

assertEquals(Optional.of("alice"), present.user);
assertEquals(Optional.of("default-user"), missing.user);

Collections and maps

import java.util.List;
import java.util.Map;

class DataModel {
    public List<String> tags;
    public Map<String, Integer> limits;
}

String lua = "return { tags = { 'a', 'b' }, limits = { api = 10 } }";
DataModel data = new LuaDeserializer().deserialize(lua, DataModel.class);

assertEquals(List.of("a", "b"), data.tags);
assertEquals(Integer.valueOf(10), data.limits.get("api"));

Nested generics

import java.util.List;
import java.util.Map;

class GenericBox<T> { public T value; }
class GenericItem<T> { public T payload; }
class StringConstructedGenericKey<T> {
    public final String value;
    public StringConstructedGenericKey(String value) { this.value = value; }
}
class GenericData {
    public GenericBox<List<GenericItem<String>>> foo;
    public Map<StringConstructedGenericKey<Integer>, List<String>> values;
}

String lua = "return { foo = { value = { { payload = 'a' } } }, values = { k1 = { 'x', 'y' } } }";
GenericData data = new LuaDeserializer().deserialize(lua, GenericData.class);

Class references

interface Service {}
final class ServiceImpl implements Service {}
class DataModel {
    public Class<Service> impl;
}

String lua = "return { impl = '" + ServiceImpl.class.getName() + "' }";
DataModel data = new LuaDeserializer().deserialize(lua, DataModel.class);

assertEquals(ServiceImpl.class, data.impl);

Nil semantics

import java.util.Optional;

class DataModel {
    public Optional<String> user = Optional.of("default-user");
}

DataModel data = new LuaDeserializer().deserialize("return { user = nil }", DataModel.class);

// Lua table `user = nil` removes key, so field behaves as missing.
assertEquals(Optional.of("default-user"), data.user);

Environment and globals

class DataModel {
    public String name;
    public Mode mode;
    enum Mode { DEV, PROD }
}

Map<String, String> env = Map.of("DATA2JAVA_TEST_APP_ENV", "prod");
Map<String, Object> globals = Map.of("defaultName", "worker-default");

// Pass injected environment and global maps to deserialize.

String lua = """
local c = { mode = 'DEV', name = defaultName }
if os.getenv('DATA2JAVA_TEST_APP_ENV') == 'prod' then c.mode = 'PROD' end
return c
""";

DataModel data = new LuaDeserializer().deserialize(lua, DataModel.class, env, globals);
assertEquals(DataModel.Mode.PROD, data.mode);
assertEquals("worker-default", data.name);

os.getenv resolves from injected values first, then falls back to system environment variables.

File loading

Lua scripts can load other Lua files with dofile(...).

local data = dofile('/absolute/path/to/base-data.lua')
data.mode = 'PROD'
data.name = data.name .. '-updated'
return data

Use an absolute path when you need deterministic resolution.

See ../../errors.md for error details.