// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chromeos/ash/services/secure_channel/pending_ble_initiator_connection_request.h"
#include <memory>
#include <utility>
#include "base/memory/raw_ptr.h"
#include "base/run_loop.h"
#include "base/test/task_environment.h"
#include "base/unguessable_token.h"
#include "chromeos/ash/services/secure_channel/fake_client_connection_parameters.h"
#include "chromeos/ash/services/secure_channel/fake_pending_connection_request_delegate.h"
#include "chromeos/ash/services/secure_channel/public/cpp/shared/connection_priority.h"
#include "chromeos/ash/services/secure_channel/public/mojom/secure_channel.mojom.h"
#include "device/bluetooth/test/mock_bluetooth_adapter.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace ash::secure_channel {
namespace {
const char kTestFeature[] = "testFeature";
} // namespace
class SecureChannelPendingBleInitiatorConnectionRequestTest
: public testing::Test {
public:
SecureChannelPendingBleInitiatorConnectionRequestTest(
const SecureChannelPendingBleInitiatorConnectionRequestTest&) = delete;
SecureChannelPendingBleInitiatorConnectionRequestTest& operator=(
const SecureChannelPendingBleInitiatorConnectionRequestTest&) = delete;
protected:
SecureChannelPendingBleInitiatorConnectionRequestTest() = default;
~SecureChannelPendingBleInitiatorConnectionRequestTest() override = default;
// testing::Test:
void SetUp() override {
fake_pending_connection_request_delegate_ =
std::make_unique<FakePendingConnectionRequestDelegate>();
auto fake_client_connection_parameters =
std::make_unique<FakeClientConnectionParameters>(kTestFeature);
fake_client_connection_parameters_ =
fake_client_connection_parameters.get();
mock_adapter_ =
base::MakeRefCounted<testing::NiceMock<device::MockBluetoothAdapter>>();
pending_ble_initiator_request_ =
PendingBleInitiatorConnectionRequest::Factory::Create(
std::move(fake_client_connection_parameters),
ConnectionPriority::kLow,
fake_pending_connection_request_delegate_.get(), mock_adapter_);
}
const std::optional<PendingConnectionRequestDelegate::FailedConnectionReason>&
GetFailedConnectionReason() {
return fake_pending_connection_request_delegate_
->GetFailedConnectionReasonForId(
pending_ble_initiator_request_->GetRequestId());
}
const std::optional<mojom::ConnectionAttemptFailureReason>&
GetConnectionAttemptFailureReason() {
return fake_client_connection_parameters_->failure_reason();
}
void HandleConnectionFailure(BleInitiatorFailureType failure_type) {
pending_ble_initiator_request_->HandleConnectionFailure(failure_type);
}
private:
base::test::TaskEnvironment task_environment_;
std::unique_ptr<FakePendingConnectionRequestDelegate>
fake_pending_connection_request_delegate_;
raw_ptr<FakeClientConnectionParameters, DanglingUntriaged>
fake_client_connection_parameters_;
scoped_refptr<testing::NiceMock<device::MockBluetoothAdapter>> mock_adapter_;
std::unique_ptr<PendingConnectionRequest<BleInitiatorFailureType>>
pending_ble_initiator_request_;
};
TEST_F(SecureChannelPendingBleInitiatorConnectionRequestTest,
HandleAuthenticationError) {
HandleConnectionFailure(BleInitiatorFailureType::kAuthenticationError);
EXPECT_EQ(
PendingConnectionRequestDelegate::FailedConnectionReason::kRequestFailed,
*GetFailedConnectionReason());
EXPECT_EQ(mojom::ConnectionAttemptFailureReason::AUTHENTICATION_ERROR,
*GetConnectionAttemptFailureReason());
}
TEST_F(SecureChannelPendingBleInitiatorConnectionRequestTest,
HandleCouldNotGenerateAdvertisement) {
HandleConnectionFailure(
BleInitiatorFailureType::kCouldNotGenerateAdvertisement);
EXPECT_EQ(
PendingConnectionRequestDelegate::FailedConnectionReason::kRequestFailed,
*GetFailedConnectionReason());
EXPECT_EQ(
mojom::ConnectionAttemptFailureReason::COULD_NOT_GENERATE_ADVERTISEMENT,
*GetConnectionAttemptFailureReason());
}
TEST_F(SecureChannelPendingBleInitiatorConnectionRequestTest,
HandleGattErrors) {
// Fail 5 times; no fatal error should occur.
for (size_t i = 0; i < 5; ++i) {
HandleConnectionFailure(BleInitiatorFailureType::kGattConnectionError);
EXPECT_FALSE(GetFailedConnectionReason());
EXPECT_FALSE(GetConnectionAttemptFailureReason());
}
// Fail a 6th time; this should be a fatal error.
HandleConnectionFailure(BleInitiatorFailureType::kGattConnectionError);
EXPECT_EQ(
PendingConnectionRequestDelegate::FailedConnectionReason::kRequestFailed,
*GetFailedConnectionReason());
EXPECT_EQ(mojom::ConnectionAttemptFailureReason::GATT_CONNECTION_ERROR,
*GetConnectionAttemptFailureReason());
}
TEST_F(SecureChannelPendingBleInitiatorConnectionRequestTest, HandleTimeouts) {
// Fail 2 times; no fatal error should occur.
for (size_t i = 0; i < 2; ++i) {
HandleConnectionFailure(
BleInitiatorFailureType::kTimeoutContactingRemoteDevice);
EXPECT_FALSE(GetFailedConnectionReason());
EXPECT_FALSE(GetConnectionAttemptFailureReason());
}
// Fail a 3rd time; this should be a fatal error.
HandleConnectionFailure(
BleInitiatorFailureType::kTimeoutContactingRemoteDevice);
EXPECT_EQ(
PendingConnectionRequestDelegate::FailedConnectionReason::kRequestFailed,
*GetFailedConnectionReason());
EXPECT_EQ(mojom::ConnectionAttemptFailureReason::TIMEOUT_FINDING_DEVICE,
*GetConnectionAttemptFailureReason());
}
TEST_F(SecureChannelPendingBleInitiatorConnectionRequestTest,
NonFailingErrors) {
// Fail 5 times due to GATT errors; no fatal error should occur.
for (size_t i = 0; i < 5; ++i) {
HandleConnectionFailure(BleInitiatorFailureType::kGattConnectionError);
EXPECT_FALSE(GetFailedConnectionReason());
EXPECT_FALSE(GetConnectionAttemptFailureReason());
}
// Fail 2 times due to timeouts; no fatal error should occur.
for (size_t i = 0; i < 2; ++i) {
HandleConnectionFailure(
BleInitiatorFailureType::kTimeoutContactingRemoteDevice);
EXPECT_FALSE(GetFailedConnectionReason());
EXPECT_FALSE(GetConnectionAttemptFailureReason());
}
// Fail due to being interrupted by a higher-priority attempt; no fatal error
// should occur.
HandleConnectionFailure(
BleInitiatorFailureType::kInterruptedByHigherPriorityConnectionAttempt);
EXPECT_FALSE(GetFailedConnectionReason());
EXPECT_FALSE(GetConnectionAttemptFailureReason());
}
} // namespace ash::secure_channel