// Copyright (c) 2019 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/browser/net/atom_url_loader_factory.h" #include #include #include "atom/browser/api/atom_api_session.h" #include "atom/browser/atom_browser_context.h" #include "atom/common/atom_constants.h" #include "atom/common/native_mate_converters/file_path_converter.h" #include "atom/common/native_mate_converters/gurl_converter.h" #include "atom/common/native_mate_converters/net_converter.h" #include "atom/common/native_mate_converters/value_converter.h" #include "base/guid.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/file_url_loader.h" #include "content/public/browser/storage_partition.h" #include "native_mate/dictionary.h" #include "net/base/filename_util.h" #include "services/network/public/cpp/url_loader_completion_status.h" #include "services/network/public/mojom/url_loader.mojom.h" #include "atom/common/node_includes.h" using content::BrowserThread; namespace atom { AtomURLLoaderFactory::AtomURLLoaderFactory(ProtocolType type, const ProtocolHandler& handler) : type_(type), handler_(handler), weak_factory_(this) {} AtomURLLoaderFactory::~AtomURLLoaderFactory() = default; void AtomURLLoaderFactory::CreateLoaderAndStart( network::mojom::URLLoaderRequest loader, int32_t routing_id, int32_t request_id, uint32_t options, const network::ResourceRequest& request, network::mojom::URLLoaderClientPtr client, const net::MutableNetworkTrafficAnnotationTag& traffic_annotation) { DCHECK_CURRENTLY_ON(content::BrowserThread::UI); v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::Locker locker(isolate); v8::HandleScope handle_scope(isolate); v8::Local context = isolate->GetCurrentContext(); v8::Context::Scope context_scope(context); switch (type_) { case ProtocolType::kBuffer: handler_.Run(request, base::BindOnce(&AtomURLLoaderFactory::SendResponseBuffer, weak_factory_.GetWeakPtr(), std::move(client), isolate)); break; case ProtocolType::kString: handler_.Run(request, base::BindOnce(&AtomURLLoaderFactory::SendResponseString, weak_factory_.GetWeakPtr(), std::move(client), isolate)); break; case ProtocolType::kFile: handler_.Run(request, base::BindOnce(&AtomURLLoaderFactory::SendResponseFile, weak_factory_.GetWeakPtr(), std::move(loader), request, std::move(client), isolate)); break; case ProtocolType::kHttp: handler_.Run( request, base::BindOnce(&AtomURLLoaderFactory::SendResponseHttp, weak_factory_.GetWeakPtr(), std::move(loader), routing_id, request_id, options, request, std::move(client), traffic_annotation, isolate)); break; default: { std::string contents = "Not Implemented"; SendContents(std::move(client), "text/html", "utf-8", contents.data(), contents.size()); } } } void AtomURLLoaderFactory::Clone( network::mojom::URLLoaderFactoryRequest request) { bindings_.AddBinding(this, std::move(request)); } void AtomURLLoaderFactory::SendResponseBuffer( network::mojom::URLLoaderClientPtr client, v8::Isolate* isolate, v8::Local response) { if (HandleError(&client, isolate, response)) return; std::string mime_type = "text/html"; std::string charset = "utf-8"; v8::Local buffer; if (node::Buffer::HasInstance(response)) { buffer = response; } else if (response->IsObject()) { mate::Dictionary dict( isolate, response->ToObject(isolate->GetCurrentContext()).ToLocalChecked()); dict.Get("mimeType", &mime_type); dict.Get("charset", &charset); dict.Get("data", &buffer); if (!node::Buffer::HasInstance(response)) buffer = v8::Local(); } if (buffer.IsEmpty()) { network::URLLoaderCompletionStatus status; status.error_code = net::ERR_NOT_IMPLEMENTED; client->OnComplete(status); return; } SendContents(std::move(client), std::move(mime_type), std::move(charset), node::Buffer::Data(buffer), node::Buffer::Length(buffer)); } void AtomURLLoaderFactory::SendResponseString( network::mojom::URLLoaderClientPtr client, v8::Isolate* isolate, v8::Local response) { if (HandleError(&client, isolate, response)) return; std::string mime_type = "text/html"; std::string charset = "utf-8"; std::string contents; if (response->IsString()) { contents = gin::V8ToString(isolate, response); } else if (response->IsObject()) { mate::Dictionary dict( isolate, response->ToObject(isolate->GetCurrentContext()).ToLocalChecked()); dict.Get("mimeType", &mime_type); dict.Get("charset", &charset); dict.Get("data", &contents); } SendContents(std::move(client), std::move(mime_type), std::move(charset), contents.data(), contents.size()); } void AtomURLLoaderFactory::SendResponseFile( network::mojom::URLLoaderRequest loader, network::ResourceRequest request, network::mojom::URLLoaderClientPtr client, v8::Isolate* isolate, v8::Local response) { if (HandleError(&client, isolate, response)) return; base::FilePath path; scoped_refptr response_headers; if (mate::ConvertFromV8(isolate, response, &path)) { request.url = net::FilePathToFileURL(path); } else if (response->IsObject()) { mate::Dictionary dict( isolate, response->ToObject(isolate->GetCurrentContext()).ToLocalChecked()); dict.Get("referrer", &request.referrer); dict.Get("method", &request.method); if (dict.Get("path", &path)) request.url = net::FilePathToFileURL(path); base::DictionaryValue headers; if (dict.Get("headers", &headers)) { response_headers = new net::HttpResponseHeaders("HTTP/1.1 200 OK"); response_headers->AddHeader(kCORSHeader); for (const auto& iter : headers.DictItems()) response_headers->AddHeader(iter.first + ": " + iter.second.GetString()); } } else { network::URLLoaderCompletionStatus status; status.error_code = net::ERR_NOT_IMPLEMENTED; client->OnComplete(status); return; } content::CreateFileURLLoader(request, std::move(loader), std::move(client), nullptr, false, response_headers); } void AtomURLLoaderFactory::SendResponseHttp( network::mojom::URLLoaderRequest loader, int32_t routing_id, int32_t request_id, uint32_t options, const network::ResourceRequest& original_request, network::mojom::URLLoaderClientPtr client, const net::MutableNetworkTrafficAnnotationTag& traffic_annotation, v8::Isolate* isolate, v8::Local response) { if (HandleError(&client, isolate, response)) return; if (!response->IsObject()) { network::URLLoaderCompletionStatus status; status.error_code = net::ERR_NOT_IMPLEMENTED; client->OnComplete(status); return; } network::ResourceRequest request; request.headers = original_request.headers; request.cors_exempt_headers = original_request.cors_exempt_headers; mate::Dictionary dict( isolate, response->ToObject(isolate->GetCurrentContext()).ToLocalChecked()); dict.Get("url", &request.url); dict.Get("referrer", &request.referrer); if (!dict.Get("method", &request.method)) request.method = original_request.method; scoped_refptr browser_context = AtomBrowserContext::From("", false); v8::Local value; if (dict.Get("session", &value)) { if (value->IsNull()) { browser_context = AtomBrowserContext::From(base::GenerateGUID(), true); } else { mate::Handle session; if (mate::ConvertFromV8(isolate, value, &session) && !session.IsEmpty()) { browser_context = session->browser_context(); } } } scoped_refptr url_loader_factory = content::BrowserContext::GetDefaultStoragePartition(browser_context.get()) ->GetURLLoaderFactoryForBrowserProcess(); url_loader_factory->CreateLoaderAndStart( std::move(loader), routing_id, request_id, options, std::move(request), std::move(client), traffic_annotation); } bool AtomURLLoaderFactory::HandleError( network::mojom::URLLoaderClientPtr* client, v8::Isolate* isolate, v8::Local response) { if (!response->IsObject()) return false; v8::Local obj = response->ToObject(isolate->GetCurrentContext()).ToLocalChecked(); network::URLLoaderCompletionStatus status; if (!mate::Dictionary(isolate, obj).Get("error", &status.error_code)) return false; std::move(*client)->OnComplete(status); return true; } void AtomURLLoaderFactory::SendContents( network::mojom::URLLoaderClientPtr client, std::string mime_type, std::string charset, const char* data, size_t ssize) { uint32_t size = base::saturated_cast(ssize); mojo::DataPipe pipe(size); MojoResult result = pipe.producer_handle->WriteData(data, &size, MOJO_WRITE_DATA_FLAG_NONE); if (result != MOJO_RESULT_OK || size < ssize) { client->OnComplete(network::URLLoaderCompletionStatus(net::ERR_FAILED)); return; } network::ResourceResponseHead head; head.mime_type = std::move(mime_type); head.charset = std::move(charset); head.headers = new net::HttpResponseHeaders("HTTP/1.1 200 OK"); head.headers->AddHeader(kCORSHeader); client->OnReceiveResponse(head); client->OnStartLoadingResponseBody(std::move(pipe.consumer_handle)); client->OnComplete(network::URLLoaderCompletionStatus(net::OK)); } } // namespace atom