[go: nahoru, domu]

blob: 740da9a469f7bccd1288355b346c13068b884776 [file] [log] [blame]
// Copyright 2019 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/download/public/common/auto_resumption_handler.h"
#include <memory>
#include "base/bind.h"
#include "base/callback.h"
#include "base/callback_helpers.h"
#include "base/guid.h"
#include "base/memory/raw_ptr.h"
#include "base/test/simple_test_clock.h"
#include "base/test/test_mock_time_task_runner.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/time/time.h"
#include "components/download/network/network_status_listener_impl.h"
#include "components/download/public/common/download_schedule.h"
#include "components/download/public/common/mock_download_item.h"
#include "components/download/public/task/mock_task_manager.h"
#include "services/network/test/test_network_connection_tracker.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
using TaskParams = download::TaskManager::TaskParams;
using network::mojom::ConnectionType;
using testing::_;
using testing::NiceMock;
using testing::Return;
using testing::ReturnRef;
using testing::ReturnRefOfCopy;
namespace download {
namespace {
const char kNow[] = "1 Sep 2020 01:00:00 GMT";
const DownloadTaskType kResumptionTaskType =
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK;
const DownloadTaskType kDownloadLaterTaskType =
DownloadTaskType::DOWNLOAD_LATER_TASK;
const int64_t kDownloadLaterTaskWindowSeconds = 15;
TaskManager::TaskParams TaskParams(int64_t window_start_time_seconds,
int64_t window_end_time_seconds,
bool require_wifi) {
TaskManager::TaskParams params;
params.window_start_time_seconds = window_start_time_seconds;
params.window_end_time_seconds = window_end_time_seconds;
params.require_unmetered_network = require_wifi;
return params;
}
base::Time GetNow() {
base::Time now;
bool success = base::Time::FromString(kNow, &now);
EXPECT_TRUE(success);
return now;
}
class AutoResumptionHandlerTest : public testing::Test {
public:
AutoResumptionHandlerTest()
: task_runner_(new base::TestMockTimeTaskRunner), handle_(task_runner_) {}
AutoResumptionHandlerTest(const AutoResumptionHandlerTest&) = delete;
AutoResumptionHandlerTest& operator=(const AutoResumptionHandlerTest&) =
delete;
~AutoResumptionHandlerTest() override = default;
protected:
void SetUp() override {
auto network_listener = std::make_unique<NetworkStatusListenerImpl>(
network::TestNetworkConnectionTracker::GetInstance());
auto task_manager = std::make_unique<download::test::MockTaskManager>();
task_manager_ = task_manager.get();
auto config = std::make_unique<AutoResumptionHandler::Config>();
config->auto_resumption_size_limit = 100;
config->is_auto_resumption_enabled_in_native = true;
clock_.SetNow(GetNow());
auto_resumption_handler_ = std::make_unique<AutoResumptionHandler>(
std::move(network_listener), std::move(task_manager), std::move(config),
&clock_);
std::vector<DownloadItem*> empty_list;
auto_resumption_handler_->SetResumableDownloads(empty_list);
task_runner_->FastForwardUntilNoTasksRemain();
}
void TearDown() override {}
void SetDownloadState(MockDownloadItem* download,
DownloadItem::DownloadState state,
bool paused,
bool allow_metered,
bool has_target_file_path = true) {
ON_CALL(*download, GetGuid())
.WillByDefault(ReturnRefOfCopy(base::GenerateGUID()));
ON_CALL(*download, GetURL())
.WillByDefault(ReturnRefOfCopy(GURL("http://example.com/foo")));
ON_CALL(*download, GetState()).WillByDefault(Return(state));
ON_CALL(*download, IsPaused()).WillByDefault(Return(paused));
ON_CALL(*download, AllowMetered()).WillByDefault(Return(allow_metered));
ON_CALL(*download, GetTargetFilePath())
.WillByDefault(ReturnRefOfCopy(
has_target_file_path ? base::FilePath(FILE_PATH_LITERAL("a.txt"))
: base::FilePath()));
auto last_reason =
state == DownloadItem::INTERRUPTED
? download::DOWNLOAD_INTERRUPT_REASON_NETWORK_DISCONNECTED
: download::DOWNLOAD_INTERRUPT_REASON_NONE;
ON_CALL(*download, GetLastReason()).WillByDefault(Return(last_reason));
ON_CALL(*download, GetDownloadSchedule())
.WillByDefault(ReturnRefOfCopy(absl::optional<DownloadSchedule>()));
// Make sure the item won't be expired and ignored.
ON_CALL(*download, GetStartTime())
.WillByDefault(Return(GetNow() - base::Days(1)));
}
void SetNetworkConnectionType(ConnectionType connection_type) {
network::TestNetworkConnectionTracker::GetInstance()->SetConnectionType(
connection_type);
}
void SetDownloadSchedule(MockDownloadItem* download,
DownloadSchedule download_schedule) {
absl::optional<DownloadSchedule> copy = download_schedule;
ON_CALL(*download, GetDownloadSchedule())
.WillByDefault(ReturnRefOfCopy(copy));
}
scoped_refptr<base::TestMockTimeTaskRunner> task_runner_;
base::ThreadTaskRunnerHandle handle_;
raw_ptr<download::test::MockTaskManager> task_manager_;
std::unique_ptr<AutoResumptionHandler> auto_resumption_handler_;
base::SimpleTestClock clock_;
};
TEST_F(AutoResumptionHandlerTest, ScheduleTaskCalledOnDownloadStart) {
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
EXPECT_CALL(*task_manager_, ScheduleTask(_, _)).Times(1);
SetDownloadState(item.get(), DownloadItem::IN_PROGRESS, false, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, TaskFinishedCalledOnDownloadCompletion) {
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
task_runner_->FastForwardUntilNoTasksRemain();
EXPECT_CALL(*task_manager_, ScheduleTask(_, _)).Times(1);
SetDownloadState(item.get(), DownloadItem::IN_PROGRESS, false, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Complete the download.
EXPECT_CALL(*task_manager_, NotifyTaskFinished(kResumptionTaskType, _));
EXPECT_CALL(*task_manager_,
NotifyTaskFinished(kDownloadLaterTaskType, false));
EXPECT_CALL(*task_manager_, UnscheduleTask(kResumptionTaskType));
EXPECT_CALL(*task_manager_, UnscheduleTask(kDownloadLaterTaskType));
SetDownloadState(item.get(), DownloadItem::COMPLETE, false, false);
auto_resumption_handler_->OnDownloadUpdated(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, TaskFinishedCalledOnDownloadRemoved) {
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
task_runner_->FastForwardUntilNoTasksRemain();
EXPECT_CALL(*task_manager_, ScheduleTask(_, _)).Times(1);
SetDownloadState(item.get(), DownloadItem::IN_PROGRESS, false, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Remove the download.
EXPECT_CALL(*task_manager_, NotifyTaskFinished(kResumptionTaskType, _));
EXPECT_CALL(*task_manager_,
NotifyTaskFinished(kDownloadLaterTaskType, false));
SetDownloadState(item.get(), DownloadItem::COMPLETE, false, false);
auto_resumption_handler_->OnDownloadRemoved(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, MultipleDownloads) {
// Start two downloads.
auto item1 = std::make_unique<NiceMock<MockDownloadItem>>();
auto item2 = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item1.get(), DownloadItem::INTERRUPTED, false, false);
SetDownloadState(item2.get(), DownloadItem::INTERRUPTED, false, false);
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
task_runner_->FastForwardUntilNoTasksRemain();
EXPECT_CALL(*task_manager_, ScheduleTask(_, _)).Times(1);
auto_resumption_handler_->OnDownloadStarted(item1.get());
auto_resumption_handler_->OnDownloadStarted(item2.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Finish item1. The task should still be running.
EXPECT_CALL(*task_manager_, UnscheduleTask(kResumptionTaskType)).Times(0);
EXPECT_CALL(*task_manager_, UnscheduleTask(kDownloadLaterTaskType));
EXPECT_CALL(*task_manager_, NotifyTaskFinished(_, _)).Times(0);
SetDownloadState(item1.get(), DownloadItem::CANCELLED, false, false);
auto_resumption_handler_->OnDownloadUpdated(item1.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Finish item2. The task should now complete.
EXPECT_CALL(*task_manager_, UnscheduleTask(kResumptionTaskType));
EXPECT_CALL(*task_manager_, UnscheduleTask(kDownloadLaterTaskType));
EXPECT_CALL(*task_manager_, NotifyTaskFinished(kResumptionTaskType, _));
EXPECT_CALL(*task_manager_,
NotifyTaskFinished(kDownloadLaterTaskType, false));
SetDownloadState(item2.get(), DownloadItem::COMPLETE, false, false);
auto_resumption_handler_->OnDownloadUpdated(item2.get());
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, DownloadResumesCorrectlyOnNetworkChange) {
// Create two downloads: item1 (unmetered), item2 (metered).
auto item1 = std::make_unique<NiceMock<MockDownloadItem>>();
auto item2 = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item1.get(), DownloadItem::INTERRUPTED, false, false);
SetDownloadState(item2.get(), DownloadItem::INTERRUPTED, false, true);
auto_resumption_handler_->OnDownloadStarted(item1.get());
auto_resumption_handler_->OnDownloadStarted(item2.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Start with disconnected network.
SetNetworkConnectionType(ConnectionType::CONNECTION_NONE);
task_runner_->FastForwardUntilNoTasksRemain();
// Connect to Wifi.
EXPECT_CALL(*item1.get(), Resume(_)).Times(1);
EXPECT_CALL(*item2.get(), Resume(_)).Times(1);
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
task_runner_->FastForwardUntilNoTasksRemain();
// Disconnect network again.
EXPECT_CALL(*item1.get(), Resume(_)).Times(0);
EXPECT_CALL(*item2.get(), Resume(_)).Times(0);
SetNetworkConnectionType(ConnectionType::CONNECTION_NONE);
task_runner_->FastForwardUntilNoTasksRemain();
// Change network to metered.
EXPECT_CALL(*item1.get(), Resume(_)).Times(0);
EXPECT_CALL(*item2.get(), Resume(_)).Times(1);
SetNetworkConnectionType(ConnectionType::CONNECTION_3G);
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, PausedDownloadsAreNotAutoResumed) {
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item.get(), DownloadItem::IN_PROGRESS, true, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
SetNetworkConnectionType(ConnectionType::CONNECTION_NONE);
task_runner_->FastForwardUntilNoTasksRemain();
// Connect to Wifi.
EXPECT_CALL(*item.get(), Resume(_)).Times(0);
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest,
OnStartScheduledTaskWillResumeAllPendingDownloads) {
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item.get(), DownloadItem::INTERRUPTED, false, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Start the task. It should resume all downloads.
EXPECT_CALL(*item.get(), Resume(_)).Times(1);
TaskFinishedCallback callback;
auto_resumption_handler_->OnStartScheduledTask(
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK, std::move(callback));
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, ExpiredDownloadNotAutoResumed) {
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
// Create a normal expired download.
base::Time expired_start_time = GetNow() - base::Days(100);
auto item0 = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item0.get(), DownloadItem::INTERRUPTED, false, false);
ON_CALL(*item0.get(), GetStartTime())
.WillByDefault(Return(expired_start_time));
// Create an expired user scheduled download.
auto item1 = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item1.get(), DownloadItem::INTERRUPTED, false, false);
SetDownloadSchedule(item1.get(),
DownloadSchedule(true /*only_on_wifi*/, absl::nullopt));
ON_CALL(*item1.get(), GetStartTime())
.WillByDefault(Return(expired_start_time));
auto_resumption_handler_->OnDownloadStarted(item0.get());
auto_resumption_handler_->OnDownloadStarted(item1.get());
task_runner_->FastForwardUntilNoTasksRemain();
// Expired downoad |item0| won't be resumed. Expired user scheduled download
// |item1| will still be resumed.
EXPECT_CALL(*item0.get(), Resume(_)).Times(0);
EXPECT_CALL(*item1.get(), Resume(_)).Times(1);
TaskFinishedCallback callback;
auto_resumption_handler_->OnStartScheduledTask(
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK, std::move(callback));
task_runner_->FastForwardUntilNoTasksRemain();
}
TEST_F(AutoResumptionHandlerTest, DownloadWithoutTargetPathNotAutoResumed) {
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item.get(), DownloadItem::INTERRUPTED, false, false, false);
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
EXPECT_CALL(*item.get(), Resume(_)).Times(0);
auto_resumption_handler_->OnStartScheduledTask(
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK, base::DoNothing());
task_runner_->FastForwardUntilNoTasksRemain();
}
// Download scheduled to start in the future should not be auto resumed now.
TEST_F(AutoResumptionHandlerTest, DownloadLaterStartFutureNotAutoResumed) {
SetNetworkConnectionType(ConnectionType::CONNECTION_WIFI);
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
auto delta = base::Days(10);
base::Time future_time = GetNow() + delta;
SetDownloadState(item.get(), DownloadItem::INTERRUPTED, false, false);
SetDownloadSchedule(item.get(),
DownloadSchedule(false /*only_on_wifi*/, future_time));
int64_t window_start = delta.InSeconds();
auto task_params = TaskParams(
window_start, window_start + kDownloadLaterTaskWindowSeconds, false);
EXPECT_CALL(*item.get(), Resume(_)).Times(0);
EXPECT_CALL(*task_manager_,
ScheduleTask(kDownloadLaterTaskType, task_params));
EXPECT_CALL(*task_manager_, ScheduleTask(kResumptionTaskType, _)).Times(0);
EXPECT_CALL(*task_manager_, UnscheduleTask(kResumptionTaskType));
EXPECT_CALL(*task_manager_, UnscheduleTask(kDownloadLaterTaskType));
EXPECT_CALL(*task_manager_, NotifyTaskFinished(kDownloadLaterTaskType, _));
EXPECT_CALL(*task_manager_, NotifyTaskFinished(kResumptionTaskType, _));
auto_resumption_handler_->OnDownloadStarted(item.get());
auto_resumption_handler_->OnStartScheduledTask(
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK, base::DoNothing());
task_runner_->FastForwardUntilNoTasksRemain();
}
// Use DownloadItem::AllowMetered() instead of DownloadSchedule::only_on_wifi()
// to determine network condition for download later.
TEST_F(AutoResumptionHandlerTest, DownloadLaterMeteredAutoResumed) {
SetNetworkConnectionType(ConnectionType::CONNECTION_3G);
auto item = std::make_unique<NiceMock<MockDownloadItem>>();
SetDownloadState(item.get(), DownloadItem::INTERRUPTED, false,
true /*allow_metered*/);
SetDownloadSchedule(item.get(),
DownloadSchedule(true /*only_on_wifi*/, absl::nullopt));
auto_resumption_handler_->OnDownloadStarted(item.get());
task_runner_->FastForwardUntilNoTasksRemain();
EXPECT_CALL(*item.get(), Resume(_));
auto_resumption_handler_->OnStartScheduledTask(
DownloadTaskType::DOWNLOAD_AUTO_RESUMPTION_TASK, base::DoNothing());
task_runner_->FastForwardUntilNoTasksRemain();
}
} // namespace
} // namespace download