// Copyright (c) 2013 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/browser/net/url_request_buffer_job.h" #include #include #include #include "base/strings/string_number_conversions.h" #include "base/strings/utf_string_conversions.h" #include "base/task/post_task.h" #include "content/public/browser/browser_task_traits.h" #include "content/public/browser/browser_thread.h" #include "net/base/mime_util.h" #include "net/base/net_errors.h" #include "shell/common/atom_constants.h" #include "shell/common/native_mate_converters/net_converter.h" #include "shell/common/native_mate_converters/v8_value_converter.h" namespace electron { namespace { std::string GetExtFromURL(const GURL& url) { std::string spec = url.spec(); size_t index = spec.find_last_of('.'); if (index == std::string::npos || index == spec.size()) return std::string(); return spec.substr(index + 1, spec.size() - index - 1); } void BeforeStartInUI(base::WeakPtr job, mate::Arguments* args) { v8::Local value; int error = net::OK; std::unique_ptr request_options = nullptr; if (args->GetNext(&value)) { V8ValueConverter converter; v8::Local context = args->isolate()->GetCurrentContext(); request_options = converter.FromV8Value(value, context); } if (request_options) { JsAsker::IsErrorOptions(request_options.get(), &error); } else { error = net::ERR_NOT_IMPLEMENTED; } base::PostTaskWithTraits(FROM_HERE, {content::BrowserThread::IO}, base::BindOnce(&URLRequestBufferJob::StartAsync, job, std::move(request_options), error)); } } // namespace URLRequestBufferJob::URLRequestBufferJob(net::URLRequest* request, net::NetworkDelegate* network_delegate) : net::URLRequestSimpleJob(request, network_delegate), status_code_(net::HTTP_NOT_IMPLEMENTED), weak_factory_(this) {} URLRequestBufferJob::~URLRequestBufferJob() = default; void URLRequestBufferJob::Start() { auto request_details = std::make_unique(); FillRequestDetails(request_details.get(), request()); base::PostTaskWithTraits( FROM_HERE, {content::BrowserThread::UI}, base::BindOnce( &JsAsker::AskForOptions, base::Unretained(isolate()), handler(), std::move(request_details), base::BindOnce(&BeforeStartInUI, weak_factory_.GetWeakPtr()))); } void URLRequestBufferJob::StartAsync(std::unique_ptr options, int error) { if (error != net::OK) { NotifyStartError( net::URLRequestStatus(net::URLRequestStatus::FAILED, error)); return; } const base::Value* binary = nullptr; if (options->is_dict()) { base::DictionaryValue* dict = static_cast(options.get()); dict->GetString("mimeType", &mime_type_); dict->GetString("charset", &charset_); dict->GetBinary("data", &binary); } else if (options->is_blob()) { binary = options.get(); } if (mime_type_.empty()) { std::string ext = GetExtFromURL(request()->url()); #if defined(OS_WIN) net::GetWellKnownMimeTypeFromExtension(base::UTF8ToUTF16(ext), &mime_type_); #else net::GetWellKnownMimeTypeFromExtension(ext, &mime_type_); #endif } if (!binary) { NotifyStartError(net::URLRequestStatus(net::URLRequestStatus::FAILED, net::ERR_NOT_IMPLEMENTED)); return; } data_ = new base::RefCountedBytes( reinterpret_cast(binary->GetBlob().data()), binary->GetBlob().size()); status_code_ = net::HTTP_OK; net::URLRequestSimpleJob::Start(); } void URLRequestBufferJob::Kill() { weak_factory_.InvalidateWeakPtrs(); net::URLRequestSimpleJob::Kill(); } void URLRequestBufferJob::GetResponseInfo(net::HttpResponseInfo* info) { std::string status("HTTP/1.1 "); status.append(base::NumberToString(status_code_)); status.append(" "); status.append(net::GetHttpReasonPhrase(status_code_)); status.append("\0\0", 2); auto* headers = new net::HttpResponseHeaders(status); headers->AddHeader(kCORSHeader); if (!mime_type_.empty()) { std::string content_type_header(net::HttpRequestHeaders::kContentType); content_type_header.append(": "); content_type_header.append(mime_type_); headers->AddHeader(content_type_header); } info->headers = headers; } int URLRequestBufferJob::GetRefCountedData( std::string* mime_type, std::string* charset, scoped_refptr* data, net::CompletionOnceCallback callback) const { *mime_type = mime_type_; *charset = charset_; *data = data_; return net::OK; } } // namespace electron