Commit 9f368f46 authored by Kyle Horimoto's avatar Kyle Horimoto Committed by Commit Bot

[CrOS MultiDevice] Clean up ConnectToDeviceOperationBase and add tests.

This CL expands upon [1], which was a quick fix to fix a crash in the
M-71 branch. This follow-up CL makes the class more robust by ensuring
that if connection priority is updated before a connection attempt is
made, the connection will be made with the updated priority when it is
performed.

Additionally, this CL adds a test for this functionality.

[1] https://chromium-review.googlesource.com/c/chromium/src/+/1338322

Change-Id: I9a881c67276c5f571b9119c27d899762b9c5f02f
Reviewed-on: https://chromium-review.googlesource.com/c/1340977Reviewed-by: default avatarRyan Hansberry <hansberry@chromium.org>
Commit-Queue: Kyle Horimoto <khorimoto@chromium.org>
Cr-Commit-Position: refs/heads/master@{#609512}
parent 5168c222
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include "base/macros.h" #include "base/macros.h"
#include "base/memory/weak_ptr.h" #include "base/memory/weak_ptr.h"
#include "base/optional.h"
#include "base/threading/thread_task_runner_handle.h" #include "base/threading/thread_task_runner_handle.h"
#include "chromeos/services/secure_channel/connect_to_device_operation.h" #include "chromeos/services/secure_channel/connect_to_device_operation.h"
#include "chromeos/services/secure_channel/device_id_pair.h" #include "chromeos/services/secure_channel/device_id_pair.h"
...@@ -41,6 +42,7 @@ class ConnectToDeviceOperationBase ...@@ -41,6 +42,7 @@ class ConnectToDeviceOperationBase
connection_priority), connection_priority),
device_id_pair_(device_id_pair), device_id_pair_(device_id_pair),
task_runner_(task_runner), task_runner_(task_runner),
pending_connection_attempt_priority_(connection_priority),
weak_ptr_factory_(this) { weak_ptr_factory_(this) {
// Attempt a connection; however, post this as a task to be run after the // Attempt a connection; however, post this as a task to be run after the
// constructor is finished. This ensures that the derived type is fully // constructor is finished. This ensures that the derived type is fully
...@@ -49,25 +51,48 @@ class ConnectToDeviceOperationBase ...@@ -49,25 +51,48 @@ class ConnectToDeviceOperationBase
FROM_HERE, FROM_HERE,
base::BindOnce(&ConnectToDeviceOperationBase< base::BindOnce(&ConnectToDeviceOperationBase<
FailureDetailType>::AttemptConnectionToDevice, FailureDetailType>::AttemptConnectionToDevice,
weak_ptr_factory_.GetWeakPtr(), connection_priority)); weak_ptr_factory_.GetWeakPtr()));
} }
~ConnectToDeviceOperationBase() override = default; ~ConnectToDeviceOperationBase() override = default;
void AttemptConnectionToDevice(ConnectionPriority connection_priority) { void AttemptConnectionToDevice() {
has_started_connection_attempt_ = true; // If there is no longer a pending connection attempt, this means that the
PerformAttemptConnectionToDevice(connection_priority); // attempt was canceled before it could be started.
if (!pending_connection_attempt_priority_)
return;
// Get the current priority, then reset the pending request.
ConnectionPriority priority = *pending_connection_attempt_priority_;
pending_connection_attempt_priority_.reset();
PerformAttemptConnectionToDevice(priority);
} }
void CancelInternal() override { void CancelInternal() override {
if (has_started_connection_attempt_) // If the attempt is still pending, it has been canceled before it even
PerformCancellation(); // was started. In this case, invalidate the pending request to ensure
// that it never ends up being attempted.
if (pending_connection_attempt_priority_) {
pending_connection_attempt_priority_.reset();
return;
}
// Otherwise, the attempt is already active, so cancel it directly.
PerformCancellation();
} }
void UpdateConnectionPriorityInternal( void UpdateConnectionPriorityInternal(
ConnectionPriority connection_priority) override { ConnectionPriority connection_priority) override {
if (has_started_connection_attempt_) // If the attempt is still pending, update the connection priorty so that
PerformUpdateConnectionPriority(connection_priority); // when the attempt is started, the correct priority will be provided.
if (pending_connection_attempt_priority_) {
pending_connection_attempt_priority_ = connection_priority;
return;
}
// Otherwise, the attempt is already active, so update its priority.
PerformUpdateConnectionPriority(connection_priority);
} }
virtual void PerformAttemptConnectionToDevice( virtual void PerformAttemptConnectionToDevice(
...@@ -81,7 +106,7 @@ class ConnectToDeviceOperationBase ...@@ -81,7 +106,7 @@ class ConnectToDeviceOperationBase
private: private:
const DeviceIdPair& device_id_pair_; const DeviceIdPair& device_id_pair_;
scoped_refptr<base::TaskRunner> task_runner_; scoped_refptr<base::TaskRunner> task_runner_;
bool has_started_connection_attempt_ = false; base::Optional<ConnectionPriority> pending_connection_attempt_priority_;
base::WeakPtrFactory<ConnectToDeviceOperationBase> weak_ptr_factory_; base::WeakPtrFactory<ConnectToDeviceOperationBase> weak_ptr_factory_;
DISALLOW_COPY_AND_ASSIGN(ConnectToDeviceOperationBase); DISALLOW_COPY_AND_ASSIGN(ConnectToDeviceOperationBase);
......
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
#include "base/memory/ptr_util.h" #include "base/memory/ptr_util.h"
#include "base/memory/ref_counted.h" #include "base/memory/ref_counted.h"
#include "base/optional.h"
#include "base/test/scoped_task_environment.h" #include "base/test/scoped_task_environment.h"
#include "base/test/test_simple_task_runner.h" #include "base/test/test_simple_task_runner.h"
#include "chromeos/services/secure_channel/device_id_pair.h" #include "chromeos/services/secure_channel/device_id_pair.h"
...@@ -34,30 +35,45 @@ class TestConnectToDeviceOperation ...@@ -34,30 +35,45 @@ class TestConnectToDeviceOperation
const ConnectToDeviceOperation<std::string>::ConnectionFailedCallback& const ConnectToDeviceOperation<std::string>::ConnectionFailedCallback&
failure_callback, failure_callback,
const DeviceIdPair& device_id_pair, const DeviceIdPair& device_id_pair,
ConnectionPriority connection_priority) { ConnectionPriority connection_priority,
bool should_attempt_connection_synchronously) {
auto test_task_runner = base::MakeRefCounted<base::TestSimpleTaskRunner>(); auto test_task_runner = base::MakeRefCounted<base::TestSimpleTaskRunner>();
auto operation = base::WrapUnique(new TestConnectToDeviceOperation( auto operation = base::WrapUnique(new TestConnectToDeviceOperation(
std::move(success_callback), std::move(failure_callback), std::move(success_callback), std::move(failure_callback),
device_id_pair, connection_priority, test_task_runner)); device_id_pair, connection_priority, test_task_runner));
test_task_runner->RunUntilIdle();
if (should_attempt_connection_synchronously)
test_task_runner->RunUntilIdle();
return operation; return operation;
} }
~TestConnectToDeviceOperation() override = default; ~TestConnectToDeviceOperation() override = default;
void RunPendingTasks() { test_task_runner_->RunUntilIdle(); }
bool has_attempted_connection() const { return has_attempted_connection_; } bool has_attempted_connection() const { return has_attempted_connection_; }
bool has_canceled_connection() const { return has_canceled_connection_; } bool has_canceled_connection() const { return has_canceled_connection_; }
const base::Optional<ConnectionPriority>& current_connection_priority() {
return current_connection_priority_;
}
// ConnectToDeviceOperationBase<std::string>: // ConnectToDeviceOperationBase<std::string>:
void PerformAttemptConnectionToDevice( void PerformAttemptConnectionToDevice(
ConnectionPriority connection_priority) override { ConnectionPriority connection_priority) override {
has_attempted_connection_ = true; has_attempted_connection_ = true;
current_connection_priority_ = connection_priority;
} }
void PerformCancellation() override { has_canceled_connection_ = true; } void PerformCancellation() override {
has_canceled_connection_ = true;
current_connection_priority_.reset();
}
void PerformUpdateConnectionPriority( void PerformUpdateConnectionPriority(
ConnectionPriority connection_priority) override {} ConnectionPriority connection_priority) override {
current_connection_priority_ = connection_priority;
}
// Make On{Successful|Failed}ConnectionAttempt() public for testing. // Make On{Successful|Failed}ConnectionAttempt() public for testing.
using ConnectToDeviceOperation<std::string>::OnSuccessfulConnectionAttempt; using ConnectToDeviceOperation<std::string>::OnSuccessfulConnectionAttempt;
...@@ -76,10 +92,13 @@ class TestConnectToDeviceOperation ...@@ -76,10 +92,13 @@ class TestConnectToDeviceOperation
std::move(failure_callback), std::move(failure_callback),
device_id_pair, device_id_pair,
connection_priority, connection_priority,
test_task_runner) {} test_task_runner),
test_task_runner_(test_task_runner) {}
scoped_refptr<base::TestSimpleTaskRunner> test_task_runner_;
bool has_attempted_connection_ = false; bool has_attempted_connection_ = false;
bool has_canceled_connection_ = false; bool has_canceled_connection_ = false;
base::Optional<ConnectionPriority> current_connection_priority_;
}; };
} // namespace } // namespace
...@@ -91,7 +110,8 @@ class SecureChannelConnectToDeviceOperationBaseTest : public testing::Test { ...@@ -91,7 +110,8 @@ class SecureChannelConnectToDeviceOperationBaseTest : public testing::Test {
~SecureChannelConnectToDeviceOperationBaseTest() override = default; ~SecureChannelConnectToDeviceOperationBaseTest() override = default;
void CreateOperation(ConnectionPriority connection_priority) { void CreateOperation(ConnectionPriority connection_priority,
bool should_attempt_connection_synchronously = true) {
test_operation_ = TestConnectToDeviceOperation::Create( test_operation_ = TestConnectToDeviceOperation::Create(
base::BindOnce(&SecureChannelConnectToDeviceOperationBaseTest:: base::BindOnce(&SecureChannelConnectToDeviceOperationBaseTest::
OnSuccessfulConnectionAttempt, OnSuccessfulConnectionAttempt,
...@@ -99,8 +119,10 @@ class SecureChannelConnectToDeviceOperationBaseTest : public testing::Test { ...@@ -99,8 +119,10 @@ class SecureChannelConnectToDeviceOperationBaseTest : public testing::Test {
base::BindRepeating(&SecureChannelConnectToDeviceOperationBaseTest:: base::BindRepeating(&SecureChannelConnectToDeviceOperationBaseTest::
OnFailedConnectionAttempt, OnFailedConnectionAttempt,
base::Unretained(this)), base::Unretained(this)),
test_device_id_pair_, connection_priority); test_device_id_pair_, connection_priority,
EXPECT_TRUE(test_operation_->has_attempted_connection()); should_attempt_connection_synchronously);
EXPECT_EQ(should_attempt_connection_synchronously,
test_operation_->has_attempted_connection());
} }
TestConnectToDeviceOperation* test_operation() { TestConnectToDeviceOperation* test_operation() {
...@@ -174,6 +196,39 @@ TEST_F(SecureChannelConnectToDeviceOperationBaseTest, Cancelation) { ...@@ -174,6 +196,39 @@ TEST_F(SecureChannelConnectToDeviceOperationBaseTest, Cancelation) {
EXPECT_TRUE(test_operation()->has_canceled_connection()); EXPECT_TRUE(test_operation()->has_canceled_connection());
} }
TEST_F(SecureChannelConnectToDeviceOperationBaseTest,
UpdateConnectionPriorityBeforeAttemptStarted) {
CreateOperation(ConnectionPriority::kLow,
false /* should_attempt_connection_synchronously */);
// Update the connection priority, then run pending tasks; the pending task
// should start the attempt.
test_operation()->UpdateConnectionPriority(ConnectionPriority::kMedium);
EXPECT_FALSE(test_operation()->has_attempted_connection());
test_operation()->RunPendingTasks();
EXPECT_TRUE(test_operation()->has_attempted_connection());
EXPECT_EQ(ConnectionPriority::kMedium,
test_operation()->current_connection_priority());
test_operation()->Cancel();
}
TEST_F(SecureChannelConnectToDeviceOperationBaseTest,
CancelBeforeAttemptStarted) {
CreateOperation(ConnectionPriority::kLow,
false /* should_attempt_connection_synchronously */);
// Update the connection priority, cancel the attempt, then run pending tasks;
// the pending task should try to start the attempt but should fail since the
// task had already been canceled.
test_operation()->UpdateConnectionPriority(ConnectionPriority::kMedium);
EXPECT_FALSE(test_operation()->has_attempted_connection());
test_operation()->Cancel();
EXPECT_FALSE(test_operation()->has_attempted_connection());
test_operation()->RunPendingTasks();
EXPECT_FALSE(test_operation()->has_attempted_connection());
}
} // namespace secure_channel } // namespace secure_channel
} // namespace chromeos } // namespace chromeos
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment