| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | // Copyright (c) 2019 Slack Technologies, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shell/renderer/api/electron_api_context_bridge.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 15:51:41 -07:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | #include <set>
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2023-06-14 13:07:18 -05:00
										 |  |  | #include <tuple>
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-30 03:28:43 -05:00
										 |  |  | #include "base/containers/contains.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-22 18:16:35 +01:00
										 |  |  | #include "base/feature_list.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | #include "base/no_destructor.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-27 14:27:34 -07:00
										 |  |  | #include "base/trace_event/trace_event.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | #include "content/public/renderer/render_frame.h"
 | 
					
						
							|  |  |  | #include "content/public/renderer/render_frame_observer.h"
 | 
					
						
							| 
									
										
										
										
											2020-05-12 18:32:04 +02:00
										 |  |  | #include "shell/common/api/object_life_monitor.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | #include "shell/common/gin_converters/blink_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_converters/callback_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_helper/dictionary.h"
 | 
					
						
							| 
									
										
										
										
											2019-11-01 15:10:32 +09:00
										 |  |  | #include "shell/common/gin_helper/promise.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | #include "shell/common/node_includes.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-23 23:33:58 -05:00
										 |  |  | #include "shell/common/world_ids.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-26 10:34:29 -07:00
										 |  |  | #include "third_party/blink/public/web/web_blob.h"
 | 
					
						
							| 
									
										
										
										
											2020-12-10 13:03:00 -08:00
										 |  |  | #include "third_party/blink/public/web/web_element.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | #include "third_party/blink/public/web/web_local_frame.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-22 18:16:35 +01:00
										 |  |  | namespace features { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const base::Feature kContextBridgeMutability{"ContextBridgeMutability", | 
					
						
							|  |  |  |                                              base::FEATURE_DISABLED_BY_DEFAULT}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-24 04:32:55 +02:00
										 |  |  | content::RenderFrame* GetRenderFrame(v8::Local<v8::Object> value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | namespace api { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  | namespace context_bridge { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 17:51:52 -07:00
										 |  |  | const char kProxyFunctionPrivateKey[] = "electron_contextBridge_proxy_fn"; | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  | const char kProxyFunctionReceiverPrivateKey[] = | 
					
						
							|  |  |  |     "electron_contextBridge_proxy_fn_receiver"; | 
					
						
							| 
									
										
										
										
											2021-07-01 17:51:52 -07:00
										 |  |  | const char kSupportsDynamicPropertiesPrivateKey[] = | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |     "electron_contextBridge_supportsDynamicProperties"; | 
					
						
							| 
									
										
										
										
											2021-07-01 17:51:52 -07:00
										 |  |  | const char kOriginalFunctionPrivateKey[] = "electron_contextBridge_original_fn"; | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | }  // namespace context_bridge
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int kMaxRecursion = 1000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  | // Returns true if |maybe| is both a value, and that value is true.
 | 
					
						
							|  |  |  | inline bool IsTrue(v8::Maybe<bool> maybe) { | 
					
						
							|  |  |  |   return maybe.IsJust() && maybe.FromJust(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | // Sourced from "extensions/renderer/v8_schema_registry.cc"
 | 
					
						
							|  |  |  | // Recursively freezes every v8 object on |object|.
 | 
					
						
							|  |  |  | bool DeepFreeze(const v8::Local<v8::Object>& object, | 
					
						
							|  |  |  |                 const v8::Local<v8::Context>& context, | 
					
						
							|  |  |  |                 std::set<int> frozen = std::set<int>()) { | 
					
						
							|  |  |  |   int hash = object->GetIdentityHash(); | 
					
						
							| 
									
										
										
										
											2023-05-30 03:28:43 -05:00
										 |  |  |   if (base::Contains(frozen, hash)) | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     return true; | 
					
						
							|  |  |  |   frozen.insert(hash); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   v8::Local<v8::Array> property_names = | 
					
						
							|  |  |  |       object->GetOwnPropertyNames(context).ToLocalChecked(); | 
					
						
							|  |  |  |   for (uint32_t i = 0; i < property_names->Length(); ++i) { | 
					
						
							|  |  |  |     v8::Local<v8::Value> child = | 
					
						
							|  |  |  |         object->Get(context, property_names->Get(context, i).ToLocalChecked()) | 
					
						
							|  |  |  |             .ToLocalChecked(); | 
					
						
							|  |  |  |     if (child->IsObject() && !child->IsTypedArray()) { | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |       if (!DeepFreeze(child.As<v8::Object>(), context, frozen)) | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  |   return IsTrue( | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       object->SetIntegrityLevel(context, v8::IntegrityLevel::kFrozen)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool IsPlainObject(const v8::Local<v8::Value>& object) { | 
					
						
							|  |  |  |   if (!object->IsObject()) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return !(object->IsNullOrUndefined() || object->IsDate() || | 
					
						
							|  |  |  |            object->IsArgumentsObject() || object->IsBigIntObject() || | 
					
						
							|  |  |  |            object->IsBooleanObject() || object->IsNumberObject() || | 
					
						
							|  |  |  |            object->IsStringObject() || object->IsSymbolObject() || | 
					
						
							|  |  |  |            object->IsNativeError() || object->IsRegExp() || | 
					
						
							|  |  |  |            object->IsPromise() || object->IsMap() || object->IsSet() || | 
					
						
							|  |  |  |            object->IsMapIterator() || object->IsSetIterator() || | 
					
						
							|  |  |  |            object->IsWeakMap() || object->IsWeakSet() || | 
					
						
							|  |  |  |            object->IsArrayBuffer() || object->IsArrayBufferView() || | 
					
						
							|  |  |  |            object->IsArray() || object->IsDataView() || | 
					
						
							| 
									
										
										
										
											2023-03-27 20:36:55 -04:00
										 |  |  |            object->IsSharedArrayBuffer() || object->IsGeneratorObject() || | 
					
						
							| 
									
										
										
										
											2021-01-28 18:20:28 -08:00
										 |  |  |            object->IsWasmModuleObject() || object->IsWasmMemoryObject() || | 
					
						
							| 
									
										
										
										
											2023-03-27 20:36:55 -04:00
										 |  |  |            object->IsModuleNamespaceObject() || object->IsProxy()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool IsPlainArray(const v8::Local<v8::Value>& arr) { | 
					
						
							|  |  |  |   if (!arr->IsArray()) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return !arr->IsTypedArray(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  | void SetPrivate(v8::Local<v8::Context> context, | 
					
						
							|  |  |  |                 v8::Local<v8::Object> target, | 
					
						
							|  |  |  |                 const std::string& key, | 
					
						
							|  |  |  |                 v8::Local<v8::Value> value) { | 
					
						
							|  |  |  |   target | 
					
						
							|  |  |  |       ->SetPrivate( | 
					
						
							|  |  |  |           context, | 
					
						
							|  |  |  |           v8::Private::ForApi(context->GetIsolate(), | 
					
						
							|  |  |  |                               gin::StringToV8(context->GetIsolate(), key)), | 
					
						
							|  |  |  |           value) | 
					
						
							|  |  |  |       .Check(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  | v8::MaybeLocal<v8::Value> GetPrivate(v8::Local<v8::Context> context, | 
					
						
							|  |  |  |                                      v8::Local<v8::Object> target, | 
					
						
							|  |  |  |                                      const std::string& key) { | 
					
						
							|  |  |  |   return target->GetPrivate( | 
					
						
							|  |  |  |       context, | 
					
						
							|  |  |  |       v8::Private::ForApi(context->GetIsolate(), | 
					
						
							|  |  |  |                           gin::StringToV8(context->GetIsolate(), key))); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | v8::MaybeLocal<v8::Value> PassValueToOtherContext( | 
					
						
							|  |  |  |     v8::Local<v8::Context> source_context, | 
					
						
							|  |  |  |     v8::Local<v8::Context> destination_context, | 
					
						
							|  |  |  |     v8::Local<v8::Value> value, | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |     v8::Local<v8::Value> parent_value, | 
					
						
							| 
									
										
										
										
											2020-03-03 23:18:22 -08:00
										 |  |  |     context_bridge::ObjectCache* object_cache, | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |     bool support_dynamic_properties, | 
					
						
							| 
									
										
										
										
											2020-07-23 14:32:38 -07:00
										 |  |  |     int recursion_depth, | 
					
						
							| 
									
										
										
										
											2021-01-26 14:23:35 -08:00
										 |  |  |     BridgeErrorTarget error_target) { | 
					
						
							| 
									
										
										
										
											2020-05-11 14:24:43 -07:00
										 |  |  |   TRACE_EVENT0("electron", "ContextBridge::PassValueToOtherContext"); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   if (recursion_depth >= kMaxRecursion) { | 
					
						
							| 
									
										
										
										
											2020-07-23 14:32:38 -07:00
										 |  |  |     v8::Context::Scope error_scope(error_target == BridgeErrorTarget::kSource | 
					
						
							|  |  |  |                                        ? source_context | 
					
						
							|  |  |  |                                        : destination_context); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     source_context->GetIsolate()->ThrowException(v8::Exception::TypeError( | 
					
						
							|  |  |  |         gin::StringToV8(source_context->GetIsolate(), | 
					
						
							|  |  |  |                         "Electron contextBridge recursion depth exceeded.  " | 
					
						
							|  |  |  |                         "Nested objects " | 
					
						
							|  |  |  |                         "deeper than 1000 are not supported."))); | 
					
						
							|  |  |  |     return v8::MaybeLocal<v8::Value>(); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-07-14 18:38:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Certain primitives always use the current contexts prototype and we can
 | 
					
						
							|  |  |  |   // pass these through directly which is significantly more performant than
 | 
					
						
							|  |  |  |   // copying them. This list of primitives is based on the classification of
 | 
					
						
							|  |  |  |   // "primitive value" as defined in the ECMA262 spec
 | 
					
						
							|  |  |  |   // https://tc39.es/ecma262/#sec-primitive-value
 | 
					
						
							|  |  |  |   if (value->IsString() || value->IsNumber() || value->IsNullOrUndefined() || | 
					
						
							|  |  |  |       value->IsBoolean() || value->IsSymbol() || value->IsBigInt()) { | 
					
						
							|  |  |  |     return v8::MaybeLocal<v8::Value>(value); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   // Check Cache
 | 
					
						
							| 
									
										
										
										
											2020-03-03 23:18:22 -08:00
										 |  |  |   auto cached_value = object_cache->GetCachedProxiedObject(value); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   if (!cached_value.IsEmpty()) { | 
					
						
							|  |  |  |     return cached_value; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Proxy functions and monitor the lifetime in the new context to release
 | 
					
						
							|  |  |  |   // the global handle at the right time.
 | 
					
						
							|  |  |  |   if (value->IsFunction()) { | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |     auto func = value.As<v8::Function>(); | 
					
						
							| 
									
										
										
										
											2021-03-24 11:43:02 -07:00
										 |  |  |     v8::MaybeLocal<v8::Value> maybe_original_fn = GetPrivate( | 
					
						
							|  |  |  |         source_context, func, context_bridge::kOriginalFunctionPrivateKey); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |       v8::Context::Scope destination_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2021-03-24 11:43:02 -07:00
										 |  |  |       v8::Local<v8::Value> proxy_func; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // If this function has already been sent over the bridge,
 | 
					
						
							|  |  |  |       // then it is being sent _back_ over the bridge and we can
 | 
					
						
							|  |  |  |       // simply return the original method here for performance reasons
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // For safety reasons we check if the destination context is the
 | 
					
						
							|  |  |  |       // creation context of the original method.  If it's not we proceed
 | 
					
						
							|  |  |  |       // with the proxy logic
 | 
					
						
							|  |  |  |       if (maybe_original_fn.ToLocal(&proxy_func) && proxy_func->IsFunction() && | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  |           proxy_func.As<v8::Object>()->GetCreationContextChecked() == | 
					
						
							| 
									
										
										
										
											2021-03-24 11:43:02 -07:00
										 |  |  |               destination_context) { | 
					
						
							|  |  |  |         return v8::MaybeLocal<v8::Value>(proxy_func); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       v8::Local<v8::Object> state = | 
					
						
							|  |  |  |           v8::Object::New(destination_context->GetIsolate()); | 
					
						
							|  |  |  |       SetPrivate(destination_context, state, | 
					
						
							|  |  |  |                  context_bridge::kProxyFunctionPrivateKey, func); | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |       SetPrivate(destination_context, state, | 
					
						
							|  |  |  |                  context_bridge::kProxyFunctionReceiverPrivateKey, | 
					
						
							|  |  |  |                  parent_value); | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       SetPrivate(destination_context, state, | 
					
						
							|  |  |  |                  context_bridge::kSupportsDynamicPropertiesPrivateKey, | 
					
						
							|  |  |  |                  gin::ConvertToV8(destination_context->GetIsolate(), | 
					
						
							|  |  |  |                                   support_dynamic_properties)); | 
					
						
							| 
									
										
										
										
											2021-03-24 11:43:02 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       if (!v8::Function::New(destination_context, ProxyFunctionWrapper, state) | 
					
						
							|  |  |  |                .ToLocal(&proxy_func)) | 
					
						
							|  |  |  |         return v8::MaybeLocal<v8::Value>(); | 
					
						
							| 
									
										
										
										
											2021-03-24 11:43:02 -07:00
										 |  |  |       SetPrivate(destination_context, proxy_func.As<v8::Object>(), | 
					
						
							|  |  |  |                  context_bridge::kOriginalFunctionPrivateKey, func); | 
					
						
							| 
									
										
										
										
											2020-03-03 23:18:22 -08:00
										 |  |  |       object_cache->CacheProxiedObject(value, proxy_func); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       return v8::MaybeLocal<v8::Value>(proxy_func); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Proxy promises as they have a safe and guaranteed memory lifecycle
 | 
					
						
							|  |  |  |   if (value->IsPromise()) { | 
					
						
							|  |  |  |     v8::Context::Scope destination_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |     auto source_promise = value.As<v8::Promise>(); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     // Make the promise a shared_ptr so that when the original promise is
 | 
					
						
							|  |  |  |     // freed the proxy promise is correctly freed as well instead of being
 | 
					
						
							|  |  |  |     // left dangling
 | 
					
						
							| 
									
										
										
										
											2021-06-07 19:00:05 -07:00
										 |  |  |     auto proxied_promise = | 
					
						
							|  |  |  |         std::make_shared<gin_helper::Promise<v8::Local<v8::Value>>>( | 
					
						
							|  |  |  |             destination_context->GetIsolate()); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     v8::Local<v8::Promise> proxied_promise_handle = | 
					
						
							|  |  |  |         proxied_promise->GetHandle(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     v8::Global<v8::Context> global_then_source_context( | 
					
						
							|  |  |  |         source_context->GetIsolate(), source_context); | 
					
						
							|  |  |  |     v8::Global<v8::Context> global_then_destination_context( | 
					
						
							|  |  |  |         destination_context->GetIsolate(), destination_context); | 
					
						
							|  |  |  |     global_then_source_context.SetWeak(); | 
					
						
							|  |  |  |     global_then_destination_context.SetWeak(); | 
					
						
							|  |  |  |     auto then_cb = base::BindOnce( | 
					
						
							|  |  |  |         [](std::shared_ptr<gin_helper::Promise<v8::Local<v8::Value>>> | 
					
						
							|  |  |  |                proxied_promise, | 
					
						
							|  |  |  |            v8::Isolate* isolate, v8::Global<v8::Context> global_source_context, | 
					
						
							|  |  |  |            v8::Global<v8::Context> global_destination_context, | 
					
						
							|  |  |  |            v8::Local<v8::Value> result) { | 
					
						
							|  |  |  |           if (global_source_context.IsEmpty() || | 
					
						
							|  |  |  |               global_destination_context.IsEmpty()) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           context_bridge::ObjectCache object_cache; | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |           v8::MaybeLocal<v8::Value> val; | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             v8::TryCatch try_catch(isolate); | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |             v8::Local<v8::Context> source_context = | 
					
						
							|  |  |  |                 global_source_context.Get(isolate); | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |             val = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |                 source_context, global_destination_context.Get(isolate), result, | 
					
						
							|  |  |  |                 source_context->Global(), &object_cache, false, 0, | 
					
						
							|  |  |  |                 BridgeErrorTarget::kDestination); | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |             if (try_catch.HasCaught()) { | 
					
						
							|  |  |  |               if (try_catch.Message().IsEmpty()) { | 
					
						
							|  |  |  |                 proxied_promise->RejectWithErrorMessage( | 
					
						
							|  |  |  |                     "An error was thrown while sending a promise result over " | 
					
						
							|  |  |  |                     "the context bridge but it was not actually an Error " | 
					
						
							|  |  |  |                     "object. This normally means that a promise was resolved " | 
					
						
							|  |  |  |                     "with a value that is not supported by the Context " | 
					
						
							|  |  |  |                     "Bridge."); | 
					
						
							|  |  |  |               } else { | 
					
						
							|  |  |  |                 proxied_promise->Reject( | 
					
						
							|  |  |  |                     v8::Exception::Error(try_catch.Message()->Get())); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           DCHECK(!val.IsEmpty()); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |           if (!val.IsEmpty()) | 
					
						
							|  |  |  |             proxied_promise->Resolve(val.ToLocalChecked()); | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         proxied_promise, destination_context->GetIsolate(), | 
					
						
							|  |  |  |         std::move(global_then_source_context), | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |         std::move(global_then_destination_context)); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     v8::Global<v8::Context> global_catch_source_context( | 
					
						
							|  |  |  |         source_context->GetIsolate(), source_context); | 
					
						
							|  |  |  |     v8::Global<v8::Context> global_catch_destination_context( | 
					
						
							|  |  |  |         destination_context->GetIsolate(), destination_context); | 
					
						
							|  |  |  |     global_catch_source_context.SetWeak(); | 
					
						
							|  |  |  |     global_catch_destination_context.SetWeak(); | 
					
						
							|  |  |  |     auto catch_cb = base::BindOnce( | 
					
						
							|  |  |  |         [](std::shared_ptr<gin_helper::Promise<v8::Local<v8::Value>>> | 
					
						
							|  |  |  |                proxied_promise, | 
					
						
							|  |  |  |            v8::Isolate* isolate, v8::Global<v8::Context> global_source_context, | 
					
						
							|  |  |  |            v8::Global<v8::Context> global_destination_context, | 
					
						
							|  |  |  |            v8::Local<v8::Value> result) { | 
					
						
							|  |  |  |           if (global_source_context.IsEmpty() || | 
					
						
							|  |  |  |               global_destination_context.IsEmpty()) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           context_bridge::ObjectCache object_cache; | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |           v8::MaybeLocal<v8::Value> val; | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             v8::TryCatch try_catch(isolate); | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |             v8::Local<v8::Context> source_context = | 
					
						
							|  |  |  |                 global_source_context.Get(isolate); | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |             val = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |                 source_context, global_destination_context.Get(isolate), result, | 
					
						
							|  |  |  |                 source_context->Global(), &object_cache, false, 0, | 
					
						
							|  |  |  |                 BridgeErrorTarget::kDestination); | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |             if (try_catch.HasCaught()) { | 
					
						
							|  |  |  |               if (try_catch.Message().IsEmpty()) { | 
					
						
							|  |  |  |                 proxied_promise->RejectWithErrorMessage( | 
					
						
							|  |  |  |                     "An error was thrown while sending a promise rejection " | 
					
						
							|  |  |  |                     "over the context bridge but it was not actually an Error " | 
					
						
							|  |  |  |                     "object. This normally means that a promise was rejected " | 
					
						
							|  |  |  |                     "with a value that is not supported by the Context " | 
					
						
							|  |  |  |                     "Bridge."); | 
					
						
							|  |  |  |               } else { | 
					
						
							|  |  |  |                 proxied_promise->Reject( | 
					
						
							|  |  |  |                     v8::Exception::Error(try_catch.Message()->Get())); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |           if (!val.IsEmpty()) | 
					
						
							|  |  |  |             proxied_promise->Reject(val.ToLocalChecked()); | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         proxied_promise, destination_context->GetIsolate(), | 
					
						
							|  |  |  |         std::move(global_catch_source_context), | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |         std::move(global_catch_destination_context)); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 18:58:52 -08:00
										 |  |  |     std::ignore = source_promise->Then( | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |         source_context, | 
					
						
							| 
									
										
										
										
											2021-11-16 13:45:32 +05:30
										 |  |  |         gin::ConvertToV8(destination_context->GetIsolate(), std::move(then_cb)) | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |             .As<v8::Function>(), | 
					
						
							| 
									
										
										
										
											2021-11-16 13:45:32 +05:30
										 |  |  |         gin::ConvertToV8(destination_context->GetIsolate(), std::move(catch_cb)) | 
					
						
							| 
									
										
										
										
											2022-02-09 18:58:52 -08:00
										 |  |  |             .As<v8::Function>()); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     object_cache->CacheProxiedObject(value, proxied_promise_handle); | 
					
						
							|  |  |  |     return v8::MaybeLocal<v8::Value>(proxied_promise_handle); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Errors aren't serializable currently, we need to pull the message out and
 | 
					
						
							|  |  |  |   // re-construct in the destination context
 | 
					
						
							|  |  |  |   if (value->IsNativeError()) { | 
					
						
							|  |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2021-01-26 14:23:35 -08:00
										 |  |  |     // We should try to pull "message" straight off of the error as a
 | 
					
						
							|  |  |  |     // v8::Message includes some pretext that can get duplicated each time it
 | 
					
						
							|  |  |  |     // crosses the bridge we fallback to the v8::Message approach if we can't
 | 
					
						
							|  |  |  |     // pull "message" for some reason
 | 
					
						
							|  |  |  |     v8::MaybeLocal<v8::Value> maybe_message = value.As<v8::Object>()->Get( | 
					
						
							|  |  |  |         source_context, | 
					
						
							|  |  |  |         gin::ConvertToV8(source_context->GetIsolate(), "message")); | 
					
						
							|  |  |  |     v8::Local<v8::Value> message; | 
					
						
							|  |  |  |     if (maybe_message.ToLocal(&message) && message->IsString()) { | 
					
						
							|  |  |  |       return v8::MaybeLocal<v8::Value>( | 
					
						
							|  |  |  |           v8::Exception::Error(message.As<v8::String>())); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     return v8::MaybeLocal<v8::Value>(v8::Exception::Error( | 
					
						
							|  |  |  |         v8::Exception::CreateMessage(destination_context->GetIsolate(), value) | 
					
						
							|  |  |  |             ->Get())); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Manually go through the array and pass each value individually into a new
 | 
					
						
							|  |  |  |   // array so that functions deep inside arrays get proxied or arrays of
 | 
					
						
							|  |  |  |   // promises are proxied correctly.
 | 
					
						
							|  |  |  |   if (IsPlainArray(value)) { | 
					
						
							|  |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |     v8::Local<v8::Array> arr = value.As<v8::Array>(); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     size_t length = arr->Length(); | 
					
						
							|  |  |  |     v8::Local<v8::Array> cloned_arr = | 
					
						
							|  |  |  |         v8::Array::New(destination_context->GetIsolate(), length); | 
					
						
							|  |  |  |     for (size_t i = 0; i < length; i++) { | 
					
						
							|  |  |  |       auto value_for_array = PassValueToOtherContext( | 
					
						
							|  |  |  |           source_context, destination_context, | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |           arr->Get(source_context, i).ToLocalChecked(), value, object_cache, | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |           support_dynamic_properties, recursion_depth + 1, error_target); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |       if (value_for_array.IsEmpty()) | 
					
						
							|  |  |  |         return v8::MaybeLocal<v8::Value>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!IsTrue(cloned_arr->Set(destination_context, static_cast<int>(i), | 
					
						
							|  |  |  |                                   value_for_array.ToLocalChecked()))) { | 
					
						
							|  |  |  |         return v8::MaybeLocal<v8::Value>(); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     object_cache->CacheProxiedObject(value, cloned_arr); | 
					
						
							|  |  |  |     return v8::MaybeLocal<v8::Value>(cloned_arr); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 13:03:00 -08:00
										 |  |  |   // Custom logic to "clone" Element references
 | 
					
						
							| 
									
										
										
										
											2023-09-18 16:44:09 -04:00
										 |  |  |   blink::WebElement elem = | 
					
						
							|  |  |  |       blink::WebElement::FromV8Value(destination_context->GetIsolate(), value); | 
					
						
							| 
									
										
										
										
											2020-12-10 13:03:00 -08:00
										 |  |  |   if (!elem.IsNull()) { | 
					
						
							|  |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2023-12-11 14:58:26 -06:00
										 |  |  |     return v8::MaybeLocal<v8::Value>( | 
					
						
							|  |  |  |         elem.ToV8Value(destination_context->GetIsolate())); | 
					
						
							| 
									
										
										
										
											2020-12-10 13:03:00 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 10:34:29 -07:00
										 |  |  |   // Custom logic to "clone" Blob references
 | 
					
						
							| 
									
										
										
										
											2023-09-18 16:44:09 -04:00
										 |  |  |   blink::WebBlob blob = | 
					
						
							|  |  |  |       blink::WebBlob::FromV8Value(destination_context->GetIsolate(), value); | 
					
						
							| 
									
										
										
										
											2021-05-26 10:34:29 -07:00
										 |  |  |   if (!blob.IsNull()) { | 
					
						
							|  |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2022-10-03 13:21:00 -07:00
										 |  |  |     return v8::MaybeLocal<v8::Value>( | 
					
						
							|  |  |  |         blob.ToV8Value(destination_context->GetIsolate())); | 
					
						
							| 
									
										
										
										
											2021-05-26 10:34:29 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   // Proxy all objects
 | 
					
						
							|  |  |  |   if (IsPlainObject(value)) { | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |     auto object_value = value.As<v8::Object>(); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |     auto passed_value = CreateProxyForAPI( | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |         object_value, source_context, destination_context, object_cache, | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |         support_dynamic_properties, recursion_depth + 1, error_target); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     if (passed_value.IsEmpty()) | 
					
						
							|  |  |  |       return v8::MaybeLocal<v8::Value>(); | 
					
						
							|  |  |  |     return v8::MaybeLocal<v8::Value>(passed_value.ToLocalChecked()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Serializable objects
 | 
					
						
							|  |  |  |   blink::CloneableMessage ret; | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-07-23 14:32:38 -07:00
										 |  |  |     v8::Local<v8::Context> error_context = | 
					
						
							|  |  |  |         error_target == BridgeErrorTarget::kSource ? source_context | 
					
						
							|  |  |  |                                                    : destination_context; | 
					
						
							|  |  |  |     v8::Context::Scope error_scope(error_context); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     // V8 serializer will throw an error if required
 | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |     if (!gin::ConvertFromV8(error_context->GetIsolate(), value, &ret)) { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |       return v8::MaybeLocal<v8::Value>(); | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     v8::Local<v8::Value> cloned_value = | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  |         gin::ConvertToV8(destination_context->GetIsolate(), ret); | 
					
						
							| 
									
										
										
										
											2020-03-03 23:18:22 -08:00
										 |  |  |     object_cache->CacheProxiedObject(value, cloned_value); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     return v8::MaybeLocal<v8::Value>(cloned_value); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  | void ProxyFunctionWrapper(const v8::FunctionCallbackInfo<v8::Value>& info) { | 
					
						
							| 
									
										
										
										
											2020-05-11 14:24:43 -07:00
										 |  |  |   TRACE_EVENT0("electron", "ContextBridge::ProxyFunctionWrapper"); | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |   CHECK(info.Data()->IsObject()); | 
					
						
							|  |  |  |   v8::Local<v8::Object> data = info.Data().As<v8::Object>(); | 
					
						
							|  |  |  |   bool support_dynamic_properties = false; | 
					
						
							|  |  |  |   gin::Arguments args(info); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   // Context the proxy function was called from
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |   v8::Local<v8::Context> calling_context = args.isolate()->GetCurrentContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Pull the original function and its context off of the data private key
 | 
					
						
							|  |  |  |   v8::MaybeLocal<v8::Value> sdp_value = | 
					
						
							|  |  |  |       GetPrivate(calling_context, data, | 
					
						
							|  |  |  |                  context_bridge::kSupportsDynamicPropertiesPrivateKey); | 
					
						
							|  |  |  |   v8::MaybeLocal<v8::Value> maybe_func = GetPrivate( | 
					
						
							|  |  |  |       calling_context, data, context_bridge::kProxyFunctionPrivateKey); | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |   v8::MaybeLocal<v8::Value> maybe_recv = GetPrivate( | 
					
						
							|  |  |  |       calling_context, data, context_bridge::kProxyFunctionReceiverPrivateKey); | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |   v8::Local<v8::Value> func_value; | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |   if (sdp_value.IsEmpty() || maybe_func.IsEmpty() || maybe_recv.IsEmpty() || | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       !gin::ConvertFromV8(args.isolate(), sdp_value.ToLocalChecked(), | 
					
						
							|  |  |  |                           &support_dynamic_properties) || | 
					
						
							|  |  |  |       !maybe_func.ToLocal(&func_value)) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |   v8::Local<v8::Function> func = func_value.As<v8::Function>(); | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  |   v8::Local<v8::Context> func_owning_context = | 
					
						
							|  |  |  |       func->GetCreationContextChecked(); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     v8::Context::Scope func_owning_context_scope(func_owning_context); | 
					
						
							|  |  |  |     context_bridge::ObjectCache object_cache; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     std::vector<v8::Local<v8::Value>> original_args; | 
					
						
							|  |  |  |     std::vector<v8::Local<v8::Value>> proxied_args; | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |     args.GetRemaining(&original_args); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto value : original_args) { | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |       auto arg = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |           calling_context, func_owning_context, value, | 
					
						
							|  |  |  |           calling_context->Global(), &object_cache, support_dynamic_properties, | 
					
						
							|  |  |  |           0, BridgeErrorTarget::kSource); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       if (arg.IsEmpty()) | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       proxied_args.push_back(arg.ToLocalChecked()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     v8::MaybeLocal<v8::Value> maybe_return_value; | 
					
						
							|  |  |  |     bool did_error = false; | 
					
						
							| 
									
										
										
										
											2021-03-30 07:26:49 +00:00
										 |  |  |     v8::Local<v8::Value> error_message; | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       v8::TryCatch try_catch(args.isolate()); | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |       maybe_return_value = | 
					
						
							|  |  |  |           func->Call(func_owning_context, maybe_recv.ToLocalChecked(), | 
					
						
							|  |  |  |                      proxied_args.size(), proxied_args.data()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       if (try_catch.HasCaught()) { | 
					
						
							|  |  |  |         did_error = true; | 
					
						
							| 
									
										
										
										
											2021-03-30 07:26:49 +00:00
										 |  |  |         v8::Local<v8::Value> exception = try_catch.Exception(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 17:51:52 -07:00
										 |  |  |         const char err_msg[] = | 
					
						
							| 
									
										
										
										
											2021-03-30 07:26:49 +00:00
										 |  |  |             "An unknown exception occurred in the isolated context, an error " | 
					
						
							|  |  |  |             "occurred but a valid exception was not thrown."; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!exception->IsNull() && exception->IsObject()) { | 
					
						
							|  |  |  |           v8::MaybeLocal<v8::Value> maybe_message = | 
					
						
							|  |  |  |               exception.As<v8::Object>()->Get( | 
					
						
							|  |  |  |                   func_owning_context, | 
					
						
							|  |  |  |                   gin::ConvertToV8(args.isolate(), "message")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (!maybe_message.ToLocal(&error_message) || | 
					
						
							|  |  |  |               !error_message->IsString()) { | 
					
						
							|  |  |  |             error_message = gin::StringToV8(args.isolate(), err_msg); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           error_message = gin::StringToV8(args.isolate(), err_msg); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (did_error) { | 
					
						
							|  |  |  |       v8::Context::Scope calling_context_scope(calling_context); | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       args.isolate()->ThrowException( | 
					
						
							| 
									
										
										
										
											2021-03-30 07:26:49 +00:00
										 |  |  |           v8::Exception::Error(error_message.As<v8::String>())); | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (maybe_return_value.IsEmpty()) | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       return; | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-12 19:51:56 -08:00
										 |  |  |     // In the case where we encountered an exception converting the return value
 | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |     // of the function we need to ensure that the exception / thrown value is
 | 
					
						
							|  |  |  |     // safely transferred from the function_owning_context (where it was thrown)
 | 
					
						
							|  |  |  |     // into the calling_context (where it needs to be thrown) To do this we pull
 | 
					
						
							|  |  |  |     // the message off the exception and later re-throw it in the right context.
 | 
					
						
							|  |  |  |     // In some cases the caught thing is not an exception i.e. it's technically
 | 
					
						
							|  |  |  |     // valid to `throw 123`.  In these cases to avoid infinite
 | 
					
						
							|  |  |  |     // PassValueToOtherContext recursion we bail early as being unable to send
 | 
					
						
							|  |  |  |     // the value from one context to the other.
 | 
					
						
							|  |  |  |     // TODO(MarshallOfSound): In this case and other cases where the error can't
 | 
					
						
							|  |  |  |     // be sent _across_ worlds we should probably log it globally in some way to
 | 
					
						
							|  |  |  |     // allow easier debugging.  This is not trivial though so is left to a
 | 
					
						
							|  |  |  |     // future change.
 | 
					
						
							|  |  |  |     bool did_error_converting_result = false; | 
					
						
							|  |  |  |     v8::MaybeLocal<v8::Value> ret; | 
					
						
							|  |  |  |     v8::Local<v8::String> exception; | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       v8::TryCatch try_catch(args.isolate()); | 
					
						
							|  |  |  |       ret = PassValueToOtherContext(func_owning_context, calling_context, | 
					
						
							|  |  |  |                                     maybe_return_value.ToLocalChecked(), | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |                                     func_owning_context->Global(), | 
					
						
							| 
									
										
										
										
											2023-04-11 02:57:48 -07:00
										 |  |  |                                     &object_cache, support_dynamic_properties, | 
					
						
							|  |  |  |                                     0, BridgeErrorTarget::kDestination); | 
					
						
							|  |  |  |       if (try_catch.HasCaught()) { | 
					
						
							|  |  |  |         did_error_converting_result = true; | 
					
						
							|  |  |  |         if (!try_catch.Message().IsEmpty()) { | 
					
						
							|  |  |  |           exception = try_catch.Message()->Get(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (did_error_converting_result) { | 
					
						
							|  |  |  |       v8::Context::Scope calling_context_scope(calling_context); | 
					
						
							|  |  |  |       if (exception.IsEmpty()) { | 
					
						
							|  |  |  |         const char err_msg[] = | 
					
						
							|  |  |  |             "An unknown exception occurred while sending a function return " | 
					
						
							|  |  |  |             "value over the context bridge, an error " | 
					
						
							|  |  |  |             "occurred but a valid exception was not thrown."; | 
					
						
							|  |  |  |         args.isolate()->ThrowException(v8::Exception::Error( | 
					
						
							|  |  |  |             gin::StringToV8(args.isolate(), err_msg).As<v8::String>())); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         args.isolate()->ThrowException(v8::Exception::Error(exception)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     DCHECK(!ret.IsEmpty()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     if (ret.IsEmpty()) | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |       return; | 
					
						
							|  |  |  |     info.GetReturnValue().Set(ret.ToLocalChecked()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | v8::MaybeLocal<v8::Object> CreateProxyForAPI( | 
					
						
							|  |  |  |     const v8::Local<v8::Object>& api_object, | 
					
						
							|  |  |  |     const v8::Local<v8::Context>& source_context, | 
					
						
							|  |  |  |     const v8::Local<v8::Context>& destination_context, | 
					
						
							| 
									
										
										
										
											2020-03-03 23:18:22 -08:00
										 |  |  |     context_bridge::ObjectCache* object_cache, | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |     bool support_dynamic_properties, | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |     int recursion_depth, | 
					
						
							|  |  |  |     BridgeErrorTarget error_target) { | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  |   gin_helper::Dictionary api(source_context->GetIsolate(), api_object); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     v8::Context::Scope destination_context_scope(destination_context); | 
					
						
							| 
									
										
										
										
											2023-10-10 12:45:44 +02:00
										 |  |  |     auto proxy = | 
					
						
							|  |  |  |         gin_helper::Dictionary::CreateEmpty(destination_context->GetIsolate()); | 
					
						
							| 
									
										
										
										
											2020-04-16 14:10:18 -07:00
										 |  |  |     object_cache->CacheProxiedObject(api.GetHandle(), proxy.GetHandle()); | 
					
						
							|  |  |  |     auto maybe_keys = api.GetHandle()->GetOwnPropertyNames( | 
					
						
							| 
									
										
										
										
											2020-07-21 15:26:16 -07:00
										 |  |  |         source_context, static_cast<v8::PropertyFilter>(v8::ONLY_ENUMERABLE)); | 
					
						
							| 
									
										
										
										
											2020-04-16 14:10:18 -07:00
										 |  |  |     if (maybe_keys.IsEmpty()) | 
					
						
							|  |  |  |       return v8::MaybeLocal<v8::Object>(proxy.GetHandle()); | 
					
						
							|  |  |  |     auto keys = maybe_keys.ToLocalChecked(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     uint32_t length = keys->Length(); | 
					
						
							|  |  |  |     for (uint32_t i = 0; i < length; i++) { | 
					
						
							|  |  |  |       v8::Local<v8::Value> key = | 
					
						
							|  |  |  |           keys->Get(destination_context, i).ToLocalChecked(); | 
					
						
							| 
									
										
										
										
											2020-07-21 15:26:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |       if (support_dynamic_properties) { | 
					
						
							|  |  |  |         v8::Context::Scope source_context_scope(source_context); | 
					
						
							|  |  |  |         auto maybe_desc = api.GetHandle()->GetOwnPropertyDescriptor( | 
					
						
							| 
									
										
										
										
											2021-05-12 09:38:21 +02:00
										 |  |  |             source_context, key.As<v8::Name>()); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |         v8::Local<v8::Value> desc_value; | 
					
						
							|  |  |  |         if (!maybe_desc.ToLocal(&desc_value) || !desc_value->IsObject()) | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         gin_helper::Dictionary desc(api.isolate(), desc_value.As<v8::Object>()); | 
					
						
							|  |  |  |         if (desc.Has("get") || desc.Has("set")) { | 
					
						
							|  |  |  |           v8::Local<v8::Value> getter; | 
					
						
							|  |  |  |           v8::Local<v8::Value> setter; | 
					
						
							|  |  |  |           desc.Get("get", &getter); | 
					
						
							|  |  |  |           desc.Get("set", &setter); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           { | 
					
						
							| 
									
										
										
										
											2022-09-07 09:46:37 +02:00
										 |  |  |             v8::Context::Scope inner_destination_context_scope( | 
					
						
							|  |  |  |                 destination_context); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |             v8::Local<v8::Value> getter_proxy; | 
					
						
							|  |  |  |             v8::Local<v8::Value> setter_proxy; | 
					
						
							|  |  |  |             if (!getter.IsEmpty()) { | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |               if (!PassValueToOtherContext( | 
					
						
							|  |  |  |                        source_context, destination_context, getter, | 
					
						
							|  |  |  |                        api.GetHandle(), object_cache, | 
					
						
							|  |  |  |                        support_dynamic_properties, 1, error_target) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |                        .ToLocal(&getter_proxy)) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (!setter.IsEmpty()) { | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |               if (!PassValueToOtherContext( | 
					
						
							|  |  |  |                        source_context, destination_context, setter, | 
					
						
							|  |  |  |                        api.GetHandle(), object_cache, | 
					
						
							|  |  |  |                        support_dynamic_properties, 1, error_target) | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |                        .ToLocal(&setter_proxy)) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 09:46:37 +02:00
										 |  |  |             v8::PropertyDescriptor prop_desc(getter_proxy, setter_proxy); | 
					
						
							| 
									
										
										
										
											2022-02-09 18:58:52 -08:00
										 |  |  |             std::ignore = proxy.GetHandle()->DefineProperty( | 
					
						
							| 
									
										
										
										
											2022-09-07 09:46:37 +02:00
										 |  |  |                 destination_context, key.As<v8::Name>(), prop_desc); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |           } | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       v8::Local<v8::Value> value; | 
					
						
							| 
									
										
										
										
											2020-07-21 15:26:16 -07:00
										 |  |  |       if (!api.Get(key, &value)) | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |         continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |       auto passed_value = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |           source_context, destination_context, value, api.GetHandle(), | 
					
						
							|  |  |  |           object_cache, support_dynamic_properties, recursion_depth + 1, | 
					
						
							|  |  |  |           error_target); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       if (passed_value.IsEmpty()) | 
					
						
							|  |  |  |         return v8::MaybeLocal<v8::Object>(); | 
					
						
							| 
									
										
										
										
											2020-07-21 15:26:16 -07:00
										 |  |  |       proxy.Set(key, passed_value.ToLocalChecked()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return proxy.GetHandle(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  | void ExposeAPIInWorld(v8::Isolate* isolate, | 
					
						
							|  |  |  |                       const int world_id, | 
					
						
							|  |  |  |                       const std::string& key, | 
					
						
							|  |  |  |                       v8::Local<v8::Value> api, | 
					
						
							|  |  |  |                       gin_helper::Arguments* args) { | 
					
						
							|  |  |  |   TRACE_EVENT2("electron", "ContextBridge::ExposeAPIInWorld", "key", key, | 
					
						
							|  |  |  |                "worldId", world_id); | 
					
						
							| 
									
										
										
										
											2020-12-04 18:43:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   auto* render_frame = GetRenderFrame(isolate->GetCurrentContext()->Global()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |   CHECK(render_frame); | 
					
						
							|  |  |  |   auto* frame = render_frame->GetWebFrame(); | 
					
						
							|  |  |  |   CHECK(frame); | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |   v8::Local<v8::Context> target_context = | 
					
						
							|  |  |  |       world_id == WorldIDs::MAIN_WORLD_ID | 
					
						
							|  |  |  |           ? frame->MainWorldScriptContext() | 
					
						
							|  |  |  |           : frame->GetScriptContextFromWorldId(isolate, world_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   gin_helper::Dictionary global(target_context->GetIsolate(), | 
					
						
							|  |  |  |                                 target_context->Global()); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (global.Has(key)) { | 
					
						
							|  |  |  |     args->ThrowError( | 
					
						
							|  |  |  |         "Cannot bind an API on top of an existing property on the window " | 
					
						
							|  |  |  |         "object"); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  |   v8::Local<v8::Context> electron_isolated_context = | 
					
						
							|  |  |  |       frame->GetScriptContextFromWorldId(args->isolate(), | 
					
						
							|  |  |  |                                          WorldIDs::ISOLATED_WORLD_ID); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  |     context_bridge::ObjectCache object_cache; | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  |     v8::Context::Scope target_context_scope(target_context); | 
					
						
							| 
									
										
										
										
											2020-05-11 13:41:42 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |     v8::MaybeLocal<v8::Value> maybe_proxy = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |         electron_isolated_context, target_context, api, | 
					
						
							|  |  |  |         electron_isolated_context->Global(), &object_cache, false, 0, | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |         BridgeErrorTarget::kSource); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |     if (maybe_proxy.IsEmpty()) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     auto proxy = maybe_proxy.ToLocalChecked(); | 
					
						
							| 
									
										
										
										
											2021-03-22 18:16:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (base::FeatureList::IsEnabled(features::kContextBridgeMutability)) { | 
					
						
							|  |  |  |       global.Set(key, proxy); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 18:43:20 +01:00
										 |  |  |     if (proxy->IsObject() && !proxy->IsTypedArray() && | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  |         !DeepFreeze(proxy.As<v8::Object>(), target_context)) | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  |       return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global.SetReadOnlyNonConfigurable(key, proxy); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  | gin_helper::Dictionary TraceKeyPath(const gin_helper::Dictionary& start, | 
					
						
							|  |  |  |                                     const std::vector<std::string>& key_path) { | 
					
						
							|  |  |  |   gin_helper::Dictionary current = start; | 
					
						
							|  |  |  |   for (size_t i = 0; i < key_path.size() - 1; i++) { | 
					
						
							|  |  |  |     CHECK(current.Get(key_path[i], ¤t)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return current; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  | void OverrideGlobalValueFromIsolatedWorld( | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |     const std::vector<std::string>& key_path, | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |     v8::Local<v8::Object> value, | 
					
						
							|  |  |  |     bool support_dynamic_properties) { | 
					
						
							| 
									
										
										
										
											2020-10-22 13:24:59 -07:00
										 |  |  |   if (key_path.empty()) | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |   auto* render_frame = GetRenderFrame(value); | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |   CHECK(render_frame); | 
					
						
							|  |  |  |   auto* frame = render_frame->GetWebFrame(); | 
					
						
							|  |  |  |   CHECK(frame); | 
					
						
							|  |  |  |   v8::Local<v8::Context> main_context = frame->MainWorldScriptContext(); | 
					
						
							|  |  |  |   gin_helper::Dictionary global(main_context->GetIsolate(), | 
					
						
							|  |  |  |                                 main_context->Global()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const std::string final_key = key_path[key_path.size() - 1]; | 
					
						
							|  |  |  |   gin_helper::Dictionary target_object = TraceKeyPath(global, key_path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     v8::Context::Scope main_context_scope(main_context); | 
					
						
							|  |  |  |     context_bridge::ObjectCache object_cache; | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |     v8::Local<v8::Context> source_context = value->GetCreationContextChecked(); | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  |     v8::MaybeLocal<v8::Value> maybe_proxy = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |         source_context, main_context, value, source_context->Global(), | 
					
						
							|  |  |  |         &object_cache, support_dynamic_properties, 1, | 
					
						
							|  |  |  |         BridgeErrorTarget::kSource); | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |     DCHECK(!maybe_proxy.IsEmpty()); | 
					
						
							|  |  |  |     auto proxy = maybe_proxy.ToLocalChecked(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     target_object.Set(final_key, proxy); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool OverrideGlobalPropertyFromIsolatedWorld( | 
					
						
							|  |  |  |     const std::vector<std::string>& key_path, | 
					
						
							|  |  |  |     v8::Local<v8::Object> getter, | 
					
						
							|  |  |  |     v8::Local<v8::Value> setter, | 
					
						
							|  |  |  |     gin_helper::Arguments* args) { | 
					
						
							| 
									
										
										
										
											2020-10-22 13:24:59 -07:00
										 |  |  |   if (key_path.empty()) | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto* render_frame = GetRenderFrame(getter); | 
					
						
							|  |  |  |   CHECK(render_frame); | 
					
						
							|  |  |  |   auto* frame = render_frame->GetWebFrame(); | 
					
						
							|  |  |  |   CHECK(frame); | 
					
						
							|  |  |  |   v8::Local<v8::Context> main_context = frame->MainWorldScriptContext(); | 
					
						
							|  |  |  |   gin_helper::Dictionary global(main_context->GetIsolate(), | 
					
						
							|  |  |  |                                 main_context->Global()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const std::string final_key = key_path[key_path.size() - 1]; | 
					
						
							|  |  |  |   v8::Local<v8::Object> target_object = | 
					
						
							|  |  |  |       TraceKeyPath(global, key_path).GetHandle(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     v8::Context::Scope main_context_scope(main_context); | 
					
						
							|  |  |  |     context_bridge::ObjectCache object_cache; | 
					
						
							|  |  |  |     v8::Local<v8::Value> getter_proxy; | 
					
						
							|  |  |  |     v8::Local<v8::Value> setter_proxy; | 
					
						
							|  |  |  |     if (!getter->IsNullOrUndefined()) { | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |       v8::Local<v8::Context> source_context = | 
					
						
							|  |  |  |           getter->GetCreationContextChecked(); | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  |       v8::MaybeLocal<v8::Value> maybe_getter_proxy = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |           source_context, main_context, getter, source_context->Global(), | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |           &object_cache, false, 1, BridgeErrorTarget::kSource); | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |       DCHECK(!maybe_getter_proxy.IsEmpty()); | 
					
						
							|  |  |  |       getter_proxy = maybe_getter_proxy.ToLocalChecked(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!setter->IsNullOrUndefined() && setter->IsObject()) { | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |       v8::Local<v8::Context> source_context = | 
					
						
							|  |  |  |           getter->GetCreationContextChecked(); | 
					
						
							| 
									
										
										
										
											2022-05-17 12:48:40 -04:00
										 |  |  |       v8::MaybeLocal<v8::Value> maybe_setter_proxy = PassValueToOtherContext( | 
					
						
							| 
									
										
										
										
											2023-10-18 07:21:42 -07:00
										 |  |  |           source_context, main_context, setter, source_context->Global(), | 
					
						
							| 
									
										
										
										
											2023-04-10 14:58:27 -07:00
										 |  |  |           &object_cache, false, 1, BridgeErrorTarget::kSource); | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |       DCHECK(!maybe_setter_proxy.IsEmpty()); | 
					
						
							|  |  |  |       setter_proxy = maybe_setter_proxy.ToLocalChecked(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     v8::PropertyDescriptor desc(getter_proxy, setter_proxy); | 
					
						
							|  |  |  |     bool success = IsTrue(target_object->DefineProperty( | 
					
						
							|  |  |  |         main_context, gin::StringToV8(args->isolate(), final_key), desc)); | 
					
						
							|  |  |  |     DCHECK(success); | 
					
						
							|  |  |  |     return success; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool IsCalledFromMainWorld(v8::Isolate* isolate) { | 
					
						
							|  |  |  |   auto* render_frame = GetRenderFrame(isolate->GetCurrentContext()->Global()); | 
					
						
							|  |  |  |   CHECK(render_frame); | 
					
						
							|  |  |  |   auto* frame = render_frame->GetWebFrame(); | 
					
						
							|  |  |  |   CHECK(frame); | 
					
						
							|  |  |  |   v8::Local<v8::Context> main_context = frame->MainWorldScriptContext(); | 
					
						
							|  |  |  |   return isolate->GetCurrentContext() == main_context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | }  // namespace api
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Initialize(v8::Local<v8::Object> exports, | 
					
						
							|  |  |  |                 v8::Local<v8::Value> unused, | 
					
						
							|  |  |  |                 v8::Local<v8::Context> context, | 
					
						
							|  |  |  |                 void* priv) { | 
					
						
							|  |  |  |   v8::Isolate* isolate = context->GetIsolate(); | 
					
						
							| 
									
										
										
										
											2019-10-31 16:56:00 +09:00
										 |  |  |   gin_helper::Dictionary dict(isolate, exports); | 
					
						
							| 
									
										
										
										
											2022-09-21 23:47:10 +05:30
										 |  |  |   dict.SetMethod("exposeAPIInWorld", &electron::api::ExposeAPIInWorld); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:46:04 -07:00
										 |  |  |   dict.SetMethod("_overrideGlobalValueFromIsolatedWorld", | 
					
						
							|  |  |  |                  &electron::api::OverrideGlobalValueFromIsolatedWorld); | 
					
						
							| 
									
										
										
										
											2020-04-22 12:42:51 -07:00
										 |  |  |   dict.SetMethod("_overrideGlobalPropertyFromIsolatedWorld", | 
					
						
							|  |  |  |                  &electron::api::OverrideGlobalPropertyFromIsolatedWorld); | 
					
						
							|  |  |  |   dict.SetMethod("_isCalledFromMainWorld", | 
					
						
							|  |  |  |                  &electron::api::IsCalledFromMainWorld); | 
					
						
							| 
									
										
										
										
											2021-06-13 19:06:09 -07:00
										 |  |  | #if DCHECK_IS_ON()
 | 
					
						
							| 
									
										
										
										
											2020-05-15 11:57:40 -07:00
										 |  |  |   dict.Set("_isDebug", true); | 
					
						
							| 
									
										
										
										
											2019-10-18 12:57:09 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-09 02:31:38 +01:00
										 |  |  | NODE_LINKED_BINDING_CONTEXT_AWARE(electron_renderer_context_bridge, Initialize) |