fbshipit-source-id: b6fc29740c6875d2e78953b8a7123890a67930f2 Co-authored-by: Sebastian McKenzie <sebmck@fb.com> Co-authored-by: John Knox <jknox@fb.com> Co-authored-by: Emil Sjölander <emilsj@fb.com> Co-authored-by: Pritesh Nandgaonkar <prit91@fb.com>
293 lines
9.9 KiB
C++
293 lines
9.9 KiB
C++
/*
|
|
* Copyright (c) 2018-present, Facebook, Inc.
|
|
*
|
|
* This source code is licensed under the MIT license found in the LICENSE
|
|
* file in the root directory of this source tree.
|
|
*
|
|
*/
|
|
|
|
#include <Sonar/SonarClient.h>
|
|
#include <SonarTestLib/SonarPluginMock.h>
|
|
#include <SonarTestLib/SonarWebSocketMock.h>
|
|
|
|
#include <folly/json.h>
|
|
#include <gtest/gtest.h>
|
|
|
|
namespace facebook {
|
|
namespace sonar {
|
|
namespace test {
|
|
|
|
using folly::dynamic;
|
|
|
|
TEST(SonarClientTests, testSaneMocks) {
|
|
SonarWebSocketMock socket;
|
|
socket.start();
|
|
EXPECT_TRUE(socket.isOpen());
|
|
socket.stop();
|
|
EXPECT_FALSE(socket.isOpen());
|
|
|
|
SonarPluginMock plugin("Test");
|
|
EXPECT_EQ(plugin.identifier(), "Test");
|
|
}
|
|
|
|
TEST(SonarClientTests, testGetPlugins) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
client.start();
|
|
|
|
client.addPlugin(std::make_shared<SonarPluginMock>("Cat"));
|
|
client.addPlugin(std::make_shared<SonarPluginMock>("Dog"));
|
|
|
|
dynamic message = dynamic::object("id", 1)("method", "getPlugins");
|
|
socket->callbacks->onMessageReceived(message);
|
|
|
|
dynamic expected = dynamic::object("id", 1)(
|
|
"success", dynamic::object("plugins", dynamic::array("Cat", "Dog")));
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testGetPlugin) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
const auto catPlugin = std::make_shared<SonarPluginMock>("Cat");
|
|
client.addPlugin(catPlugin);
|
|
const auto dogPlugin = std::make_shared<SonarPluginMock>("Dog");
|
|
client.addPlugin(dogPlugin);
|
|
|
|
EXPECT_EQ(catPlugin, client.getPlugin("Cat"));
|
|
EXPECT_EQ(dogPlugin, client.getPlugin("Dog"));
|
|
}
|
|
|
|
TEST(SonarClientTests, testGetPluginWithDowncast) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
const auto catPlugin = std::make_shared<SonarPluginMock>("Cat");
|
|
client.addPlugin(catPlugin);
|
|
EXPECT_EQ(catPlugin, client.getPlugin<SonarPluginMock>("Cat"));
|
|
}
|
|
|
|
TEST(SonarClientTests, testRemovePlugin) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
client.start();
|
|
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test");
|
|
client.addPlugin(plugin);
|
|
client.removePlugin(plugin);
|
|
|
|
dynamic message = dynamic::object("id", 1)("method", "getPlugins");
|
|
socket->callbacks->onMessageReceived(message);
|
|
|
|
dynamic expected = dynamic::object("id", 1)(
|
|
"success", dynamic::object("plugins", dynamic::array()));
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testStartStop) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
client.start();
|
|
EXPECT_TRUE(socket->isOpen());
|
|
|
|
client.stop();
|
|
EXPECT_FALSE(socket->isOpen());
|
|
}
|
|
|
|
TEST(SonarClientTests, testConnectDisconnect) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
bool pluginConnected = false;
|
|
const auto connectionCallback = [&](std::shared_ptr<SonarConnection> conn) {
|
|
pluginConnected = true;
|
|
};
|
|
const auto disconnectionCallback = [&]() { pluginConnected = false; };
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test", connectionCallback,
|
|
disconnectionCallback);
|
|
client.addPlugin(plugin);
|
|
|
|
client.start();
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
EXPECT_TRUE(pluginConnected);
|
|
|
|
client.stop();
|
|
EXPECT_FALSE(pluginConnected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testInitDeinit) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
bool pluginConnected = false;
|
|
const auto connectionCallback = [&](std::shared_ptr<SonarConnection> conn) {
|
|
pluginConnected = true;
|
|
};
|
|
const auto disconnectionCallback = [&]() { pluginConnected = false; };
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test", connectionCallback,
|
|
disconnectionCallback);
|
|
|
|
client.start();
|
|
client.addPlugin(plugin);
|
|
EXPECT_FALSE(pluginConnected);
|
|
|
|
dynamic expected = dynamic::object("method", "refreshPlugins");
|
|
EXPECT_EQ(socket->messages.front(), expected);
|
|
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
EXPECT_TRUE(pluginConnected);
|
|
|
|
dynamic messageDeinit = dynamic::object("method", "deinit")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageDeinit);
|
|
EXPECT_FALSE(pluginConnected);
|
|
|
|
dynamic messageReinit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageReinit);
|
|
EXPECT_TRUE(pluginConnected);
|
|
|
|
client.stop();
|
|
EXPECT_FALSE(pluginConnected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testRemovePluginWhenConnected) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
bool pluginConnected = false;
|
|
const auto connectionCallback = [&](std::shared_ptr<SonarConnection> conn) {
|
|
pluginConnected = true;
|
|
};
|
|
const auto disconnectionCallback = [&]() { pluginConnected = false; };
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test", connectionCallback,
|
|
disconnectionCallback);
|
|
|
|
client.addPlugin(plugin);
|
|
client.start();
|
|
client.removePlugin(plugin);
|
|
EXPECT_FALSE(pluginConnected);
|
|
|
|
dynamic expected = dynamic::object("method", "refreshPlugins");
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testUnhandleableMethod) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test");
|
|
client.addPlugin(plugin);
|
|
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
|
|
dynamic messageExecute = dynamic::object("id", 1)("method", "unexpected");
|
|
socket->callbacks->onMessageReceived(messageExecute);
|
|
|
|
dynamic expected = dynamic::object("id", 1)(
|
|
"error",
|
|
dynamic::object("message", "Received unknown method: unexpected"));
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testExecute) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
client.start();
|
|
|
|
const auto connectionCallback = [](std::shared_ptr<SonarConnection> conn) {
|
|
const auto receiver = [](const dynamic ¶ms,
|
|
std::unique_ptr<SonarResponder> responder) {
|
|
dynamic payload = dynamic::object("message", "yes_i_hear_u");
|
|
responder->success(payload);
|
|
};
|
|
conn->receive("plugin_can_u_hear_me", receiver);
|
|
};
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test", connectionCallback);
|
|
client.addPlugin(plugin);
|
|
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
|
|
dynamic messageUnexpected = dynamic::object("id", 1)("method", "execute")(
|
|
"params",
|
|
dynamic::object("api", "Test")("method", "plugin_can_u_hear_me"));
|
|
socket->callbacks->onMessageReceived(messageUnexpected);
|
|
|
|
dynamic expected = dynamic::object("id", 1)(
|
|
"success", dynamic::object("message", "yes_i_hear_u"));
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testExecuteWithParams) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
|
|
const auto connectionCallback = [&](std::shared_ptr<SonarConnection> conn) {
|
|
const auto receiver = [](const dynamic ¶ms,
|
|
std::unique_ptr<SonarResponder> responder) {
|
|
const auto &first = params["first"].asString();
|
|
const auto &second = params["second"].asString();
|
|
std::map<std::string, std::string> m{{"dog", "woof"}, {"cat", "meow"}};
|
|
dynamic payload = dynamic::object(first, m[first])(second, m[second]);
|
|
responder->success(payload);
|
|
};
|
|
conn->receive("animal_sounds", receiver);
|
|
};
|
|
auto plugin = std::make_shared<SonarPluginMock>("Test", connectionCallback);
|
|
client.addPlugin(plugin);
|
|
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Test"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
|
|
dynamic messageExecute = dynamic::object("id", 1)("method", "execute")(
|
|
"params",
|
|
dynamic::object("api", "Test")("method", "animal_sounds")(
|
|
"params", dynamic::object("first", "dog")("second", "cat")));
|
|
socket->callbacks->onMessageReceived(messageExecute);
|
|
|
|
dynamic expected = dynamic::object("id", 1)(
|
|
"success", dynamic::object("dog", "woof")("cat", "meow"));
|
|
EXPECT_EQ(socket->messages.back(), expected);
|
|
}
|
|
|
|
TEST(SonarClientTests, testExceptionUnknownPlugin) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
client.start();
|
|
|
|
dynamic messageInit = dynamic::object("method", "init")(
|
|
"params", dynamic::object("plugin", "Unknown"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
|
|
EXPECT_EQ(socket->messages.back()["error"]["message"],
|
|
"plugin Unknown not found for method init");
|
|
}
|
|
|
|
TEST(SonarClientTests, testExceptionUnknownApi) {
|
|
auto socket = new SonarWebSocketMock;
|
|
SonarClient client(std::unique_ptr<SonarWebSocketMock>{socket});
|
|
client.start();
|
|
|
|
dynamic messageInit = dynamic::object("method", "execute")(
|
|
"params", dynamic::object("api", "Unknown"));
|
|
socket->callbacks->onMessageReceived(messageInit);
|
|
|
|
EXPECT_EQ(socket->messages.back()["error"]["message"],
|
|
"connection Unknown not found for method execute");
|
|
}
|
|
|
|
} // namespace test
|
|
} // namespace sonar
|
|
} // namespace facebook
|