Chromium Code Reviews| Index: components/proximity_auth/ble/bluetooth_low_energy_weave_client_connection_unittest.cc |
| diff --git a/components/proximity_auth/ble/bluetooth_low_energy_weave_client_connection_unittest.cc b/components/proximity_auth/ble/bluetooth_low_energy_weave_client_connection_unittest.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..a00f51e0779394a2420255fe109c55023e118e8e |
| --- /dev/null |
| +++ b/components/proximity_auth/ble/bluetooth_low_energy_weave_client_connection_unittest.cc |
| @@ -0,0 +1,960 @@ |
| +// Copyright 2016 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/ble/bluetooth_low_energy_weave_client_connection.h" |
| + |
| +#include <stdint.h> |
| + |
| +#include <memory> |
| +#include <utility> |
| + |
| +#include "base/bind.h" |
| +#include "base/memory/ptr_util.h" |
| +#include "base/message_loop/message_loop.h" |
| +#include "base/run_loop.h" |
| +#include "base/test/test_simple_task_runner.h" |
| +#include "components/proximity_auth/bluetooth_throttler.h" |
| +#include "components/proximity_auth/connection_finder.h" |
| +#include "components/proximity_auth/proximity_auth_test_util.h" |
| +#include "components/proximity_auth/remote_device.h" |
| +#include "components/proximity_auth/wire_message.h" |
| +#include "device/bluetooth/bluetooth_adapter_factory.h" |
| +#include "device/bluetooth/test/mock_bluetooth_adapter.h" |
| +#include "device/bluetooth/test/mock_bluetooth_device.h" |
| +#include "device/bluetooth/test/mock_bluetooth_discovery_session.h" |
| +#include "device/bluetooth/test/mock_bluetooth_gatt_characteristic.h" |
| +#include "device/bluetooth/test/mock_bluetooth_gatt_connection.h" |
| +#include "device/bluetooth/test/mock_bluetooth_gatt_notify_session.h" |
| +#include "testing/gmock/include/gmock/gmock.h" |
| +#include "testing/gtest/include/gtest/gtest.h" |
| + |
| +using testing::_; |
| +using testing::AtLeast; |
| +using testing::NiceMock; |
| +using testing::Return; |
| +using testing::StrictMock; |
| +using testing::SaveArg; |
| + |
| +namespace proximity_auth { |
| +namespace { |
| + |
| +class MockBluetoothThrottler : public BluetoothThrottler { |
| + public: |
| + MockBluetoothThrottler() {} |
| + ~MockBluetoothThrottler() override {} |
| + |
| + MOCK_CONST_METHOD0(GetDelay, base::TimeDelta()); |
| + MOCK_METHOD1(OnConnection, void(Connection* connection)); |
| + |
| + private: |
| + DISALLOW_COPY_AND_ASSIGN(MockBluetoothThrottler); |
| +}; |
| + |
| +class MockBluetoothLowEnergyCharacteristicsFinder |
| + : public BluetoothLowEnergyCharacteristicsFinder { |
| + public: |
| + MockBluetoothLowEnergyCharacteristicsFinder() {} |
| + ~MockBluetoothLowEnergyCharacteristicsFinder() override {} |
| + |
| + private: |
| + DISALLOW_COPY_AND_ASSIGN(MockBluetoothLowEnergyCharacteristicsFinder); |
| +}; |
| + |
| +} // namespace |
| + |
| +namespace weave { |
| +namespace { |
| + |
| +typedef BluetoothLowEnergyWeaveClientConnection::SubStatus SubStatus; |
| +typedef BluetoothLowEnergyWeavePacketReceiver::State ReceiverState; |
| +typedef BluetoothLowEnergyWeavePacketReceiver::ReceiverError ReceiverError; |
| +typedef BluetoothLowEnergyWeavePacketReceiver::ReceiverType ReceiverType; |
| + |
| +const char kServiceUUID[] = "DEADBEEF-CAFE-FEED-FOOD-D15EA5EBEEEF"; |
| +const char kTXCharacteristicUUID[] = "977c6674-1239-4e72-993b-502369b8bb5a"; |
| +const char kRXCharacteristicUUID[] = "f4b904a2-a030-43b3-98a8-221c536c03cb"; |
| + |
| +const char kServiceID[] = "service id"; |
| +const char kTXCharacteristicID[] = "TX characteristic id"; |
| +const char kRXCharacteristicID[] = "RX characteristic id"; |
| + |
| +const device::BluetoothRemoteGattCharacteristic::Properties |
| + kCharacteristicProperties = |
| + device::BluetoothRemoteGattCharacteristic::PROPERTY_BROADCAST | |
| + device::BluetoothRemoteGattCharacteristic::PROPERTY_READ | |
| + device::BluetoothRemoteGattCharacteristic:: |
| + PROPERTY_WRITE_WITHOUT_RESPONSE | |
| + device::BluetoothRemoteGattCharacteristic::PROPERTY_INDICATE; |
| + |
| +const int kMaxNumberOfTries = 3; |
| +const uint16_t kLargeMaxPacketSize = 30; |
| + |
| +const uint8_t kDataHeader = 0; |
| +const uint8_t kConnectionRequestHeader = 1; |
| +const uint8_t kSmallConnectionResponseHeader = 2; |
| +const uint8_t kLargeConnectionResponseHeader = 3; |
| +const uint8_t kConnectionCloseHeader = 4; |
| +const uint8_t kErroneousHeader = 5; |
| + |
| +const std::string kSmallMessage = "bb"; |
| +const std::string kLargeMessage = "aaabbb"; |
| +const std::string kLargeMessage0 = "aaa"; |
| +const std::string kLargeMessage1 = "bbb"; |
| + |
| +const Packet kConnectionRequest{kConnectionRequestHeader}; |
| +const Packet kSmallConnectionResponse{kSmallConnectionResponseHeader}; |
| +const Packet kLargeConnectionResponse{kLargeConnectionResponseHeader}; |
| +const Packet kConnectionCloseSuccess{kConnectionCloseHeader, |
| + ReasonForClose::CLOSE_WITHOUT_ERROR}; |
| +const Packet kConnectionCloseUnknownError{kConnectionCloseHeader, |
| + ReasonForClose::UNKNOWN_ERROR}; |
| +const Packet kConnectionCloseApplicationError{ |
| + kConnectionCloseHeader, ReasonForClose::APPLICATION_ERROR}; |
| + |
| +const Packet kSmallPackets0 = Packet{kDataHeader, 'b', 'b'}; |
| +const Packet kLargePackets0 = Packet{kDataHeader, 'a', 'a', 'a'}; |
| +const Packet kLargePackets1 = Packet{kDataHeader, 'b', 'b', 'b'}; |
| +const Packet kErroneousPacket = Packet{kErroneousHeader}; |
| + |
| +const std::vector<Packet> kSmallPackets{kSmallPackets0}; |
| +const std::vector<Packet> kLargePackets{kLargePackets0, kLargePackets1}; |
| + |
| +class MockBluetoothLowEnergyWeavePacketGenerator |
| + : public BluetoothLowEnergyWeavePacketGenerator { |
| + public: |
| + MockBluetoothLowEnergyWeavePacketGenerator() |
| + : max_packet_size_(kDefaultMaxPacketSize) {} |
| + |
| + Packet CreateConnectionRequest() override { return kConnectionRequest; } |
| + |
| + Packet CreateConnectionResponse() override { |
| + NOTIMPLEMENTED(); |
| + return Packet(); |
| + } |
| + |
| + Packet CreateConnectionClose(ReasonForClose reason_for_close) override { |
| + return Packet{kConnectionCloseHeader, |
| + static_cast<uint8_t>(reason_for_close)}; |
| + } |
| + |
| + void SetMaxPacketSize(uint16_t size) override { max_packet_size_ = size; } |
| + |
| + std::vector<Packet> EncodeDataMessage(std::string message) override { |
| + if (message == kSmallMessage && max_packet_size_ == kDefaultMaxPacketSize) { |
| + return kSmallPackets; |
| + } else if (message == kLargeMessage && |
| + max_packet_size_ == kLargeMaxPacketSize) { |
| + return kLargePackets; |
| + } else { |
| + NOTREACHED(); |
| + return std::vector<Packet>(); |
| + } |
| + } |
| + |
| + uint16_t GetMaxPacketSize() { return max_packet_size_; } |
| + |
| + private: |
| + uint16_t max_packet_size_; |
| +}; |
| + |
| +class MockBluetoothLowEnergyWeavePacketReceiver |
| + : public BluetoothLowEnergyWeavePacketReceiver { |
| + public: |
| + MockBluetoothLowEnergyWeavePacketReceiver() |
| + : BluetoothLowEnergyWeavePacketReceiver(ReceiverType::CLIENT), |
| + state_(State::CONNECTING), |
| + max_packet_size_(kDefaultMaxPacketSize), |
| + reason_for_close_(ReasonForClose::CLOSE_WITHOUT_ERROR), |
| + reason_to_close_(ReasonForClose::CLOSE_WITHOUT_ERROR) {} |
| + |
| + ReceiverState GetState() override { return state_; } |
| + |
| + uint16_t GetMaxPacketSize() override { return max_packet_size_; } |
| + |
| + ReasonForClose GetReasonForClose() override { return reason_for_close_; } |
| + |
| + ReasonForClose GetReasonToClose() override { return reason_to_close_; } |
| + |
| + std::string GetDataMessage() override { |
| + if (max_packet_size_ == kDefaultMaxPacketSize) { |
| + return kSmallMessage; |
| + } else { |
| + return kLargeMessage; |
| + } |
| + } |
| + |
| + ReceiverError GetReceiverError() override { |
| + return ReceiverError::NO_ERROR_DETECTED; |
| + } |
| + |
| + ReceiverState ReceivePacket(const Packet& packet) override { |
| + switch (packet[0]) { |
| + case kSmallConnectionResponseHeader: |
| + max_packet_size_ = kDefaultMaxPacketSize; |
| + state_ = ReceiverState::WAITING; |
| + break; |
| + case kLargeConnectionResponseHeader: |
| + max_packet_size_ = kLargeMaxPacketSize; |
| + state_ = ReceiverState::WAITING; |
| + break; |
| + case kConnectionCloseHeader: |
| + state_ = ReceiverState::CONNECTION_CLOSED; |
| + reason_for_close_ = static_cast<ReasonForClose>(packet[1]); |
| + break; |
| + case kDataHeader: |
| + if (packet == kSmallPackets0 || packet == kLargePackets1) { |
| + state_ = ReceiverState::DATA_READY; |
| + } else { |
| + state_ = ReceiverState::RECEIVING_DATA; |
| + } |
| + break; |
| + default: |
| + reason_to_close_ = ReasonForClose::APPLICATION_ERROR; |
| + state_ = ReceiverState::ERROR_DETECTED; |
| + } |
| + return state_; |
| + } |
| + |
| + private: |
| + ReceiverState state_; |
| + uint16_t max_packet_size_; |
| + ReasonForClose reason_for_close_; |
| + ReasonForClose reason_to_close_; |
| +}; |
| + |
| +class MockBluetoothLowEnergyWeavePacketGeneratorFactory |
| + : public BluetoothLowEnergyWeavePacketGenerator::Factory { |
| + public: |
| + // most_recent_instance_ will be obsolete after the connection class |
| + // destructs. Do not use if that's the case. |
| + MockBluetoothLowEnergyWeavePacketGenerator* GetMostRecentInstance() { |
| + return most_recent_instance_; |
| + } |
| + |
| + private: |
| + std::unique_ptr<BluetoothLowEnergyWeavePacketGenerator> BuildInstance() |
| + override { |
| + most_recent_instance_ = new MockBluetoothLowEnergyWeavePacketGenerator(); |
| + return std::unique_ptr<BluetoothLowEnergyWeavePacketGenerator>( |
| + most_recent_instance_); |
| + } |
| + |
| + MockBluetoothLowEnergyWeavePacketGenerator* most_recent_instance_; |
| +}; |
| + |
| +class MockBluetoothLowEnergyWeavePacketReceiverFactory |
| + : public BluetoothLowEnergyWeavePacketReceiver::Factory { |
| + public: |
| + // most_recent_instance_ will be obsolete after the connection class |
| + // destructs. Do not use if that's the case. |
| + MockBluetoothLowEnergyWeavePacketReceiver* GetMostRecentInstance() { |
| + return most_recent_instance_; |
| + } |
| + |
| + private: |
| + std::unique_ptr<BluetoothLowEnergyWeavePacketReceiver> BuildInstance( |
| + ReceiverType receiver_type) override { |
| + most_recent_instance_ = new MockBluetoothLowEnergyWeavePacketReceiver(); |
| + return std::unique_ptr<BluetoothLowEnergyWeavePacketReceiver>( |
| + most_recent_instance_); |
| + } |
| + |
| + MockBluetoothLowEnergyWeavePacketReceiver* most_recent_instance_; |
| +}; |
| + |
| +class TestBluetoothLowEnergyWeaveClientConnection |
| + : public BluetoothLowEnergyWeaveClientConnection { |
| + public: |
| + TestBluetoothLowEnergyWeaveClientConnection( |
| + const RemoteDevice& remote_device, |
| + scoped_refptr<device::BluetoothAdapter> adapter, |
| + const device::BluetoothUUID remote_service_uuid, |
| + BluetoothThrottler* bluetooth_throttler, |
| + int max_number_of_write_attempts) |
| + : BluetoothLowEnergyWeaveClientConnection(remote_device, |
| + adapter, |
| + remote_service_uuid, |
| + bluetooth_throttler, |
| + max_number_of_write_attempts) {} |
| + |
| + ~TestBluetoothLowEnergyWeaveClientConnection() override {} |
| + |
| + MOCK_METHOD2( |
| + CreateCharacteristicsFinder, |
| + BluetoothLowEnergyCharacteristicsFinder*( |
| + const BluetoothLowEnergyCharacteristicsFinder::SuccessCallback& |
| + success, |
| + const BluetoothLowEnergyCharacteristicsFinder::ErrorCallback& error)); |
| + |
| + MOCK_METHOD2(OnDidSendMessage, |
| + void(const WireMessage& message, bool success)); |
| + MOCK_METHOD1(OnBytesReceived, void(const std::string& bytes)); |
| + |
| + // Exposing inherited protected methods for testing. |
| + using BluetoothLowEnergyWeaveClientConnection::GattCharacteristicValueChanged; |
| + using BluetoothLowEnergyWeaveClientConnection::SetTaskRunnerForTesting; |
| + using BluetoothLowEnergyWeaveClientConnection::DestroyConnection; |
| + |
| + // Exposing inherited protected fields for testing. |
| + using BluetoothLowEnergyWeaveClientConnection::status; |
| + using BluetoothLowEnergyWeaveClientConnection::sub_status; |
| + |
| + private: |
| + DISALLOW_COPY_AND_ASSIGN(TestBluetoothLowEnergyWeaveClientConnection); |
| +}; |
| + |
| +} // namespace |
| + |
| +class ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest |
| + : public testing::Test { |
| + public: |
| + ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest() |
| + : adapter_(new NiceMock<device::MockBluetoothAdapter>), |
| + remote_device_(CreateLERemoteDeviceForTest()), |
| + service_uuid_(device::BluetoothUUID(kServiceUUID)), |
| + tx_characteristic_uuid_(device::BluetoothUUID(kTXCharacteristicUUID)), |
| + rx_characteristic_uuid_(device::BluetoothUUID(kRXCharacteristicUUID)), |
| + notify_session_alias_(NULL), |
| + bluetooth_throttler_(new NiceMock<MockBluetoothThrottler>), |
| + task_runner_(new base::TestSimpleTaskRunner) { |
| + BluetoothLowEnergyWeavePacketGenerator::Factory::SetInstanceForTesting( |
| + &generator_factory_); |
| + BluetoothLowEnergyWeavePacketReceiver::Factory::SetInstanceForTesting( |
| + &receiver_factory_); |
| + } |
| + |
| + void SetUp() override { |
| + device_ = base::WrapUnique(new NiceMock<device::MockBluetoothDevice>( |
| + adapter_.get(), 0, kTestRemoteDeviceName, |
| + kTestRemoteDeviceBluetoothAddress, false, false)); |
| + |
| + service_ = base::WrapUnique(new NiceMock<device::MockBluetoothGattService>( |
| + device_.get(), kServiceID, service_uuid_, true, false)); |
| + tx_characteristic_ = |
| + base::WrapUnique(new NiceMock<device::MockBluetoothGattCharacteristic>( |
| + service_.get(), kTXCharacteristicID, tx_characteristic_uuid_, false, |
| + kCharacteristicProperties, |
| + device::BluetoothRemoteGattCharacteristic::PERMISSION_NONE)); |
| + |
| + rx_characteristic_ = |
| + base::WrapUnique(new NiceMock<device::MockBluetoothGattCharacteristic>( |
| + service_.get(), kRXCharacteristicID, rx_characteristic_uuid_, false, |
| + kCharacteristicProperties, |
| + device::BluetoothRemoteGattCharacteristic::PERMISSION_NONE)); |
| + |
| + device::BluetoothAdapterFactory::SetAdapterForTesting(adapter_); |
| + |
| + std::vector<const device::BluetoothDevice*> devices; |
| + devices.push_back(device_.get()); |
| + ON_CALL(*adapter_, GetDevices()).WillByDefault(Return(devices)); |
| + ON_CALL(*adapter_, GetDevice(kTestRemoteDeviceBluetoothAddress)) |
| + .WillByDefault(Return(device_.get())); |
| + ON_CALL(*device_, GetGattService(kServiceID)) |
| + .WillByDefault(Return(service_.get())); |
| + ON_CALL(*service_, GetCharacteristic(kRXCharacteristicID)) |
| + .WillByDefault(Return(rx_characteristic_.get())); |
| + ON_CALL(*service_, GetCharacteristic(kTXCharacteristicID)) |
| + .WillByDefault(Return(tx_characteristic_.get())); |
| + } |
| + |
| + // Creates a BluetoothLowEnergyWeaveClientConnection and verifies it's in |
| + // DISCONNECTED state. |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> |
| + CreateConnection() { |
| + EXPECT_CALL(*adapter_, AddObserver(_)); |
| + EXPECT_CALL(*adapter_, RemoveObserver(_)); |
| + |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + new TestBluetoothLowEnergyWeaveClientConnection( |
| + remote_device_, adapter_, service_uuid_, bluetooth_throttler_.get(), |
| + kMaxNumberOfTries)); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| + |
| + connection->SetTaskRunnerForTesting(task_runner_); |
| + |
| + return connection; |
| + } |
| + |
| + // Transitions |connection| from DISCONNECTED to WAITING_CHARACTERISTICS |
| + // state, without an existing GATT connection. |
| + void ConnectGatt(TestBluetoothLowEnergyWeaveClientConnection* connection) { |
| + // Preparing |connection| for a CreateGattConnection call. |
| + EXPECT_CALL(*device_, CreateGattConnection(_, _)) |
| + .WillOnce(DoAll(SaveArg<0>(&create_gatt_connection_success_callback_), |
| + SaveArg<1>(&create_gatt_connection_error_callback_))); |
| + |
| + // No throttling by default |
| + EXPECT_CALL(*bluetooth_throttler_, GetDelay()) |
| + .WillOnce(Return(base::TimeDelta())); |
| + |
| + connection->Connect(); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::WAITING_GATT_CONNECTION); |
| + EXPECT_EQ(connection->status(), Connection::IN_PROGRESS); |
| + |
| + // Preparing |connection| to run |create_gatt_connection_success_callback_|. |
| + EXPECT_FALSE(create_gatt_connection_error_callback_.is_null()); |
| + ASSERT_FALSE(create_gatt_connection_success_callback_.is_null()); |
| + EXPECT_CALL(*connection, CreateCharacteristicsFinder(_, _)) |
| + .WillOnce(DoAll( |
| + SaveArg<0>(&characteristics_finder_success_callback_), |
| + SaveArg<1>(&characteristics_finder_error_callback_), |
| + Return(new NiceMock<MockBluetoothLowEnergyCharacteristicsFinder>))); |
| + |
| + create_gatt_connection_success_callback_.Run( |
| + base::WrapUnique(new NiceMock<device::MockBluetoothGattConnection>( |
| + adapter_, kTestRemoteDeviceBluetoothAddress))); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::WAITING_CHARACTERISTICS); |
| + EXPECT_EQ(connection->status(), Connection::IN_PROGRESS); |
| + } |
| + |
| + // Transitions |connection| from WAITING_CHARACTERISTICS to |
| + // WAITING_NOTIFY_SESSION state. |
| + void CharacteristicsFound( |
| + TestBluetoothLowEnergyWeaveClientConnection* connection) { |
| + EXPECT_CALL(*rx_characteristic_, StartNotifySession(_, _)) |
| + .WillOnce(DoAll(SaveArg<0>(¬ify_session_success_callback_), |
| + SaveArg<1>(¬ify_session_error_callback_))); |
| + EXPECT_FALSE(characteristics_finder_error_callback_.is_null()); |
| + ASSERT_FALSE(characteristics_finder_success_callback_.is_null()); |
| + |
| + characteristics_finder_success_callback_.Run( |
| + {service_uuid_, kServiceID}, |
| + {tx_characteristic_uuid_, kTXCharacteristicID}, |
| + {rx_characteristic_uuid_, kRXCharacteristicID}); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::WAITING_NOTIFY_SESSION); |
| + EXPECT_EQ(connection->status(), Connection::IN_PROGRESS); |
| + } |
| + |
| + // Transitions |connection| from WAITING_NOTIFY_SESSION to |
| + // WAITING_CONNECTION_RESPONSE state. |
| + void NotifySessionStarted( |
| + TestBluetoothLowEnergyWeaveClientConnection* connection) { |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + EXPECT_FALSE(notify_session_error_callback_.is_null()); |
| + ASSERT_FALSE(notify_session_success_callback_.is_null()); |
| + |
| + // Store an alias for the notify session passed |connection|. |
| + std::unique_ptr<device::MockBluetoothGattNotifySession> notify_session( |
| + new NiceMock<device::MockBluetoothGattNotifySession>( |
| + kTXCharacteristicID)); |
| + notify_session_alias_ = notify_session.get(); |
| + |
| + notify_session_success_callback_.Run(std::move(notify_session)); |
| + task_runner_->RunUntilIdle(); |
| + |
| + // Written value contains only the mock Connection Request. |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, kConnectionRequest); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::WAITING_CONNECTION_RESPONSE); |
| + EXPECT_EQ(connection->status(), Connection::IN_PROGRESS); |
| + } |
| + |
| + // Transitions |connection| from WAITING_CONNECTION_RESPONSE to CONNECTED. |
| + void ConnectionResponseReceived( |
| + TestBluetoothLowEnergyWeaveClientConnection* connection, |
| + uint16_t selected_packet_size) { |
| + // Written value contains only the mock Connection Request. |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, kConnectionRequest); |
| + |
| + EXPECT_CALL(*connection, OnDidSendMessage(_, _)).Times(0); |
| + RunWriteCharacteristicSuccessCallback(); |
| + |
| + // Received Connection Response. |
| + if (selected_packet_size == kDefaultMaxPacketSize) { |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kSmallConnectionResponse); |
| + EXPECT_EQ(receiver_factory_.GetMostRecentInstance()->GetMaxPacketSize(), |
| + kDefaultMaxPacketSize); |
| + EXPECT_EQ(generator_factory_.GetMostRecentInstance()->GetMaxPacketSize(), |
| + kDefaultMaxPacketSize); |
| + } else if (selected_packet_size == kLargeMaxPacketSize) { |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kLargeConnectionResponse); |
| + EXPECT_EQ(receiver_factory_.GetMostRecentInstance()->GetMaxPacketSize(), |
| + kLargeMaxPacketSize); |
| + EXPECT_EQ(generator_factory_.GetMostRecentInstance()->GetMaxPacketSize(), |
| + kLargeMaxPacketSize); |
| + } else { |
| + NOTREACHED(); |
| + } |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::CONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::CONNECTED); |
| + } |
| + |
| + // Transitions |connection| to a DISCONNECTED state regardless of its initial |
| + // state. |
| + void Disconnect(TestBluetoothLowEnergyWeaveClientConnection* connection) { |
| + // A notify session was previously set. |
| + if (notify_session_alias_) |
| + EXPECT_CALL(*notify_session_alias_, Stop(_)); |
| + |
| + if (connection->sub_status() == SubStatus::CONNECTED) { |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + } |
| + |
| + connection->Disconnect(); |
| + |
| + if (connection->sub_status() == SubStatus::CONNECTED) { |
| + connection->DestroyConnection(); |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, |
| + kConnectionCloseSuccess); |
| + RunWriteCharacteristicSuccessCallback(); |
| + } |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| + } |
| + |
| + void InitializeConnection( |
| + TestBluetoothLowEnergyWeaveClientConnection* connection, |
| + uint32_t selected_packet_size) { |
| + ConnectGatt(connection); |
| + CharacteristicsFound(connection); |
| + NotifySessionStarted(connection); |
| + ConnectionResponseReceived(connection, selected_packet_size); |
| + } |
| + |
| + void RunWriteCharacteristicSuccessCallback() { |
| + EXPECT_FALSE(write_remote_characteristic_error_callback_.is_null()); |
| + ASSERT_FALSE(write_remote_characteristic_success_callback_.is_null()); |
| + write_remote_characteristic_success_callback_.Run(); |
| + } |
| + |
| + protected: |
| + scoped_refptr<device::MockBluetoothAdapter> adapter_; |
| + RemoteDevice remote_device_; |
| + device::BluetoothUUID service_uuid_; |
| + device::BluetoothUUID tx_characteristic_uuid_; |
| + device::BluetoothUUID rx_characteristic_uuid_; |
| + std::unique_ptr<device::MockBluetoothDevice> device_; |
| + std::unique_ptr<device::MockBluetoothGattService> service_; |
| + std::unique_ptr<device::MockBluetoothGattCharacteristic> tx_characteristic_; |
| + std::unique_ptr<device::MockBluetoothGattCharacteristic> rx_characteristic_; |
| + std::vector<uint8_t> last_value_written_on_tx_characteristic_; |
| + device::MockBluetoothGattNotifySession* notify_session_alias_; |
| + std::unique_ptr<MockBluetoothThrottler> bluetooth_throttler_; |
| + scoped_refptr<base::TestSimpleTaskRunner> task_runner_; |
| + base::MessageLoop message_loop_; |
| + |
| + // Callbacks |
| + device::BluetoothDevice::GattConnectionCallback |
| + create_gatt_connection_success_callback_; |
| + device::BluetoothDevice::ConnectErrorCallback |
| + create_gatt_connection_error_callback_; |
| + |
| + BluetoothLowEnergyCharacteristicsFinder::SuccessCallback |
| + characteristics_finder_success_callback_; |
| + BluetoothLowEnergyCharacteristicsFinder::ErrorCallback |
| + characteristics_finder_error_callback_; |
| + |
| + device::BluetoothRemoteGattCharacteristic::NotifySessionCallback |
| + notify_session_success_callback_; |
| + device::BluetoothRemoteGattCharacteristic::ErrorCallback |
| + notify_session_error_callback_; |
| + |
| + base::Closure write_remote_characteristic_success_callback_; |
| + device::BluetoothRemoteGattCharacteristic::ErrorCallback |
| + write_remote_characteristic_error_callback_; |
| + |
| + MockBluetoothLowEnergyWeavePacketGeneratorFactory generator_factory_; |
| + MockBluetoothLowEnergyWeavePacketReceiverFactory receiver_factory_; |
| +}; |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + CreateAndDestroyWithoutConnectCallDoesntCrash) { |
| + BluetoothLowEnergyWeaveClientConnection connection( |
| + remote_device_, adapter_, service_uuid_, bluetooth_throttler_.get(), |
| + kMaxNumberOfTries); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + DisconectWithoutConnectDoesntCrash) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + Disconnect(connection.get()); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectSuccess) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + CharacteristicsFound(connection.get()); |
| + NotifySessionStarted(connection.get()); |
| + ConnectionResponseReceived(connection.get(), kDefaultMaxPacketSize); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectSuccessDisconnect) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::CONNECTED); |
| + Disconnect(connection.get()); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectIncompleteDisconnectFromWaitingCharacteristicsState) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + Disconnect(connection.get()); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectIncompleteDisconnectFromWaitingNotifySessionState) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + CharacteristicsFound(connection.get()); |
| + Disconnect(connection.get()); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectIncompleteDisconnectFromWaitingConnectionResponseState) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + CharacteristicsFound(connection.get()); |
| + NotifySessionStarted(connection.get()); |
| + Disconnect(connection.get()); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectFailsCharacteristicsNotFound) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + |
| + EXPECT_CALL(*rx_characteristic_, StartNotifySession(_, _)).Times(0); |
| + EXPECT_FALSE(characteristics_finder_success_callback_.is_null()); |
| + ASSERT_FALSE(characteristics_finder_error_callback_.is_null()); |
| + |
| + characteristics_finder_error_callback_.Run( |
| + {tx_characteristic_uuid_, kTXCharacteristicID}, |
| + {rx_characteristic_uuid_, kRXCharacteristicID}); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectFailsNotifySessionError) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + CharacteristicsFound(connection.get()); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)).Times(0); |
| + EXPECT_FALSE(notify_session_success_callback_.is_null()); |
| + ASSERT_FALSE(notify_session_error_callback_.is_null()); |
| + |
| + notify_session_error_callback_.Run( |
| + device::BluetoothRemoteGattService::GATT_ERROR_UNKNOWN); |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectFailsErrorSendingConnectionRequest) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + ConnectGatt(connection.get()); |
| + CharacteristicsFound(connection.get()); |
| + NotifySessionStarted(connection.get()); |
| + |
| + // |connection| will call WriteRemoteCharacteristics(_,_) to try to send the |
| + // message |kMaxNumberOfTries| times. There is alredy one EXPECTA_CALL for |
| + // WriteRemoteCharacteristic(_,_,_) in NotifySessionStated, that's why we use |
| + // |kMaxNumberOfTries-1| in the EXPECT_CALL statement. |
| + EXPECT_CALL(*connection, OnDidSendMessage(_, _)).Times(0); |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .Times(kMaxNumberOfTries - 1) |
| + .WillRepeatedly( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + for (int i = 0; i < kMaxNumberOfTries; i++) { |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, kConnectionRequest); |
| + ASSERT_FALSE(write_remote_characteristic_error_callback_.is_null()); |
| + EXPECT_FALSE(write_remote_characteristic_success_callback_.is_null()); |
| + write_remote_characteristic_error_callback_.Run( |
| + device::BluetoothRemoteGattService::GATT_ERROR_UNKNOWN); |
| + } |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ReceiveMessageSmallerThanCharacteristicSize) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + |
| + std::string received_bytes; |
| + EXPECT_CALL(*connection, OnBytesReceived(_)) |
| + .WillOnce(SaveArg<0>(&received_bytes)); |
| + |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kSmallPackets0); |
| + |
| + EXPECT_EQ(received_bytes, kSmallMessage); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ReceiveMessageLargerThanCharacteristicSize) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + InitializeConnection(connection.get(), kLargeMaxPacketSize); |
| + |
| + std::string received_bytes; |
| + EXPECT_CALL(*connection, OnBytesReceived(_)) |
| + .WillOnce(SaveArg<0>(&received_bytes)); |
| + |
| + std::vector<Packet> packets = kLargePackets; |
| + |
| + for (auto packet : packets) { |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), packet); |
| + } |
| + EXPECT_EQ(received_bytes, kLargeMessage); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + SendMessageSmallerThanCharacteristicSize) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + |
| + // Expecting a first call of WriteRemoteCharacteristic, after SendMessage is |
| + // called. |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + connection->SendMessage(base::WrapUnique(new FakeWireMessage(kSmallMessage))); |
| + |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, kSmallPackets0); |
| + EXPECT_CALL(*connection, OnDidSendMessage(_, _)); |
| + |
| + RunWriteCharacteristicSuccessCallback(); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + SendMessageLargerThanCharacteristicSize) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + InitializeConnection(connection.get(), kLargeMaxPacketSize); |
| + |
| + // Expecting a first call of WriteRemoteCharacteristic, after SendMessage is |
| + // called. |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + connection->SendMessage(base::WrapUnique(new FakeWireMessage(kLargeMessage))); |
| + |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, kLargePackets0); |
| + std::vector<uint8_t> bytes_received( |
| + last_value_written_on_tx_characteristic_.begin() + 1, |
| + last_value_written_on_tx_characteristic_.end()); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + RunWriteCharacteristicSuccessCallback(); |
| + bytes_received.insert(bytes_received.end(), |
| + last_value_written_on_tx_characteristic_.begin() + 1, |
| + last_value_written_on_tx_characteristic_.end()); |
| + |
| + std::vector<uint8_t> expected(kLargeMessage.begin(), kLargeMessage.end()); |
| + EXPECT_EQ(expected, bytes_received); |
| + |
| + EXPECT_CALL(*connection, OnDidSendMessage(_, _)); |
| + RunWriteCharacteristicSuccessCallback(); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ReceiveCloseConnectionTest) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kConnectionCloseUnknownError); |
| + |
| + EXPECT_EQ(receiver_factory_.GetMostRecentInstance()->GetReasonForClose(), |
| + ReasonForClose::UNKNOWN_ERROR); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ReceiverErrorTest) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kErroneousPacket); |
| + |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, |
| + kConnectionCloseApplicationError); |
| + EXPECT_EQ(receiver_factory_.GetMostRecentInstance()->GetReasonToClose(), |
| + ReasonForClose::APPLICATION_ERROR); |
| + |
| + RunWriteCharacteristicSuccessCallback(); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ReceiverErrorWithPendingWritesTest) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + InitializeConnection(connection.get(), kLargeMaxPacketSize); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + connection->SendMessage(base::WrapUnique(new FakeWireMessage(kLargeMessage))); |
| + |
| + connection->GattCharacteristicValueChanged( |
| + adapter_.get(), rx_characteristic_.get(), kErroneousPacket); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + |
| + RunWriteCharacteristicSuccessCallback(); |
| + |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, |
|
Kyle Horimoto
2016/07/12 01:03:16
Also test that the in-flight message is also sent.
jingxuy
2016/07/15 18:25:00
Done.
|
| + kConnectionCloseApplicationError); |
| + EXPECT_EQ(receiver_factory_.GetMostRecentInstance()->GetReasonToClose(), |
| + ReasonForClose::APPLICATION_ERROR); |
| + |
| + RunWriteCharacteristicSuccessCallback(); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + WriteConnectionCloseMaxNumberOfTimes) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + InitializeConnection(connection.get(), kDefaultMaxPacketSize); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::CONNECTED); |
| + |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + connection->Disconnect(); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::CONNECTED); |
| + |
| + for (int i = 0; i < kMaxNumberOfTries; i++) { |
| + EXPECT_EQ(last_value_written_on_tx_characteristic_, |
| + kConnectionCloseSuccess); |
| + ASSERT_FALSE(write_remote_characteristic_error_callback_.is_null()); |
| + EXPECT_FALSE(write_remote_characteristic_success_callback_.is_null()); |
| + |
| + if (i != kMaxNumberOfTries - 1) { |
| + EXPECT_CALL(*tx_characteristic_, WriteRemoteCharacteristic(_, _, _)) |
| + .WillOnce( |
| + DoAll(SaveArg<0>(&last_value_written_on_tx_characteristic_), |
| + SaveArg<1>(&write_remote_characteristic_success_callback_), |
| + SaveArg<2>(&write_remote_characteristic_error_callback_))); |
| + } |
| + |
| + write_remote_characteristic_error_callback_.Run( |
| + device::BluetoothRemoteGattService::GATT_ERROR_UNKNOWN); |
| + } |
| + |
| + EXPECT_EQ(connection->sub_status(), SubStatus::DISCONNECTED); |
| + EXPECT_EQ(connection->status(), Connection::DISCONNECTED); |
| +} |
| + |
| +TEST_F(ProximityAuthBluetoothLowEnergyWeaveClientConnectionTest, |
| + ConnectAfterADelayWhenThrottled) { |
| + std::unique_ptr<TestBluetoothLowEnergyWeaveClientConnection> connection( |
| + CreateConnection()); |
| + |
| + EXPECT_CALL(*bluetooth_throttler_, GetDelay()) |
| + .WillOnce(Return(base::TimeDelta(base::TimeDelta::FromSeconds(1)))); |
| + EXPECT_CALL(*device_, CreateGattConnection(_, _)) |
| + .WillOnce(DoAll(SaveArg<0>(&create_gatt_connection_success_callback_), |
| + SaveArg<1>(&create_gatt_connection_error_callback_))); |
| + |
| + // No GATT connection should be created before the delay. |
| + connection->Connect(); |
| + EXPECT_EQ(connection->sub_status(), SubStatus::WAITING_GATT_CONNECTION); |
| + EXPECT_EQ(connection->status(), Connection::IN_PROGRESS); |
| + EXPECT_TRUE(create_gatt_connection_error_callback_.is_null()); |
| + EXPECT_TRUE(create_gatt_connection_success_callback_.is_null()); |
| + |
| + // A GATT connection should be created after the delay. |
| + task_runner_->RunUntilIdle(); |
| + EXPECT_FALSE(create_gatt_connection_error_callback_.is_null()); |
| + ASSERT_FALSE(create_gatt_connection_success_callback_.is_null()); |
| + |
| + // Preparing |connection| to run |create_gatt_connection_success_callback_|. |
| + EXPECT_CALL(*connection, CreateCharacteristicsFinder(_, _)) |
| + .WillOnce(DoAll( |
| + SaveArg<0>(&characteristics_finder_success_callback_), |
| + SaveArg<1>(&characteristics_finder_error_callback_), |
| + Return(new NiceMock<MockBluetoothLowEnergyCharacteristicsFinder>))); |
| + |
| + create_gatt_connection_success_callback_.Run( |
| + base::WrapUnique(new NiceMock<device::MockBluetoothGattConnection>( |
| + adapter_, kTestRemoteDeviceBluetoothAddress))); |
| + |
| + CharacteristicsFound(connection.get()); |
| + NotifySessionStarted(connection.get()); |
| + ConnectionResponseReceived(connection.get(), kDefaultMaxPacketSize); |
| +} |
| + |
| +} // namespace weave |
| + |
| +} // namespace proximity_auth |