mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-15 22:49:48 +00:00
0f331a4fc5
Fix potential memory leak in LuaUtil::Callback See merge request OpenMW/openmw!2733
469 lines
17 KiB
C++
469 lines
17 KiB
C++
#include "gmock/gmock.h"
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <components/esm/luascripts.hpp>
|
|
|
|
#include <components/lua/asyncpackage.hpp>
|
|
#include <components/lua/luastate.hpp>
|
|
#include <components/lua/scriptscontainer.hpp>
|
|
|
|
#include "../testing_util.hpp"
|
|
|
|
namespace
|
|
{
|
|
using namespace testing;
|
|
using namespace TestingOpenMW;
|
|
|
|
VFSTestFile invalidScript("not a script");
|
|
VFSTestFile incorrectScript(
|
|
"return { incorrectSection = {}, engineHandlers = { incorrectHandler = function() end } }");
|
|
VFSTestFile emptyScript("");
|
|
|
|
VFSTestFile testScript(R"X(
|
|
return {
|
|
engineHandlers = {
|
|
onUpdate = function(dt) print(' update ' .. tostring(dt)) end,
|
|
onLoad = function() print('load') end,
|
|
},
|
|
eventHandlers = {
|
|
Event1 = function(eventData) print(' event1 ' .. tostring(eventData.x)) end,
|
|
Event2 = function(eventData) print(' event2 ' .. tostring(eventData.x)) end,
|
|
Print = function() print('print') end
|
|
}
|
|
}
|
|
)X");
|
|
|
|
VFSTestFile stopEventScript(R"X(
|
|
return {
|
|
eventHandlers = {
|
|
Event1 = function(eventData)
|
|
print(' event1 ' .. tostring(eventData.x))
|
|
return eventData.x >= 1
|
|
end
|
|
}
|
|
}
|
|
)X");
|
|
|
|
VFSTestFile loadSaveScript(R"X(
|
|
x = 0
|
|
y = 0
|
|
return {
|
|
engineHandlers = {
|
|
onSave = function(state)
|
|
return {x = x, y = y}
|
|
end,
|
|
onLoad = function(state)
|
|
x, y = state.x, state.y
|
|
end
|
|
},
|
|
eventHandlers = {
|
|
Set = function(eventData)
|
|
eventData.n = eventData.n - 1
|
|
if eventData.n == 0 then
|
|
x, y = eventData.x, eventData.y
|
|
end
|
|
end,
|
|
Print = function()
|
|
print(x, y)
|
|
end
|
|
}
|
|
}
|
|
)X");
|
|
|
|
VFSTestFile interfaceScript(R"X(
|
|
return {
|
|
interfaceName = "TestInterface",
|
|
interface = {
|
|
fn = function(x) print('FN', x) end,
|
|
value = 3.5
|
|
},
|
|
}
|
|
)X");
|
|
|
|
VFSTestFile overrideInterfaceScript(R"X(
|
|
local old = nil
|
|
local interface = {
|
|
fn = function(x)
|
|
print('NEW FN', x)
|
|
old.fn(x)
|
|
end,
|
|
value,
|
|
}
|
|
return {
|
|
interfaceName = "TestInterface",
|
|
interface = interface,
|
|
engineHandlers = {
|
|
onInit = function() print('init') end,
|
|
onLoad = function() print('load') end,
|
|
onInterfaceOverride = function(oldInterface)
|
|
print('override')
|
|
old = oldInterface
|
|
interface.value = oldInterface.value + 1
|
|
end
|
|
},
|
|
}
|
|
)X");
|
|
|
|
VFSTestFile useInterfaceScript(R"X(
|
|
local interfaces = require('openmw.interfaces')
|
|
return {
|
|
engineHandlers = {
|
|
onUpdate = function()
|
|
interfaces.TestInterface.fn(interfaces.TestInterface.value)
|
|
end,
|
|
},
|
|
}
|
|
)X");
|
|
|
|
struct LuaScriptsContainerTest : Test
|
|
{
|
|
std::unique_ptr<VFS::Manager> mVFS = createTestVFS({
|
|
{ "invalid.lua", &invalidScript },
|
|
{ "incorrect.lua", &incorrectScript },
|
|
{ "empty.lua", &emptyScript },
|
|
{ "test1.lua", &testScript },
|
|
{ "test2.lua", &testScript },
|
|
{ "stopEvent.lua", &stopEventScript },
|
|
{ "loadSave1.lua", &loadSaveScript },
|
|
{ "loadSave2.lua", &loadSaveScript },
|
|
{ "testInterface.lua", &interfaceScript },
|
|
{ "overrideInterface.lua", &overrideInterfaceScript },
|
|
{ "useInterface.lua", &useInterfaceScript },
|
|
});
|
|
|
|
LuaUtil::ScriptsConfiguration mCfg;
|
|
LuaUtil::LuaState mLua{ mVFS.get(), &mCfg };
|
|
|
|
LuaScriptsContainerTest()
|
|
{
|
|
ESM::LuaScriptsCfg cfg;
|
|
LuaUtil::parseOMWScripts(cfg, R"X(
|
|
CUSTOM: invalid.lua
|
|
CUSTOM: incorrect.lua
|
|
CUSTOM: empty.lua
|
|
CUSTOM: test1.lua
|
|
CUSTOM: stopEvent.lua
|
|
CUSTOM: test2.lua
|
|
NPC: loadSave1.lua
|
|
CUSTOM, NPC: loadSave2.lua
|
|
CUSTOM, PLAYER: testInterface.lua
|
|
CUSTOM, PLAYER: overrideInterface.lua
|
|
CUSTOM, PLAYER: useInterface.lua
|
|
)X");
|
|
mCfg.init(std::move(cfg));
|
|
}
|
|
};
|
|
|
|
TEST_F(LuaScriptsContainerTest, VerifyStructure)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_FALSE(scripts.addCustomScript(*mCfg.findId("invalid.lua")));
|
|
std::string output = testing::internal::GetCapturedStdout();
|
|
EXPECT_THAT(output, HasSubstr("Can't start Test[invalid.lua]"));
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("incorrect.lua")));
|
|
std::string output = testing::internal::GetCapturedStdout();
|
|
EXPECT_THAT(output, HasSubstr("Not supported handler 'incorrectHandler' in Test[incorrect.lua]"));
|
|
EXPECT_THAT(output, HasSubstr("Not supported section 'incorrectSection' in Test[incorrect.lua]"));
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("empty.lua")));
|
|
EXPECT_FALSE(scripts.addCustomScript(*mCfg.findId("empty.lua"))); // already present
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "");
|
|
}
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, CallHandler)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test1.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("stopEvent.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test2.lua")));
|
|
scripts.update(1.5f);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test1.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t update 1.5\n");
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, CallEvent)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test1.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("stopEvent.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test2.lua")));
|
|
|
|
std::string X0 = LuaUtil::serialize(mLua.sol().create_table_with("x", 0.5));
|
|
std::string X1 = LuaUtil::serialize(mLua.sol().create_table_with("x", 1.5));
|
|
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.receiveEvent("SomeEvent", X1);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test has received event 'SomeEvent', but there are no handlers for this event\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.receiveEvent("Event1", X1);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test2.lua]:\t event1 1.5\n"
|
|
"Test[stopEvent.lua]:\t event1 1.5\n"
|
|
"Test[test1.lua]:\t event1 1.5\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.receiveEvent("Event2", X1);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test2.lua]:\t event2 1.5\n"
|
|
"Test[test1.lua]:\t event2 1.5\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.receiveEvent("Event1", X0);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test2.lua]:\t event1 0.5\n"
|
|
"Test[stopEvent.lua]:\t event1 0.5\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.receiveEvent("Event2", X0);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test2.lua]:\t event2 0.5\n"
|
|
"Test[test1.lua]:\t event2 0.5\n");
|
|
}
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, RemoveScript)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test1.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("stopEvent.lua")));
|
|
EXPECT_TRUE(scripts.addCustomScript(*mCfg.findId("test2.lua")));
|
|
std::string X = LuaUtil::serialize(mLua.sol().create_table_with("x", 0.5));
|
|
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.update(1.5f);
|
|
scripts.receiveEvent("Event1", X);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test1.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t event1 0.5\n"
|
|
"Test[stopEvent.lua]:\t event1 0.5\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
int stopEventScriptId = *mCfg.findId("stopEvent.lua");
|
|
EXPECT_TRUE(scripts.hasScript(stopEventScriptId));
|
|
scripts.removeScript(stopEventScriptId);
|
|
EXPECT_FALSE(scripts.hasScript(stopEventScriptId));
|
|
scripts.update(1.5f);
|
|
scripts.receiveEvent("Event1", X);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test1.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t event1 0.5\n"
|
|
"Test[test1.lua]:\t event1 0.5\n");
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts.removeScript(*mCfg.findId("test1.lua"));
|
|
scripts.update(1.5f);
|
|
scripts.receiveEvent("Event1", X);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test2.lua]:\t update 1.5\n"
|
|
"Test[test2.lua]:\t event1 0.5\n");
|
|
}
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, AutoStart)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
scripts.setAutoStartConf(mCfg.getPlayerConf());
|
|
testing::internal::CaptureStdout();
|
|
scripts.addAutoStartedScripts();
|
|
scripts.update(1.5f);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[overrideInterface.lua]:\toverride\n"
|
|
"Test[overrideInterface.lua]:\tinit\n"
|
|
"Test[overrideInterface.lua]:\tNEW FN\t4.5\n"
|
|
"Test[testInterface.lua]:\tFN\t4.5\n");
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, Interface)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
scripts.setAutoStartConf(mCfg.getLocalConf(ESM::REC_CREA, ESM::RefId(), ESM::RefNum()));
|
|
int addIfaceId = *mCfg.findId("testInterface.lua");
|
|
int overrideIfaceId = *mCfg.findId("overrideInterface.lua");
|
|
int useIfaceId = *mCfg.findId("useInterface.lua");
|
|
|
|
testing::internal::CaptureStdout();
|
|
scripts.addAutoStartedScripts();
|
|
scripts.update(1.5f);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "");
|
|
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_TRUE(scripts.addCustomScript(addIfaceId));
|
|
EXPECT_TRUE(scripts.addCustomScript(overrideIfaceId));
|
|
EXPECT_TRUE(scripts.addCustomScript(useIfaceId));
|
|
scripts.update(1.5f);
|
|
scripts.removeScript(overrideIfaceId);
|
|
scripts.update(1.5f);
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[overrideInterface.lua]:\toverride\n"
|
|
"Test[overrideInterface.lua]:\tinit\n"
|
|
"Test[overrideInterface.lua]:\tNEW FN\t4.5\n"
|
|
"Test[testInterface.lua]:\tFN\t4.5\n"
|
|
"Test[testInterface.lua]:\tFN\t3.5\n");
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, LoadSave)
|
|
{
|
|
LuaUtil::ScriptsContainer scripts1(&mLua, "Test");
|
|
LuaUtil::ScriptsContainer scripts2(&mLua, "Test");
|
|
LuaUtil::ScriptsContainer scripts3(&mLua, "Test");
|
|
scripts1.setAutoStartConf(mCfg.getLocalConf(ESM::REC_NPC_, ESM::RefId(), ESM::RefNum()));
|
|
scripts2.setAutoStartConf(mCfg.getLocalConf(ESM::REC_NPC_, ESM::RefId(), ESM::RefNum()));
|
|
scripts3.setAutoStartConf(mCfg.getPlayerConf());
|
|
|
|
scripts1.addAutoStartedScripts();
|
|
EXPECT_TRUE(scripts1.addCustomScript(*mCfg.findId("test1.lua")));
|
|
|
|
scripts1.receiveEvent("Set", LuaUtil::serialize(mLua.sol().create_table_with("n", 1, "x", 0.5, "y", 3.5)));
|
|
scripts1.receiveEvent("Set", LuaUtil::serialize(mLua.sol().create_table_with("n", 2, "x", 2.5, "y", 1.5)));
|
|
|
|
ESM::LuaScripts data;
|
|
scripts1.save(data);
|
|
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts2.load(data);
|
|
scripts2.receiveEvent("Print", "");
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Test[test1.lua]:\tload\n"
|
|
"Test[loadSave2.lua]:\t0.5\t3.5\n"
|
|
"Test[loadSave1.lua]:\t2.5\t1.5\n"
|
|
"Test[test1.lua]:\tprint\n");
|
|
EXPECT_FALSE(scripts2.hasScript(*mCfg.findId("testInterface.lua")));
|
|
}
|
|
{
|
|
testing::internal::CaptureStdout();
|
|
scripts3.load(data);
|
|
scripts3.receiveEvent("Print", "");
|
|
EXPECT_EQ(internal::GetCapturedStdout(),
|
|
"Ignoring Test[loadSave1.lua]; this script is not allowed here\n"
|
|
"Test[test1.lua]:\tload\n"
|
|
"Test[overrideInterface.lua]:\toverride\n"
|
|
"Test[overrideInterface.lua]:\tinit\n"
|
|
"Test[loadSave2.lua]:\t0.5\t3.5\n"
|
|
"Test[test1.lua]:\tprint\n");
|
|
EXPECT_TRUE(scripts3.hasScript(*mCfg.findId("testInterface.lua")));
|
|
}
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, Timers)
|
|
{
|
|
using TimerType = LuaUtil::ScriptsContainer::TimerType;
|
|
LuaUtil::ScriptsContainer scripts(&mLua, "Test");
|
|
int test1Id = *mCfg.findId("test1.lua");
|
|
int test2Id = *mCfg.findId("test2.lua");
|
|
|
|
testing::internal::CaptureStdout();
|
|
EXPECT_TRUE(scripts.addCustomScript(test1Id));
|
|
EXPECT_TRUE(scripts.addCustomScript(test2Id));
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "");
|
|
|
|
int counter1 = 0, counter2 = 0, counter3 = 0, counter4 = 0;
|
|
sol::function fn1 = sol::make_object(mLua.sol(), [&]() { counter1++; });
|
|
sol::function fn2 = sol::make_object(mLua.sol(), [&]() { counter2++; });
|
|
sol::function fn3 = sol::make_object(mLua.sol(), [&](int d) { counter3 += d; });
|
|
sol::function fn4 = sol::make_object(mLua.sol(), [&](int d) { counter4 += d; });
|
|
|
|
scripts.registerTimerCallback(test1Id, "A", fn3);
|
|
scripts.registerTimerCallback(test1Id, "B", fn4);
|
|
scripts.registerTimerCallback(test2Id, "B", fn3);
|
|
scripts.registerTimerCallback(test2Id, "A", fn4);
|
|
|
|
scripts.processTimers(1, 2);
|
|
|
|
scripts.setupSerializableTimer(TimerType::SIMULATION_TIME, 10, test1Id, "B", sol::make_object(mLua.sol(), 3));
|
|
scripts.setupSerializableTimer(TimerType::GAME_TIME, 10, test2Id, "B", sol::make_object(mLua.sol(), 4));
|
|
scripts.setupSerializableTimer(TimerType::SIMULATION_TIME, 5, test1Id, "A", sol::make_object(mLua.sol(), 1));
|
|
scripts.setupSerializableTimer(TimerType::GAME_TIME, 5, test2Id, "A", sol::make_object(mLua.sol(), 2));
|
|
scripts.setupSerializableTimer(TimerType::SIMULATION_TIME, 15, test1Id, "A", sol::make_object(mLua.sol(), 10));
|
|
scripts.setupSerializableTimer(TimerType::SIMULATION_TIME, 15, test1Id, "B", sol::make_object(mLua.sol(), 20));
|
|
|
|
scripts.setupUnsavableTimer(TimerType::SIMULATION_TIME, 10, test2Id, fn2);
|
|
scripts.setupUnsavableTimer(TimerType::GAME_TIME, 10, test1Id, fn2);
|
|
scripts.setupUnsavableTimer(TimerType::SIMULATION_TIME, 5, test2Id, fn1);
|
|
scripts.setupUnsavableTimer(TimerType::GAME_TIME, 5, test1Id, fn1);
|
|
scripts.setupUnsavableTimer(TimerType::SIMULATION_TIME, 15, test2Id, fn1);
|
|
|
|
EXPECT_EQ(counter1, 0);
|
|
EXPECT_EQ(counter3, 0);
|
|
|
|
scripts.processTimers(6, 4);
|
|
|
|
EXPECT_EQ(counter1, 1);
|
|
EXPECT_EQ(counter3, 1);
|
|
EXPECT_EQ(counter4, 0);
|
|
|
|
scripts.processTimers(6, 8);
|
|
|
|
EXPECT_EQ(counter1, 2);
|
|
EXPECT_EQ(counter2, 0);
|
|
EXPECT_EQ(counter3, 1);
|
|
EXPECT_EQ(counter4, 2);
|
|
|
|
scripts.processTimers(11, 12);
|
|
|
|
EXPECT_EQ(counter1, 2);
|
|
EXPECT_EQ(counter2, 2);
|
|
EXPECT_EQ(counter3, 5);
|
|
EXPECT_EQ(counter4, 5);
|
|
|
|
testing::internal::CaptureStdout();
|
|
ESM::LuaScripts data;
|
|
scripts.save(data);
|
|
scripts.load(data);
|
|
scripts.registerTimerCallback(test1Id, "B", fn4);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "Test[test1.lua]:\tload\nTest[test2.lua]:\tload\n");
|
|
|
|
testing::internal::CaptureStdout();
|
|
scripts.processTimers(20, 20);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "Test[test1.lua] callTimer failed: Callback 'A' doesn't exist\n");
|
|
|
|
EXPECT_EQ(counter1, 2);
|
|
EXPECT_EQ(counter2, 2);
|
|
EXPECT_EQ(counter3, 5);
|
|
EXPECT_EQ(counter4, 25);
|
|
}
|
|
|
|
TEST_F(LuaScriptsContainerTest, CallbackWrapper)
|
|
{
|
|
LuaUtil::Callback callback{ mLua.sol()["print"], mLua.newTable() };
|
|
callback.mHiddenData[LuaUtil::ScriptsContainer::sScriptDebugNameKey] = "some_script.lua";
|
|
callback.mHiddenData[LuaUtil::ScriptsContainer::sScriptIdKey] = LuaUtil::ScriptId{ nullptr, 0 };
|
|
|
|
testing::internal::CaptureStdout();
|
|
callback.call(1.5);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "1.5\n");
|
|
|
|
testing::internal::CaptureStdout();
|
|
callback.call(1.5, 2.5);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "1.5\t2.5\n");
|
|
|
|
testing::internal::CaptureStdout();
|
|
callback.mHiddenData[LuaUtil::ScriptsContainer::sScriptIdKey] = sol::nil;
|
|
callback.call(1.5, 2.5);
|
|
EXPECT_EQ(internal::GetCapturedStdout(), "Ignored callback to the removed script some_script.lua\n");
|
|
}
|
|
|
|
}
|