// Copyright © 2015 The CefSharp 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 "Stdafx.h" #include "V8Serialization.h" #include "JavascriptCallbackRegistry.h" #include "../CefSharp.Core.Runtime/Internals/Serialization/Primitives.h" #include using namespace std; using namespace CefSharp::Internals::Serialization; namespace CefSharp { namespace BrowserSubprocess { namespace Serialization { typedef deque> value_deque; template void SerializeV8Object(const CefRefPtr &obj, const CefRefPtr& list, const TIndex& index, JavascriptCallbackRegistry^ callbackRegistry, value_deque &seen) { for (value_deque::const_iterator it = seen.begin(); it != seen.end(); ++it) { if (obj->IsSame(*it)) { throw exception("Cycle found"); } } seen.push_back(obj); if (obj->IsNull() || obj->IsUndefined()) { list->SetNull(index); } else if (obj->IsBool()) { list->SetBool(index, obj->GetBoolValue()); } else if (obj->IsInt()) { // CEF doesn't differentiate between UINT and INT // so we have to do some additional bounds checking // To make sure the correct value is returned // CEF IPC doesn't support UINT so we can only // support int and double. // https://github.com/cefsharp/CefSharp/issues/3858 auto dValue = obj->GetDoubleValue(); if (dValue < INT_MIN || dValue > INT_MAX) { list->SetDouble(index, dValue); } else { list->SetInt(index, obj->GetIntValue()); } } else if (obj->IsDouble()) { list->SetDouble(index, obj->GetDoubleValue()); } else if (obj->IsString()) { list->SetString(index, obj->GetStringValue()); } else if (obj->IsDate()) { SetCefTime(list, index, obj->GetDateValue().val); } else if (obj->IsArray()) { int arrLength = obj->GetArrayLength(); auto array = CefListValue::Create(); if (arrLength > 0) { for (int i = 0; i < arrLength; i++) { SerializeV8Object(obj->GetValue(i), array, i, callbackRegistry, seen); } } list->SetList(index, array); } else if (obj->IsFunction()) { auto context = CefV8Context::GetCurrentContext(); auto jsCallback = callbackRegistry->Register(context, obj); SetJsCallback(list, index, jsCallback); } else if (obj->IsObject()) { std::vector keys; if (obj->GetKeys(keys) && keys.size() > 0) { auto result = CefDictionaryValue::Create(); for (size_t i = 0; i < keys.size(); i++) { auto p_keyStr = StringUtils::ToClr(keys[i].ToString()); if ((obj->HasValue(keys[i])) && (!p_keyStr->StartsWith("__"))) { SerializeV8Object(obj->GetValue(keys[i]), result, keys[i], callbackRegistry, seen); } } list->SetDictionary(index, result); } } else { list->SetNull(index); } seen.pop_back(); } template void SerializeV8Object(const CefRefPtr &obj, const CefRefPtr& list, const TIndex& index, JavascriptCallbackRegistry^ callbackRegistry) { try { value_deque seen; SerializeV8Object(obj, list, index, callbackRegistry, seen); } catch (const exception&) { list->SetNull(index); } } template CefRefPtr DeserializeV8Object(const CefRefPtr& list, const TIndex& index) { auto type = list->GetType(index); if (type == VTYPE_BOOL) { return CefV8Value::CreateBool(list->GetBool(index)); } if (type == VTYPE_INT) { return CefV8Value::CreateInt(list->GetInt(index)); } if (type == VTYPE_DOUBLE) { return CefV8Value::CreateDouble(list->GetDouble(index)); } if (type == VTYPE_STRING) { return CefV8Value::CreateString(list->GetString(index)); } if (IsCefTime(list, index)) { auto time = GetCefTime(list, index); return CefV8Value::CreateDate(time); } if (type == VTYPE_LIST) { auto subList = list->GetList(index); auto size = static_cast(subList->GetSize()); auto result = CefV8Value::CreateArray(size); for (int i = 0; i < size; i++) { result->SetValue(i, DeserializeV8Object(subList, i)); } return result; } if (type == VTYPE_DICTIONARY) { auto subDict = list->GetDictionary(index); auto size = subDict->GetSize(); std::vector keys; subDict->GetKeys(keys); auto result = CefV8Value::CreateObject(nullptr, nullptr); for (size_t i = 0; i < size; i++) { result->SetValue(keys[i], DeserializeV8Object(subDict, keys[i]), V8_PROPERTY_ATTRIBUTE_NONE); } return result; } return CefV8Value::CreateNull(); } template void SerializeV8Object(const CefRefPtr &value, const CefRefPtr& list, const int& index, JavascriptCallbackRegistry^ callbackRegistry); template void SerializeV8Object(const CefRefPtr &value, const CefRefPtr& list, const size_t& index, JavascriptCallbackRegistry^ callbackRegistry); template void SerializeV8Object(const CefRefPtr &value, const CefRefPtr& list, const CefString& index, JavascriptCallbackRegistry^ callbackRegistry); template void SerializeV8Object(const CefRefPtr &value, const CefRefPtr& list, const size_t& index, JavascriptCallbackRegistry^ callbackRegistry, value_deque &visited); template void SerializeV8Object(const CefRefPtr &value, const CefRefPtr& list, const CefString& index, JavascriptCallbackRegistry^ callbackRegistry, value_deque &visited); template CefRefPtr DeserializeV8Object(const CefRefPtr& list, const int& index); template CefRefPtr DeserializeV8Object(const CefRefPtr& list, const CefString& index); } } }