Don't use private access in FlipperRSocketResponder

Summary:
Changing FlipperRSocketResponder to only use public parts of FlipperConnectionManagerImpl.

This means we can test it by injecting a FCM mock, and it can use its public interface.

Reviewed By: passy

Differential Revision: D14000078

fbshipit-source-id: c0431a888b0ca041807631c81b99fb8b947274d6
This commit is contained in:
John Knox
2019-02-11 14:01:32 -08:00
committed by Facebook Github Bot
parent c48c1a728a
commit 5da8f35ee3
8 changed files with 142 additions and 47 deletions

View File

@@ -45,6 +45,13 @@ class FlipperConnectionManager {
The callbacks should be set before a connection is established. The callbacks should be set before a connection is established.
*/ */
virtual void setCallbacks(Callbacks* callbacks) = 0; virtual void setCallbacks(Callbacks* callbacks) = 0;
/**
Called by ws server when a message has been received.
*/
virtual void onMessageReceived(
const folly::dynamic& message,
std::unique_ptr<FlipperResponder> responder) = 0;
}; };
class FlipperConnectionManager::Callbacks { class FlipperConnectionManager::Callbacks {

View File

@@ -193,7 +193,7 @@ void FlipperConnectionManagerImpl::connectSecurely() {
std::move(address), std::move(address),
std::move(sslContext)), std::move(sslContext)),
std::move(parameters), std::move(parameters),
std::make_shared<FlipperRSocketResponder>(this), std::make_shared<FlipperRSocketResponder>(this, connectionEventBase_),
std::chrono::seconds(connectionKeepaliveSeconds), // keepaliveInterval std::chrono::seconds(connectionKeepaliveSeconds), // keepaliveInterval
nullptr, // stats nullptr, // stats
std::make_shared<ConnectionEvents>(this)) std::make_shared<ConnectionEvents>(this))
@@ -241,6 +241,12 @@ void FlipperConnectionManagerImpl::sendMessage(const folly::dynamic& message) {
}); });
} }
void FlipperConnectionManagerImpl::onMessageReceived(
const folly::dynamic& message,
std::unique_ptr<FlipperResponder> responder) {
callbacks_->onMessageReceived(message, std::move(responder));
}
bool FlipperConnectionManagerImpl::isCertificateExchangeNeeded() { bool FlipperConnectionManagerImpl::isCertificateExchangeNeeded() {
if (failedConnectionAttempts_ >= 2) { if (failedConnectionAttempts_ >= 2) {
return true; return true;

View File

@@ -25,7 +25,6 @@ rsocket::Payload toRSocketPayload(folly::dynamic data);
class FlipperConnectionManagerImpl : public FlipperConnectionManager { class FlipperConnectionManagerImpl : public FlipperConnectionManager {
friend ConnectionEvents; friend ConnectionEvents;
friend FlipperRSocketResponder;
public: public:
FlipperConnectionManagerImpl(FlipperInitConfig config, std::shared_ptr<FlipperState> state, std::shared_ptr<ConnectionContextStore> contextStore); FlipperConnectionManagerImpl(FlipperInitConfig config, std::shared_ptr<FlipperState> state, std::shared_ptr<ConnectionContextStore> contextStore);
@@ -42,6 +41,10 @@ class FlipperConnectionManagerImpl : public FlipperConnectionManager {
void sendMessage(const folly::dynamic& message) override; void sendMessage(const folly::dynamic& message) override;
void onMessageReceived(
const folly::dynamic& message,
std::unique_ptr<FlipperResponder> responder) override;
void reconnect(); void reconnect();
private: private:

View File

@@ -31,7 +31,7 @@ void FlipperRSocketResponder::handleFireAndForget(
std::make_unique<FireAndForgetBasedFlipperResponder>(websocket_, id); std::make_unique<FireAndForgetBasedFlipperResponder>(websocket_, id);
} }
websocket_->callbacks_->onMessageReceived( websocket_->onMessageReceived(
folly::parseJson(payload), std::move(responder)); folly::parseJson(payload), std::move(responder));
} }
@@ -44,7 +44,7 @@ FlipperRSocketResponder::handleRequestResponse(
auto dynamicSingle = yarpl::single::Single<folly::dynamic>::create( auto dynamicSingle = yarpl::single::Single<folly::dynamic>::create(
[payload = std::move(requestString), this](auto observer) { [payload = std::move(requestString), this](auto observer) {
auto responder = std::make_unique<FlipperResponderImpl>(observer); auto responder = std::make_unique<FlipperResponderImpl>(observer);
websocket_->callbacks_->onMessageReceived( websocket_->onMessageReceived(
folly::parseJson(payload), std::move(responder)); folly::parseJson(payload), std::move(responder));
}); });
@@ -53,19 +53,18 @@ FlipperRSocketResponder::handleRequestResponse(
observer->onSubscribe(yarpl::single::SingleSubscriptions::empty()); observer->onSubscribe(yarpl::single::SingleSubscriptions::empty());
dynamicSingle->subscribe( dynamicSingle->subscribe(
[observer, this](folly::dynamic d) { [observer, this](folly::dynamic d) {
websocket_->connectionEventBase_->runInEventBaseThread( eventBase_->runInEventBaseThread([observer, d]() {
[observer, d]() { try {
try { observer->onSuccess(toRSocketPayload(d));
observer->onSuccess(toRSocketPayload(d));
} catch (std::exception& e) { } catch (std::exception& e) {
log(e.what()); log(e.what());
observer->onError(e); observer->onError(e);
} }
}); });
}, },
[observer, this](folly::exception_wrapper e) { [observer, this](folly::exception_wrapper e) {
websocket_->connectionEventBase_->runInEventBaseThread( eventBase_->runInEventBaseThread(
[observer, e]() { observer->onError(e); }); [observer, e]() { observer->onError(e); });
}); });
}); });

View File

@@ -9,15 +9,18 @@
namespace facebook { namespace facebook {
namespace flipper { namespace flipper {
class FlipperConnectionManagerImpl; class FlipperConnectionManager;
class FlipperRSocketResponder : public rsocket::RSocketResponder { class FlipperRSocketResponder : public rsocket::RSocketResponder {
private: private:
FlipperConnectionManagerImpl* websocket_; FlipperConnectionManager* websocket_;
folly::EventBase* eventBase_;
public: public:
FlipperRSocketResponder(FlipperConnectionManagerImpl* websocket) FlipperRSocketResponder(
: websocket_(websocket){}; FlipperConnectionManager* websocket,
folly::EventBase* eventBase)
: websocket_(websocket), eventBase_(eventBase){};
void handleFireAndForget( void handleFireAndForget(
rsocket::Payload request, rsocket::Payload request,

View File

@@ -1,11 +1,9 @@
/* /**
* Copyright (c) 2018-present, Facebook, Inc. * Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the LICENSE
* file in the root directory of this source tree.
* *
* This source code is licensed under the MIT license found in the LICENSE
* file in the root directory of this source tree.
*/ */
#pragma once #pragma once
#include <Flipper/FlipperConnectionManager.h> #include <Flipper/FlipperConnectionManager.h>
@@ -40,6 +38,16 @@ class FlipperConnectionManagerMock : public FlipperConnectionManager {
messages.push_back(message); messages.push_back(message);
} }
void onMessageReceived(
const folly::dynamic& message,
std::unique_ptr<FlipperResponder> responder) override {
if (responder) {
respondersReceived++;
}
callbacks->onMessageReceived(message, std::move(responder));
messagesReceived.push_back(message);
}
void setCallbacks(Callbacks* aCallbacks) override { void setCallbacks(Callbacks* aCallbacks) override {
callbacks = aCallbacks; callbacks = aCallbacks;
} }
@@ -48,6 +56,8 @@ class FlipperConnectionManagerMock : public FlipperConnectionManager {
bool open = false; bool open = false;
Callbacks* callbacks; Callbacks* callbacks;
std::vector<folly::dynamic> messages; std::vector<folly::dynamic> messages;
std::vector<folly::dynamic> messagesReceived;
int respondersReceived = 0;
}; };
} // namespace test } // namespace test

View File

@@ -61,7 +61,7 @@ TEST_F(FlipperClientTest, testGetPlugins) {
client->addPlugin(std::make_shared<FlipperPluginMock>("Dog")); client->addPlugin(std::make_shared<FlipperPluginMock>("Dog"));
dynamic message = dynamic::object("id", 1)("method", "getPlugins"); dynamic message = dynamic::object("id", 1)("method", "getPlugins");
socket->callbacks->onMessageReceived(message, getResponder()); socket->onMessageReceived(message, getResponder());
dynamic expected = dynamic::object("plugins", dynamic::array("Cat", "Dog")); dynamic expected = dynamic::object("plugins", dynamic::array("Cat", "Dog"));
EXPECT_EQ(successes[0], expected); EXPECT_EQ(successes[0], expected);
@@ -92,8 +92,8 @@ TEST_F(FlipperClientTest, testRemovePlugin) {
client->removePlugin(plugin); client->removePlugin(plugin);
dynamic message = dynamic::object("id", 1)("method", "getPlugins"); dynamic message = dynamic::object("id", 1)("method", "getPlugins");
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(message, getResponder()); socket->onMessageReceived(message, getResponder());
dynamic expected = dynamic::object("plugins", dynamic::array()); dynamic expected = dynamic::object("plugins", dynamic::array());
EXPECT_EQ(successes[0], expected); EXPECT_EQ(successes[0], expected);
@@ -148,24 +148,24 @@ TEST_F(FlipperClientTest, testInitDeinit) {
{ {
dynamic messageInit = dynamic::object("method", "init")( dynamic messageInit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
EXPECT_TRUE(pluginConnected); EXPECT_TRUE(pluginConnected);
} }
{ {
dynamic messageDeinit = dynamic::object("method", "deinit")( dynamic messageDeinit = dynamic::object("method", "deinit")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageDeinit, getResponder()); socket->onMessageReceived(messageDeinit, getResponder());
EXPECT_FALSE(pluginConnected); EXPECT_FALSE(pluginConnected);
} }
{ {
dynamic messageReinit = dynamic::object("method", "init")( dynamic messageReinit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageReinit, getResponder()); socket->onMessageReceived(messageReinit, getResponder());
EXPECT_TRUE(pluginConnected); EXPECT_TRUE(pluginConnected);
} }
@@ -198,14 +198,14 @@ TEST_F(FlipperClientTest, testUnhandleableMethod) {
{ {
dynamic messageInit = dynamic::object("method", "init")( dynamic messageInit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
} }
{ {
dynamic messageExecute = dynamic::object("id", 1)("method", "unexpected"); dynamic messageExecute = dynamic::object("id", 1)("method", "unexpected");
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageExecute, getResponder()); socket->onMessageReceived(messageExecute, getResponder());
} }
dynamic expected = dynamic expected =
@@ -231,8 +231,8 @@ TEST_F(FlipperClientTest, testExecute) {
{ {
dynamic messageInit = dynamic::object("method", "init")( dynamic messageInit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
} }
{ {
@@ -266,8 +266,8 @@ TEST_F(FlipperClientTest, testExecuteWithParams) {
{ {
dynamic messageInit = dynamic::object("method", "init")( dynamic messageInit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Test")); "params", dynamic::object("plugin", "Test"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
} }
{ {
@@ -275,8 +275,8 @@ TEST_F(FlipperClientTest, testExecuteWithParams) {
"params", "params",
dynamic::object("api", "Test")("method", "animal_sounds")( dynamic::object("api", "Test")("method", "animal_sounds")(
"params", dynamic::object("first", "dog")("second", "cat"))); "params", dynamic::object("first", "dog")("second", "cat")));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageExecute, getResponder()); socket->onMessageReceived(messageExecute, getResponder());
} }
dynamic expected = dynamic::object("dog", "woof")("cat", "meow"); dynamic expected = dynamic::object("dog", "woof")("cat", "meow");
@@ -289,8 +289,8 @@ TEST_F(FlipperClientTest, testExceptionUnknownPlugin) {
dynamic messageInit = dynamic::object("method", "init")( dynamic messageInit = dynamic::object("method", "init")(
"params", dynamic::object("plugin", "Unknown")); "params", dynamic::object("plugin", "Unknown"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
auto failure = failures[0]; auto failure = failures[0];
EXPECT_EQ(failure["message"], "Plugin Unknown not found for method init"); EXPECT_EQ(failure["message"], "Plugin Unknown not found for method init");
@@ -302,8 +302,8 @@ TEST_F(FlipperClientTest, testExceptionUnknownApi) {
dynamic messageInit = dynamic::object("method", "execute")( dynamic messageInit = dynamic::object("method", "execute")(
"params", dynamic::object("api", "Unknown")); "params", dynamic::object("api", "Unknown"));
auto responder = std::make_shared<FlipperResponderMock>(); auto responder = std::make_unique<FlipperResponderMock>();
socket->callbacks->onMessageReceived(messageInit, getResponder()); socket->onMessageReceived(messageInit, getResponder());
auto failure = failures[0]; auto failure = failures[0];
EXPECT_EQ( EXPECT_EQ(
failure["message"], "Connection Unknown not found for method execute"); failure["message"], "Connection Unknown not found for method execute");

View File

@@ -0,0 +1,67 @@
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the LICENSE
* file in the root directory of this source tree.
*/
#include <Flipper/FlipperRSocketResponder.h>
#include <Flipper/Log.h>
#include <FlipperTestLib/FlipperConnectionManagerMock.h>
#include <folly/json.h>
#include <gtest/gtest.h>
namespace facebook {
namespace flipper {
namespace test {
using folly::dynamic;
class Callbacks
: public facebook::flipper::FlipperConnectionManager::Callbacks {
public:
void onConnected() {}
void onDisconnected() {}
void onMessageReceived(
const folly::dynamic& message,
std::unique_ptr<FlipperResponder> responder) {
message_ = message;
responder_ = std::move(responder);
}
folly::dynamic message_;
std::unique_ptr<FlipperResponder> responder_;
};
TEST(FlipperRSocketResponderTests, testFireAndForgetWithoutIdParam) {
auto socket = facebook::flipper::test::FlipperConnectionManagerMock();
auto callbacks = new Callbacks();
socket.setCallbacks(callbacks);
folly::EventBase* eb = new folly::EventBase();
auto responder = facebook::flipper::FlipperRSocketResponder(&socket, eb);
dynamic d = dynamic::object("my", "message");
auto json = folly::toJson(d);
responder.handleFireAndForget(rsocket::Payload(json), rsocket::StreamId(1));
EXPECT_EQ(socket.messagesReceived.size(), 1);
EXPECT_EQ(socket.messagesReceived[0]["my"], "message");
EXPECT_EQ(socket.respondersReceived, 0);
}
TEST(FlipperRSocketResponderTests, testFireAndForgetWithIdParam) {
auto socket = facebook::flipper::test::FlipperConnectionManagerMock();
auto callbacks = new Callbacks();
socket.setCallbacks(callbacks);
folly::EventBase* eb = new folly::EventBase();
auto responder = facebook::flipper::FlipperRSocketResponder(&socket, eb);
dynamic d = dynamic::object("my", "message")("id", 7);
auto json = folly::toJson(d);
responder.handleFireAndForget(rsocket::Payload(json), rsocket::StreamId(1));
EXPECT_EQ(socket.messagesReceived.size(), 1);
EXPECT_EQ(socket.messagesReceived[0]["my"], "message");
EXPECT_EQ(socket.messagesReceived[0]["id"], 7);
EXPECT_EQ(socket.respondersReceived, 1);
}
} // namespace test
} // namespace flipper
} // namespace facebook