// 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 "shell/common/gin_converters/net_converter.h" #include #include #include #include #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/values.h" #include "gin/dictionary.h" #include "net/base/upload_bytes_element_reader.h" #include "net/base/upload_data_stream.h" #include "net/base/upload_element_reader.h" #include "net/base/upload_file_element_reader.h" #include "net/cert/x509_certificate.h" #include "net/cert/x509_util.h" #include "net/http/http_response_headers.h" #include "services/network/public/cpp/resource_request.h" #include "shell/common/gin_converters/gurl_converter.h" #include "shell/common/gin_converters/std_converter.h" #include "shell/common/gin_converters/string16_converter.h" #include "shell/common/gin_converters/value_converter_gin_adapter.h" #include "shell/common/node_includes.h" namespace gin { namespace { bool CertFromData(const std::string& data, scoped_refptr* out) { auto cert_list = net::X509Certificate::CreateCertificateListFromBytes( data.c_str(), data.length(), net::X509Certificate::FORMAT_SINGLE_CERTIFICATE); if (cert_list.empty()) return false; auto leaf_cert = cert_list.front(); if (!leaf_cert) return false; *out = leaf_cert; return true; } } // namespace // static v8::Local Converter::ToV8( v8::Isolate* isolate, const net::AuthChallengeInfo& val) { gin::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("isProxy", val.is_proxy); dict.Set("scheme", val.scheme); dict.Set("host", val.challenger.host()); dict.Set("port", static_cast(val.challenger.port())); dict.Set("realm", val.realm); return gin::ConvertToV8(isolate, dict); } // static v8::Local Converter>::ToV8( v8::Isolate* isolate, const scoped_refptr& val) { gin::Dictionary dict(isolate, v8::Object::New(isolate)); std::string encoded_data; net::X509Certificate::GetPEMEncoded(val->cert_buffer(), &encoded_data); dict.Set("data", encoded_data); dict.Set("issuer", val->issuer()); dict.Set("issuerName", val->issuer().GetDisplayName()); dict.Set("subject", val->subject()); dict.Set("subjectName", val->subject().GetDisplayName()); dict.Set("serialNumber", base::HexEncode(val->serial_number().data(), val->serial_number().size())); dict.Set("validStart", val->valid_start().ToDoubleT()); dict.Set("validExpiry", val->valid_expiry().ToDoubleT()); dict.Set("fingerprint", net::HashValue(val->CalculateFingerprint256(val->cert_buffer())) .ToString()); const auto& intermediate_buffers = val->intermediate_buffers(); if (!intermediate_buffers.empty()) { std::vector> issuer_intermediates; issuer_intermediates.reserve(intermediate_buffers.size() - 1); for (size_t i = 1; i < intermediate_buffers.size(); ++i) { issuer_intermediates.push_back( bssl::UpRef(intermediate_buffers[i].get())); } const scoped_refptr& issuer_cert = net::X509Certificate::CreateFromBuffer( bssl::UpRef(intermediate_buffers[0].get()), std::move(issuer_intermediates)); dict.Set("issuerCert", issuer_cert); } return ConvertToV8(isolate, dict); } bool Converter>::FromV8( v8::Isolate* isolate, v8::Local val, scoped_refptr* out) { gin::Dictionary dict(nullptr); if (!ConvertFromV8(isolate, val, &dict)) return false; std::string data; dict.Get("data", &data); scoped_refptr leaf_cert; if (!CertFromData(data, &leaf_cert)) return false; scoped_refptr issuer_cert; if (dict.Get("issuerCert", &issuer_cert)) { std::vector> intermediates; intermediates.push_back(bssl::UpRef(issuer_cert->cert_buffer())); auto cert = net::X509Certificate::CreateFromBuffer( bssl::UpRef(leaf_cert->cert_buffer()), std::move(intermediates)); if (!cert) return false; *out = cert; } else { *out = leaf_cert; } return true; } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const net::CertPrincipal& val) { gin::Dictionary dict(isolate, v8::Object::New(isolate)); dict.Set("commonName", val.common_name); dict.Set("organizations", val.organization_names); dict.Set("organizationUnits", val.organization_unit_names); dict.Set("locality", val.locality_name); dict.Set("state", val.state_or_province_name); dict.Set("country", val.country_name); return ConvertToV8(isolate, dict); } // static v8::Local Converter::ToV8( v8::Isolate* isolate, net::HttpResponseHeaders* headers) { base::DictionaryValue response_headers; if (headers) { size_t iter = 0; std::string key; std::string value; while (headers->EnumerateHeaderLines(&iter, &key, &value)) { key = base::ToLowerASCII(key); if (response_headers.FindKey(key)) { base::ListValue* values = nullptr; if (response_headers.GetList(key, &values)) values->AppendString(value); } else { auto values = std::make_unique(); values->AppendString(value); response_headers.Set(key, std::move(values)); } } } return ConvertToV8(isolate, response_headers); } bool Converter::FromV8( v8::Isolate* isolate, v8::Local val, net::HttpResponseHeaders* out) { if (!val->IsObject()) { return false; } auto addHeaderFromValue = [&isolate, &out]( const std::string& key, const v8::Local& localVal) { auto context = isolate->GetCurrentContext(); v8::Local localStrVal; if (!localVal->ToString(context).ToLocal(&localStrVal)) { return false; } std::string value; gin::ConvertFromV8(isolate, localStrVal, &value); out->AddHeader(key + ": " + value); return true; }; auto context = isolate->GetCurrentContext(); auto headers = v8::Local::Cast(val); auto keys = headers->GetOwnPropertyNames(context).ToLocalChecked(); for (uint32_t i = 0; i < keys->Length(); i++) { v8::Local keyVal; if (!keys->Get(context, i).ToLocal(&keyVal)) { return false; } std::string key; gin::ConvertFromV8(isolate, keyVal, &key); auto localVal = headers->Get(context, keyVal).ToLocalChecked(); if (localVal->IsArray()) { auto values = v8::Local::Cast(localVal); for (uint32_t j = 0; j < values->Length(); j++) { if (!addHeaderFromValue(key, values->Get(context, j).ToLocalChecked())) { return false; } } } else { if (!addHeaderFromValue(key, localVal)) { return false; } } } return true; } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const net::HttpRequestHeaders& val) { gin::Dictionary headers(isolate, v8::Object::New(isolate)); for (net::HttpRequestHeaders::Iterator it(val); it.GetNext();) headers.Set(it.name(), it.value()); return ConvertToV8(isolate, headers); } // static bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, net::HttpRequestHeaders* out) { base::DictionaryValue dict; if (!ConvertFromV8(isolate, val, &dict)) return false; for (base::DictionaryValue::Iterator it(dict); !it.IsAtEnd(); it.Advance()) { if (it.value().is_string()) { std::string value = it.value().GetString(); out->SetHeader(it.key(), value); } } return true; } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const network::ResourceRequest& val) { gin::Dictionary dict(isolate, v8::Object::New(isolate)); dict.Set("method", val.method); dict.Set("url", val.url.spec()); dict.Set("referrer", val.referrer.spec()); dict.Set("headers", val.headers); if (val.request_body) { const auto& elements = *val.request_body->elements(); v8::Local arr = v8::Array::New(isolate, elements.size()); for (size_t i = 0; i < elements.size(); ++i) { const auto& element = elements[i]; gin::Dictionary upload_data(isolate, v8::Object::New(isolate)); switch (element.type()) { case network::mojom::DataElementType::kFile: upload_data.Set("file", element.path().value()); break; case network::mojom::DataElementType::kBytes: upload_data.Set("bytes", node::Buffer::Copy(isolate, element.bytes(), element.length()) .ToLocalChecked()); break; case network::mojom::DataElementType::kBlob: upload_data.Set("blobUUID", element.blob_uuid()); break; default: NOTREACHED() << "Found unsupported data element"; } arr->Set(isolate->GetCurrentContext(), static_cast(i), ConvertToV8(isolate, upload_data)) .Check(); } dict.Set("uploadData", arr); } return ConvertToV8(isolate, dict); } // static v8::Local Converter::ToV8( v8::Isolate* isolate, electron::VerifyRequestParams val) { gin::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("hostname", val.hostname); dict.Set("certificate", val.certificate); dict.Set("verificationResult", val.default_result); dict.Set("errorCode", val.error_code); return ConvertToV8(isolate, dict); } } // namespace gin namespace electron { void FillRequestDetails(base::DictionaryValue* details, const net::URLRequest* request) { details->SetString("method", request->method()); std::string url; if (!request->url_chain().empty()) url = request->url().spec(); details->SetKey("url", base::Value(url)); details->SetString("referrer", request->referrer()); auto list = std::make_unique(); GetUploadData(list.get(), request); if (!list->empty()) details->Set("uploadData", std::move(list)); auto headers_value = std::make_unique(); for (net::HttpRequestHeaders::Iterator it(request->extra_request_headers()); it.GetNext();) { headers_value->SetString(it.name(), it.value()); } details->Set("headers", std::move(headers_value)); } void GetUploadData(base::ListValue* upload_data_list, const net::URLRequest* request) { const net::UploadDataStream* upload_data = request->get_upload(); if (!upload_data) return; const std::vector>* readers = upload_data->GetElementReaders(); for (const auto& reader : *readers) { auto upload_data_dict = std::make_unique(); if (reader->AsBytesReader()) { const net::UploadBytesElementReader* bytes_reader = reader->AsBytesReader(); auto bytes = std::make_unique( std::vector(bytes_reader->bytes(), bytes_reader->bytes() + bytes_reader->length())); upload_data_dict->Set("bytes", std::move(bytes)); } else if (reader->AsFileReader()) { const net::UploadFileElementReader* file_reader = reader->AsFileReader(); auto file_path = file_reader->path().AsUTF8Unsafe(); upload_data_dict->SetKey("file", base::Value(file_path)); } // else { // const storage::UploadBlobElementReader* blob_reader = // static_cast(reader.get()); // upload_data_dict->SetString("blobUUID", blob_reader->uuid()); // } upload_data_list->Append(std::move(upload_data_dict)); } } } // namespace electron