blob: ca700e12d9c18454f1fec2a849b26c14deb0364a [file] [log] [blame]
// Copyright 2013 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.
#include "chrome/browser/extensions/api/image_writer_private/write_from_url_operation.h"
#include "base/files/file_util.h"
#include "chrome/browser/extensions/api/image_writer_private/error_messages.h"
#include "chrome/browser/extensions/api/image_writer_private/operation_manager.h"
#include "content/public/browser/browser_thread.h"
#include "net/traffic_annotation/network_traffic_annotation.h"
#include "net/url_request/url_fetcher.h"
namespace extensions {
namespace image_writer {
using content::BrowserThread;
WriteFromUrlOperation::WriteFromUrlOperation(
base::WeakPtr<OperationManager> manager,
const ExtensionId& extension_id,
net::URLRequestContextGetter* request_context,
GURL url,
const std::string& hash,
const std::string& device_path,
const base::FilePath& download_folder)
: Operation(manager, extension_id, device_path, download_folder),
request_context_(request_context),
url_(url),
hash_(hash),
download_continuation_() {}
WriteFromUrlOperation::~WriteFromUrlOperation() {
}
void WriteFromUrlOperation::StartImpl() {
DCHECK(IsRunningInCorrectSequence());
GetDownloadTarget(base::BindOnce(
&WriteFromUrlOperation::Download, this,
base::BindOnce(
&WriteFromUrlOperation::VerifyDownload, this,
base::BindOnce(
&WriteFromUrlOperation::Unzip, this,
base::Bind(&WriteFromUrlOperation::Write, this,
base::Bind(&WriteFromUrlOperation::VerifyWrite, this,
base::Bind(&WriteFromUrlOperation::Finish,
this)))))));
}
void WriteFromUrlOperation::GetDownloadTarget(base::OnceClosure continuation) {
DCHECK(IsRunningInCorrectSequence());
if (IsCancelled()) {
return;
}
if (url_.ExtractFileName().empty()) {
if (!base::CreateTemporaryFileInDir(temp_dir_.GetPath(), &image_path_)) {
Error(error::kTempFileError);
return;
}
} else {
base::FilePath file_name =
base::FilePath::FromUTF8Unsafe(url_.ExtractFileName());
image_path_ = temp_dir_.GetPath().Append(file_name);
}
PostTask(std::move(continuation));
}
void WriteFromUrlOperation::Download(base::OnceClosure continuation) {
DCHECK(IsRunningInCorrectSequence());
if (IsCancelled()) {
return;
}
download_continuation_ = std::move(continuation);
SetStage(image_writer_api::STAGE_DOWNLOAD);
// Create traffic annotation tag.
net::NetworkTrafficAnnotationTag traffic_annotation =
net::DefineNetworkTrafficAnnotation("cros_recovery_image_download", R"(
semantics {
sender: "Chrome OS Recovery Utility"
description:
"The Google Chrome OS recovery utility downloads the recovery "
"image from Google Download Server."
trigger:
"User uses the Chrome OS Recovery Utility app/extension, selects "
"a Chrome OS recovery image, and clicks the Create button to write "
"the image to a USB or SD card."
data:
"URL of the image file to be downloaded. No other data or user "
"identifier is sent."
destination: GOOGLE_OWNED_SERVICE
}
policy {
cookies_allowed: YES
cookies_store: "user"
setting:
"This feature cannot be disabled by settings, it can only be used "
"by whitelisted apps/extension."
policy_exception_justification:
"Not implemented, considered not useful."
})");
// Store the URL fetcher on this object so that it is destroyed before this
// object is.
url_fetcher_ = net::URLFetcher::Create(url_, net::URLFetcher::GET, this,
traffic_annotation);
url_fetcher_->SetRequestContext(request_context_);
url_fetcher_->SaveResponseToFileAtPath(image_path_, task_runner());
AddCleanUpFunction(
base::BindOnce(&WriteFromUrlOperation::DestroyUrlFetcher, this));
url_fetcher_->Start();
}
void WriteFromUrlOperation::DestroyUrlFetcher() { url_fetcher_.reset(); }
void WriteFromUrlOperation::OnURLFetchUploadProgress(
const net::URLFetcher* source,
int64_t current,
int64_t total) {
// No-op
}
void WriteFromUrlOperation::OnURLFetchDownloadProgress(
const net::URLFetcher* source,
int64_t current,
int64_t total,
int64_t current_network_bytes) {
DCHECK(IsRunningInCorrectSequence());
if (IsCancelled()) {
url_fetcher_.reset(NULL);
}
int progress = (kProgressComplete * current) / total;
SetProgress(progress);
}
void WriteFromUrlOperation::OnURLFetchComplete(const net::URLFetcher* source) {
DCHECK(IsRunningInCorrectSequence());
if (source->GetStatus().is_success() && source->GetResponseCode() == 200) {
SetProgress(kProgressComplete);
std::move(download_continuation_).Run();
} else {
Error(error::kDownloadInterrupted);
}
}
void WriteFromUrlOperation::VerifyDownload(base::OnceClosure continuation) {
DCHECK(IsRunningInCorrectSequence());
if (IsCancelled()) {
return;
}
// Skip verify if no hash.
if (hash_.empty()) {
PostTask(std::move(continuation));
return;
}
SetStage(image_writer_api::STAGE_VERIFYDOWNLOAD);
GetMD5SumOfFile(image_path_, 0, 0, kProgressComplete,
base::BindOnce(&WriteFromUrlOperation::VerifyDownloadCompare,
this, std::move(continuation)));
}
void WriteFromUrlOperation::VerifyDownloadCompare(
base::OnceClosure continuation,
const std::string& download_hash) {
DCHECK(IsRunningInCorrectSequence());
if (download_hash != hash_) {
Error(error::kDownloadHashError);
return;
}
PostTask(base::BindOnce(&WriteFromUrlOperation::VerifyDownloadComplete, this,
std::move(continuation)));
}
void WriteFromUrlOperation::VerifyDownloadComplete(
base::OnceClosure continuation) {
DCHECK(IsRunningInCorrectSequence());
if (IsCancelled()) {
return;
}
SetProgress(kProgressComplete);
PostTask(std::move(continuation));
}
} // namespace image_writer
} // namespace extensions