// Copyright 2014 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 "components/proximity_auth/connection.h"
#include "components/proximity_auth/connection_observer.h"
#include "components/proximity_auth/remote_device.h"
#include "components/proximity_auth/wire_message.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
using testing::_;
using testing::DoAll;
using testing::NiceMock;
using testing::Return;
using testing::SetArgPointee;
using testing::StrictMock;
namespace proximity_auth {
namespace {
class MockConnection : public Connection {
public:
MockConnection() : Connection(RemoteDevice()) {}
~MockConnection() {}
MOCK_METHOD1(SetPaused, void(bool paused));
MOCK_METHOD0(Connect, void());
MOCK_METHOD0(Disconnect, void());
MOCK_METHOD0(CancelConnectionAttempt, void());
MOCK_METHOD1(SendMessageImplProxy, void(WireMessage* message));
MOCK_METHOD1(DeserializeWireMessageProxy,
WireMessage*(bool* is_incomplete_message));
// Gmock only supports copyable types, so create simple wrapper methods for
// ease of mocking.
virtual void SendMessageImpl(scoped_ptr<WireMessage> message) OVERRIDE {
SendMessageImplProxy(message.get());
}
virtual scoped_ptr<WireMessage> DeserializeWireMessage(
bool* is_incomplete_message) OVERRIDE {
return make_scoped_ptr(DeserializeWireMessageProxy(is_incomplete_message));
}
using Connection::status;
using Connection::SetStatus;
using Connection::OnDidSendMessage;
using Connection::OnBytesReceived;
private:
DISALLOW_COPY_AND_ASSIGN(MockConnection);
};
class MockConnectionObserver : public ConnectionObserver {
public:
MockConnectionObserver() {}
virtual ~MockConnectionObserver() {}
MOCK_METHOD3(OnConnectionStatusChanged,
void(const Connection& connection,
Connection::Status old_status,
Connection::Status new_status));
MOCK_METHOD2(OnMessageReceived,
void(const Connection& connection, const WireMessage& message));
MOCK_METHOD3(OnSendCompleted,
void(const Connection& connection,
const WireMessage& message,
bool success));
private:
DISALLOW_COPY_AND_ASSIGN(MockConnectionObserver);
};
// Unlike WireMessage, offers a public constructor.
class TestWireMessage : public WireMessage {
public:
TestWireMessage() : WireMessage(std::string(), std::string()) {}
virtual ~TestWireMessage() {}
private:
DISALLOW_COPY_AND_ASSIGN(TestWireMessage);
};
} // namespace
TEST(ProximityAuthConnectionTest, IsConnected) {
StrictMock<MockConnection> connection;
EXPECT_FALSE(connection.IsConnected());
connection.SetStatus(Connection::CONNECTED);
EXPECT_TRUE(connection.IsConnected());
connection.SetStatus(Connection::DISCONNECTED);
EXPECT_FALSE(connection.IsConnected());
connection.SetStatus(Connection::IN_PROGRESS);
EXPECT_FALSE(connection.IsConnected());
}
TEST(ProximityAuthConnectionTest, SendMessage_FailsWhenNotConnected) {
StrictMock<MockConnection> connection;
connection.SetStatus(Connection::IN_PROGRESS);
EXPECT_CALL(connection, SendMessageImplProxy(_)).Times(0);
connection.SendMessage(scoped_ptr<WireMessage>());
}
TEST(ProximityAuthConnectionTest,
SendMessage_FailsWhenAnotherMessageSendIsInProgress) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
connection.SendMessage(scoped_ptr<WireMessage>());
EXPECT_CALL(connection, SendMessageImplProxy(_)).Times(0);
connection.SendMessage(scoped_ptr<WireMessage>());
}
TEST(ProximityAuthConnectionTest, SendMessage_SucceedsWhenConnected) {
StrictMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
EXPECT_CALL(connection, SendMessageImplProxy(_));
connection.SendMessage(scoped_ptr<WireMessage>());
}
TEST(ProximityAuthConnectionTest,
SendMessage_SucceedsAfterPreviousMessageSendCompletes) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
connection.SendMessage(scoped_ptr<WireMessage>());
connection.OnDidSendMessage(TestWireMessage(), true /* success */);
EXPECT_CALL(connection, SendMessageImplProxy(_));
connection.SendMessage(scoped_ptr<WireMessage>());
}
TEST(ProximityAuthConnectionTest, SetStatus_NotifiesObserversOfStatusChange) {
StrictMock<MockConnection> connection;
EXPECT_EQ(Connection::DISCONNECTED, connection.status());
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
EXPECT_CALL(
observer,
OnConnectionStatusChanged(
Ref(connection), Connection::DISCONNECTED, Connection::CONNECTED));
connection.SetStatus(Connection::CONNECTED);
}
TEST(ProximityAuthConnectionTest,
SetStatus_DoesntNotifyObserversIfStatusUnchanged) {
StrictMock<MockConnection> connection;
EXPECT_EQ(Connection::DISCONNECTED, connection.status());
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
EXPECT_CALL(observer, OnConnectionStatusChanged(_, _, _)).Times(0);
connection.SetStatus(Connection::DISCONNECTED);
}
TEST(ProximityAuthConnectionTest,
OnDidSendMessage_NotifiesObserversIfMessageSendInProgress) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
connection.SendMessage(scoped_ptr<WireMessage>());
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
EXPECT_CALL(observer, OnSendCompleted(Ref(connection), _, true));
connection.OnDidSendMessage(TestWireMessage(), true /* success */);
}
TEST(ProximityAuthConnectionTest,
OnDidSendMessage_DoesntNotifyObserversIfNoMessageSendInProgress) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
EXPECT_CALL(observer, OnSendCompleted(_, _, _)).Times(0);
connection.OnDidSendMessage(TestWireMessage(), true /* success */);
}
TEST(ProximityAuthConnectionTest,
OnBytesReceived_NotifiesObserversOnValidMessage) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
ON_CALL(connection, DeserializeWireMessageProxy(_))
.WillByDefault(
DoAll(SetArgPointee<0>(false), Return(new TestWireMessage)));
EXPECT_CALL(observer, OnMessageReceived(Ref(connection), _));
connection.OnBytesReceived(std::string());
}
TEST(ProximityAuthConnectionTest,
OnBytesReceived_DoesntNotifyObserversIfNotConnected) {
StrictMock<MockConnection> connection;
connection.SetStatus(Connection::IN_PROGRESS);
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
EXPECT_CALL(observer, OnMessageReceived(_, _)).Times(0);
connection.OnBytesReceived(std::string());
}
TEST(ProximityAuthConnectionTest,
OnBytesReceived_DoesntNotifyObserversIfMessageIsIncomplete) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
ON_CALL(connection, DeserializeWireMessageProxy(_))
.WillByDefault(DoAll(SetArgPointee<0>(true),
Return(static_cast<WireMessage*>(NULL))));
EXPECT_CALL(observer, OnMessageReceived(_, _)).Times(0);
connection.OnBytesReceived(std::string());
}
TEST(ProximityAuthConnectionTest,
OnBytesReceived_DoesntNotifyObserversIfMessageIsInvalid) {
NiceMock<MockConnection> connection;
connection.SetStatus(Connection::CONNECTED);
StrictMock<MockConnectionObserver> observer;
connection.AddObserver(&observer);
ON_CALL(connection, DeserializeWireMessageProxy(_))
.WillByDefault(DoAll(SetArgPointee<0>(false),
Return(static_cast<WireMessage*>(NULL))));
EXPECT_CALL(observer, OnMessageReceived(_, _)).Times(0);
connection.OnBytesReceived(std::string());
}
} // namespace proximity_auth