#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, Basic) { // Note: LuaUtil::Callback can be used only if Lua is initialized via LuaUtil::LuaState LuaUtil::LuaState luaState{ nullptr, nullptr }; sol::state_view& mLua = luaState.sol(); LuaUtil::LuaStorage::initLuaBindings(mLua); LuaUtil::LuaStorage storage(mLua); std::vector callbackCalls; LuaUtil::Callback callback{ sol::make_object(mLua, [&](const std::string& section, const sol::optional& key) { if (key) callbackCalls.push_back(section + "_" + *key); else callbackCalls.push_back(section + "_*"); }), sol::table(mLua, sol::create) }; callback.mHiddenData[LuaUtil::ScriptsContainer::sScriptIdKey] = LuaUtil::ScriptId{}; mLua["mutable"] = storage.getMutableSection("test"); mLua["ro"] = storage.getReadOnlySection("test"); mLua["ro"]["subscribe"](mLua["ro"], callback); mLua.safe_script("mutable:set('x', 5)"); EXPECT_EQ(get(mLua, "mutable:get('x')"), 5); EXPECT_EQ(get(mLua, "ro:get('x')"), 5); EXPECT_THROW(mLua.safe_script("ro:set('y', 3)"), std::exception); mLua.safe_script("t1 = mutable:asTable()"); mLua.safe_script("t2 = ro:asTable()"); EXPECT_EQ(get(mLua, "t1.x"), 5); EXPECT_EQ(get(mLua, "t2.x"), 5); mLua.safe_script("mutable:reset()"); EXPECT_TRUE(get(mLua, "ro:get('x') == nil")); mLua.safe_script("mutable:reset({x=4, y=7})"); EXPECT_EQ(get(mLua, "ro:get('x')"), 4); EXPECT_EQ(get(mLua, "ro:get('y')"), 7); EXPECT_THAT(callbackCalls, ::testing::ElementsAre("test_x", "test_*", "test_*")); } TEST(LuaUtilStorageTest, Table) { sol::state mLua; LuaUtil::LuaStorage::initLuaBindings(mLua); LuaUtil::LuaStorage storage(mLua); mLua["mutable"] = storage.getMutableSection("test"); mLua["ro"] = storage.getReadOnlySection("test"); mLua.safe_script("mutable:set('x', { y = 'abc', z = 7 })"); EXPECT_EQ(get(mLua, "mutable:get('x').z"), 7); EXPECT_THROW(mLua.safe_script("mutable:get('x').z = 3"), std::exception); EXPECT_NO_THROW(mLua.safe_script("mutable:getCopy('x').z = 3")); EXPECT_EQ(get(mLua, "mutable:get('x').z"), 7); EXPECT_EQ(get(mLua, "ro:get('x').z"), 7); EXPECT_EQ(get(mLua, "ro:get('x').y"), "abc"); } TEST(LuaUtilStorageTest, Saving) { sol::state mLua; LuaUtil::LuaStorage::initLuaBindings(mLua); LuaUtil::LuaStorage storage(mLua); mLua["permanent"] = storage.getMutableSection("permanent"); mLua["temporary"] = storage.getMutableSection("temporary"); mLua.safe_script("temporary:removeOnExit()"); mLua.safe_script("permanent:set('x', 1)"); mLua.safe_script("temporary:set('y', 2)"); const auto tmpFile = std::filesystem::temp_directory_path() / "test_storage.bin"; storage.save(tmpFile); EXPECT_EQ(get(mLua, "permanent:get('x')"), 1); EXPECT_EQ(get(mLua, "temporary:get('y')"), 2); storage.clearTemporaryAndRemoveCallbacks(); mLua["permanent"] = storage.getMutableSection("permanent"); mLua["temporary"] = storage.getMutableSection("temporary"); EXPECT_EQ(get(mLua, "permanent:get('x')"), 1); EXPECT_TRUE(get(mLua, "temporary:get('y') == nil")); mLua.safe_script("permanent:set('x', 3)"); mLua.safe_script("permanent:set('z', 4)"); LuaUtil::LuaStorage storage2(mLua); storage2.load(tmpFile); mLua["permanent"] = storage2.getMutableSection("permanent"); mLua["temporary"] = storage2.getMutableSection("temporary"); EXPECT_EQ(get(mLua, "permanent:get('x')"), 1); EXPECT_TRUE(get(mLua, "permanent:get('z') == nil")); EXPECT_TRUE(get(mLua, "temporary:get('y') == nil")); } }