// Copyright (c) 2019 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #ifndef ELECTRON_SHELL_COMMON_GIN_CONVERTERS_STD_CONVERTER_H_ #define ELECTRON_SHELL_COMMON_GIN_CONVERTERS_STD_CONVERTER_H_ #include #include #include #include #include #include #include "gin/converter.h" #include "base/strings/string_util.h" #if BUILDFLAG(IS_WIN) #include "base/strings/string_util_win.h" #endif namespace gin { // Make it possible to convert move-only types. template v8::Local ConvertToV8(v8::Isolate* isolate, T&& input) { return Converter::type>::ToV8( isolate, std::forward(input)); } #if !BUILDFLAG(IS_LINUX) template <> struct Converter { // NOLINT(runtime/int) static v8::Local ToV8(v8::Isolate* isolate, unsigned long val) { // NOLINT(runtime/int) return v8::Integer::New(isolate, val); } static bool FromV8(v8::Isolate* isolate, v8::Local val, unsigned long* out) { // NOLINT(runtime/int) auto maybe = val->IntegerValue(isolate->GetCurrentContext()); if (maybe.IsNothing()) return false; *out = maybe.FromJust(); return true; } }; #endif template <> struct Converter { static v8::Local ToV8(v8::Isolate* isolate, std::nullptr_t val) { return v8::Null(isolate); } }; template <> struct Converter { static v8::Local ToV8(v8::Isolate* isolate, const char* val) { return v8::String::NewFromUtf8(isolate, val, v8::NewStringType::kNormal) .ToLocalChecked(); } }; template struct Converter { static v8::Local ToV8(v8::Isolate* isolate, const char (&val)[N]) { return v8::String::NewFromUtf8Literal(isolate, val); } }; template <> struct Converter> { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val) { return val; } static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out) { if (!val->IsArray()) return false; *out = val.As(); return true; } }; template <> struct Converter> { static v8::Local ToV8(v8::Isolate* isolate, v8::Local val) { return val; } static bool FromV8(v8::Isolate* isolate, v8::Local val, v8::Local* out) { if (!val->IsString()) return false; *out = val.As(); return true; } }; template struct Converter> { static v8::Local ToV8(v8::Isolate* isolate, const std::set& val) { v8::Local result( v8::Array::New(isolate, static_cast(val.size()))); auto context = isolate->GetCurrentContext(); typename std::set::const_iterator it; int i; for (i = 0, it = val.begin(); it != val.end(); ++it, ++i) result->Set(context, i, Converter::ToV8(isolate, *it)).Check(); return result; } static bool FromV8(v8::Isolate* isolate, v8::Local val, std::set* out) { if (!val->IsArray()) return false; auto context = isolate->GetCurrentContext(); std::set result; v8::Local array = val.As(); uint32_t length = array->Length(); for (uint32_t i = 0; i < length; ++i) { T item; if (!Converter::FromV8(isolate, array->Get(context, i).ToLocalChecked(), &item)) return false; result.insert(item); } out->swap(result); return true; } }; template struct Converter> { static bool FromV8(v8::Isolate* isolate, v8::Local value, std::map* out) { if (!value->IsObject()) return false; out->clear(); v8::Local context = isolate->GetCurrentContext(); v8::Local obj = value.As(); v8::Local keys = obj->GetPropertyNames(context).ToLocalChecked(); for (uint32_t i = 0; i < keys->Length(); ++i) { v8::MaybeLocal maybe_v8key = keys->Get(context, i); if (maybe_v8key.IsEmpty()) return false; v8::Local v8key = maybe_v8key.ToLocalChecked(); v8::MaybeLocal maybe_v8value = obj->Get(context, v8key); if (maybe_v8value.IsEmpty()) return false; K key; V out_value; if (!ConvertFromV8(isolate, v8key, &key) || !ConvertFromV8(isolate, maybe_v8value.ToLocalChecked(), &out_value)) return false; (*out)[key] = std::move(out_value); } return true; } static v8::Local ToV8(v8::Isolate* isolate, const std::map& dict) { v8::Local obj = v8::Object::New(isolate); v8::Local context = isolate->GetCurrentContext(); for (const auto& it : dict) { if (obj->Set(context, ConvertToV8(isolate, it.first), ConvertToV8(isolate, it.second)) .IsNothing()) break; } return obj; } }; #if BUILDFLAG(IS_WIN) template <> struct Converter { static v8::Local ToV8(v8::Isolate* isolate, const std::wstring& val) { return Converter::ToV8(isolate, base::AsString16(val)); } static bool FromV8(v8::Isolate* isolate, v8::Local val, std::wstring* out) { if (!val->IsString()) return false; std::u16string str; if (Converter::FromV8(isolate, val, &str)) { *out = base::AsWString(str); return true; } else { return false; } } }; #endif namespace detail { // Get a key from `key_val` and check `lookup` for a matching entry. // Return true iff a match is found, and set `*out` to the entry's value. template bool FromV8WithLookup(v8::Isolate* isolate, v8::Local key_val, const Map& table, Out* out, std::function key_transform = {}) { static_assert(std::is_same_v); auto key = KeyType{}; if (!ConvertFromV8(isolate, key_val, &key)) return false; if (key_transform) key_transform(key); if (const auto* iter = table.find(key); iter != table.end()) { *out = iter->second; return true; } return false; } } // namespace detail // Convert `key_val to a string key and check `lookup` for a matching entry. // Return true iff a match is found, and set `*out` to the entry's value. template bool FromV8WithLookup(v8::Isolate* isolate, v8::Local key_val, const Map& table, Out* out) { return detail::FromV8WithLookup(isolate, key_val, table, out); } // Convert `key_val` to a lowercase string key and check `lookup` for a matching // entry. Return true iff a match is found, and set `*out` to the entry's value. template bool FromV8WithLowerLookup(v8::Isolate* isolate, v8::Local key_val, const Map& table, Out* out) { static constexpr auto to_lower_ascii_inplace = [](std::string& str) { for (auto& ch : str) ch = base::ToLowerASCII(ch); }; return detail::FromV8WithLookup(isolate, key_val, table, out, to_lower_ascii_inplace); } } // namespace gin #endif // ELECTRON_SHELL_COMMON_GIN_CONVERTERS_STD_CONVERTER_H_