#include #include #include #include #include namespace { using namespace testing; template T get(sol::state_view& lua, std::string luaCode) { return lua.safe_script("return " + luaCode).get(); } TEST(LuaUtilStorageTest, Subscribe) { // Note: LuaUtil::Callback can be used only if Lua is initialized via LuaUtil::LuaState LuaUtil::LuaState luaState{ nullptr, nullptr }; luaState.protectedCall([](LuaUtil::LuaView& view) { sol::state_view& lua = view.sol(); LuaUtil::LuaStorage::initLuaBindings(view); LuaUtil::LuaStorage storage; storage.setActive(true); sol::table callbackHiddenData(lua, sol::create); callbackHiddenData[LuaUtil::ScriptsContainer::sScriptIdKey] = LuaUtil::ScriptId{}; LuaUtil::getAsyncPackageInitializer( lua.lua_state(), []() { return 0.0; }, []() { return 0.0; })(callbackHiddenData); lua["async"] = LuaUtil::AsyncPackageId{ nullptr, 0, callbackHiddenData }; lua["mutable"] = storage.getMutableSection(lua, "test"); lua["ro"] = storage.getReadOnlySection(lua, "test"); lua.safe_script(R"( callbackCalls = {} ro:subscribe(async:callback(function(section, key) table.insert(callbackCalls, section .. '_' .. (key or '*')) end)) )"); lua.safe_script("mutable:set('x', 5)"); EXPECT_EQ(get(lua, "mutable:get('x')"), 5); EXPECT_EQ(get(lua, "ro:get('x')"), 5); EXPECT_THROW(lua.safe_script("ro:set('y', 3)"), std::exception); lua.safe_script("t1 = mutable:asTable()"); lua.safe_script("t2 = ro:asTable()"); EXPECT_EQ(get(lua, "t1.x"), 5); EXPECT_EQ(get(lua, "t2.x"), 5); lua.safe_script("mutable:reset()"); EXPECT_TRUE(get(lua, "ro:get('x') == nil")); lua.safe_script("mutable:reset({x=4, y=7})"); EXPECT_EQ(get(lua, "ro:get('x')"), 4); EXPECT_EQ(get(lua, "ro:get('y')"), 7); EXPECT_THAT(get(lua, "table.concat(callbackCalls, ', ')"), "test_x, test_*, test_*"); }); } TEST(LuaUtilStorageTest, Table) { LuaUtil::LuaState luaState{ nullptr, nullptr }; luaState.protectedCall([](LuaUtil::LuaView& view) { LuaUtil::LuaStorage::initLuaBindings(view); LuaUtil::LuaStorage storage; auto& lua = view.sol(); storage.setActive(true); lua["mutable"] = storage.getMutableSection(lua, "test"); lua["ro"] = storage.getReadOnlySection(lua, "test"); lua.safe_script("mutable:set('x', { y = 'abc', z = 7 })"); EXPECT_EQ(get(lua, "mutable:get('x').z"), 7); EXPECT_THROW(lua.safe_script("mutable:get('x').z = 3"), std::exception); EXPECT_NO_THROW(lua.safe_script("mutable:getCopy('x').z = 3")); EXPECT_EQ(get(lua, "mutable:get('x').z"), 7); EXPECT_EQ(get(lua, "ro:get('x').z"), 7); EXPECT_EQ(get(lua, "ro:get('x').y"), "abc"); }); } TEST(LuaUtilStorageTest, Saving) { LuaUtil::LuaState luaState{ nullptr, nullptr }; luaState.protectedCall([](LuaUtil::LuaView& view) { LuaUtil::LuaStorage::initLuaBindings(view); LuaUtil::LuaStorage storage; auto& lua = view.sol(); storage.setActive(true); lua["permanent"] = storage.getMutableSection(lua, "permanent"); lua["temporary"] = storage.getMutableSection(lua, "temporary"); lua.safe_script("temporary:removeOnExit()"); lua.safe_script("permanent:set('x', 1)"); lua.safe_script("temporary:set('y', 2)"); const auto tmpFile = std::filesystem::temp_directory_path() / "test_storage.bin"; storage.save(lua, tmpFile); EXPECT_EQ(get(lua, "permanent:get('x')"), 1); EXPECT_EQ(get(lua, "temporary:get('y')"), 2); storage.clearTemporaryAndRemoveCallbacks(); lua["permanent"] = storage.getMutableSection(lua, "permanent"); lua["temporary"] = storage.getMutableSection(lua, "temporary"); EXPECT_EQ(get(lua, "permanent:get('x')"), 1); EXPECT_TRUE(get(lua, "temporary:get('y') == nil")); lua.safe_script("permanent:set('x', 3)"); lua.safe_script("permanent:set('z', 4)"); LuaUtil::LuaStorage storage2; storage2.setActive(true); storage2.load(lua, tmpFile); lua["permanent"] = storage2.getMutableSection(lua, "permanent"); lua["temporary"] = storage2.getMutableSection(lua, "temporary"); EXPECT_EQ(get(lua, "permanent:get('x')"), 1); EXPECT_TRUE(get(lua, "permanent:get('z') == nil")); EXPECT_TRUE(get(lua, "temporary:get('y') == nil")); }); } }