mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-13 19:49:51 +01:00
Add plugin tests
This commit is contained in:
parent
b338d822fa
commit
de6e2cc1fd
|
@ -59,6 +59,8 @@ struct PluginMeta {
|
|||
}
|
||||
|
||||
explicit PluginMeta(const QJsonObject &obj);
|
||||
// This is for tests
|
||||
PluginMeta() = default;
|
||||
};
|
||||
|
||||
class Plugin
|
||||
|
@ -153,6 +155,7 @@ private:
|
|||
int lastTimerId = 0;
|
||||
|
||||
friend class PluginController;
|
||||
friend class PluginControllerAccess; // this is for tests
|
||||
};
|
||||
} // namespace chatterino
|
||||
#endif
|
||||
|
|
|
@ -74,6 +74,9 @@ private:
|
|||
static void loadChatterinoLib(lua_State *l);
|
||||
bool tryLoadFromDir(const QDir &pluginDir);
|
||||
std::map<QString, std::unique_ptr<Plugin>> plugins_;
|
||||
|
||||
// This is for tests, pay no attention
|
||||
friend class PluginControllerAccess;
|
||||
};
|
||||
|
||||
} // namespace chatterino
|
||||
|
|
|
@ -10,6 +10,8 @@ namespace chatterino {
|
|||
|
||||
struct PluginPermission {
|
||||
explicit PluginPermission(const QJsonObject &obj);
|
||||
// This is for tests
|
||||
PluginPermission() = default;
|
||||
|
||||
enum class Type {
|
||||
FilesystemRead,
|
||||
|
|
|
@ -48,6 +48,7 @@ set(test_SOURCES
|
|||
${CMAKE_CURRENT_LIST_DIR}/src/FlagsEnum.cpp
|
||||
${CMAKE_CURRENT_LIST_DIR}/src/MessageLayoutContainer.cpp
|
||||
${CMAKE_CURRENT_LIST_DIR}/src/CancellationToken.cpp
|
||||
${CMAKE_CURRENT_LIST_DIR}/src/Plugins.cpp
|
||||
# Add your new file above this line!
|
||||
)
|
||||
|
||||
|
|
550
tests/src/Plugins.cpp
Normal file
550
tests/src/Plugins.cpp
Normal file
|
@ -0,0 +1,550 @@
|
|||
#ifdef CHATTERINO_HAVE_PLUGINS
|
||||
# include "Application.hpp"
|
||||
# include "common/Channel.hpp"
|
||||
# include "common/network/NetworkCommon.hpp"
|
||||
# include "controllers/commands/Command.hpp" // IT IS USED
|
||||
# include "controllers/commands/CommandController.hpp"
|
||||
# include "controllers/plugins/api/ChannelRef.hpp"
|
||||
# include "controllers/plugins/Plugin.hpp"
|
||||
# include "controllers/plugins/PluginController.hpp"
|
||||
# include "controllers/plugins/PluginPermission.hpp"
|
||||
# include "controllers/plugins/SolTypes.hpp" // IT IS USE
|
||||
# include "mocks/BaseApplication.hpp"
|
||||
# include "mocks/Channel.hpp"
|
||||
# include "mocks/Emotes.hpp"
|
||||
# include "mocks/Logging.hpp"
|
||||
# include "mocks/TwitchIrcServer.hpp"
|
||||
# include "singletons/Logging.hpp"
|
||||
# include "Test.hpp"
|
||||
|
||||
# include <gtest/gtest.h>
|
||||
# include <lauxlib.h>
|
||||
# include <sol/state_view.hpp>
|
||||
|
||||
# include <memory>
|
||||
# include <optional>
|
||||
# include <utility>
|
||||
|
||||
using namespace chatterino;
|
||||
using chatterino::mock::MockChannel;
|
||||
|
||||
namespace {
|
||||
const QString TEST_SETTINGS = R"(
|
||||
{
|
||||
"plugins": {
|
||||
"supportEnabled": true,
|
||||
"enabledPlugins": [
|
||||
"test"
|
||||
]
|
||||
}
|
||||
}
|
||||
)";
|
||||
|
||||
class MockTwitch : public mock::MockTwitchIrcServer
|
||||
{
|
||||
public:
|
||||
ChannelPtr mm2pl = std::make_shared<MockChannel>("mm2pl");
|
||||
|
||||
ChannelPtr getChannelOrEmpty(const QString &dirtyChannelName) override
|
||||
{
|
||||
if (dirtyChannelName == "mm2pl")
|
||||
{
|
||||
return this->mm2pl;
|
||||
}
|
||||
return Channel::getEmpty();
|
||||
}
|
||||
|
||||
std::shared_ptr<Channel> getChannelOrEmptyByID(
|
||||
const QString &channelID) override
|
||||
{
|
||||
if (channelID == "117691339")
|
||||
{
|
||||
return this->mm2pl;
|
||||
}
|
||||
return Channel::getEmpty();
|
||||
}
|
||||
};
|
||||
|
||||
class MockApplication : public mock::BaseApplication
|
||||
{
|
||||
public:
|
||||
MockApplication()
|
||||
: mock::BaseApplication(TEST_SETTINGS)
|
||||
, plugins(this->paths_)
|
||||
, commands(this->paths_)
|
||||
{
|
||||
}
|
||||
|
||||
PluginController *getPlugins() override
|
||||
{
|
||||
return &this->plugins;
|
||||
}
|
||||
|
||||
CommandController *getCommands() override
|
||||
{
|
||||
return &this->commands;
|
||||
}
|
||||
|
||||
IEmotes *getEmotes() override
|
||||
{
|
||||
return &this->emotes;
|
||||
}
|
||||
|
||||
mock::MockTwitchIrcServer *getTwitch() override
|
||||
{
|
||||
return &this->twitch;
|
||||
}
|
||||
|
||||
ILogging *getChatLogger() override
|
||||
{
|
||||
return &this->logging;
|
||||
}
|
||||
|
||||
PluginController plugins;
|
||||
mock::EmptyLogging logging;
|
||||
CommandController commands;
|
||||
mock::Emotes emotes;
|
||||
MockTwitch twitch;
|
||||
};
|
||||
|
||||
} // namespace
|
||||
|
||||
namespace chatterino {
|
||||
class PluginControllerAccess
|
||||
{
|
||||
public:
|
||||
static bool tryLoadFromDir(const QDir &pluginDir)
|
||||
{
|
||||
return getApp()->getPlugins()->tryLoadFromDir(pluginDir);
|
||||
}
|
||||
|
||||
static void openLibrariesFor(Plugin *plugin)
|
||||
{
|
||||
return PluginController::openLibrariesFor(plugin);
|
||||
}
|
||||
|
||||
static std::map<QString, std::unique_ptr<Plugin>> &plugins()
|
||||
{
|
||||
return getApp()->getPlugins()->plugins_;
|
||||
}
|
||||
|
||||
static lua_State *state(Plugin *pl)
|
||||
{
|
||||
return pl->state_;
|
||||
}
|
||||
};
|
||||
} // namespace chatterino
|
||||
|
||||
class PluginTest : public ::testing::Test
|
||||
{
|
||||
protected:
|
||||
void configure(std::vector<PluginPermission> permissions = {})
|
||||
{
|
||||
this->app = std::make_unique<MockApplication>();
|
||||
|
||||
auto &plugins = PluginControllerAccess::plugins();
|
||||
{
|
||||
PluginMeta meta;
|
||||
meta.name = "Test";
|
||||
meta.license = "MIT";
|
||||
meta.homepage = "https://github.com/Chatterino/chatterino2";
|
||||
meta.description = "Plugin for tests";
|
||||
meta.permissions = std::move(permissions);
|
||||
|
||||
QDir plugindir =
|
||||
QDir(app->paths_.pluginsDirectory).absoluteFilePath("test");
|
||||
|
||||
plugindir.mkpath(".");
|
||||
auto temp = std::make_unique<Plugin>("test", luaL_newstate(), meta,
|
||||
plugindir);
|
||||
this->rawpl = temp.get();
|
||||
plugins.insert({"test", std::move(temp)});
|
||||
}
|
||||
|
||||
// XXX: this skips PluginController::load()
|
||||
PluginControllerAccess::openLibrariesFor(rawpl);
|
||||
this->lua = {PluginControllerAccess::state(rawpl)};
|
||||
|
||||
this->channel = app->twitch.mm2pl;
|
||||
this->rawpl->dataDirectory().mkpath(".");
|
||||
}
|
||||
|
||||
void TearDown() override
|
||||
{
|
||||
// perform safe destruction of the plugin
|
||||
this->lua = std::nullopt;
|
||||
PluginControllerAccess::plugins().clear();
|
||||
this->rawpl = nullptr;
|
||||
this->app.reset();
|
||||
}
|
||||
|
||||
Plugin *rawpl = nullptr;
|
||||
std::unique_ptr<MockApplication> app;
|
||||
std::optional<sol::state_view> lua;
|
||||
ChannelPtr channel;
|
||||
};
|
||||
|
||||
TEST_F(PluginTest, testCommands)
|
||||
{
|
||||
configure();
|
||||
|
||||
lua->script(R"lua(
|
||||
_G.called = false
|
||||
_G.words = nil
|
||||
_G.channel = nil
|
||||
c2.register_command("/test", function(ctx)
|
||||
_G.called = true
|
||||
_G.words = ctx.words
|
||||
_G.channel = ctx.channel
|
||||
end)
|
||||
)lua");
|
||||
|
||||
EXPECT_EQ(app->commands.pluginCommands(), QStringList{"/test"});
|
||||
app->commands.execCommand("/test with arguments", channel, false);
|
||||
bool called = (*lua)["called"];
|
||||
EXPECT_EQ(called, true);
|
||||
|
||||
EXPECT_NE((*lua)["words"], sol::nil);
|
||||
{
|
||||
sol::table tbl = (*lua)["words"];
|
||||
std::vector<std::string> words;
|
||||
for (auto &o : tbl)
|
||||
{
|
||||
words.push_back(o.second.as<std::string>());
|
||||
}
|
||||
EXPECT_EQ(words,
|
||||
std::vector<std::string>({"/test", "with", "arguments"}));
|
||||
}
|
||||
|
||||
sol::object chnobj = (*lua)["channel"];
|
||||
EXPECT_EQ(chnobj.get_type(), sol::type::userdata);
|
||||
lua::api::ChannelRef ref = chnobj.as<lua::api::ChannelRef>();
|
||||
EXPECT_EQ(ref.get_name(), channel->getName());
|
||||
}
|
||||
|
||||
TEST_F(PluginTest, testCompletion)
|
||||
{
|
||||
configure();
|
||||
|
||||
lua->script(R"lua(
|
||||
_G.called = false
|
||||
_G.query = nil
|
||||
_G.full_text_content = nil
|
||||
_G.cursor_position = nil
|
||||
_G.is_first_word = nil
|
||||
|
||||
c2.register_callback(
|
||||
c2.EventType.CompletionRequested,
|
||||
function(ev)
|
||||
_G.called = true
|
||||
_G.query = ev.query
|
||||
_G.full_text_content = ev.full_text_content
|
||||
_G.cursor_position = ev.cursor_position
|
||||
_G.is_first_word = ev.is_first_word
|
||||
if ev.query == "exclusive" then
|
||||
return {
|
||||
hide_others = true,
|
||||
values = {"Completion1", "Completion2"}
|
||||
}
|
||||
end
|
||||
return {
|
||||
hide_others = false,
|
||||
values = {"Completion"},
|
||||
}
|
||||
end
|
||||
)
|
||||
)lua");
|
||||
|
||||
bool done{};
|
||||
QStringList results;
|
||||
std::tie(done, results) =
|
||||
app->plugins.updateCustomCompletions("foo", "foo", 3, true);
|
||||
ASSERT_EQ(done, false);
|
||||
ASSERT_EQ(results, QStringList{"Completion"});
|
||||
|
||||
ASSERT_EQ((*lua).get<std::string>("query"), "foo");
|
||||
ASSERT_EQ((*lua).get<std::string>("full_text_content"), "foo");
|
||||
ASSERT_EQ((*lua).get<int>("cursor_position"), 3);
|
||||
ASSERT_EQ((*lua).get<bool>("is_first_word"), true);
|
||||
|
||||
std::tie(done, results) = app->plugins.updateCustomCompletions(
|
||||
"exclusive", "foo exclusive", 13, false);
|
||||
ASSERT_EQ(done, true);
|
||||
ASSERT_EQ(results, QStringList({"Completion1", "Completion2"}));
|
||||
|
||||
ASSERT_EQ((*lua).get<std::string>("query"), "exclusive");
|
||||
ASSERT_EQ((*lua).get<std::string>("full_text_content"), "foo exclusive");
|
||||
ASSERT_EQ((*lua).get<int>("cursor_position"), 13);
|
||||
ASSERT_EQ((*lua).get<bool>("is_first_word"), false);
|
||||
}
|
||||
|
||||
TEST_F(PluginTest, testChannel)
|
||||
{
|
||||
configure();
|
||||
lua->script(R"lua(
|
||||
chn = c2.Channel.by_name("mm2pl")
|
||||
)lua");
|
||||
|
||||
ASSERT_EQ(lua->script(R"lua( return chn:get_name() )lua").get<QString>(0),
|
||||
"mm2pl");
|
||||
ASSERT_EQ(
|
||||
lua->script(R"lua( return chn:get_type() )lua").get<Channel::Type>(0),
|
||||
Channel::Type::Twitch);
|
||||
ASSERT_EQ(
|
||||
lua->script(R"lua( return chn:get_display_name() )lua").get<QString>(0),
|
||||
"mm2pl");
|
||||
// TODO: send_message, add_system_message
|
||||
|
||||
ASSERT_EQ(
|
||||
lua->script(R"lua( return chn:is_twitch_channel() )lua").get<bool>(0),
|
||||
true);
|
||||
|
||||
// this is not a TwitchChannel
|
||||
EXPECT_ANY_THROW(lua->script(R"lua( return chn:is_broadcaster() )lua"));
|
||||
EXPECT_ANY_THROW(lua->script(R"lua( return chn:is_mod() )lua"));
|
||||
EXPECT_ANY_THROW(lua->script(R"lua( return chn:is_vip() )lua"));
|
||||
EXPECT_ANY_THROW(lua->script(R"lua( return chn:get_twitch_id() )lua"));
|
||||
}
|
||||
|
||||
# ifdef CHATTERINO_TEST_USE_PUBLIC_HTTPBIN
|
||||
// Not using httpbin.org, since it can be really slow and cause timeouts.
|
||||
// postman-echo has the same API.
|
||||
const char *const HTTPBIN_BASE_URL = "https://postman-echo.com";
|
||||
# else
|
||||
const char *const HTTPBIN_BASE_URL = "http://127.0.0.1:9051";
|
||||
# endif
|
||||
|
||||
class RequestWaiter
|
||||
{
|
||||
public:
|
||||
void requestDone()
|
||||
{
|
||||
{
|
||||
std::unique_lock lck(this->mutex_);
|
||||
ASSERT_FALSE(this->requestDone_);
|
||||
this->requestDone_ = true;
|
||||
}
|
||||
this->condition_.notify_one();
|
||||
}
|
||||
|
||||
void waitForRequest()
|
||||
{
|
||||
using namespace std::chrono_literals;
|
||||
|
||||
while (true)
|
||||
{
|
||||
{
|
||||
std::unique_lock lck(this->mutex_);
|
||||
bool done = this->condition_.wait_for(lck, 10ms, [this] {
|
||||
return this->requestDone_;
|
||||
});
|
||||
if (done)
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
QCoreApplication::processEvents(QEventLoop::AllEvents);
|
||||
QCoreApplication::sendPostedEvents(nullptr, QEvent::DeferredDelete);
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
std::mutex mutex_;
|
||||
std::condition_variable condition_;
|
||||
bool requestDone_ = false;
|
||||
};
|
||||
TEST_F(PluginTest, testHttp)
|
||||
{
|
||||
{
|
||||
PluginPermission net;
|
||||
net.type = PluginPermission::Type::Network;
|
||||
configure({net});
|
||||
}
|
||||
|
||||
lua->script(R"lua(
|
||||
function DoReq(url, postdata)
|
||||
r = c2.HTTPRequest.create(method, url)
|
||||
r:on_success(function(res)
|
||||
status = res:status()
|
||||
data = res:data()
|
||||
error = res:error()
|
||||
success = true
|
||||
end)
|
||||
r:on_error(function(res)
|
||||
status = res:status()
|
||||
data = res:data()
|
||||
error = res:error()
|
||||
failure = true
|
||||
end)
|
||||
r:finally(function()
|
||||
finally = true
|
||||
done()
|
||||
end)
|
||||
if postdata ~= "" then
|
||||
r:set_payload(postdata)
|
||||
r:set_header("Content-Type", "text/plain")
|
||||
end
|
||||
r:set_timeout(1000)
|
||||
r:execute()
|
||||
end
|
||||
)lua");
|
||||
|
||||
struct RequestCase {
|
||||
QString url;
|
||||
bool success;
|
||||
bool failure;
|
||||
|
||||
int status;
|
||||
QString error;
|
||||
|
||||
NetworkRequestType meth = NetworkRequestType::Get;
|
||||
QByteArray data;
|
||||
};
|
||||
|
||||
std::vector<RequestCase> cases{
|
||||
{"/status/200", true, false, 200, "200"},
|
||||
{"/delay/2", false, true, 0, "TimeoutError"},
|
||||
{"/post", true, false, 200, "200", NetworkRequestType::Post,
|
||||
"Example data"},
|
||||
};
|
||||
|
||||
for (const auto &c : cases)
|
||||
{
|
||||
lua->script(R"lua(
|
||||
success = false
|
||||
failure = false
|
||||
finally = false
|
||||
|
||||
status = nil
|
||||
data = nil
|
||||
error = nil
|
||||
)lua");
|
||||
RequestWaiter waiter;
|
||||
(*lua)["method"] = c.meth;
|
||||
(*lua)["done"] = [&waiter]() {
|
||||
waiter.requestDone();
|
||||
};
|
||||
|
||||
(*lua)["DoReq"](HTTPBIN_BASE_URL + c.url, c.data);
|
||||
waiter.waitForRequest();
|
||||
|
||||
EXPECT_EQ(lua->get<bool>("success"), c.success);
|
||||
EXPECT_EQ(lua->get<bool>("failure"), c.failure);
|
||||
EXPECT_EQ(lua->get<bool>("finally"), true);
|
||||
|
||||
if (c.status != 0)
|
||||
{
|
||||
EXPECT_EQ(lua->get<int>("status"), c.status);
|
||||
}
|
||||
else
|
||||
{
|
||||
EXPECT_EQ((*lua)["status"], sol::nil);
|
||||
}
|
||||
EXPECT_EQ(lua->get<QString>("error"), c.error);
|
||||
EXPECT_EQ(lua->get<QByteArray>("data"), c.data);
|
||||
}
|
||||
}
|
||||
const QByteArray TEST_FILE_DATA = "Test file data\nWith a new line.\n";
|
||||
|
||||
TEST_F(PluginTest, ioTest)
|
||||
{
|
||||
{
|
||||
PluginPermission ioread;
|
||||
PluginPermission iowrite;
|
||||
ioread.type = PluginPermission::Type::FilesystemRead;
|
||||
iowrite.type = PluginPermission::Type::FilesystemWrite;
|
||||
configure({ioread, iowrite});
|
||||
}
|
||||
|
||||
lua->set("TEST_DATA", TEST_FILE_DATA);
|
||||
|
||||
lua->script(R"lua(
|
||||
f, err = io.open("testfile", "w")
|
||||
print(f, err)
|
||||
f:write(TEST_DATA)
|
||||
f:close()
|
||||
|
||||
f, err = io.open("testfile", "r")
|
||||
out = f:read("a")
|
||||
f:close()
|
||||
)lua");
|
||||
EXPECT_EQ(lua->get<QByteArray>("out"), TEST_FILE_DATA);
|
||||
|
||||
lua->script(R"lua(
|
||||
io.input("testfile")
|
||||
out = io.read("a")
|
||||
)lua");
|
||||
EXPECT_EQ(lua->get<QByteArray>("out"), TEST_FILE_DATA);
|
||||
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
io.popen("/bin/sh", "-c", "notify-send \"This should not execute.\"")
|
||||
)lua"));
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
io.tmpfile()
|
||||
)lua"));
|
||||
}
|
||||
|
||||
TEST_F(PluginTest, ioNoPerms)
|
||||
{
|
||||
configure();
|
||||
auto file = rawpl->dataDirectory().filePath("testfile");
|
||||
QFile f(file);
|
||||
f.open(QFile::WriteOnly);
|
||||
f.write(TEST_FILE_DATA);
|
||||
f.close();
|
||||
|
||||
EXPECT_EQ(
|
||||
// clang-format off
|
||||
lua->script(R"lua(
|
||||
f, err = io.open("testfile", "r")
|
||||
return err
|
||||
)lua").get<QString>(0),
|
||||
"Plugin does not have permissions to access given file."
|
||||
// clang-format on
|
||||
);
|
||||
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
io.input("testfile")
|
||||
)lua"));
|
||||
|
||||
EXPECT_EQ(
|
||||
// clang-format off
|
||||
lua->script(R"lua(
|
||||
f, err = io.open("testfile", "w")
|
||||
return err
|
||||
)lua").get<QString>(0),
|
||||
"Plugin does not have permissions to access given file."
|
||||
// clang-format on
|
||||
);
|
||||
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
io.output("testfile")
|
||||
)lua"));
|
||||
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
io.lines("testfile")
|
||||
)lua"));
|
||||
}
|
||||
|
||||
TEST_F(PluginTest, requireNoData)
|
||||
{
|
||||
{
|
||||
PluginPermission ioread;
|
||||
PluginPermission iowrite;
|
||||
ioread.type = PluginPermission::Type::FilesystemRead;
|
||||
iowrite.type = PluginPermission::Type::FilesystemWrite;
|
||||
configure({ioread, iowrite});
|
||||
}
|
||||
|
||||
auto file = rawpl->dataDirectory().filePath("thisiscode.lua");
|
||||
QFile f(file);
|
||||
f.open(QFile::WriteOnly);
|
||||
f.write(R"lua(print("Data was executed"))lua");
|
||||
f.close();
|
||||
|
||||
EXPECT_ANY_THROW(lua->script(R"lua(
|
||||
require("data.thisiscode")
|
||||
)lua"));
|
||||
}
|
||||
|
||||
#endif
|
Loading…
Reference in a new issue