• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 //
5 // Each download is represented by a DownloadItem, and all DownloadItems
6 // are owned by the DownloadManager which maintains a global list of all
7 // downloads. DownloadItems are created when a user initiates a download,
8 // and exist for the duration of the browser life time.
9 //
10 // Download observers:
11 //   DownloadItem::Observer:
12 //     - allows observers to receive notifications about one download from start
13 //       to completion
14 // Use AddObserver() / RemoveObserver() on the appropriate download object to
15 // receive state updates.
16 
17 #ifndef CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_
18 #define CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_
19 #pragma once
20 
21 #include <string>
22 
23 #include "base/basictypes.h"
24 #include "base/file_path.h"
25 #include "base/observer_list.h"
26 #include "base/time.h"
27 #include "base/timer.h"
28 #include "googleurl/src/gurl.h"
29 
30 class DownloadFileManager;
31 class DownloadManager;
32 struct DownloadCreateInfo;
33 
34 // One DownloadItem per download. This is the model class that stores all the
35 // state for a download. Multiple views, such as a tab's download shelf and the
36 // Destination tab's download view, may refer to a given DownloadItem.
37 //
38 // This is intended to be used only on the UI thread.
39 class DownloadItem {
40  public:
41   enum DownloadState {
42     // Download is actively progressing.
43     IN_PROGRESS,
44 
45     // Download is completely finished.
46     COMPLETE,
47 
48     // Download has been cancelled.
49     CANCELLED,
50 
51     // This state indicates that the download item is about to be destroyed,
52     // and observers seeing this state should release all references.
53     REMOVING,
54 
55     // This state indicates that the download has been interrupted.
56     INTERRUPTED
57   };
58 
59   enum SafetyState {
60     SAFE = 0,
61     DANGEROUS,
62     DANGEROUS_BUT_VALIDATED  // Dangerous but the user confirmed the download.
63   };
64 
65   // This enum is used by histograms.  Do not change the ordering or remove
66   // items.
67   enum DangerType {
68     NOT_DANGEROUS = 0,
69 
70     // A dangerous file to the system (e.g.: an executable or extension from
71     // places other than gallery).
72     DANGEROUS_FILE,
73 
74     // Safebrowsing service shows this URL leads to malicious file download.
75     DANGEROUS_URL,
76 
77     // Memory space for histograms is determined by the max.
78     // ALWAYS ADD NEW VALUES BEFORE THIS ONE.
79     DANGEROUS_TYPE_MAX
80   };
81 
82   // Reason for deleting the download.  Passed to Delete().
83   enum DeleteReason {
84     DELETE_DUE_TO_BROWSER_SHUTDOWN = 0,
85     DELETE_DUE_TO_USER_DISCARD
86   };
87 
88   // Interface that observers of a particular download must implement in order
89   // to receive updates to the download's status.
90   class Observer {
91    public:
92     virtual void OnDownloadUpdated(DownloadItem* download) = 0;
93 
94     // Called when a downloaded file has been opened.
95     virtual void OnDownloadOpened(DownloadItem* download) = 0;
96 
97    protected:
~Observer()98     virtual ~Observer() {}
99   };
100 
101   // Constructing from persistent store:
102   DownloadItem(DownloadManager* download_manager,
103                const DownloadCreateInfo& info);
104 
105   // Constructing for a regular download:
106   DownloadItem(DownloadManager* download_manager,
107                const DownloadCreateInfo& info,
108                bool is_otr);
109 
110   // Constructing for the "Save Page As..." feature:
111   DownloadItem(DownloadManager* download_manager,
112                const FilePath& path,
113                const GURL& url,
114                bool is_otr);
115 
116   ~DownloadItem();
117 
118   void AddObserver(Observer* observer);
119   void RemoveObserver(Observer* observer);
120 
121   // Notifies our observers periodically.
122   void UpdateObservers();
123 
124   // Whether it is OK to open this download.
125   bool CanOpenDownload();
126 
127   // Tests if a file type should be opened automatically.
128   bool ShouldOpenFileBasedOnExtension();
129 
130   // Registers this file extension for automatic opening upon download
131   // completion if 'open' is true, or prevents the extension from automatic
132   // opening if 'open' is false.
133   void OpenFilesBasedOnExtension(bool open);
134 
135   // Open the file associated with this download (wait for the download to
136   // complete if it is in progress).
137   void OpenDownload();
138 
139   // Show the download via the OS shell.
140   void ShowDownloadInShell();
141 
142   // Called when the user has validated the download of a dangerous file.
143   void DangerousDownloadValidated();
144 
145   // Received a new chunk of data
146   void Update(int64 bytes_so_far);
147 
148   // Cancel the download operation. We need to distinguish between cancels at
149   // exit (DownloadManager destructor) from user interface initiated cancels
150   // because at exit, the history system may not exist, and any updates to it
151   // require AddRef'ing the DownloadManager in the destructor which results in
152   // a DCHECK failure. Set 'update_history' to false when canceling from at
153   // exit to prevent this crash. This may result in a difference between the
154   // downloaded file's size on disk, and what the history system's last record
155   // of it is. At worst, we'll end up re-downloading a small portion of the file
156   // when resuming a download (assuming the server supports byte ranges).
157   void Cancel(bool update_history);
158 
159   // Called when all data has been saved.  Only has display effects.
160   void OnAllDataSaved(int64 size);
161 
162   // Called by external code (SavePackage) using the DownloadItem interface
163   // to display progress when the DownloadItem should be considered complete.
164   void MarkAsComplete();
165 
166   // Download operation had an error.
167   // |size| is the amount of data received so far, and |os_error| is the error
168   // code that the operation received.
169   void Interrupted(int64 size, int os_error);
170 
171   // Deletes the file from disk and removes the download from the views and
172   // history.  |user| should be true if this is the result of the user clicking
173   // the discard button, and false if it is being deleted for other reasons like
174   // browser shutdown.
175   void Delete(DeleteReason reason);
176 
177   // Removes the download from the views and history.
178   void Remove();
179 
180   // Simple calculation of the amount of time remaining to completion. Fills
181   // |*remaining| with the amount of time remaining if successful. Fails and
182   // returns false if we do not have the number of bytes or the speed so can
183   // not estimate.
184   bool TimeRemaining(base::TimeDelta* remaining) const;
185 
186   // Simple speed estimate in bytes/s
187   int64 CurrentSpeed() const;
188 
189   // Rough percent complete, -1 means we don't know (since we didn't receive a
190   // total size).
191   int PercentComplete() const;
192 
193   // Whether or not this download has saved all of its data.
all_data_saved()194   bool all_data_saved() const { return all_data_saved_; }
195 
196   // Update the fields that may have changed in DownloadCreateInfo as a
197   // result of analyzing the file and figuring out its type, location, etc.
198   // May only be called once.
199   void SetFileCheckResults(const FilePath& path,
200                            bool is_dangerous_file,
201                            bool is_dangerous_url,
202                            int path_uniquifier,
203                            bool prompt,
204                            bool is_extension_install,
205                            const FilePath& original_name);
206 
207   // Update the download's path, the actual file is renamed on the download
208   // thread.
209   void Rename(const FilePath& full_path);
210 
211   // Allow the user to temporarily pause a download or resume a paused download.
212   void TogglePause();
213 
214   // Called when the name of the download is finalized.
215   void OnNameFinalized();
216 
217   // Called when the download is ready to complete.
218   // This may perform final rename if necessary and will eventually call
219   // DownloadItem::Completed().
220   void OnDownloadCompleting(DownloadFileManager* file_manager);
221 
222   // Called when the file name for the download is renamed to its final name.
223   void OnDownloadRenamedToFinalName(const FilePath& full_path);
224 
225   // Returns true if this item matches |query|. |query| must be lower-cased.
226   bool MatchesQuery(const string16& query) const;
227 
228   // Returns true if the download needs more data.
229   bool IsPartialDownload() const;
230 
231   // Returns true if the download is still receiving data.
232   bool IsInProgress() const;
233 
234   // Returns true if the download has been cancelled or was interrupted.
235   bool IsCancelled() const;
236 
237   // Returns true if the download was interrupted.
238   bool IsInterrupted() const;
239 
240   // Returns true if we have all the data and know the final file name.
241   bool IsComplete() const;
242 
243   // Accessors
state()244   DownloadState state() const { return state_; }
full_path()245   FilePath full_path() const { return full_path_; }
set_path_uniquifier(int uniquifier)246   void set_path_uniquifier(int uniquifier) { path_uniquifier_ = uniquifier; }
url()247   const GURL& url() const { return url_chain_.back(); }
url_chain()248   const std::vector<GURL>& url_chain() const { return url_chain_; }
original_url()249   const GURL& original_url() const { return url_chain_.front(); }
referrer_url()250   const GURL& referrer_url() const { return referrer_url_; }
mime_type()251   std::string mime_type() const { return mime_type_; }
original_mime_type()252   std::string original_mime_type() const { return original_mime_type_; }
total_bytes()253   int64 total_bytes() const { return total_bytes_; }
set_total_bytes(int64 total_bytes)254   void set_total_bytes(int64 total_bytes) { total_bytes_ = total_bytes; }
received_bytes()255   int64 received_bytes() const { return received_bytes_; }
id()256   int32 id() const { return id_; }
start_time()257   base::Time start_time() const { return start_time_; }
set_db_handle(int64 handle)258   void set_db_handle(int64 handle) { db_handle_ = handle; }
db_handle()259   int64 db_handle() const { return db_handle_; }
is_paused()260   bool is_paused() const { return is_paused_; }
open_when_complete()261   bool open_when_complete() const { return open_when_complete_; }
set_open_when_complete(bool open)262   void set_open_when_complete(bool open) { open_when_complete_ = open; }
render_process_id()263   int render_process_id() const { return render_process_id_; }
request_id()264   int request_id() const { return request_id_; }
safety_state()265   SafetyState safety_state() const { return safety_state_; }
set_safety_state(SafetyState safety_state)266   void set_safety_state(SafetyState safety_state) {
267     safety_state_ = safety_state;
268   }
danger_type()269   DangerType danger_type() { return danger_type_;}
auto_opened()270   bool auto_opened() { return auto_opened_; }
target_name()271   FilePath target_name() const { return target_name_; }
save_as()272   bool save_as() const { return save_as_; }
is_otr()273   bool is_otr() const { return is_otr_; }
is_extension_install()274   bool is_extension_install() const { return is_extension_install_; }
name_finalized()275   bool name_finalized() const { return name_finalized_; }
is_temporary()276   bool is_temporary() const { return is_temporary_; }
set_opened(bool opened)277   void set_opened(bool opened) { opened_ = opened; }
opened()278   bool opened() const { return opened_; }
279 
280   // Returns the final target file path for the download.
281   FilePath GetTargetFilePath() const;
282 
283   // Returns the file-name that should be reported to the user, which is
284   // target_name_ possibly with the uniquifier number.
285   FilePath GetFileNameToReportUser() const;
286 
287   // Returns the user-verified target file path for the download.
288   // This returns the same path as GetTargetFilePath() for safe downloads
289   // but does not for dangerous downloads until the name is verified.
290   FilePath GetUserVerifiedFilePath() const;
291 
292   // Returns true if the current file name is not the final target name yet.
NeedsRename()293   bool NeedsRename() const {
294     return target_name_ != full_path_.BaseName();
295   }
296 
297   std::string DebugString(bool verbose) const;
298 
299  private:
300   void Init(bool start_timer);
301 
302   // Internal helper for maintaining consistent received and total sizes.
303   void UpdateSize(int64 size);
304 
305   // Called when the entire download operation (including renaming etc)
306   // is completed.
307   void Completed();
308 
309   // Start/stop sending periodic updates to our observers
310   void StartProgressTimer();
311   void StopProgressTimer();
312 
313   // Request ID assigned by the ResourceDispatcherHost.
314   int32 id_;
315 
316   // Full path to the downloaded or downloading file.
317   FilePath full_path_;
318 
319   // A number that should be appended to the path to make it unique, or 0 if the
320   // path should be used as is.
321   int path_uniquifier_;
322 
323   // The chain of redirects that leading up to and including the final URL.
324   std::vector<GURL> url_chain_;
325 
326   // The URL of the page that initiated the download.
327   GURL referrer_url_;
328 
329   // The mimetype of the download
330   std::string mime_type_;
331 
332   // The value of the content type header received when downloading
333   // this item.  |mime_type_| may be different because of type sniffing.
334   std::string original_mime_type_;
335 
336   // Total bytes expected
337   int64 total_bytes_;
338 
339   // Current received bytes
340   int64 received_bytes_;
341 
342   // Last error.
343   int last_os_error_;
344 
345   // Start time for calculating remaining time
346   base::TimeTicks start_tick_;
347 
348   // The current state of this download
349   DownloadState state_;
350 
351   // The views of this item in the download shelf and download tab
352   ObserverList<Observer> observers_;
353 
354   // Time the download was started
355   base::Time start_time_;
356 
357   // Our persistent store handle
358   int64 db_handle_;
359 
360   // Timer for regularly updating our observers
361   base::RepeatingTimer<DownloadItem> update_timer_;
362 
363   // Our owning object
364   DownloadManager* download_manager_;
365 
366   // In progress downloads may be paused by the user, we note it here
367   bool is_paused_;
368 
369   // A flag for indicating if the download should be opened at completion.
370   bool open_when_complete_;
371 
372   // Whether the download is considered potentially safe or dangerous
373   // (executable files are typically considered dangerous).
374   SafetyState safety_state_;
375 
376   // Why |safety_state_| is not SAFE.
377   DangerType danger_type_;
378 
379   // Whether the download was auto-opened. We set this rather than using
380   // an observer as it's frequently possible for the download to be auto opened
381   // before the observer is added.
382   bool auto_opened_;
383 
384   // Dangerous downloads or ongoing downloads are given temporary names until
385   // the user approves them or the downloads finish.
386   // This stores their final target name.
387   FilePath target_name_;
388 
389   // For canceling or pausing requests.
390   int render_process_id_;
391   int request_id_;
392 
393   // True if the item was downloaded as a result of 'save as...'
394   bool save_as_;
395 
396   // True if the download was initiated in an incognito window.
397   bool is_otr_;
398 
399   // True if the item was downloaded for an extension installation.
400   bool is_extension_install_;
401 
402   // True if the filename is finalized.
403   bool name_finalized_;
404 
405   // True if the item was downloaded temporarily.
406   bool is_temporary_;
407 
408   // True if we've saved all the data for the download.
409   bool all_data_saved_;
410 
411   // Did the user open the item either directly or indirectly (such as by
412   // setting always open files of this type)? The shelf also sets this field
413   // when the user closes the shelf before the item has been opened but should
414   // be treated as though the user opened it.
415   bool opened_;
416 
417   DISALLOW_COPY_AND_ASSIGN(DownloadItem);
418 };
419 
420 #endif  // CHROME_BROWSER_DOWNLOAD_DOWNLOAD_ITEM_H_
421