// Copyright (c) 2012 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.
#ifndef CHROME_BROWSER_DRIVE_DRIVE_UPLOADER_H_
#define CHROME_BROWSER_DRIVE_DRIVE_UPLOADER_H_
#include <string>
#include "base/basictypes.h"
#include "base/callback_forward.h"
#include "base/memory/ref_counted.h"
#include "base/memory/weak_ptr.h"
#include "chrome/browser/drive/drive_service_interface.h"
#include "google_apis/drive/gdata_errorcode.h"
#include "google_apis/drive/gdata_wapi_parser.h"
class GURL;
namespace base {
class FilePath;
class TaskRunner;
}
namespace google_apis {
struct UploadRangeResponse;
}
namespace drive {
class DriveServiceInterface;
// Callback to be invoked once the upload has completed.
// |upload_location| will be returned when the uploading process is started but
// terminated before the completion due to some errors. It can be used to
// resume it.
typedef base::Callback<void(
google_apis::GDataErrorCode error,
const GURL& upload_location,
scoped_ptr<google_apis::ResourceEntry> resource_entry)>
UploadCompletionCallback;
class DriveUploaderInterface {
public:
virtual ~DriveUploaderInterface() {}
// Uploads a new file to a directory specified by |upload_location|.
// Returns a callback for cancelling the uploading job.
//
// parent_resource_id:
// resource id of the destination directory.
//
// local_file_path:
// The path to the local file to be uploaded.
//
// title:
// The title (file name) of the file to be uploaded.
//
// content_type:
// The content type of the file to be uploaded.
//
// callback:
// Called when an upload is done regardless of it was successful or not.
// Must not be null.
//
// progress_callback:
// Periodically called back with the total number of bytes sent so far.
// May be null if the information is not needed.
virtual google_apis::CancelCallback UploadNewFile(
const std::string& parent_resource_id,
const base::FilePath& local_file_path,
const std::string& title,
const std::string& content_type,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) = 0;
// Uploads an existing file (a file that already exists on Drive).
//
// See comments at UploadNewFile about common parameters and the return value.
//
// resource_id:
// resource id of the existing file to be overwritten.
//
// etag:
// Expected ETag for the destination file. If it does not match, the upload
// fails with UPLOAD_ERROR_CONFLICT.
// If |etag| is empty, the test is skipped.
virtual google_apis::CancelCallback UploadExistingFile(
const std::string& resource_id,
const base::FilePath& local_file_path,
const std::string& content_type,
const std::string& etag,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) = 0;
// Resumes the uploading process terminated before the completion.
// |upload_location| should be the one returned via UploadCompletionCallback
// for previous invocation. |drive_file_path|, |local_file_path| and
// |content_type| must be set to the same ones for previous invocation.
//
// See comments at UploadNewFile about common parameters and the return value.
virtual google_apis::CancelCallback ResumeUploadFile(
const GURL& upload_location,
const base::FilePath& local_file_path,
const std::string& content_type,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) = 0;
};
class DriveUploader : public DriveUploaderInterface {
public:
DriveUploader(DriveServiceInterface* drive_service,
base::TaskRunner* blocking_task_runner);
virtual ~DriveUploader();
// DriveUploaderInterface overrides.
virtual google_apis::CancelCallback UploadNewFile(
const std::string& parent_resource_id,
const base::FilePath& local_file_path,
const std::string& title,
const std::string& content_type,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) OVERRIDE;
virtual google_apis::CancelCallback UploadExistingFile(
const std::string& resource_id,
const base::FilePath& local_file_path,
const std::string& content_type,
const std::string& etag,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) OVERRIDE;
virtual google_apis::CancelCallback ResumeUploadFile(
const GURL& upload_location,
const base::FilePath& local_file_path,
const std::string& content_type,
const UploadCompletionCallback& callback,
const google_apis::ProgressCallback& progress_callback) OVERRIDE;
private:
struct UploadFileInfo;
typedef base::Callback<void(scoped_ptr<UploadFileInfo> upload_file_info)>
StartInitiateUploadCallback;
// Starts uploading a file with |upload_file_info|.
google_apis::CancelCallback StartUploadFile(
scoped_ptr<UploadFileInfo> upload_file_info,
const StartInitiateUploadCallback& start_initiate_upload_callback);
void StartUploadFileAfterGetFileSize(
scoped_ptr<UploadFileInfo> upload_file_info,
const StartInitiateUploadCallback& start_initiate_upload_callback,
bool get_file_size_result);
// Starts to initiate the new file uploading.
// Upon completion, OnUploadLocationReceived should be called.
void StartInitiateUploadNewFile(
const std::string& parent_resource_id,
const std::string& title,
scoped_ptr<UploadFileInfo> upload_file_info);
// Starts to initiate the existing file uploading.
// Upon completion, OnUploadLocationReceived should be called.
void StartInitiateUploadExistingFile(
const std::string& resource_id,
const std::string& etag,
scoped_ptr<UploadFileInfo> upload_file_info);
// DriveService callback for InitiateUpload.
void OnUploadLocationReceived(scoped_ptr<UploadFileInfo> upload_file_info,
google_apis::GDataErrorCode code,
const GURL& upload_location);
// Starts to get the current upload status for the file uploading.
// Upon completion, OnUploadRangeResponseReceived should be called.
void StartGetUploadStatus(scoped_ptr<UploadFileInfo> upload_file_info);
// Uploads the next chunk of data from the file.
void UploadNextChunk(scoped_ptr<UploadFileInfo> upload_file_info);
// DriveService callback for ResumeUpload.
void OnUploadRangeResponseReceived(
scoped_ptr<UploadFileInfo> upload_file_info,
const google_apis::UploadRangeResponse& response,
scoped_ptr<google_apis::ResourceEntry> entry);
void OnUploadProgress(const google_apis::ProgressCallback& callback,
int64 start_position,
int64 total_size,
int64 progress_of_chunk,
int64 total_of_chunk);
// Handle failed uploads.
void UploadFailed(scoped_ptr<UploadFileInfo> upload_file_info,
google_apis::GDataErrorCode error);
// The lifetime of this object should be guaranteed to exceed that of the
// DriveUploader instance.
DriveServiceInterface* drive_service_; // Not owned by this class.
scoped_refptr<base::TaskRunner> blocking_task_runner_;
// Note: This should remain the last member so it'll be destroyed and
// invalidate its weak pointers before any other members are destroyed.
base::WeakPtrFactory<DriveUploader> weak_ptr_factory_;
DISALLOW_COPY_AND_ASSIGN(DriveUploader);
};
} // namespace drive
#endif // CHROME_BROWSER_DRIVE_DRIVE_UPLOADER_H_