// Copyright (c) 2012 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 "content/browser/download/download_manager_impl.h" #include #include #include #include #include #include "base/bind.h" #include "base/files/scoped_temp_dir.h" #include "base/guid.h" #include "base/macros.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/message_loop/message_loop.h" #include "base/stl_util.h" #include "base/strings/string16.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "build/build_config.h" #include "content/browser/byte_stream.h" #include "content/browser/download/download_create_info.h" #include "content/browser/download/download_file_factory.h" #include "content/browser/download/download_item_factory.h" #include "content/browser/download/download_item_impl.h" #include "content/browser/download/download_item_impl_delegate.h" #include "content/browser/download/download_request_handle.h" #include "content/browser/download/mock_download_file.h" #include "content/public/browser/browser_context.h" #include "content/public/browser/download_interrupt_reasons.h" #include "content/public/browser/download_item.h" #include "content/public/browser/download_manager_delegate.h" #include "content/public/browser/zoom_level_delegate.h" #include "content/public/test/mock_download_item.h" #include "content/public/test/test_browser_context.h" #include "content/public/test/test_browser_thread.h" #include "net/log/net_log.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gmock_mutant.h" #include "testing/gtest/include/gtest/gtest.h" #include "url/origin.h" using ::testing::AllOf; using ::testing::DoAll; using ::testing::Eq; using ::testing::Ref; using ::testing::Return; using ::testing::ReturnRef; using ::testing::ReturnRefOfCopy; using ::testing::SetArgPointee; using ::testing::StrictMock; using ::testing::_; ACTION_TEMPLATE(RunCallback, HAS_1_TEMPLATE_PARAMS(int, k), AND_1_VALUE_PARAMS(p0)) { return ::std::tr1::get(args).Run(p0); } namespace content { class ByteStreamReader; namespace { // Matches a DownloadCreateInfo* that points to the same object as |info| and // has a |default_download_directory| that matches |download_directory|. MATCHER_P2(DownloadCreateInfoWithDefaultPath, info, download_directory, "") { return arg == info && arg->default_download_directory == download_directory; } class MockDownloadItemImpl : public DownloadItemImpl { public: // Use history constructor for minimal base object. explicit MockDownloadItemImpl(DownloadItemImplDelegate* delegate) : DownloadItemImpl(delegate, std::string("7d122682-55b5-4a47-a253-36cadc3e5bee"), content::DownloadItem::kInvalidId, base::FilePath(), base::FilePath(), std::vector(), GURL(), "application/octet-stream", "application/octet-stream", base::Time(), base::Time(), std::string(), std::string(), 0, 0, std::string(), DownloadItem::COMPLETE, DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, DOWNLOAD_INTERRUPT_REASON_NONE, false, net::BoundNetLog()) {} virtual ~MockDownloadItemImpl() {} MOCK_METHOD4(OnDownloadTargetDetermined, void(const base::FilePath&, TargetDisposition, DownloadDangerType, const base::FilePath&)); MOCK_METHOD1(AddObserver, void(DownloadItem::Observer*)); MOCK_METHOD1(RemoveObserver, void(DownloadItem::Observer*)); MOCK_METHOD0(UpdateObservers, void()); MOCK_METHOD0(CanShowInFolder, bool()); MOCK_METHOD0(CanOpenDownload, bool()); MOCK_METHOD0(ShouldOpenFileBasedOnExtension, bool()); MOCK_METHOD0(OpenDownload, void()); MOCK_METHOD0(ShowDownloadInShell, void()); MOCK_METHOD0(ValidateDangerousDownload, void()); MOCK_METHOD1(StealDangerousDownload, void(const AcquireFileCallback&)); MOCK_METHOD3(UpdateProgress, void(int64_t, int64_t, const std::string&)); MOCK_METHOD1(Cancel, void(bool)); MOCK_METHOD0(MarkAsComplete, void()); void OnAllDataSaved(int64_t, scoped_ptr) override { NOTREACHED(); } MOCK_METHOD0(OnDownloadedFileRemoved, void()); void Start(scoped_ptr download_file, scoped_ptr req_handle, const DownloadCreateInfo& create_info) override { MockStart(download_file.get(), req_handle.get()); } MOCK_METHOD2(MockStart, void(DownloadFile*, DownloadRequestHandleInterface*)); MOCK_METHOD0(Remove, void()); MOCK_CONST_METHOD1(TimeRemaining, bool(base::TimeDelta*)); MOCK_CONST_METHOD0(CurrentSpeed, int64_t()); MOCK_CONST_METHOD0(PercentComplete, int()); MOCK_CONST_METHOD0(AllDataSaved, bool()); MOCK_CONST_METHOD1(MatchesQuery, bool(const base::string16& query)); MOCK_CONST_METHOD0(IsDone, bool()); MOCK_CONST_METHOD0(GetFullPath, const base::FilePath&()); MOCK_CONST_METHOD0(GetTargetFilePath, const base::FilePath&()); MOCK_CONST_METHOD0(GetTargetDisposition, TargetDisposition()); MOCK_METHOD1(OnContentCheckCompleted, void(DownloadDangerType)); MOCK_CONST_METHOD0(GetState, DownloadState()); MOCK_CONST_METHOD0(GetUrlChain, const std::vector&()); MOCK_METHOD1(SetTotalBytes, void(int64_t)); MOCK_CONST_METHOD0(GetURL, const GURL&()); MOCK_CONST_METHOD0(GetOriginalUrl, const GURL&()); MOCK_CONST_METHOD0(GetReferrerUrl, const GURL&()); MOCK_CONST_METHOD0(GetTabUrl, const GURL&()); MOCK_CONST_METHOD0(GetTabReferrerUrl, const GURL&()); MOCK_CONST_METHOD0(GetSuggestedFilename, std::string()); MOCK_CONST_METHOD0(GetContentDisposition, std::string()); MOCK_CONST_METHOD0(GetMimeType, std::string()); MOCK_CONST_METHOD0(GetOriginalMimeType, std::string()); MOCK_CONST_METHOD0(GetReferrerCharset, std::string()); MOCK_CONST_METHOD0(GetRemoteAddress, std::string()); MOCK_CONST_METHOD0(GetTotalBytes, int64_t()); MOCK_CONST_METHOD0(GetReceivedBytes, int64_t()); MOCK_CONST_METHOD0(GetHashState, const std::string&()); MOCK_CONST_METHOD0(GetHash, const std::string&()); MOCK_CONST_METHOD0(GetId, uint32_t()); MOCK_CONST_METHOD0(GetGuid, const std::string&()); MOCK_CONST_METHOD0(GetStartTime, base::Time()); MOCK_CONST_METHOD0(GetEndTime, base::Time()); MOCK_METHOD0(GetDownloadManager, DownloadManager*()); MOCK_CONST_METHOD0(IsPaused, bool()); MOCK_CONST_METHOD0(GetOpenWhenComplete, bool()); MOCK_METHOD1(SetOpenWhenComplete, void(bool)); MOCK_CONST_METHOD0(GetFileExternallyRemoved, bool()); MOCK_CONST_METHOD0(GetDangerType, DownloadDangerType()); MOCK_CONST_METHOD0(IsDangerous, bool()); MOCK_METHOD0(GetAutoOpened, bool()); MOCK_CONST_METHOD0(GetForcedFilePath, const base::FilePath&()); MOCK_CONST_METHOD0(HasUserGesture, bool()); MOCK_CONST_METHOD0(GetTransitionType, ui::PageTransition()); MOCK_CONST_METHOD0(IsTemporary, bool()); MOCK_METHOD1(SetIsTemporary, void(bool)); MOCK_METHOD1(SetOpened, void(bool)); MOCK_CONST_METHOD0(GetOpened, bool()); MOCK_CONST_METHOD0(GetLastModifiedTime, const std::string&()); MOCK_CONST_METHOD0(GetETag, const std::string&()); MOCK_CONST_METHOD0(GetLastReason, DownloadInterruptReason()); MOCK_CONST_METHOD0(GetBrowserContext, BrowserContext*()); MOCK_CONST_METHOD0(GetWebContents, WebContents*()); MOCK_CONST_METHOD0(GetFileNameToReportUser, base::FilePath()); MOCK_METHOD1(SetDisplayName, void(const base::FilePath&)); MOCK_METHOD0(NotifyRemoved, void()); // May be called when vlog is on. std::string DebugString(bool verbose) const override { return std::string(); } }; class MockDownloadManagerDelegate : public DownloadManagerDelegate { public: MockDownloadManagerDelegate(); virtual ~MockDownloadManagerDelegate(); MOCK_METHOD0(Shutdown, void()); MOCK_METHOD1(GetNextId, void(const DownloadIdCallback&)); MOCK_METHOD2(DetermineDownloadTarget, bool(DownloadItem* item, const DownloadTargetCallback&)); MOCK_METHOD1(ShouldOpenFileBasedOnExtension, bool(const base::FilePath&)); MOCK_METHOD2(ShouldCompleteDownload, bool(DownloadItem*, const base::Closure&)); MOCK_METHOD2(ShouldOpenDownload, bool(DownloadItem*, const DownloadOpenDelayedCallback&)); MOCK_METHOD4(GetSaveDir, void(BrowserContext*, base::FilePath*, base::FilePath*, bool*)); MOCK_METHOD5(ChooseSavePath, void( WebContents*, const base::FilePath&, const base::FilePath::StringType&, bool, const SavePackagePathPickedCallback&)); MOCK_CONST_METHOD0(ApplicationClientIdForFileScanning, std::string()); }; MockDownloadManagerDelegate::MockDownloadManagerDelegate() {} MockDownloadManagerDelegate::~MockDownloadManagerDelegate() {} class MockDownloadItemFactory : public DownloadItemFactory, public base::SupportsWeakPtr { public: MockDownloadItemFactory(); ~MockDownloadItemFactory() override; // Access to map of created items. // TODO(rdsmith): Could add type (save page, persisted, etc.) // functionality if it's ever needed by consumers. // Returns NULL if no item of that id is present. MockDownloadItemImpl* GetItem(int id); // Remove and return an item made by the factory. // Generally used during teardown. MockDownloadItemImpl* PopItem(); // Should be called when the item of this id is removed so that // we don't keep dangling pointers. void RemoveItem(int id); // Overridden methods from DownloadItemFactory. DownloadItemImpl* CreatePersistedItem( DownloadItemImplDelegate* delegate, const std::string& guid, uint32_t download_id, const base::FilePath& current_path, const base::FilePath& target_path, const std::vector& url_chain, const GURL& referrer_url, const std::string& mime_type, const std::string& original_mime_type, const base::Time& start_time, const base::Time& end_time, const std::string& etag, const std::string& last_modofied, int64_t received_bytes, int64_t total_bytes, const std::string& hash, DownloadItem::DownloadState state, DownloadDangerType danger_type, DownloadInterruptReason interrupt_reason, bool opened, const net::BoundNetLog& bound_net_log) override; DownloadItemImpl* CreateActiveItem( DownloadItemImplDelegate* delegate, uint32_t download_id, const DownloadCreateInfo& info, const net::BoundNetLog& bound_net_log) override; DownloadItemImpl* CreateSavePageItem( DownloadItemImplDelegate* delegate, uint32_t download_id, const base::FilePath& path, const GURL& url, const std::string& mime_type, scoped_ptr request_handle, const net::BoundNetLog& bound_net_log) override; private: std::map items_; DownloadItemImplDelegate item_delegate_; DISALLOW_COPY_AND_ASSIGN(MockDownloadItemFactory); }; MockDownloadItemFactory::MockDownloadItemFactory() {} MockDownloadItemFactory::~MockDownloadItemFactory() {} MockDownloadItemImpl* MockDownloadItemFactory::GetItem(int id) { if (items_.find(id) == items_.end()) return NULL; return items_[id]; } MockDownloadItemImpl* MockDownloadItemFactory::PopItem() { if (items_.empty()) return NULL; std::map::iterator first_item = items_.begin(); MockDownloadItemImpl* result = first_item->second; items_.erase(first_item); return result; } void MockDownloadItemFactory::RemoveItem(int id) { DCHECK(items_.find(id) != items_.end()); items_.erase(id); } DownloadItemImpl* MockDownloadItemFactory::CreatePersistedItem( DownloadItemImplDelegate* delegate, const std::string& guid, uint32_t download_id, const base::FilePath& current_path, const base::FilePath& target_path, const std::vector& url_chain, const GURL& referrer_url, const std::string& mime_type, const std::string& original_mime_type, const base::Time& start_time, const base::Time& end_time, const std::string& etag, const std::string& last_modified, int64_t received_bytes, int64_t total_bytes, const std::string& hash, DownloadItem::DownloadState state, DownloadDangerType danger_type, DownloadInterruptReason interrupt_reason, bool opened, const net::BoundNetLog& bound_net_log) { DCHECK(items_.find(download_id) == items_.end()); MockDownloadItemImpl* result = new StrictMock(&item_delegate_); EXPECT_CALL(*result, GetId()) .WillRepeatedly(Return(download_id)); EXPECT_CALL(*result, GetGuid()).WillRepeatedly(ReturnRefOfCopy(guid)); items_[download_id] = result; return result; } DownloadItemImpl* MockDownloadItemFactory::CreateActiveItem( DownloadItemImplDelegate* delegate, uint32_t download_id, const DownloadCreateInfo& info, const net::BoundNetLog& bound_net_log) { DCHECK(items_.find(download_id) == items_.end()); MockDownloadItemImpl* result = new StrictMock(&item_delegate_); EXPECT_CALL(*result, GetId()) .WillRepeatedly(Return(download_id)); EXPECT_CALL(*result, GetGuid()) .WillRepeatedly( ReturnRefOfCopy(base::ToUpperASCII(base::GenerateGUID()))); items_[download_id] = result; // Active items are created and then immediately are called to start // the download. EXPECT_CALL(*result, MockStart(_, _)); return result; } DownloadItemImpl* MockDownloadItemFactory::CreateSavePageItem( DownloadItemImplDelegate* delegate, uint32_t download_id, const base::FilePath& path, const GURL& url, const std::string& mime_type, scoped_ptr request_handle, const net::BoundNetLog& bound_net_log) { DCHECK(items_.find(download_id) == items_.end()); MockDownloadItemImpl* result = new StrictMock(&item_delegate_); EXPECT_CALL(*result, GetId()) .WillRepeatedly(Return(download_id)); items_[download_id] = result; return result; } class MockDownloadFileFactory : public DownloadFileFactory, public base::SupportsWeakPtr { public: MockDownloadFileFactory() {} virtual ~MockDownloadFileFactory() {} // Overridden method from DownloadFileFactory MOCK_METHOD2(MockCreateFile, MockDownloadFile*(const DownloadSaveInfo&, ByteStreamReader*)); virtual DownloadFile* CreateFile( scoped_ptr save_info, const base::FilePath& default_download_directory, scoped_ptr byte_stream, const net::BoundNetLog& bound_net_log, base::WeakPtr observer) override { return MockCreateFile(*save_info, byte_stream.get()); } }; class MockBrowserContext : public BrowserContext { public: MockBrowserContext() { content::BrowserContext::Initialize(this, base::FilePath()); } ~MockBrowserContext() {} MOCK_CONST_METHOD0(GetPath, base::FilePath()); MOCK_METHOD1(CreateZoomLevelDelegateMock, ZoomLevelDelegate*(const base::FilePath&)); MOCK_CONST_METHOD0(IsOffTheRecord, bool()); MOCK_METHOD0(GetRequestContext, net::URLRequestContextGetter*()); MOCK_METHOD1(GetRequestContextForRenderProcess, net::URLRequestContextGetter*(int renderer_child_id)); MOCK_METHOD0(GetMediaRequestContext, net::URLRequestContextGetter*()); MOCK_METHOD1(GetMediaRequestContextForRenderProcess, net::URLRequestContextGetter*(int renderer_child_id)); MOCK_METHOD2(GetMediaRequestContextForStoragePartition, net::URLRequestContextGetter*( const base::FilePath& partition_path, bool in_memory)); MOCK_METHOD0(GetResourceContext, ResourceContext*()); MOCK_METHOD0(GetDownloadManagerDelegate, DownloadManagerDelegate*()); MOCK_METHOD0(GetGuestManager, BrowserPluginGuestManager* ()); MOCK_METHOD0(GetSpecialStoragePolicy, storage::SpecialStoragePolicy*()); MOCK_METHOD0(GetPushMessagingService, PushMessagingService*()); MOCK_METHOD0(GetSSLHostStateDelegate, SSLHostStateDelegate*()); MOCK_METHOD0(GetPermissionManager, PermissionManager*()); MOCK_METHOD0(GetBackgroundSyncController, BackgroundSyncController*()); scoped_ptr CreateZoomLevelDelegate( const base::FilePath& path) override { return scoped_ptr(CreateZoomLevelDelegateMock(path)); } }; class MockDownloadManagerObserver : public DownloadManager::Observer { public: MockDownloadManagerObserver() {} ~MockDownloadManagerObserver() {} MOCK_METHOD2(OnDownloadCreated, void( DownloadManager*, DownloadItem*)); MOCK_METHOD1(ManagerGoingDown, void(DownloadManager*)); MOCK_METHOD2(SelectFileDialogDisplayed, void(DownloadManager*, int32_t)); }; class MockByteStreamReader : public ByteStreamReader { public: virtual ~MockByteStreamReader() {} MOCK_METHOD2(Read, StreamState(scoped_refptr*, size_t*)); MOCK_CONST_METHOD0(GetStatus, int()); MOCK_METHOD1(RegisterCallback, void(const base::Closure&)); }; } // namespace class DownloadManagerTest : public testing::Test { public: static const char* kTestData; static const size_t kTestDataLen; DownloadManagerTest() : callback_called_(false), ui_thread_(BrowserThread::UI, &message_loop_), file_thread_(BrowserThread::FILE, &message_loop_), next_download_id_(0) { } // We tear down everything in TearDown(). ~DownloadManagerTest() override {} // Create a MockDownloadItemFactory and MockDownloadManagerDelegate, // then create a DownloadManager that points // at all of those. void SetUp() override { DCHECK(!download_manager_); mock_download_item_factory_ = (new MockDownloadItemFactory())->AsWeakPtr(); mock_download_file_factory_ = (new MockDownloadFileFactory())->AsWeakPtr(); mock_download_manager_delegate_.reset( new StrictMock); EXPECT_CALL(*mock_download_manager_delegate_.get(), Shutdown()) .WillOnce(Return()); mock_browser_context_.reset(new StrictMock); EXPECT_CALL(*mock_browser_context_.get(), IsOffTheRecord()) .WillRepeatedly(Return(false)); download_manager_.reset(new DownloadManagerImpl( NULL, mock_browser_context_.get())); download_manager_->SetDownloadItemFactoryForTesting( scoped_ptr(mock_download_item_factory_.get())); download_manager_->SetDownloadFileFactoryForTesting( scoped_ptr(mock_download_file_factory_.get())); observer_.reset(new MockDownloadManagerObserver()); download_manager_->AddObserver(observer_.get()); download_manager_->SetDelegate(mock_download_manager_delegate_.get()); download_urls_.push_back(GURL("http://www.url1.com")); download_urls_.push_back(GURL("http://www.url2.com")); download_urls_.push_back(GURL("http://www.url3.com")); download_urls_.push_back(GURL("http://www.url4.com")); } void TearDown() override { while (MockDownloadItemImpl* item = mock_download_item_factory_->PopItem()) { EXPECT_CALL(*item, GetState()) .WillOnce(Return(DownloadItem::CANCELLED)); } EXPECT_CALL(GetMockObserver(), ManagerGoingDown(download_manager_.get())) .WillOnce(Return()); download_manager_->Shutdown(); download_manager_.reset(); message_loop_.RunUntilIdle(); ASSERT_EQ(NULL, mock_download_item_factory_.get()); ASSERT_EQ(NULL, mock_download_file_factory_.get()); message_loop_.RunUntilIdle(); mock_download_manager_delegate_.reset(); mock_browser_context_.reset(); download_urls_.clear(); } // Returns download id. MockDownloadItemImpl& AddItemToManager() { DownloadCreateInfo info; // Args are ignored except for download id, so everything else can be // null. uint32_t id = next_download_id_; ++next_download_id_; info.request_handle.reset(new DownloadRequestHandle); download_manager_->CreateActiveItem(id, info); DCHECK(mock_download_item_factory_->GetItem(id)); MockDownloadItemImpl& item(*mock_download_item_factory_->GetItem(id)); // Satisfy expectation. If the item is created in StartDownload(), // we call Start on it immediately, so we need to set that expectation // in the factory. scoped_ptr req_handle; item.Start(scoped_ptr(), std::move(req_handle), info); DCHECK(id < download_urls_.size()); EXPECT_CALL(item, GetURL()).WillRepeatedly(ReturnRef(download_urls_[id])); return item; } MockDownloadItemImpl& GetMockDownloadItem(int id) { MockDownloadItemImpl* itemp = mock_download_item_factory_->GetItem(id); DCHECK(itemp); return *itemp; } void RemoveMockDownloadItem(int id) { // Owned by DownloadManager; should be deleted there. mock_download_item_factory_->RemoveItem(id); } MockDownloadManagerDelegate& GetMockDownloadManagerDelegate() { return *mock_download_manager_delegate_; } MockDownloadManagerObserver& GetMockObserver() { return *observer_; } void DownloadTargetDeterminedCallback( const base::FilePath& target_path, DownloadItem::TargetDisposition disposition, DownloadDangerType danger_type, const base::FilePath& intermediate_path) { callback_called_ = true; target_path_ = target_path; target_disposition_ = disposition; danger_type_ = danger_type; intermediate_path_ = intermediate_path; } void DetermineDownloadTarget(DownloadItemImpl* item) { download_manager_->DetermineDownloadTarget( item, base::Bind( &DownloadManagerTest::DownloadTargetDeterminedCallback, base::Unretained(this))); } protected: // Key test variable; we'll keep it available to sub-classes. scoped_ptr download_manager_; base::WeakPtr mock_download_file_factory_; // Target detetermined callback. bool callback_called_; base::FilePath target_path_; DownloadItem::TargetDisposition target_disposition_; DownloadDangerType danger_type_; base::FilePath intermediate_path_; std::vector download_urls_; private: base::MessageLoopForUI message_loop_; TestBrowserThread ui_thread_; TestBrowserThread file_thread_; base::WeakPtr mock_download_item_factory_; scoped_ptr mock_download_manager_delegate_; scoped_ptr mock_browser_context_; scoped_ptr observer_; uint32_t next_download_id_; DISALLOW_COPY_AND_ASSIGN(DownloadManagerTest); }; // Confirm the appropriate invocations occur when you start a download. TEST_F(DownloadManagerTest, StartDownload) { scoped_ptr info(new DownloadCreateInfo); scoped_ptr stream(new MockByteStreamReader); uint32_t local_id(5); // Random value base::FilePath download_path(FILE_PATH_LITERAL("download/path")); EXPECT_FALSE(download_manager_->GetDownload(local_id)); EXPECT_CALL(GetMockObserver(), OnDownloadCreated(download_manager_.get(), _)) .WillOnce(Return()); EXPECT_CALL(GetMockDownloadManagerDelegate(), GetNextId(_)) .WillOnce(RunCallback<0>(local_id)); // Doing nothing will set the default download directory to null. EXPECT_CALL(GetMockDownloadManagerDelegate(), GetSaveDir(_, _, _, _)); EXPECT_CALL(GetMockDownloadManagerDelegate(), ApplicationClientIdForFileScanning()) .WillRepeatedly(Return("client-id")); MockDownloadFile* mock_file = new MockDownloadFile; EXPECT_CALL(*mock_download_file_factory_.get(), MockCreateFile(Ref(*info->save_info.get()), stream.get())) .WillOnce(Return(mock_file)); download_manager_->StartDownload(std::move(info), std::move(stream), DownloadUrlParameters::OnStartedCallback()); EXPECT_TRUE(download_manager_->GetDownload(local_id)); } // Confirm that calling DetermineDownloadTarget behaves properly if the delegate // blocks starting. TEST_F(DownloadManagerTest, DetermineDownloadTarget_True) { // Put a mock we have a handle to on the download manager. MockDownloadItemImpl& item(AddItemToManager()); EXPECT_CALL(item, GetState()) .WillRepeatedly(Return(DownloadItem::IN_PROGRESS)); EXPECT_CALL(GetMockDownloadManagerDelegate(), DetermineDownloadTarget(&item, _)) .WillOnce(Return(true)); DetermineDownloadTarget(&item); } // Confirm that calling DetermineDownloadTarget behaves properly if the delegate // allows starting. This also tests OnDownloadTargetDetermined. TEST_F(DownloadManagerTest, DetermineDownloadTarget_False) { // Put a mock we have a handle to on the download manager. MockDownloadItemImpl& item(AddItemToManager()); base::FilePath path(FILE_PATH_LITERAL("random_filepath.txt")); EXPECT_CALL(GetMockDownloadManagerDelegate(), DetermineDownloadTarget(&item, _)) .WillOnce(Return(false)); EXPECT_CALL(item, GetForcedFilePath()) .WillOnce(ReturnRef(path)); // Confirm that the callback was called with the right values in this case. callback_called_ = false; DetermineDownloadTarget(&item); EXPECT_TRUE(callback_called_); EXPECT_EQ(path, target_path_); EXPECT_EQ(DownloadItem::TARGET_DISPOSITION_OVERWRITE, target_disposition_); EXPECT_EQ(DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, danger_type_); EXPECT_EQ(path, intermediate_path_); } // Confirm the DownloadManagerImpl::RemoveAllDownloads() functionality TEST_F(DownloadManagerTest, RemoveAllDownloads) { base::Time now(base::Time::Now()); for (uint32_t i = 0; i < 4; ++i) { MockDownloadItemImpl& item(AddItemToManager()); EXPECT_EQ(i, item.GetId()); EXPECT_CALL(item, GetStartTime()) .WillRepeatedly(Return(now)); } // Specify states for each. EXPECT_CALL(GetMockDownloadItem(0), GetState()) .WillRepeatedly(Return(DownloadItem::COMPLETE)); EXPECT_CALL(GetMockDownloadItem(1), GetState()) .WillRepeatedly(Return(DownloadItem::CANCELLED)); EXPECT_CALL(GetMockDownloadItem(2), GetState()) .WillRepeatedly(Return(DownloadItem::INTERRUPTED)); EXPECT_CALL(GetMockDownloadItem(3), GetState()) .WillRepeatedly(Return(DownloadItem::IN_PROGRESS)); // Expectations for whether or not they'll actually be removed. EXPECT_CALL(GetMockDownloadItem(0), Remove()) .WillOnce(Return()); EXPECT_CALL(GetMockDownloadItem(1), Remove()) .WillOnce(Return()); EXPECT_CALL(GetMockDownloadItem(2), Remove()) .WillOnce(Return()); EXPECT_CALL(GetMockDownloadItem(3), Remove()) .Times(0); download_manager_->RemoveAllDownloads(); // Because we're mocking the download item, the Remove call doesn't // result in them being removed from the DownloadManager list. } TEST_F(DownloadManagerTest, GetDownloadByGuid) { for (uint32_t i = 0; i < 4; ++i) AddItemToManager(); MockDownloadItemImpl& item = GetMockDownloadItem(0); DownloadItem* result = download_manager_->GetDownloadByGuid(item.GetGuid()); ASSERT_TRUE(result); ASSERT_EQ(static_cast(&item), result); ASSERT_FALSE(download_manager_->GetDownloadByGuid("")); const char kGuid[] = "8DF158E8-C980-4618-BB03-EBA3242EB48B"; DownloadItem* persisted_item = download_manager_->CreateDownloadItem( kGuid, 10, base::FilePath(), base::FilePath(), std::vector(), GURL("http://example.com/a"), "application/octet-stream", "application/octet-stream", base::Time::Now(), base::Time::Now(), std::string(), std::string(), 10, 10, std::string(), DownloadItem::INTERRUPTED, DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, DOWNLOAD_INTERRUPT_REASON_SERVER_FAILED, false); ASSERT_TRUE(persisted_item); ASSERT_EQ(persisted_item, download_manager_->GetDownloadByGuid(kGuid)); } namespace { base::Callback GetSingleURLFilter(const GURL& url) { return base::Bind(&GURL::operator==, base::Owned(new GURL(url))); } } // namespace // Confirm that only downloads with the specified URL are removed. TEST_F(DownloadManagerTest, RemoveDownloadsByURL) { base::Time now(base::Time::Now()); for (uint32_t i = 0; i < 2; ++i) { MockDownloadItemImpl& item(AddItemToManager()); EXPECT_CALL(item, GetStartTime()).WillRepeatedly(Return(now)); EXPECT_CALL(item, GetState()) .WillRepeatedly(Return(DownloadItem::COMPLETE)); } EXPECT_CALL(GetMockDownloadItem(0), Remove()); EXPECT_CALL(GetMockDownloadItem(1), Remove()).Times(0); base::Callback url_filter = GetSingleURLFilter(download_urls_[0]); int remove_count = download_manager_->RemoveDownloadsByURLAndTime( url_filter, base::Time(), base::Time::Max()); EXPECT_EQ(remove_count, 1); } } // namespace content