blob: 26a8e44bd41b3d4eb90e8d641e17b4d5e3eb6d70 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_BROWSER_DOWNLOAD_DOWNLOAD_MANAGER_IMPL_H_
#define CONTENT_BROWSER_DOWNLOAD_DOWNLOAD_MANAGER_IMPL_H_
#include <stdint.h>
#include <memory>
#include <optional>
#include <set>
#include <string>
#include <unordered_map>
#include <vector>
#include "base/functional/callback_forward.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/observer_list.h"
#include "base/synchronization/lock.h"
#include "base/task/sequenced_task_runner.h"
#include "base/task/sequenced_task_runner_helpers.h"
#include "build/build_config.h"
#include "components/download/public/common/download_item_impl_delegate.h"
#include "components/download/public/common/download_job.h"
#include "components/download/public/common/download_url_parameters.h"
#include "components/download/public/common/in_progress_download_manager.h"
#include "components/download/public/common/url_download_handler.h"
#include "content/browser/loader/navigation_url_loader.h"
#include "content/common/content_export.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/download_manager.h"
#include "content/public/browser/download_manager_delegate.h"
#include "content/public/browser/ssl_status.h"
#include "mojo/public/cpp/system/data_pipe.h"
#include "services/metrics/public/cpp/ukm_source_id.h"
#include "services/network/public/mojom/url_loader.mojom.h"
#include "url/origin.h"
namespace download {
class DownloadFileFactory;
class DownloadItemFactory;
class DownloadItemImpl;
}
// These values are persisted to logs (Download.InitiatedByWindowOpener).
// Entries should not be renumbered and numeric values should never be reused.
// Openee is the tab over which the download is initiated.
// Opener is the tab that opened the openee tab and initiated a download on it.
enum class InitiatedByWindowOpenerType {
kSameOrigin = 0,
// Openee and opener are cross origin.
kCrossOrigin = 1,
// Openee and opener are cross origin but same site (i.e. same eTLD+1).
kSameSite = 2,
// Either the openee or the opener is not HTTP or HTTPS, e.g. about:blank.
kNonHTTPOrHTTPS = 3,
kMaxValue = kNonHTTPOrHTTPS
};
namespace content {
class CONTENT_EXPORT DownloadManagerImpl
: public DownloadManager,
public download::InProgressDownloadManager::Delegate,
private download::DownloadItemImplDelegate {
public:
using DownloadItemImplCreated =
base::OnceCallback<void(download::DownloadItemImpl*)>;
// Caller guarantees that |net_log| will remain valid
// for the lifetime of DownloadManagerImpl (until Shutdown() is called).
explicit DownloadManagerImpl(BrowserContext* browser_context);
DownloadManagerImpl(const DownloadManagerImpl&) = delete;
DownloadManagerImpl& operator=(const DownloadManagerImpl&) = delete;
~DownloadManagerImpl() override;
// Implementation functions (not part of the DownloadManager interface).
// Creates a download item for the SavePackage system.
// Must be called on the UI thread. Note that the DownloadManager
// retains ownership.
void CreateSavePackageDownloadItem(
const base::FilePath& main_file_path,
const base::FilePath& main_file_display_name,
const GURL& page_url,
const std::string& mime_type,
int render_process_id,
int render_frame_id,
download::DownloadJob::CancelRequestCallback cancel_request_callback,
DownloadItemImplCreated item_created);
// DownloadManager functions.
void SetDelegate(DownloadManagerDelegate* delegate) override;
DownloadManagerDelegate* GetDelegate() override;
void Shutdown() override;
void GetAllDownloads(
download::SimpleDownloadManager::DownloadVector* result) override;
void GetUninitializedActiveDownloadsIfAny(
download::SimpleDownloadManager::DownloadVector* result) override;
int RemoveDownloadsByURLAndTime(
const base::RepeatingCallback<bool(const GURL&)>& url_filter,
base::Time remove_begin,
base::Time remove_end) override;
bool CanDownload(download::DownloadUrlParameters* parameters) override;
void DownloadUrl(
std::unique_ptr<download::DownloadUrlParameters> parameters) override;
void DownloadUrl(std::unique_ptr<download::DownloadUrlParameters> params,
scoped_refptr<network::SharedURLLoaderFactory>
blob_url_loader_factory) override;
void AddObserver(Observer* observer) override;
void RemoveObserver(Observer* observer) override;
download::DownloadItem* CreateDownloadItem(
const std::string& guid,
uint32_t id,
const base::FilePath& current_path,
const base::FilePath& target_path,
const std::vector<GURL>& url_chain,
const GURL& referrer_url,
const StoragePartitionConfig& storage_partition_config,
const GURL& tab_url,
const GURL& tab_refererr_url,
const std::optional<url::Origin>& request_initiator,
const std::string& mime_type,
const std::string& original_mime_type,
base::Time start_time,
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,
download::DownloadItem::DownloadState state,
download::DownloadDangerType danger_type,
download::DownloadInterruptReason interrupt_reason,
bool opened,
base::Time last_access_time,
bool transient,
const std::vector<download::DownloadItem::ReceivedSlice>& received_slices)
override;
void PostInitialization(DownloadInitializationDependency dependency) override;
bool IsManagerInitialized() override;
int InProgressCount() override;
int BlockingShutdownCount() override;
BrowserContext* GetBrowserContext() override;
void CheckForHistoryFilesRemoval() override;
void OnHistoryQueryComplete(
base::OnceClosure load_history_downloads_cb) override;
download::DownloadItem* GetDownload(uint32_t id) override;
download::DownloadItem* GetDownloadByGuid(const std::string& guid) override;
void GetNextId(GetNextIdCallback callback) override;
std::string StoragePartitionConfigToSerializedEmbedderDownloadData(
const StoragePartitionConfig& storage_partition_config) override;
StoragePartitionConfig SerializedEmbedderDownloadDataToStoragePartitionConfig(
const std::string& serialized_embedder_download_data) override;
StoragePartitionConfig GetStoragePartitionConfigForSiteUrl(
const GURL& site_url) override;
void StartDownload(
std::unique_ptr<download::DownloadCreateInfo> info,
std::unique_ptr<download::InputStream> stream,
download::DownloadUrlParameters::OnStartedCallback on_started);
// For testing; specifically, accessed from TestFileErrorInjector.
void SetDownloadItemFactoryForTesting(
std::unique_ptr<download::DownloadItemFactory> item_factory);
void SetDownloadFileFactoryForTesting(
std::unique_ptr<download::DownloadFileFactory> file_factory);
virtual download::DownloadFileFactory* GetDownloadFileFactoryForTesting();
// Continue a navigation that ends up to be a download after it reaches the
// OnResponseStarted() step. It has to be called on the UI thread.
void InterceptNavigation(
std::unique_ptr<network::ResourceRequest> resource_request,
std::vector<GURL> url_chain,
network::mojom::URLResponseHeadPtr response_head,
mojo::ScopedDataPipeConsumerHandle response_body,
network::mojom::URLLoaderClientEndpointsPtr url_loader_client_endpoints,
net::CertStatus cert_status,
FrameTreeNodeId frame_tree_node_id,
bool from_download_cross_origin_redirect);
// DownloadItemImplDelegate overrides.
download::QuarantineConnectionCallback GetQuarantineConnectionCallback()
override;
std::string GetApplicationClientIdForFileScanning() const override;
std::unique_ptr<download::DownloadItemRenameHandler>
GetRenameHandlerForDownload(
download::DownloadItemImpl* download_item) override;
private:
using DownloadSet = std::set<download::DownloadItem*>;
using DownloadGuidMap =
std::unordered_map<std::string,
raw_ptr<download::DownloadItemImpl, CtnExperimental>>;
using DownloadItemImplVector = std::vector<download::DownloadItemImpl*>;
// For testing.
friend class DownloadManagerTest;
friend class DownloadTest;
void CreateSavePackageDownloadItemWithId(
const base::FilePath& main_file_path,
const base::FilePath& main_file_display_name,
const GURL& page_url,
const std::string& mime_type,
int render_process_id,
int render_frame_id,
download::DownloadJob::CancelRequestCallback cancel_request_callback,
DownloadItemImplCreated on_started,
uint32_t id);
// InProgressDownloadManager::Delegate implementations.
void OnDownloadsInitialized() override;
bool InterceptDownload(const download::DownloadCreateInfo& info) override;
base::FilePath GetDefaultDownloadDirectory() override;
void StartDownloadItem(
std::unique_ptr<download::DownloadCreateInfo> info,
download::DownloadUrlParameters::OnStartedCallback on_started,
download::InProgressDownloadManager::StartDownloadItemCallback callback)
override;
// Creates a new download item and call |callback|.
void CreateNewDownloadItemToStart(
std::unique_ptr<download::DownloadCreateInfo> info,
download::DownloadUrlParameters::OnStartedCallback on_started,
download::InProgressDownloadManager::StartDownloadItemCallback callback,
uint32_t id,
const base::FilePath& duplicate_download_file_path,
bool duplicate_file_exists);
// Sets the |next_download_id_| if the |next_id| is larger. Runs all the
// |id_callbacks_| if both the ID from both history db and in-progress db
// are retrieved.
void SetNextId(uint32_t next_id);
// Called when the next ID from history db is retrieved.
void OnHistoryNextIdRetrieved(uint32_t next_id);
// Create a new active item based on the info. Separate from
// StartDownload() for testing.
download::DownloadItemImpl* CreateActiveItem(
uint32_t id,
const download::DownloadCreateInfo& info);
// Called with the result of CheckForFileExistence. Updates the state of the
// file and then notifies this update to the file's observer.
void OnFileExistenceChecked(const std::string& guid, bool result);
// Overridden from DownloadItemImplDelegate
void DetermineDownloadTarget(
download::DownloadItemImpl* item,
download::DownloadTargetCallback callback) override;
bool ShouldCompleteDownload(download::DownloadItemImpl* item,
base::OnceClosure complete_callback) override;
bool ShouldAutomaticallyOpenFile(const GURL& url,
const base::FilePath& path) override;
bool ShouldAutomaticallyOpenFileByPolicy(const GURL& url,
const base::FilePath& path) override;
bool ShouldOpenDownload(download::DownloadItemImpl* item,
ShouldOpenDownloadCallback callback) override;
void CheckForFileRemoval(download::DownloadItemImpl* download_item) override;
void ResumeInterruptedDownload(
std::unique_ptr<download::DownloadUrlParameters> params,
const std::string& serialized_embedder_download_data) override;
void OpenDownload(download::DownloadItemImpl* download) override;
void ShowDownloadInShell(download::DownloadItemImpl* download) override;
void DownloadRemoved(download::DownloadItemImpl* download) override;
void DownloadInterrupted(download::DownloadItemImpl* download) override;
bool IsOffTheRecord() const override;
void ReportBytesWasted(download::DownloadItemImpl* download) override;
void BindWakeLockProvider(
mojo::PendingReceiver<device::mojom::WakeLockProvider> receiver) override;
// Drops a download before it is created.
void DropDownload();
// Helper method to start or resume a download.
void BeginDownloadInternal(
std::unique_ptr<download::DownloadUrlParameters> params,
scoped_refptr<network::SharedURLLoaderFactory> blob_url_loader_factory,
bool is_new_download,
const std::string& serialized_embedder_download_data);
void InterceptNavigationOnChecksComplete(
FrameTreeNodeId frame_tree_node_id,
std::unique_ptr<network::ResourceRequest> resource_request,
std::vector<GURL> url_chain,
net::CertStatus cert_status,
network::mojom::URLResponseHeadPtr response_head,
mojo::ScopedDataPipeConsumerHandle response_body,
network::mojom::URLLoaderClientEndpointsPtr url_loader_client_endpoints,
bool is_download_allowed);
void BeginResourceDownloadOnChecksComplete(
std::unique_ptr<download::DownloadUrlParameters> params,
scoped_refptr<network::SharedURLLoaderFactory> blob_url_loader_factory,
bool is_new_download,
const StoragePartitionConfig& storage_partition_config,
bool is_download_allowed);
// Whether |next_download_id_| is initialized.
bool IsNextIdInitialized() const;
// Called when a new download is created.
void OnDownloadCreated(std::unique_ptr<download::DownloadItemImpl> download);
// Retrieves a download from |in_progress_downloads_|.
std::unique_ptr<download::DownloadItemImpl> RetrieveInProgressDownload(
uint32_t id);
// Import downloads from |in_progress_downloads_| into |downloads_|, resolve
// missing download IDs.
void ImportInProgressDownloads(uint32_t next_id);
// Called when this object is considered initialized.
void OnDownloadManagerInitialized();
// Check whether a download should be cleared from history. Cancelled and
// non-resumable interrupted download will be cleaned up to save memory.
bool ShouldClearDownloadFromDB(const GURL& url,
download::DownloadItem::DownloadState state,
download::DownloadInterruptReason reason,
const base::Time& start_time);
// Called when Id for a new download item is retrieved.
void OnNewDownloadIdRetrieved(
std::unique_ptr<download::DownloadCreateInfo> info,
download::DownloadUrlParameters::OnStartedCallback on_started,
download::InProgressDownloadManager::StartDownloadItemCallback callback,
uint32_t id);
// Factory for creation of downloads items.
std::unique_ptr<download::DownloadItemFactory> item_factory_;
// |downloads_| is the owning set for all downloads known to the
// DownloadManager. This includes downloads started by the user in
// this session, downloads initialized from the history system, and
// "save page as" downloads.
// TODO(asanka): Remove this container in favor of downloads_by_guid_ as a
// part of http://crbug.com/593020.
std::unordered_map<uint32_t, std::unique_ptr<download::DownloadItemImpl>>
downloads_;
// Same as the above, but maps from GUID to download item. Note that the
// container is case sensitive. Hence the key needs to be normalized to
// upper-case when inserting new elements here. Fortunately for us,
// DownloadItemImpl already normalizes the string GUID.
DownloadGuidMap downloads_by_guid_;
// True if the download manager has been initialized and requires a shutdown.
bool shutdown_needed_;
// Whether the history db and/or in progress cache are initialized.
bool history_db_initialized_;
bool in_progress_cache_initialized_;
// Observers that want to be notified of changes to the set of downloads.
base::ObserverList<Observer>::Unchecked observers_;
// Stores information about in-progress download items.
std::unique_ptr<download::DownloadItem::Observer>
in_progress_download_observer_;
// The current active browser context.
raw_ptr<BrowserContext> browser_context_;
// Allows an embedder to control behavior. Guaranteed to outlive this object.
raw_ptr<DownloadManagerDelegate> delegate_;
std::unique_ptr<download::InProgressDownloadManager> in_progress_manager_;
// Callback to run to load all history downloads.
base::OnceClosure load_history_downloads_cb_;
// The next download id to issue to new downloads. The |next_download_id_| can
// only be used when both history and in-progress db have provided their
// values.
uint32_t next_download_id_;
// Whether next download ID from history DB is being retrieved.
bool is_history_download_id_retrieved_;
// Whether new download should be persisted to the in progress download
// database.
bool should_persist_new_download_;
// The download GUIDs that are cleared up on startup.
std::set<std::string> cleared_download_guids_on_startup_;
// In progress downloads returned by |in_progress_manager_| that are not yet
// added to |downloads_|. If a download was started without launching full
// browser process, its ID will be invalid. DownloadManager will assign new
// ID to it when importing all downloads.
std::vector<std::unique_ptr<download::DownloadItemImpl>>
in_progress_downloads_;
// Callbacks to run once download ID is determined.
using IdCallbackVector = std::vector<std::unique_ptr<GetNextIdCallback>>;
IdCallbackVector id_callbacks_;
// SequencedTaskRunner to check for file existence. A sequence is used so
// that a large download history doesn't cause a large number of concurrent
// disk operations.
const scoped_refptr<base::SequencedTaskRunner> disk_access_task_runner_;
// DownloadItem for which a query is queued in the |disk_access_task_runner_|.
std::set<std::string> pending_disk_access_query_;
base::WeakPtrFactory<DownloadManagerImpl> weak_factory_{this};
};
} // namespace content
#endif // CONTENT_BROWSER_DOWNLOAD_DOWNLOAD_MANAGER_IMPL_H_