// Copyright (c) 2011 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. // // Each download is represented by a DownloadItem, and all DownloadItems // are owned by the DownloadManager which maintains a global list of all // downloads. DownloadItems are created when a user initiates a download, // and exist for the duration of the browser life time. // // Download observers: // DownloadItem::Observer: // - allows observers to receive notifications about one download from start // to completion // Use AddObserver() / RemoveObserver() on the appropriate download object to // receive state updates. #ifndef CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_ #define CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_ #pragma once #include <string> #include "base/basictypes.h" #include "base/file_path.h" #include "base/observer_list.h" #include "base/time.h" #include "base/timer.h" #include "googleurl/src/gurl.h" class DownloadFileManager; class DownloadManager; struct DownloadCreateInfo; // One DownloadItem per download. This is the model class that stores all the // state for a download. Multiple views, such as a tab's download shelf and the // Destination tab's download view, may refer to a given DownloadItem. // // This is intended to be used only on the UI thread. class DownloadItem { public: enum DownloadState { // Download is actively progressing. IN_PROGRESS, // Download is completely finished. COMPLETE, // Download has been cancelled. CANCELLED, // This state indicates that the download item is about to be destroyed, // and observers seeing this state should release all references. REMOVING, // This state indicates that the download has been interrupted. INTERRUPTED }; enum SafetyState { SAFE = 0, DANGEROUS, DANGEROUS_BUT_VALIDATED // Dangerous but the user confirmed the download. }; // This enum is used by histograms. Do not change the ordering or remove // items. enum DangerType { NOT_DANGEROUS = 0, // A dangerous file to the system (e.g.: an executable or extension from // places other than gallery). DANGEROUS_FILE, // Safebrowsing service shows this URL leads to malicious file download. DANGEROUS_URL, // Memory space for histograms is determined by the max. // ALWAYS ADD NEW VALUES BEFORE THIS ONE. DANGEROUS_TYPE_MAX }; // Reason for deleting the download. Passed to Delete(). enum DeleteReason { DELETE_DUE_TO_BROWSER_SHUTDOWN = 0, DELETE_DUE_TO_USER_DISCARD }; // Interface that observers of a particular download must implement in order // to receive updates to the download's status. class Observer { public: virtual void OnDownloadUpdated(DownloadItem* download) = 0; // Called when a downloaded file has been opened. virtual void OnDownloadOpened(DownloadItem* download) = 0; protected: virtual ~Observer() {} }; // Constructing from persistent store: DownloadItem(DownloadManager* download_manager, const DownloadCreateInfo& info); // Constructing for a regular download: DownloadItem(DownloadManager* download_manager, const DownloadCreateInfo& info, bool is_otr); // Constructing for the "Save Page As..." feature: DownloadItem(DownloadManager* download_manager, const FilePath& path, const GURL& url, bool is_otr); ~DownloadItem(); void AddObserver(Observer* observer); void RemoveObserver(Observer* observer); // Notifies our observers periodically. void UpdateObservers(); // Whether it is OK to open this download. bool CanOpenDownload(); // Tests if a file type should be opened automatically. bool ShouldOpenFileBasedOnExtension(); // Registers this file extension for automatic opening upon download // completion if 'open' is true, or prevents the extension from automatic // opening if 'open' is false. void OpenFilesBasedOnExtension(bool open); // Open the file associated with this download (wait for the download to // complete if it is in progress). void OpenDownload(); // Show the download via the OS shell. void ShowDownloadInShell(); // Called when the user has validated the download of a dangerous file. void DangerousDownloadValidated(); // Received a new chunk of data void Update(int64 bytes_so_far); // Cancel the download operation. We need to distinguish between cancels at // exit (DownloadManager destructor) from user interface initiated cancels // because at exit, the history system may not exist, and any updates to it // require AddRef'ing the DownloadManager in the destructor which results in // a DCHECK failure. Set 'update_history' to false when canceling from at // exit to prevent this crash. This may result in a difference between the // downloaded file's size on disk, and what the history system's last record // of it is. At worst, we'll end up re-downloading a small portion of the file // when resuming a download (assuming the server supports byte ranges). void Cancel(bool update_history); // Called when all data has been saved. Only has display effects. void OnAllDataSaved(int64 size); // Called by external code (SavePackage) using the DownloadItem interface // to display progress when the DownloadItem should be considered complete. void MarkAsComplete(); // Download operation had an error. // |size| is the amount of data received so far, and |os_error| is the error // code that the operation received. void Interrupted(int64 size, int os_error); // Deletes the file from disk and removes the download from the views and // history. |user| should be true if this is the result of the user clicking // the discard button, and false if it is being deleted for other reasons like // browser shutdown. void Delete(DeleteReason reason); // Removes the download from the views and history. void Remove(); // Simple calculation of the amount of time remaining to completion. Fills // |*remaining| with the amount of time remaining if successful. Fails and // returns false if we do not have the number of bytes or the speed so can // not estimate. bool TimeRemaining(base::TimeDelta* remaining) const; // Simple speed estimate in bytes/s int64 CurrentSpeed() const; // Rough percent complete, -1 means we don't know (since we didn't receive a // total size). int PercentComplete() const; // Whether or not this download has saved all of its data. bool all_data_saved() const { return all_data_saved_; } // Update the fields that may have changed in DownloadCreateInfo as a // result of analyzing the file and figuring out its type, location, etc. // May only be called once. void SetFileCheckResults(const FilePath& path, bool is_dangerous_file, bool is_dangerous_url, int path_uniquifier, bool prompt, bool is_extension_install, const FilePath& original_name); // Update the download's path, the actual file is renamed on the download // thread. void Rename(const FilePath& full_path); // Allow the user to temporarily pause a download or resume a paused download. void TogglePause(); // Called when the name of the download is finalized. void OnNameFinalized(); // Called when the download is ready to complete. // This may perform final rename if necessary and will eventually call // DownloadItem::Completed(). void OnDownloadCompleting(DownloadFileManager* file_manager); // Called when the file name for the download is renamed to its final name. void OnDownloadRenamedToFinalName(const FilePath& full_path); // Returns true if this item matches |query|. |query| must be lower-cased. bool MatchesQuery(const string16& query) const; // Returns true if the download needs more data. bool IsPartialDownload() const; // Returns true if the download is still receiving data. bool IsInProgress() const; // Returns true if the download has been cancelled or was interrupted. bool IsCancelled() const; // Returns true if the download was interrupted. bool IsInterrupted() const; // Returns true if we have all the data and know the final file name. bool IsComplete() const; // Accessors DownloadState state() const { return state_; } FilePath full_path() const { return full_path_; } void set_path_uniquifier(int uniquifier) { path_uniquifier_ = uniquifier; } const GURL& url() const { return url_chain_.back(); } const std::vector<GURL>& url_chain() const { return url_chain_; } const GURL& original_url() const { return url_chain_.front(); } const GURL& referrer_url() const { return referrer_url_; } std::string mime_type() const { return mime_type_; } std::string original_mime_type() const { return original_mime_type_; } int64 total_bytes() const { return total_bytes_; } void set_total_bytes(int64 total_bytes) { total_bytes_ = total_bytes; } int64 received_bytes() const { return received_bytes_; } int32 id() const { return id_; } base::Time start_time() const { return start_time_; } void set_db_handle(int64 handle) { db_handle_ = handle; } int64 db_handle() const { return db_handle_; } bool is_paused() const { return is_paused_; } bool open_when_complete() const { return open_when_complete_; } void set_open_when_complete(bool open) { open_when_complete_ = open; } int render_process_id() const { return render_process_id_; } int request_id() const { return request_id_; } SafetyState safety_state() const { return safety_state_; } void set_safety_state(SafetyState safety_state) { safety_state_ = safety_state; } DangerType danger_type() { return danger_type_;} bool auto_opened() { return auto_opened_; } FilePath target_name() const { return target_name_; } bool save_as() const { return save_as_; } bool is_otr() const { return is_otr_; } bool is_extension_install() const { return is_extension_install_; } bool name_finalized() const { return name_finalized_; } bool is_temporary() const { return is_temporary_; } void set_opened(bool opened) { opened_ = opened; } bool opened() const { return opened_; } // Returns the final target file path for the download. FilePath GetTargetFilePath() const; // Returns the file-name that should be reported to the user, which is // target_name_ possibly with the uniquifier number. FilePath GetFileNameToReportUser() const; // Returns the user-verified target file path for the download. // This returns the same path as GetTargetFilePath() for safe downloads // but does not for dangerous downloads until the name is verified. FilePath GetUserVerifiedFilePath() const; // Returns true if the current file name is not the final target name yet. bool NeedsRename() const { return target_name_ != full_path_.BaseName(); } std::string DebugString(bool verbose) const; private: void Init(bool start_timer); // Internal helper for maintaining consistent received and total sizes. void UpdateSize(int64 size); // Called when the entire download operation (including renaming etc) // is completed. void Completed(); // Start/stop sending periodic updates to our observers void StartProgressTimer(); void StopProgressTimer(); // Request ID assigned by the ResourceDispatcherHost. int32 id_; // Full path to the downloaded or downloading file. FilePath full_path_; // A number that should be appended to the path to make it unique, or 0 if the // path should be used as is. int path_uniquifier_; // The chain of redirects that leading up to and including the final URL. std::vector<GURL> url_chain_; // The URL of the page that initiated the download. GURL referrer_url_; // The mimetype of the download std::string mime_type_; // The value of the content type header received when downloading // this item. |mime_type_| may be different because of type sniffing. std::string original_mime_type_; // Total bytes expected int64 total_bytes_; // Current received bytes int64 received_bytes_; // Last error. int last_os_error_; // Start time for calculating remaining time base::TimeTicks start_tick_; // The current state of this download DownloadState state_; // The views of this item in the download shelf and download tab ObserverList<Observer> observers_; // Time the download was started base::Time start_time_; // Our persistent store handle int64 db_handle_; // Timer for regularly updating our observers base::RepeatingTimer<DownloadItem> update_timer_; // Our owning object DownloadManager* download_manager_; // In progress downloads may be paused by the user, we note it here bool is_paused_; // A flag for indicating if the download should be opened at completion. bool open_when_complete_; // Whether the download is considered potentially safe or dangerous // (executable files are typically considered dangerous). SafetyState safety_state_; // Why |safety_state_| is not SAFE. DangerType danger_type_; // Whether the download was auto-opened. We set this rather than using // an observer as it's frequently possible for the download to be auto opened // before the observer is added. bool auto_opened_; // Dangerous downloads or ongoing downloads are given temporary names until // the user approves them or the downloads finish. // This stores their final target name. FilePath target_name_; // For canceling or pausing requests. int render_process_id_; int request_id_; // True if the item was downloaded as a result of 'save as...' bool save_as_; // True if the download was initiated in an incognito window. bool is_otr_; // True if the item was downloaded for an extension installation. bool is_extension_install_; // True if the filename is finalized. bool name_finalized_; // True if the item was downloaded temporarily. bool is_temporary_; // True if we've saved all the data for the download. bool all_data_saved_; // Did the user open the item either directly or indirectly (such as by // setting always open files of this type)? The shelf also sets this field // when the user closes the shelf before the item has been opened but should // be treated as though the user opened it. bool opened_; DISALLOW_COPY_AND_ASSIGN(DownloadItem); }; #endif // CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_