blob: f033eeeab431b1c11691fbc3da5e21a34fc877ce [file] [log] [blame]
// Copyright (c) 2012 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 "remoting/host/chromoting_host.h"
#include <utility>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/memory/scoped_ptr.h"
#include "remoting/base/auto_thread_task_runner.h"
#include "remoting/host/audio_capturer.h"
#include "remoting/host/chromoting_host_context.h"
#include "remoting/host/fake_desktop_environment.h"
#include "remoting/host/fake_mouse_cursor_monitor.h"
#include "remoting/host/host_mock_objects.h"
#include "remoting/proto/video.pb.h"
#include "remoting/protocol/errors.h"
#include "remoting/protocol/fake_connection_to_client.h"
#include "remoting/protocol/fake_desktop_capturer.h"
#include "remoting/protocol/protocol_mock_objects.h"
#include "remoting/protocol/session_config.h"
#include "remoting/protocol/transport_context.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gmock_mutant.h"
#include "testing/gtest/include/gtest/gtest.h"
using ::remoting::protocol::MockClientStub;
using ::remoting::protocol::MockConnectionToClientEventHandler;
using ::remoting::protocol::MockHostStub;
using ::remoting::protocol::MockSession;
using ::remoting::protocol::MockVideoStub;
using ::remoting::protocol::Session;
using ::remoting::protocol::SessionConfig;
using testing::_;
using testing::AnyNumber;
using testing::AtLeast;
using testing::AtMost;
using testing::DeleteArg;
using testing::DoAll;
using testing::Expectation;
using testing::InSequence;
using testing::Invoke;
using testing::InvokeArgument;
using testing::InvokeWithoutArgs;
using testing::Return;
using testing::ReturnRef;
using testing::SaveArg;
using testing::Sequence;
namespace remoting {
class ChromotingHostTest : public testing::Test {
public:
ChromotingHostTest() {}
void SetUp() override {
task_runner_ = new AutoThreadTaskRunner(message_loop_.task_runner(),
base::Bind(&base::DoNothing));
desktop_environment_factory_.reset(new FakeDesktopEnvironmentFactory());
session_manager_ = new protocol::MockSessionManager();
host_.reset(new ChromotingHost(
desktop_environment_factory_.get(), make_scoped_ptr(session_manager_),
protocol::TransportContext::ForTests(protocol::TransportRole::SERVER),
task_runner_, // Audio
task_runner_, // Input
task_runner_, // Video capture
task_runner_, // Video encode
task_runner_, // Network
task_runner_)); // UI
host_->AddStatusObserver(&host_status_observer_);
xmpp_login_ = "host@domain";
session1_ = new MockSession();
session2_ = new MockSession();
session_unowned1_.reset(new MockSession());
session_unowned2_.reset(new MockSession());
session_config1_ = SessionConfig::ForTest();
session_jid1_ = "user@domain/rest-of-jid";
session_config2_ = SessionConfig::ForTest();
session_jid2_ = "user2@domain/rest-of-jid";
session_unowned_jid1_ = "user3@doman/rest-of-jid";
session_unowned_jid2_ = "user4@doman/rest-of-jid";
EXPECT_CALL(*session1_, jid())
.WillRepeatedly(ReturnRef(session_jid1_));
EXPECT_CALL(*session2_, jid())
.WillRepeatedly(ReturnRef(session_jid2_));
EXPECT_CALL(*session_unowned1_, jid())
.WillRepeatedly(ReturnRef(session_unowned_jid1_));
EXPECT_CALL(*session_unowned2_, jid())
.WillRepeatedly(ReturnRef(session_unowned_jid2_));
EXPECT_CALL(*session_unowned1_, SetEventHandler(_))
.Times(AnyNumber())
.WillRepeatedly(SaveArg<0>(&session_unowned1_event_handler_));
EXPECT_CALL(*session_unowned2_, SetEventHandler(_))
.Times(AnyNumber())
.WillRepeatedly(SaveArg<0>(&session_unowned2_event_handler_));
EXPECT_CALL(*session1_, config())
.WillRepeatedly(ReturnRef(*session_config1_));
EXPECT_CALL(*session2_, config())
.WillRepeatedly(ReturnRef(*session_config2_));
EXPECT_CALL(*session_unowned1_, config())
.WillRepeatedly(ReturnRef(*session_config1_));
EXPECT_CALL(*session_unowned2_, config())
.WillRepeatedly(ReturnRef(*session_config2_));
owned_connection1_.reset(
new protocol::FakeConnectionToClient(make_scoped_ptr(session1_)));
owned_connection1_->set_host_stub(&host_stub1_);
connection1_ = owned_connection1_.get();
connection1_->set_client_stub(&client_stub1_);
owned_connection2_.reset(
new protocol::FakeConnectionToClient(make_scoped_ptr(session2_)));
owned_connection2_->set_host_stub(&host_stub2_);
connection2_ = owned_connection2_.get();
connection2_->set_client_stub(&client_stub2_);
}
// Helper method to pretend a client is connected to ChromotingHost.
void SimulateClientConnection(int connection_index, bool authenticate,
bool reject) {
scoped_ptr<protocol::ConnectionToClient> connection = std::move(
(connection_index == 0) ? owned_connection1_ : owned_connection2_);
protocol::ConnectionToClient* connection_ptr = connection.get();
scoped_ptr<ClientSession> client(new ClientSession(
host_.get(),
task_runner_, // Audio
task_runner_, // Input
task_runner_, // Video capture
task_runner_, // Video encode
task_runner_, // Network
task_runner_, // UI
std::move(connection), desktop_environment_factory_.get(),
base::TimeDelta(), nullptr, std::vector<HostExtension*>()));
ClientSession* client_ptr = client.get();
connection_ptr->set_host_stub(client.get());
get_client(connection_index) = client_ptr;
// |host| is responsible for deleting |client| from now on.
host_->clients_.push_back(client.release());
if (authenticate) {
client_ptr->OnConnectionAuthenticated(connection_ptr);
if (!reject)
client_ptr->OnConnectionChannelsConnected(connection_ptr);
} else {
client_ptr->OnConnectionClosed(connection_ptr,
protocol::AUTHENTICATION_FAILED);
}
}
void TearDown() override {
if (host_)
ShutdownHost();
task_runner_ = nullptr;
message_loop_.RunUntilIdle();
}
void DisconnectAllClients() {
host_->DisconnectAllClients();
}
void NotifyConnectionClosed1() {
if (session_unowned1_event_handler_) {
session_unowned1_event_handler_->OnSessionStateChange(Session::CLOSED);
}
}
void NotifyConnectionClosed2() {
if (session_unowned2_event_handler_) {
session_unowned2_event_handler_->OnSessionStateChange(Session::CLOSED);
}
}
void ShutdownHost() {
EXPECT_CALL(host_status_observer_, OnShutdown());
host_.reset();
desktop_environment_factory_.reset();
}
// Starts the host.
void StartHost() {
EXPECT_CALL(host_status_observer_, OnStart(xmpp_login_));
EXPECT_CALL(*session_manager_, AcceptIncoming(_));
host_->Start(xmpp_login_);
}
// Expect a client to connect.
// Return an expectation that a session has started.
Expectation ExpectClientConnected(int connection_index) {
const std::string& session_jid = get_session_jid(connection_index);
Expectation client_authenticated =
EXPECT_CALL(host_status_observer_, OnClientAuthenticated(session_jid));
return EXPECT_CALL(host_status_observer_, OnClientConnected(session_jid))
.After(client_authenticated);
}
// Expect that a client is disconnected. The given action will be done after
// the status observer is notified that the session has finished.
Expectation ExpectClientDisconnected(int connection_index) {
return EXPECT_CALL(host_status_observer_,
OnClientDisconnected(get_session_jid(connection_index)))
.RetiresOnSaturation();
}
protected:
base::MessageLoop message_loop_;
scoped_refptr<AutoThreadTaskRunner> task_runner_;
MockConnectionToClientEventHandler handler_;
scoped_ptr<FakeDesktopEnvironmentFactory> desktop_environment_factory_;
MockHostStatusObserver host_status_observer_;
scoped_ptr<ChromotingHost> host_;
protocol::MockSessionManager* session_manager_;
std::string xmpp_login_;
protocol::FakeConnectionToClient* connection1_;
scoped_ptr<protocol::FakeConnectionToClient> owned_connection1_;
ClientSession* client1_;
std::string session_jid1_;
MockSession* session1_; // Owned by |connection_|.
scoped_ptr<SessionConfig> session_config1_;
MockClientStub client_stub1_;
MockHostStub host_stub1_;
protocol::FakeConnectionToClient* connection2_;
scoped_ptr<protocol::FakeConnectionToClient> owned_connection2_;
ClientSession* client2_;
std::string session_jid2_;
MockSession* session2_; // Owned by |connection2_|.
scoped_ptr<SessionConfig> session_config2_;
MockClientStub client_stub2_;
MockHostStub host_stub2_;
scoped_ptr<MockSession> session_unowned1_; // Not owned by a connection.
std::string session_unowned_jid1_;
scoped_ptr<MockSession> session_unowned2_; // Not owned by a connection.
std::string session_unowned_jid2_;
protocol::Session::EventHandler* session_unowned1_event_handler_;
protocol::Session::EventHandler* session_unowned2_event_handler_;
protocol::FakeConnectionToClient*& get_connection(int connection_index) {
return (connection_index == 0) ? connection1_ : connection2_;
}
// Returns the cached client pointers client1_ or client2_.
ClientSession*& get_client(int connection_index) {
return (connection_index == 0) ? client1_ : client2_;
}
// Returns the list of clients of the host_.
std::list<ClientSession*>& get_clients_from_host() {
return host_->clients_;
}
const std::string& get_session_jid(int connection_index) {
return (connection_index == 0) ? session_jid1_ : session_jid2_;
}
};
TEST_F(ChromotingHostTest, StartAndShutdown) {
StartHost();
}
TEST_F(ChromotingHostTest, Connect) {
StartHost();
// Shut down the host when the first video packet is received.
ExpectClientConnected(0);
SimulateClientConnection(0, true, false);
}
TEST_F(ChromotingHostTest, AuthenticationFailed) {
StartHost();
EXPECT_CALL(host_status_observer_, OnAccessDenied(session_jid1_));
SimulateClientConnection(0, false, false);
}
TEST_F(ChromotingHostTest, Reconnect) {
StartHost();
// Connect first client.
ExpectClientConnected(0);
SimulateClientConnection(0, true, false);
// Disconnect first client.
ExpectClientDisconnected(0);
client1_->OnConnectionClosed(connection1_, protocol::OK);
// Connect second client.
ExpectClientConnected(1);
SimulateClientConnection(1, true, false);
// Disconnect second client.
ExpectClientDisconnected(1);
client2_->OnConnectionClosed(connection2_, protocol::OK);
}
TEST_F(ChromotingHostTest, ConnectWhenAnotherClientIsConnected) {
StartHost();
// Connect first client.
ExpectClientConnected(0);
SimulateClientConnection(0, true, false);
// Connect second client. First client should be disconnected automatically.
{
InSequence s;
ExpectClientDisconnected(0);
ExpectClientConnected(1);
}
SimulateClientConnection(1, true, false);
// Disconnect second client.
ExpectClientDisconnected(1);
client2_->OnConnectionClosed(connection2_, protocol::OK);
}
TEST_F(ChromotingHostTest, IncomingSessionAccepted) {
StartHost();
MockSession* session = session_unowned1_.get();
protocol::SessionManager::IncomingSessionResponse response =
protocol::SessionManager::DECLINE;
host_->OnIncomingSession(session_unowned1_.release(), &response);
EXPECT_EQ(protocol::SessionManager::ACCEPT, response);
EXPECT_CALL(*session, Close(_)).WillOnce(InvokeWithoutArgs(
this, &ChromotingHostTest::NotifyConnectionClosed1));
ShutdownHost();
}
TEST_F(ChromotingHostTest, LoginBackOffUponConnection) {
StartHost();
protocol::SessionManager::IncomingSessionResponse response =
protocol::SessionManager::DECLINE;
EXPECT_CALL(*session_unowned1_, Close(_)).WillOnce(
InvokeWithoutArgs(this, &ChromotingHostTest::NotifyConnectionClosed1));
host_->OnIncomingSession(session_unowned1_.release(), &response);
EXPECT_EQ(protocol::SessionManager::ACCEPT, response);
host_->OnSessionAuthenticating(get_clients_from_host().front());
host_->OnIncomingSession(session_unowned2_.get(), &response);
EXPECT_EQ(protocol::SessionManager::OVERLOAD, response);
}
TEST_F(ChromotingHostTest, LoginBackOffUponAuthenticating) {
StartHost();
EXPECT_CALL(*session_unowned1_, Close(_)).WillOnce(
InvokeWithoutArgs(this, &ChromotingHostTest::NotifyConnectionClosed1));
EXPECT_CALL(*session_unowned2_, Close(_)).WillOnce(
InvokeWithoutArgs(this, &ChromotingHostTest::NotifyConnectionClosed2));
protocol::SessionManager::IncomingSessionResponse response =
protocol::SessionManager::DECLINE;
host_->OnIncomingSession(session_unowned1_.release(), &response);
EXPECT_EQ(protocol::SessionManager::ACCEPT, response);
host_->OnIncomingSession(session_unowned2_.release(), &response);
EXPECT_EQ(protocol::SessionManager::ACCEPT, response);
// This will set the backoff.
host_->OnSessionAuthenticating(get_clients_from_host().front());
// This should disconnect client2.
host_->OnSessionAuthenticating(get_clients_from_host().back());
// Verify that the host only has 1 client at this point.
EXPECT_EQ(get_clients_from_host().size(), 1U);
}
TEST_F(ChromotingHostTest, OnSessionRouteChange) {
StartHost();
ExpectClientConnected(0);
SimulateClientConnection(0, true, false);
std::string channel_name("ChannelName");
protocol::TransportRoute route;
EXPECT_CALL(host_status_observer_,
OnClientRouteChange(session_jid1_, channel_name, _));
host_->OnSessionRouteChange(get_client(0), channel_name, route);
}
TEST_F(ChromotingHostTest, DisconnectAllClients) {
StartHost();
ExpectClientConnected(0);
SimulateClientConnection(0, true, false);
ExpectClientDisconnected(0);
DisconnectAllClients();
testing::Mock::VerifyAndClearExpectations(&host_status_observer_);
}
} // namespace remoting