Home | History | Annotate | Download | only in download
      1 // Copyright (c) 2012 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 #include "content/browser/download/save_item.h"
      6 
      7 #include "base/logging.h"
      8 #include "base/strings/string_util.h"
      9 #include "content/browser/download/save_file.h"
     10 #include "content/browser/download/save_file_manager.h"
     11 #include "content/browser/download/save_package.h"
     12 
     13 namespace content {
     14 
     15 // Constructor for SaveItem when creating each saving job.
     16 SaveItem::SaveItem(const GURL& url,
     17                    const Referrer& referrer,
     18                    SavePackage* package,
     19                    SaveFileCreateInfo::SaveFileSource save_source)
     20   : save_id_(-1),
     21     url_(url),
     22     referrer_(referrer),
     23     total_bytes_(0),
     24     received_bytes_(0),
     25     state_(WAIT_START),
     26     has_final_name_(false),
     27     is_success_(false),
     28     save_source_(save_source),
     29     package_(package) {
     30   DCHECK(package);
     31 }
     32 
     33 SaveItem::~SaveItem() {
     34 }
     35 
     36 // Set start state for save item.
     37 void SaveItem::Start() {
     38   DCHECK(state_ == WAIT_START);
     39   state_ = IN_PROGRESS;
     40 }
     41 
     42 // If we've received more data than we were expecting (bad server info?),
     43 // revert to 'unknown size mode'.
     44 void SaveItem::UpdateSize(int64 bytes_so_far) {
     45   received_bytes_ = bytes_so_far;
     46   if (received_bytes_ >= total_bytes_)
     47     total_bytes_ = 0;
     48 }
     49 
     50 // Updates from the file thread may have been posted while this saving job
     51 // was being canceled in the UI thread, so we'll accept them unless we're
     52 // complete.
     53 void SaveItem::Update(int64 bytes_so_far) {
     54   if (state_ != IN_PROGRESS) {
     55     NOTREACHED();
     56     return;
     57   }
     58   UpdateSize(bytes_so_far);
     59 }
     60 
     61 // Cancel this saving item job. If the job is not in progress, ignore
     62 // this command. The SavePackage will each in-progress SaveItem's cancel
     63 // when canceling whole saving page job.
     64 void SaveItem::Cancel() {
     65   // If item is in WAIT_START mode, which means no request has been sent.
     66   // So we need not to cancel it.
     67   if (state_ != IN_PROGRESS) {
     68     // Small downloads might be complete before method has a chance to run.
     69     return;
     70   }
     71   state_ = CANCELED;
     72   is_success_ = false;
     73   Finish(received_bytes_, false);
     74   package_->SaveCanceled(this);
     75 }
     76 
     77 // Set finish state for a save item
     78 void SaveItem::Finish(int64 size, bool is_success) {
     79   // When this function is called, the SaveItem should be one of following
     80   // three situations.
     81   // a) The data of this SaveItem is finished saving. So it should have
     82   // generated final name.
     83   // b) Error happened before the start of saving process. So no |save_id_| is
     84   // generated for this SaveItem and the |is_success_| should be false.
     85   // c) Error happened in the start of saving process, the SaveItem has a save
     86   // id, |is_success_| should be false, and the |size| should be 0.
     87   DCHECK(has_final_name() || (save_id_ == -1 && !is_success_) ||
     88          (save_id_ != -1 && !is_success_ && !size));
     89   state_ = COMPLETE;
     90   is_success_ = is_success;
     91   UpdateSize(size);
     92 }
     93 
     94 // Calculate the percentage of the save item
     95 int SaveItem::PercentComplete() const {
     96   switch (state_) {
     97     case COMPLETE:
     98     case CANCELED:
     99       return 100;
    100     case WAIT_START:
    101       return 0;
    102     case IN_PROGRESS: {
    103       int percent = 0;
    104       if (total_bytes_ > 0)
    105         percent = static_cast<int>(received_bytes_ * 100.0 / total_bytes_);
    106       return percent;
    107     }
    108     default: {
    109       NOTREACHED();
    110       return -1;
    111     }
    112   }
    113 }
    114 
    115 // Rename the save item with new path.
    116 void SaveItem::Rename(const base::FilePath& full_path) {
    117   DCHECK(!full_path.empty() && !has_final_name());
    118   full_path_ = full_path;
    119   file_name_ = full_path_.BaseName();
    120   has_final_name_ = true;
    121 }
    122 
    123 void SaveItem::SetSaveId(int32 save_id) {
    124   DCHECK_EQ(-1, save_id_);
    125   save_id_ = save_id;
    126 }
    127 
    128 void SaveItem::SetTotalBytes(int64 total_bytes) {
    129   DCHECK_EQ(0, total_bytes_);
    130   total_bytes_ = total_bytes;
    131 }
    132 
    133 }  // namespace content
    134