15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
57d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)#include "chrome/browser/drive/drive_uploader.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <algorithm>
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/callback.h"
11a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/file_util.h"
122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/strings/string_number_conversions.h"
13a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/task_runner_util.h"
147d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)#include "chrome/browser/drive/drive_service_interface.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "chrome/browser/google_apis/gdata_wapi_parser.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/public/browser/browser_thread.h"
172a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "content/public/browser/power_save_blocker.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)using content::BrowserThread;
20eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::CancelCallback;
21eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::GDATA_CANCELLED;
22eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::GDataErrorCode;
23eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::GDATA_NO_SPACE;
24eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_CONFLICT;
25eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_CREATED;
26eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_FORBIDDEN;
27eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_NOT_FOUND;
28eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_PRECONDITION;
29eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_RESUME_INCOMPLETE;
30eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::HTTP_SUCCESS;
31eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::ProgressCallback;
32eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::ResourceEntry;
33eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochusing google_apis::UploadRangeResponse;
34eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch
35eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochnamespace drive {
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3758e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdochnamespace {
3858e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch// Upload data is split to multiple HTTP request each conveying kUploadChunkSize
3958e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch// bytes (except the request for uploading the last chunk of data).
4058e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch// The value must be a multiple of 512KB according to the spec of GData WAPI and
4158e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch// Drive API v2. It is set to a smaller value than 2^31 for working around
4258e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch// server side error (crbug.com/264089).
4358e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdochconst int64 kUploadChunkSize = (1LL << 30);  // 1GB
4458e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch}  // namespace
4558e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch
462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// Structure containing current upload information of file, passed between
472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// DriveServiceInterface methods and callbacks.
482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)struct DriveUploader::UploadFileInfo {
49eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  UploadFileInfo(const base::FilePath& local_path,
502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 const std::string& content_type,
51c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 const UploadCompletionCallback& callback,
52c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 const ProgressCallback& progress_callback)
53eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      : file_path(local_path),
542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        content_type(content_type),
552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        completion_callback(callback),
56c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        progress_callback(progress_callback),
572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        content_length(0),
582385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch        next_start_position(-1),
592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        power_save_blocker(content::PowerSaveBlocker::Create(
602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)            content::PowerSaveBlocker::kPowerSaveBlockPreventAppSuspension,
617d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)            "Upload in progress")),
627d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)        cancelled(false),
637d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)        weak_ptr_factory_(this) {
642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  ~UploadFileInfo() {
672a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Useful for printf debugging.
702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  std::string DebugString() const {
712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return "file_path=[" + file_path.AsUTF8Unsafe() +
722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)           "], content_type=[" + content_type +
732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)           "], content_length=[" + base::UintToString(content_length) +
742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)           "]";
752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
777d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // Returns the callback to cancel the upload represented by this struct.
787d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  CancelCallback GetCancelCallback() {
797d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    return base::Bind(&UploadFileInfo::Cancel, weak_ptr_factory_.GetWeakPtr());
807d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  }
817d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // The local file path of the file to be uploaded.
832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const base::FilePath file_path;
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Content-Type of file.
862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const std::string content_type;
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
882a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Callback to be invoked once the upload has finished.
892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  const UploadCompletionCallback completion_callback;
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
91c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  // Callback to periodically notify the upload progress.
92c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  const ProgressCallback progress_callback;
93c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Location URL where file is to be uploaded to, returned from
952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // InitiateUpload. Used for the subsequent ResumeUpload requests.
962a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  GURL upload_location;
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Header content-Length.
992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  int64 content_length;
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1012385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  int64 next_start_position;
1022385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch
1032a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // Blocks system suspend while upload is in progress.
1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  scoped_ptr<content::PowerSaveBlocker> power_save_blocker;
1057d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
1067d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // Fields for implementing cancellation. |cancel_callback| is non-null if
1077d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // there is an in-flight HTTP request. In that case, |cancell_callback| will
1087d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // cancel the operation. |cancelled| is initially false and turns to true
1097d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // once Cancel() is called. DriveUploader will check this field before after
1107d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // an async task other than HTTP requests and cancels the subsequent requests
1117d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // if this is flagged to true.
1127d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  CancelCallback cancel_callback;
1137d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  bool cancelled;
1147d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
1157d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles) private:
1167d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  // Cancels the upload represented by this struct.
1177d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  void Cancel() {
1187d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    cancelled = true;
1197d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    if (!cancel_callback.is_null())
1207d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)      cancel_callback.Run();
1217d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  }
1227d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
1237d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  base::WeakPtrFactory<UploadFileInfo> weak_ptr_factory_;
1247d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(UploadFileInfo);
1252a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)};
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
127eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen MurdochDriveUploader::DriveUploader(DriveServiceInterface* drive_service,
128eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch                             base::TaskRunner* blocking_task_runner)
1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    : drive_service_(drive_service),
130eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      blocking_task_runner_(blocking_task_runner),
131c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      weak_ptr_factory_(this) {
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)DriveUploader::~DriveUploader() {}
1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1367d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)CancelCallback DriveUploader::UploadNewFile(
1377d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const std::string& parent_resource_id,
1387d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const base::FilePath& local_file_path,
1397d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const std::string& title,
1407d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const std::string& content_type,
1417d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const UploadCompletionCallback& callback,
1427d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    const ProgressCallback& progress_callback) {
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!parent_resource_id.empty());
1452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!local_file_path.empty());
1462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!title.empty());
1472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!content_type.empty());
1482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!callback.is_null());
1492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1507d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  return StartUploadFile(
151eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      scoped_ptr<UploadFileInfo>(new UploadFileInfo(local_file_path,
1522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                                    content_type,
153c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                                    callback,
154c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                                    progress_callback)),
1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&DriveUploader::StartInitiateUploadNewFile,
1562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
1572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 parent_resource_id,
1582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 title));
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1617d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)CancelCallback DriveUploader::UploadExistingFile(
1622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& resource_id,
1632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const base::FilePath& local_file_path,
1642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& content_type,
1652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& etag,
166c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    const UploadCompletionCallback& callback,
167c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    const ProgressCallback& progress_callback) {
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
1692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!resource_id.empty());
1702a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!local_file_path.empty());
1712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!content_type.empty());
1722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(!callback.is_null());
1732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1747d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  return StartUploadFile(
175eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      scoped_ptr<UploadFileInfo>(new UploadFileInfo(local_file_path,
1762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                                                    content_type,
177c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                                    callback,
178c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                                    progress_callback)),
1792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&DriveUploader::StartInitiateUploadExistingFile,
1802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
1812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 resource_id,
1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 etag));
1835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1857d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)CancelCallback DriveUploader::ResumeUploadFile(
18690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const GURL& upload_location,
18790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const base::FilePath& local_file_path,
18890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const std::string& content_type,
18990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const UploadCompletionCallback& callback,
19090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const ProgressCallback& progress_callback) {
19190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
19290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(!local_file_path.empty());
19390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(!content_type.empty());
19490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(!callback.is_null());
19590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
19690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  scoped_ptr<UploadFileInfo> upload_file_info(new UploadFileInfo(
197eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      local_file_path, content_type,
19890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      callback, progress_callback));
19990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  upload_file_info->upload_location = upload_location;
20090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
2017d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  return StartUploadFile(
20290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      upload_file_info.Pass(),
20390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      base::Bind(&DriveUploader::StartGetUploadStatus,
20490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr()));
20590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)}
20690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
2077d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)CancelCallback DriveUploader::StartUploadFile(
2082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info,
2092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const StartInitiateUploadCallback& start_initiate_upload_callback) {
2102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
2112a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DVLOG(1) << "Uploading file: " << upload_file_info->DebugString();
2122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  UploadFileInfo* info_ptr = upload_file_info.get();
2142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  base::PostTaskAndReplyWithResult(
215eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      blocking_task_runner_.get(),
2162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      FROM_HERE,
217eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      base::Bind(&file_util::GetFileSize,
218eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch                 info_ptr->file_path,
219a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                 &info_ptr->content_length),
220a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      base::Bind(&DriveUploader::StartUploadFileAfterGetFileSize,
2212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
2222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 base::Passed(&upload_file_info),
2232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 start_initiate_upload_callback));
2247d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  return info_ptr->GetCancelCallback();
2255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
227a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void DriveUploader::StartUploadFileAfterGetFileSize(
2282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info,
2292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const StartInitiateUploadCallback& start_initiate_upload_callback,
230a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    bool get_file_size_result) {
2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
233a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!get_file_size_result) {
234c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    UploadFailed(upload_file_info.Pass(), HTTP_NOT_FOUND);
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
2362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
237a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK_GE(upload_file_info->content_length, 0);
2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2397d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  if (upload_file_info->cancelled) {
2407d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    UploadFailed(upload_file_info.Pass(), GDATA_CANCELLED);
2417d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    return;
2427d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  }
2432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  start_initiate_upload_callback.Run(upload_file_info.Pass());
2442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2462a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void DriveUploader::StartInitiateUploadNewFile(
2472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& parent_resource_id,
2482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& title,
2492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info) {
2502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
2515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2522a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  UploadFileInfo* info_ptr = upload_file_info.get();
2537d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  info_ptr->cancel_callback = drive_service_->InitiateUploadNewFile(
2542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_type,
2552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_length,
2562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      parent_resource_id,
2572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      title,
2582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&DriveUploader::OnUploadLocationReceived,
2592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
2602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 base::Passed(&upload_file_info)));
2615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2632a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void DriveUploader::StartInitiateUploadExistingFile(
2642a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& resource_id,
2652a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const std::string& etag,
2662a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info) {
2675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
2685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2692a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  UploadFileInfo* info_ptr = upload_file_info.get();
2707d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  info_ptr->cancel_callback = drive_service_->InitiateUploadExistingFile(
2712a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_type,
2722a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_length,
2732a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      resource_id,
2742a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      etag,
2752a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&DriveUploader::OnUploadLocationReceived,
2762a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
2772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)                 base::Passed(&upload_file_info)));
2782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2792a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
2802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void DriveUploader::OnUploadLocationReceived(
2812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info,
2822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    GDataErrorCode code,
2832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const GURL& upload_location) {
2842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
2855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DVLOG(1) << "Got upload location [" << upload_location.spec()
287eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch           << "] for [" << upload_file_info->file_path.value() << "]";
2885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (code != HTTP_SUCCESS) {
2905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // TODO(achuith): Handle error codes from Google Docs server.
2912a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    if (code == HTTP_PRECONDITION) {
2922a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      // ETag mismatch.
293c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      UploadFailed(upload_file_info.Pass(), HTTP_CONFLICT);
2942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      return;
2952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    }
296c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    UploadFailed(upload_file_info.Pass(), code);
2975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
2985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  upload_file_info->upload_location = upload_location;
3012385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  upload_file_info->next_start_position = 0;
3022385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  UploadNextChunk(upload_file_info.Pass());
3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
30590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)void DriveUploader::StartGetUploadStatus(
30690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info) {
30790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
30890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  DCHECK(upload_file_info);
30990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
31090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  UploadFileInfo* info_ptr = upload_file_info.get();
3117d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  info_ptr->cancel_callback = drive_service_->GetUploadStatus(
31290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      info_ptr->upload_location,
31390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      info_ptr->content_length,
31490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      base::Bind(&DriveUploader::OnUploadRangeResponseReceived,
31590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
31690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                 base::Passed(&upload_file_info)));
31790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)}
31890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
319a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void DriveUploader::UploadNextChunk(
3202385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch    scoped_ptr<UploadFileInfo> upload_file_info) {
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
322a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(upload_file_info);
3232385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  DCHECK_GE(upload_file_info->next_start_position, 0);
3242385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  DCHECK_LE(upload_file_info->next_start_position,
3252385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch            upload_file_info->content_length);
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3277d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  if (upload_file_info->cancelled) {
3287d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    UploadFailed(upload_file_info.Pass(), GDATA_CANCELLED);
3297d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    return;
3307d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  }
3317d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
33258e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch  // Limit the size of data uploaded per each request by kUploadChunkSize.
3332385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  const int64 end_position = std::min(
3342385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch      upload_file_info->content_length,
3352385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch      upload_file_info->next_start_position + kUploadChunkSize);
33658e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch
3372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  UploadFileInfo* info_ptr = upload_file_info.get();
3387d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  info_ptr->cancel_callback = drive_service_->ResumeUpload(
3392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->upload_location,
3402385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch      info_ptr->next_start_position,
34158e6fbe4ee35d65e14b626c557d37565bf8ad179Ben Murdoch      end_position,
3422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_length,
3432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      info_ptr->content_type,
344a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      info_ptr->file_path,
3452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      base::Bind(&DriveUploader::OnUploadRangeResponseReceived,
3465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
347c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 base::Passed(&upload_file_info)),
348c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      base::Bind(&DriveUploader::OnUploadProgress,
349c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 weak_ptr_factory_.GetWeakPtr(),
350c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 info_ptr->progress_callback,
3512385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch                 info_ptr->next_start_position,
352c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                 info_ptr->content_length));
3535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
3545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void DriveUploader::OnUploadRangeResponseReceived(
3562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<UploadFileInfo> upload_file_info,
3572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    const UploadRangeResponse& response,
3582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    scoped_ptr<ResourceEntry> entry) {
3595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
3605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
361c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (response.code == HTTP_CREATED || response.code == HTTP_SUCCESS) {
36290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // When uploading a new file, we expect HTTP_CREATED, and when uploading
36390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // an existing file (to overwrite), we expect HTTP_SUCCESS.
36490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // There is an exception: if we uploading an empty file, uploading a new
36590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // file also returns HTTP_SUCCESS on Drive API v2. The correct way of the
36690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // fix should be uploading the metadata only. However, to keep the
367c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    // compatibility with GData WAPI during the migration period, we just
368c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    // relax the condition here.
369c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    // TODO(hidehiko): Upload metadata only for empty files, after GData WAPI
370c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    // code is gone.
3715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    DVLOG(1) << "Successfully created uploaded file=["
372eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch             << upload_file_info->file_path.value() << "]";
3735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Done uploading.
37590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    upload_file_info->completion_callback.Run(
37690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        HTTP_SUCCESS, GURL(), entry.Pass());
3772a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return;
3782a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
3795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3802a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  // ETag mismatch.
3812a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (response.code == HTTP_PRECONDITION) {
382c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    UploadFailed(upload_file_info.Pass(), HTTP_CONFLICT);
3835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
3845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
3855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
386a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // If code is 308 (RESUME_INCOMPLETE) and |range_received| starts with 0
387a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // (meaning that the data is uploaded from the beginning of the file),
388a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // proceed to upload the next chunk.
3895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (response.code != HTTP_RESUME_INCOMPLETE ||
390a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      response.start_position_received != 0) {
391a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    LOG(ERROR)
392a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        << "UploadNextChunk http code=" << response.code
3932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        << ", start_position_received=" << response.start_position_received
394a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        << ", end_position_received=" << response.end_position_received;
395a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    UploadFailed(
396a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        upload_file_info.Pass(),
397a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        response.code == HTTP_FORBIDDEN ? GDATA_NO_SPACE : response.code);
3985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
3995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
4005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DVLOG(1) << "Received range " << response.start_position_received
4022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)           << "-" << response.end_position_received
403eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch           << " for [" << upload_file_info->file_path.value() << "]";
4045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4052385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  upload_file_info->next_start_position = response.end_position_received;
4062385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  UploadNextChunk(upload_file_info.Pass());
4075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
409c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)void DriveUploader::OnUploadProgress(const ProgressCallback& callback,
410c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                     int64 start_position,
411c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                     int64 total_size,
412c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                     int64 progress_of_chunk,
413c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                     int64 total_of_chunk) {
414c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (!callback.is_null())
415c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    callback.Run(start_position + progress_of_chunk, total_size);
416c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}
417c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
4182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void DriveUploader::UploadFailed(scoped_ptr<UploadFileInfo> upload_file_info,
419c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                 GDataErrorCode error) {
4205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
4215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  LOG(ERROR) << "Upload failed " << upload_file_info->DebugString();
4235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
4242385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  if (upload_file_info->next_start_position < 0) {
4252385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch    // Discard the upload location because no request could succeed with it.
4262385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch    // Maybe it's obsolete.
4272385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch    upload_file_info->upload_location = GURL();
4282385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch  }
4292385ea399aae016c0806a4f9ef3c9cfe3d2a39dfBen Murdoch
43090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  upload_file_info->completion_callback.Run(
43190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      error, upload_file_info->upload_location, scoped_ptr<ResourceEntry>());
4325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
4335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
434eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch}  // namespace drive
435