blob: 7a66e3d69a6dfedaff4d0600a549aef14ffc8d33 [file] [log] [blame]
// Copyright 2018 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "osp/impl/quic/quic_client.h"
#include <memory>
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "osp/impl/quic/quic_service_common.h"
#include "osp/impl/quic/testing/fake_quic_connection_factory.h"
#include "osp/impl/quic/testing/quic_test_support.h"
#include "osp/public/network_metrics.h"
#include "osp/public/network_service_manager.h"
#include "osp/public/testing/message_demuxer_test_support.h"
#include "platform/base/error.h"
#include "platform/test/fake_clock.h"
#include "platform/test/fake_task_runner.h"
#include "util/osp_logging.h"
namespace openscreen {
namespace osp {
namespace {
using ::testing::_;
using ::testing::Invoke;
class MockConnectionObserver final : public ProtocolConnection::Observer {
public:
~MockConnectionObserver() override = default;
MOCK_METHOD1(OnConnectionClosed, void(const ProtocolConnection& connection));
};
class ConnectionCallback final
: public ProtocolConnectionClient::ConnectionRequestCallback {
public:
explicit ConnectionCallback(std::unique_ptr<ProtocolConnection>* connection)
: connection_(connection) {}
~ConnectionCallback() override = default;
void OnConnectionOpened(
uint64_t request_id,
std::unique_ptr<ProtocolConnection> connection) override {
OSP_DCHECK(!failed_ && !*connection_);
*connection_ = std::move(connection);
}
void OnConnectionFailed(uint64_t request_id) override {
OSP_DCHECK(!failed_ && !*connection_);
failed_ = true;
}
private:
bool failed_ = false;
std::unique_ptr<ProtocolConnection>* const connection_;
};
class QuicClientTest : public ::testing::Test {
public:
QuicClientTest() {
fake_clock_ = std::make_unique<FakeClock>(
Clock::time_point(std::chrono::milliseconds(1298424)));
task_runner_ = std::make_unique<FakeTaskRunner>(fake_clock_.get());
quic_bridge_ =
std::make_unique<FakeQuicBridge>(task_runner_.get(), FakeClock::now);
}
protected:
void SetUp() override {
client_ = quic_bridge_->quic_client.get();
NetworkServiceManager::Create(nullptr, nullptr,
std::move(quic_bridge_->quic_client),
std::move(quic_bridge_->quic_server));
}
void TearDown() override { NetworkServiceManager::Dispose(); }
void SendTestMessage(ProtocolConnection* connection) {
MockMessageCallback mock_message_callback;
MessageDemuxer::MessageWatch message_watch =
quic_bridge_->receiver_demuxer->WatchMessageType(
0, msgs::Type::kPresentationConnectionMessage,
&mock_message_callback);
msgs::CborEncodeBuffer buffer;
msgs::PresentationConnectionMessage message;
message.connection_id = 7;
message.message.which = decltype(message.message.which)::kString;
new (&message.message.str) std::string("message from client");
ASSERT_TRUE(msgs::EncodePresentationConnectionMessage(message, &buffer));
connection->Write(buffer.data(), buffer.size());
connection->CloseWriteEnd();
ssize_t decode_result = 0;
msgs::PresentationConnectionMessage received_message;
EXPECT_CALL(
mock_message_callback,
OnStreamMessage(0, connection->id(),
msgs::Type::kPresentationConnectionMessage, _, _, _))
.WillOnce(Invoke([&decode_result, &received_message](
uint64_t endpoint_id, uint64_t connection_id,
msgs::Type message_type, const uint8_t* buffer,
size_t buffer_size, Clock::time_point now) {
decode_result = msgs::DecodePresentationConnectionMessage(
buffer, buffer_size, &received_message);
if (decode_result < 0)
return ErrorOr<size_t>(Error::Code::kCborParsing);
return ErrorOr<size_t>(decode_result);
}));
quic_bridge_->RunTasksUntilIdle();
ASSERT_GT(decode_result, 0);
EXPECT_EQ(decode_result, static_cast<ssize_t>(buffer.size() - 1));
EXPECT_EQ(received_message.connection_id, message.connection_id);
ASSERT_EQ(received_message.message.which,
decltype(received_message.message.which)::kString);
EXPECT_EQ(received_message.message.str, message.message.str);
}
std::unique_ptr<FakeClock> fake_clock_;
std::unique_ptr<FakeTaskRunner> task_runner_;
std::unique_ptr<FakeQuicBridge> quic_bridge_;
QuicClient* client_;
};
} // namespace
TEST_F(QuicClientTest, Connect) {
client_->Start();
std::unique_ptr<ProtocolConnection> connection;
ConnectionCallback connection_callback(&connection);
ProtocolConnectionClient::ConnectRequest request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
ASSERT_TRUE(request);
quic_bridge_->RunTasksUntilIdle();
ASSERT_TRUE(connection);
SendTestMessage(connection.get());
client_->Stop();
}
TEST_F(QuicClientTest, DoubleConnect) {
client_->Start();
std::unique_ptr<ProtocolConnection> connection1;
ConnectionCallback connection_callback1(&connection1);
ProtocolConnectionClient::ConnectRequest request1 =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback1);
ASSERT_TRUE(request1);
ASSERT_FALSE(connection1);
std::unique_ptr<ProtocolConnection> connection2;
ConnectionCallback connection_callback2(&connection2);
ProtocolConnectionClient::ConnectRequest request2 =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback2);
ASSERT_TRUE(request2);
quic_bridge_->RunTasksUntilIdle();
ASSERT_TRUE(connection1);
ASSERT_TRUE(connection2);
SendTestMessage(connection1.get());
client_->Stop();
}
TEST_F(QuicClientTest, OpenImmediate) {
client_->Start();
std::unique_ptr<ProtocolConnection> connection1;
std::unique_ptr<ProtocolConnection> connection2;
connection2 = client_->CreateProtocolConnection(1);
EXPECT_FALSE(connection2);
ConnectionCallback connection_callback(&connection1);
ProtocolConnectionClient::ConnectRequest request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
ASSERT_TRUE(request);
connection2 = client_->CreateProtocolConnection(1);
EXPECT_FALSE(connection2);
quic_bridge_->RunTasksUntilIdle();
ASSERT_TRUE(connection1);
connection2 = client_->CreateProtocolConnection(connection1->endpoint_id());
ASSERT_TRUE(connection2);
SendTestMessage(connection2.get());
client_->Stop();
}
TEST_F(QuicClientTest, States) {
client_->Stop();
std::unique_ptr<ProtocolConnection> connection1;
ConnectionCallback connection_callback(&connection1);
ProtocolConnectionClient::ConnectRequest request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
EXPECT_FALSE(request);
std::unique_ptr<ProtocolConnection> connection2 =
client_->CreateProtocolConnection(1);
EXPECT_FALSE(connection2);
EXPECT_CALL(quic_bridge_->mock_client_observer, OnRunning());
EXPECT_TRUE(client_->Start());
EXPECT_FALSE(client_->Start());
request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
ASSERT_TRUE(request);
quic_bridge_->RunTasksUntilIdle();
ASSERT_TRUE(connection1);
MockConnectionObserver mock_connection_observer1;
connection1->SetObserver(&mock_connection_observer1);
connection2 = client_->CreateProtocolConnection(connection1->endpoint_id());
ASSERT_TRUE(connection2);
MockConnectionObserver mock_connection_observer2;
connection2->SetObserver(&mock_connection_observer2);
EXPECT_CALL(mock_connection_observer1, OnConnectionClosed(_));
EXPECT_CALL(mock_connection_observer2, OnConnectionClosed(_));
EXPECT_CALL(quic_bridge_->mock_client_observer, OnStopped());
EXPECT_TRUE(client_->Stop());
EXPECT_FALSE(client_->Stop());
request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
EXPECT_FALSE(request);
connection2 = client_->CreateProtocolConnection(1);
EXPECT_FALSE(connection2);
}
TEST_F(QuicClientTest, RequestIds) {
client_->Start();
EXPECT_CALL(quic_bridge_->mock_server_observer, OnIncomingConnectionMock(_))
.WillOnce(Invoke([](std::unique_ptr<ProtocolConnection>& connection) {
connection->CloseWriteEnd();
}));
std::unique_ptr<ProtocolConnection> connection;
ConnectionCallback connection_callback(&connection);
ProtocolConnectionClient::ConnectRequest request =
client_->Connect(quic_bridge_->kReceiverEndpoint, &connection_callback);
ASSERT_TRUE(request);
quic_bridge_->RunTasksUntilIdle();
ASSERT_TRUE(connection);
const uint64_t endpoint_id = connection->endpoint_id();
EXPECT_EQ(0u, client_->endpoint_request_ids()->GetNextRequestId(endpoint_id));
EXPECT_EQ(2u, client_->endpoint_request_ids()->GetNextRequestId(endpoint_id));
connection->CloseWriteEnd();
connection.reset();
quic_bridge_->RunTasksUntilIdle();
EXPECT_EQ(4u, client_->endpoint_request_ids()->GetNextRequestId(endpoint_id));
client_->Stop();
EXPECT_EQ(0u, client_->endpoint_request_ids()->GetNextRequestId(endpoint_id));
}
} // namespace osp
} // namespace openscreen