renderer_client_base.cc 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650
  1. // Copyright (c) 2017 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/renderer/renderer_client_base.h"
  5. #include <memory>
  6. #include <string>
  7. #include <utility>
  8. #include <vector>
  9. #include "base/command_line.h"
  10. #include "base/strings/string_split.h"
  11. #include "components/network_hints/renderer/web_prescient_networking_impl.h"
  12. #include "content/common/buildflags.h"
  13. #include "content/public/common/content_constants.h"
  14. #include "content/public/common/content_switches.h"
  15. #include "content/public/renderer/render_frame.h"
  16. #include "content/public/renderer/render_thread.h"
  17. #include "electron/buildflags/buildflags.h"
  18. #include "electron/fuses.h"
  19. #include "printing/buildflags/buildflags.h"
  20. #include "shell/browser/api/electron_api_protocol.h"
  21. #include "shell/common/api/electron_api_native_image.h"
  22. #include "shell/common/color_util.h"
  23. #include "shell/common/gin_helper/dictionary.h"
  24. #include "shell/common/node_includes.h"
  25. #include "shell/common/node_util.h"
  26. #include "shell/common/options_switches.h"
  27. #include "shell/common/plugin.mojom.h"
  28. #include "shell/common/world_ids.h"
  29. #include "shell/renderer/api/context_bridge/object_cache.h"
  30. #include "shell/renderer/api/electron_api_context_bridge.h"
  31. #include "shell/renderer/browser_exposed_renderer_interfaces.h"
  32. #include "shell/renderer/content_settings_observer.h"
  33. #include "shell/renderer/electron_api_service_impl.h"
  34. #include "shell/renderer/electron_autofill_agent.h"
  35. #include "third_party/abseil-cpp/absl/strings/str_format.h"
  36. #include "third_party/blink/public/common/associated_interfaces/associated_interface_provider.h"
  37. #include "third_party/blink/public/common/associated_interfaces/associated_interface_registry.h"
  38. #include "third_party/blink/public/common/web_preferences/web_preferences.h"
  39. #include "third_party/blink/public/platform/web_runtime_features.h"
  40. #include "third_party/blink/public/web/blink.h"
  41. #include "third_party/blink/public/web/web_custom_element.h" // NOLINT(build/include_alpha)
  42. #include "third_party/blink/public/web/web_frame_widget.h"
  43. #include "third_party/blink/public/web/web_local_frame.h"
  44. #include "third_party/blink/public/web/web_plugin_params.h"
  45. #include "third_party/blink/public/web/web_script_source.h"
  46. #include "third_party/blink/public/web/web_security_policy.h"
  47. #include "third_party/blink/public/web/web_view.h"
  48. #include "third_party/blink/renderer/platform/media/multi_buffer_data_source.h" // nogncheck
  49. #include "third_party/blink/renderer/platform/weborigin/scheme_registry.h" // nogncheck
  50. #include "third_party/widevine/cdm/buildflags.h"
  51. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  52. #include "components/pdf/renderer/internal_plugin_renderer_helpers.h"
  53. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  54. #if BUILDFLAG(IS_WIN)
  55. #include <shlobj.h>
  56. #endif
  57. #if BUILDFLAG(ENABLE_WIDEVINE)
  58. #include "chrome/renderer/media/chrome_key_systems.h" // nogncheck
  59. #endif
  60. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  61. #include "components/spellcheck/renderer/spellcheck.h"
  62. #include "components/spellcheck/renderer/spellcheck_provider.h"
  63. #endif
  64. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  65. #include "components/pdf/common/constants.h" // nogncheck
  66. #include "components/pdf/common/pdf_util.h" // nogncheck
  67. #include "components/pdf/renderer/pdf_internal_plugin_delegate.h"
  68. #include "shell/common/electron_constants.h"
  69. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  70. #if BUILDFLAG(ENABLE_PLUGINS)
  71. #include "shell/common/plugin_info.h"
  72. #endif // BUILDFLAG(ENABLE_PLUGINS)
  73. #if BUILDFLAG(ENABLE_PRINTING)
  74. #include "components/printing/renderer/print_render_frame_helper.h"
  75. #include "printing/metafile_agent.h" // nogncheck
  76. #include "shell/renderer/printing/print_render_frame_helper_delegate.h"
  77. #endif // BUILDFLAG(ENABLE_PRINTING)
  78. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  79. #include "base/strings/utf_string_conversions.h"
  80. #include "content/public/common/webplugininfo.h"
  81. #include "extensions/common/constants.h"
  82. #include "extensions/common/extensions_client.h"
  83. #include "extensions/renderer/api/core_extensions_renderer_api_provider.h"
  84. #include "extensions/renderer/dispatcher.h"
  85. #include "extensions/renderer/extension_frame_helper.h"
  86. #include "extensions/renderer/extension_web_view_helper.h"
  87. #include "extensions/renderer/guest_view/mime_handler_view/mime_handler_view_container_manager.h"
  88. #include "shell/common/extensions/electron_extensions_client.h"
  89. #include "shell/renderer/extensions/electron_extensions_renderer_api_provider.h"
  90. #include "shell/renderer/extensions/electron_extensions_renderer_client.h"
  91. #endif // BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  92. namespace electron {
  93. content::RenderFrame* GetRenderFrame(v8::Local<v8::Object> value);
  94. namespace {
  95. void SetIsWebView(v8::Isolate* isolate, v8::Local<v8::Object> object) {
  96. gin_helper::Dictionary dict(isolate, object);
  97. dict.SetHidden("isWebView", true);
  98. }
  99. std::vector<std::string> ParseSchemesCLISwitch(
  100. base::CommandLine* command_line,
  101. const std::string_view switch_name) {
  102. std::string custom_schemes = command_line->GetSwitchValueASCII(switch_name);
  103. return base::SplitString(custom_schemes, ",", base::TRIM_WHITESPACE,
  104. base::SPLIT_WANT_NONEMPTY);
  105. }
  106. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  107. class ChromePdfInternalPluginDelegate final
  108. : public pdf::PdfInternalPluginDelegate {
  109. public:
  110. ChromePdfInternalPluginDelegate() = default;
  111. ChromePdfInternalPluginDelegate(const ChromePdfInternalPluginDelegate&) =
  112. delete;
  113. ChromePdfInternalPluginDelegate& operator=(
  114. const ChromePdfInternalPluginDelegate&) = delete;
  115. ~ChromePdfInternalPluginDelegate() override = default;
  116. // `pdf::PdfInternalPluginDelegate`:
  117. [[nodiscard]] bool IsAllowedOrigin(const url::Origin& origin) const override {
  118. return origin.scheme() == extensions::kExtensionScheme &&
  119. origin.host() == extension_misc::kPdfExtensionId;
  120. }
  121. };
  122. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  123. // static
  124. RendererClientBase* g_renderer_client_base = nullptr;
  125. bool IsDevTools(content::RenderFrame* render_frame) {
  126. return render_frame->GetWebFrame()->GetDocument().Url().ProtocolIs(
  127. "devtools");
  128. }
  129. bool IsDevToolsExtension(content::RenderFrame* render_frame) {
  130. return render_frame->GetWebFrame()->GetDocument().Url().ProtocolIs(
  131. "chrome-extension");
  132. }
  133. } // namespace
  134. RendererClientBase::RendererClientBase() {
  135. auto* command_line = base::CommandLine::ForCurrentProcess();
  136. // Parse --service-worker-schemes=scheme1,scheme2
  137. std::vector<std::string> service_worker_schemes_list =
  138. ParseSchemesCLISwitch(command_line, switches::kServiceWorkerSchemes);
  139. for (const std::string& scheme : service_worker_schemes_list)
  140. electron::api::AddServiceWorkerScheme(scheme);
  141. // Parse --standard-schemes=scheme1,scheme2
  142. std::vector<std::string> standard_schemes_list =
  143. ParseSchemesCLISwitch(command_line, switches::kStandardSchemes);
  144. for (const std::string& scheme : standard_schemes_list)
  145. url::AddStandardScheme(scheme.c_str(), url::SCHEME_WITH_HOST);
  146. // Parse --cors-schemes=scheme1,scheme2
  147. std::vector<std::string> cors_schemes_list =
  148. ParseSchemesCLISwitch(command_line, switches::kCORSSchemes);
  149. for (const std::string& scheme : cors_schemes_list)
  150. url::AddCorsEnabledScheme(scheme.c_str());
  151. // Parse --streaming-schemes=scheme1,scheme2
  152. std::vector<std::string> streaming_schemes_list =
  153. ParseSchemesCLISwitch(command_line, switches::kStreamingSchemes);
  154. for (const std::string& scheme : streaming_schemes_list)
  155. blink::AddStreamingScheme(scheme.c_str());
  156. // Parse --secure-schemes=scheme1,scheme2
  157. std::vector<std::string> secure_schemes_list =
  158. ParseSchemesCLISwitch(command_line, switches::kSecureSchemes);
  159. for (const std::string& scheme : secure_schemes_list)
  160. url::AddSecureScheme(scheme.data());
  161. // We rely on the unique process host id which is notified to the
  162. // renderer process via command line switch from the content layer,
  163. // if this switch is removed from the content layer for some reason,
  164. // we should define our own.
  165. DCHECK(command_line->HasSwitch(::switches::kRendererClientId));
  166. renderer_client_id_ =
  167. command_line->GetSwitchValueASCII(::switches::kRendererClientId);
  168. g_renderer_client_base = this;
  169. }
  170. RendererClientBase::~RendererClientBase() {
  171. g_renderer_client_base = nullptr;
  172. }
  173. // static
  174. RendererClientBase* RendererClientBase::Get() {
  175. DCHECK(g_renderer_client_base);
  176. return g_renderer_client_base;
  177. }
  178. void RendererClientBase::BindProcess(v8::Isolate* isolate,
  179. gin_helper::Dictionary* process,
  180. content::RenderFrame* render_frame) {
  181. auto context_id = absl::StrFormat("%s-%" PRId64, renderer_client_id_.c_str(),
  182. ++next_context_id_);
  183. process->SetReadOnly("isMainFrame", render_frame->IsMainFrame());
  184. process->SetReadOnly("contextIsolated",
  185. render_frame->GetBlinkPreferences().context_isolation);
  186. process->SetReadOnly("contextId", context_id);
  187. }
  188. bool RendererClientBase::ShouldLoadPreload(
  189. v8::Local<v8::Context> context,
  190. content::RenderFrame* render_frame) const {
  191. auto prefs = render_frame->GetBlinkPreferences();
  192. bool is_main_frame = render_frame->IsMainFrame();
  193. bool is_devtools =
  194. IsDevTools(render_frame) || IsDevToolsExtension(render_frame);
  195. bool allow_node_in_sub_frames = prefs.node_integration_in_sub_frames;
  196. return (is_main_frame || is_devtools || allow_node_in_sub_frames) &&
  197. !IsWebViewFrame(context, render_frame);
  198. }
  199. void RendererClientBase::RenderThreadStarted() {
  200. auto* command_line = base::CommandLine::ForCurrentProcess();
  201. // Enable MessagePort close event by default.
  202. // The feature got reverted from stable to test in
  203. // https://chromium-review.googlesource.com/c/chromium/src/+/5276821
  204. // We had the event supported through patch before upstream support,
  205. // this is an alternative option than restoring our patch.
  206. blink::WebRuntimeFeatures::EnableFeatureFromString("MessagePortCloseEvent",
  207. true);
  208. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  209. extensions_client_ = std::make_unique<ElectronExtensionsClient>();
  210. extensions::ExtensionsClient::Set(extensions_client_.get());
  211. extensions_renderer_client_ =
  212. std::make_unique<ElectronExtensionsRendererClient>();
  213. extensions_renderer_client_->AddAPIProvider(
  214. std::make_unique<extensions::CoreExtensionsRendererAPIProvider>());
  215. extensions_renderer_client_->AddAPIProvider(
  216. std::make_unique<ElectronExtensionsRendererAPIProvider>());
  217. extensions::ExtensionsRendererClient::Set(extensions_renderer_client_.get());
  218. extensions_renderer_client_->RenderThreadStarted();
  219. WTF::String extension_scheme(extensions::kExtensionScheme);
  220. // Extension resources are HTTP-like and safe to expose to the fetch API. The
  221. // rules for the fetch API are consistent with XHR.
  222. blink::SchemeRegistry::RegisterURLSchemeAsSupportingFetchAPI(
  223. extension_scheme);
  224. // Extension resources, when loaded as the top-level document, should bypass
  225. // Blink's strict first-party origin checks.
  226. blink::SchemeRegistry::RegisterURLSchemeAsFirstPartyWhenTopLevel(
  227. extension_scheme);
  228. blink::SchemeRegistry::RegisterURLSchemeAsBypassingContentSecurityPolicy(
  229. extension_scheme);
  230. #endif
  231. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  232. spellcheck_ = std::make_unique<SpellCheck>(this);
  233. #endif
  234. blink::WebCustomElement::AddEmbedderCustomElementName("webview");
  235. blink::WebCustomElement::AddEmbedderCustomElementName("browserplugin");
  236. std::vector<std::string> fetch_enabled_schemes =
  237. ParseSchemesCLISwitch(command_line, switches::kFetchSchemes);
  238. for (const std::string& scheme : fetch_enabled_schemes) {
  239. blink::WebSecurityPolicy::RegisterURLSchemeAsSupportingFetchAPI(
  240. blink::WebString::FromASCII(scheme));
  241. }
  242. std::vector<std::string> service_worker_schemes =
  243. ParseSchemesCLISwitch(command_line, switches::kServiceWorkerSchemes);
  244. for (const std::string& scheme : service_worker_schemes)
  245. blink::WebSecurityPolicy::RegisterURLSchemeAsAllowingServiceWorkers(
  246. blink::WebString::FromASCII(scheme));
  247. std::vector<std::string> csp_bypassing_schemes =
  248. ParseSchemesCLISwitch(command_line, switches::kBypassCSPSchemes);
  249. for (const std::string& scheme : csp_bypassing_schemes)
  250. blink::SchemeRegistry::RegisterURLSchemeAsBypassingContentSecurityPolicy(
  251. WTF::String::FromUTF8(scheme));
  252. std::vector<std::string> code_cache_schemes_list =
  253. ParseSchemesCLISwitch(command_line, switches::kCodeCacheSchemes);
  254. for (const auto& scheme : code_cache_schemes_list) {
  255. blink::WebSecurityPolicy::RegisterURLSchemeAsCodeCacheWithHashing(
  256. blink::WebString::FromASCII(scheme));
  257. }
  258. // Allow file scheme to handle service worker by default.
  259. // FIXME(zcbenz): Can this be moved elsewhere?
  260. if (electron::fuses::IsGrantFileProtocolExtraPrivilegesEnabled()) {
  261. blink::WebSecurityPolicy::RegisterURLSchemeAsAllowingServiceWorkers("file");
  262. blink::SchemeRegistry::RegisterURLSchemeAsSupportingFetchAPI("file");
  263. }
  264. #if BUILDFLAG(IS_WIN)
  265. // Set ApplicationUserModelID in renderer process.
  266. std::wstring app_id =
  267. command_line->GetSwitchValueNative(switches::kAppUserModelId);
  268. if (!app_id.empty()) {
  269. SetCurrentProcessExplicitAppUserModelID(app_id.c_str());
  270. }
  271. #endif
  272. }
  273. void RendererClientBase::ExposeInterfacesToBrowser(mojo::BinderMap* binders) {
  274. // NOTE: Do not add binders directly within this method. Instead, modify the
  275. // definition of |ExposeElectronRendererInterfacesToBrowser()| to ensure
  276. // security review coverage.
  277. ExposeElectronRendererInterfacesToBrowser(this, binders);
  278. }
  279. void RendererClientBase::RenderFrameCreated(
  280. content::RenderFrame* render_frame) {
  281. #if defined(TOOLKIT_VIEWS)
  282. new AutofillAgent(render_frame,
  283. render_frame->GetAssociatedInterfaceRegistry());
  284. #endif
  285. new ContentSettingsObserver(render_frame);
  286. #if BUILDFLAG(ENABLE_PRINTING)
  287. new printing::PrintRenderFrameHelper(
  288. render_frame,
  289. std::make_unique<electron::PrintRenderFrameHelperDelegate>());
  290. #endif
  291. // Note: ElectronApiServiceImpl has to be created now to capture the
  292. // DidCreateDocumentElement event.
  293. new ElectronApiServiceImpl(render_frame, this);
  294. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  295. auto* dispatcher = extensions_renderer_client_->dispatcher();
  296. // ExtensionFrameHelper destroys itself when the RenderFrame is destroyed.
  297. new extensions::ExtensionFrameHelper(render_frame, dispatcher);
  298. dispatcher->OnRenderFrameCreated(render_frame);
  299. render_frame->GetAssociatedInterfaceRegistry()
  300. ->AddInterface<extensions::mojom::MimeHandlerViewContainerManager>(
  301. base::BindRepeating(
  302. &extensions::MimeHandlerViewContainerManager::BindReceiver,
  303. base::Unretained(render_frame)));
  304. #endif
  305. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  306. if (render_frame->GetBlinkPreferences().enable_spellcheck) {
  307. new SpellCheckProvider(render_frame, spellcheck_.get());
  308. }
  309. #endif
  310. }
  311. #if BUILDFLAG(ENABLE_BUILTIN_SPELLCHECKER)
  312. void RendererClientBase::GetInterface(
  313. const std::string& interface_name,
  314. mojo::ScopedMessagePipeHandle interface_pipe) {
  315. // TODO(crbug.com/977637): Get rid of the use of this implementation of
  316. // |service_manager::LocalInterfaceProvider|. This was done only to avoid
  317. // churning spellcheck code while eliminating the "chrome" and
  318. // "chrome_renderer" services. Spellcheck is (and should remain) the only
  319. // consumer of this implementation.
  320. content::RenderThread::Get()->BindHostReceiver(
  321. mojo::GenericPendingReceiver(interface_name, std::move(interface_pipe)));
  322. }
  323. #endif
  324. void RendererClientBase::DidClearWindowObject(
  325. content::RenderFrame* render_frame) {
  326. // Make sure every page will get a script context created.
  327. render_frame->GetWebFrame()->ExecuteScript(blink::WebScriptSource("void 0"));
  328. }
  329. bool RendererClientBase::OverrideCreatePlugin(
  330. content::RenderFrame* render_frame,
  331. const blink::WebPluginParams& params,
  332. blink::WebPlugin** plugin) {
  333. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  334. if (params.mime_type.Utf8() == pdf::kInternalPluginMimeType) {
  335. *plugin = pdf::CreateInternalPlugin(
  336. std::move(params), render_frame,
  337. std::make_unique<ChromePdfInternalPluginDelegate>());
  338. return true;
  339. }
  340. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  341. if (params.mime_type.Utf8() == content::kBrowserPluginMimeType ||
  342. render_frame->GetBlinkPreferences().enable_plugins)
  343. return false;
  344. *plugin = nullptr;
  345. return true;
  346. }
  347. void RendererClientBase::DidSetUserAgent(const std::string& user_agent) {
  348. #if BUILDFLAG(ENABLE_PRINTING)
  349. printing::SetAgent(user_agent);
  350. #endif
  351. }
  352. bool RendererClientBase::IsPluginHandledExternally(
  353. content::RenderFrame* render_frame,
  354. const blink::WebElement& plugin_element,
  355. const GURL& original_url,
  356. const std::string& mime_type) {
  357. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  358. DCHECK(plugin_element.HasHTMLTagName("object") ||
  359. plugin_element.HasHTMLTagName("embed"));
  360. mojo::AssociatedRemote<mojom::ElectronPluginInfoHost> plugin_info_host;
  361. render_frame->GetRemoteAssociatedInterfaces()->GetInterface(
  362. &plugin_info_host);
  363. mojom::PluginInfoPtr plugin_info = mojom::PluginInfo::New();
  364. plugin_info_host->GetPluginInfo(
  365. original_url, render_frame->GetWebFrame()->Top()->GetSecurityOrigin(),
  366. mime_type, &plugin_info);
  367. if (plugin_info->actual_mime_type == pdf::kInternalPluginMimeType) {
  368. if (IsPdfInternalPluginAllowedOrigin(
  369. render_frame->GetWebFrame()->GetSecurityOrigin())) {
  370. return true;
  371. }
  372. }
  373. return extensions::MimeHandlerViewContainerManager::Get(
  374. content::RenderFrame::FromWebFrame(
  375. plugin_element.GetDocument().GetFrame()),
  376. true /* create_if_does_not_exist */)
  377. ->CreateFrameContainer(plugin_element, original_url,
  378. plugin_info->actual_mime_type,
  379. plugin_info->plugin);
  380. #else
  381. return false;
  382. #endif
  383. }
  384. v8::Local<v8::Object> RendererClientBase::GetScriptableObject(
  385. const blink::WebElement& plugin_element,
  386. v8::Isolate* isolate) {
  387. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  388. // If there is a MimeHandlerView that can provide the scriptable object then
  389. // MaybeCreateMimeHandlerView must have been called before and a container
  390. // manager should exist.
  391. auto* container_manager = extensions::MimeHandlerViewContainerManager::Get(
  392. content::RenderFrame::FromWebFrame(
  393. plugin_element.GetDocument().GetFrame()),
  394. false /* create_if_does_not_exist */);
  395. if (container_manager)
  396. return container_manager->GetScriptableObject(plugin_element, isolate);
  397. #endif
  398. return {};
  399. }
  400. std::unique_ptr<blink::WebPrescientNetworking>
  401. RendererClientBase::CreatePrescientNetworking(
  402. content::RenderFrame* render_frame) {
  403. return std::make_unique<network_hints::WebPrescientNetworkingImpl>(
  404. render_frame);
  405. }
  406. void RendererClientBase::RunScriptsAtDocumentStart(
  407. content::RenderFrame* render_frame) {
  408. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  409. extensions_renderer_client_.get()->RunScriptsAtDocumentStart(render_frame);
  410. #endif
  411. }
  412. void RendererClientBase::RunScriptsAtDocumentIdle(
  413. content::RenderFrame* render_frame) {
  414. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  415. extensions_renderer_client_.get()->RunScriptsAtDocumentIdle(render_frame);
  416. #endif
  417. }
  418. void RendererClientBase::RunScriptsAtDocumentEnd(
  419. content::RenderFrame* render_frame) {
  420. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  421. extensions_renderer_client_.get()->RunScriptsAtDocumentEnd(render_frame);
  422. #endif
  423. }
  424. bool RendererClientBase::AllowScriptExtensionForServiceWorker(
  425. const url::Origin& script_origin) {
  426. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  427. return script_origin.scheme() == extensions::kExtensionScheme;
  428. #else
  429. return false;
  430. #endif
  431. }
  432. void RendererClientBase::DidInitializeServiceWorkerContextOnWorkerThread(
  433. blink::WebServiceWorkerContextProxy* context_proxy,
  434. const GURL& service_worker_scope,
  435. const GURL& script_url) {
  436. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  437. extensions_renderer_client_->dispatcher()
  438. ->DidInitializeServiceWorkerContextOnWorkerThread(
  439. context_proxy, service_worker_scope, script_url);
  440. #endif
  441. }
  442. void RendererClientBase::WillEvaluateServiceWorkerOnWorkerThread(
  443. blink::WebServiceWorkerContextProxy* context_proxy,
  444. v8::Local<v8::Context> v8_context,
  445. int64_t service_worker_version_id,
  446. const GURL& service_worker_scope,
  447. const GURL& script_url,
  448. const blink::ServiceWorkerToken& service_worker_token) {
  449. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  450. extensions_renderer_client_->dispatcher()
  451. ->WillEvaluateServiceWorkerOnWorkerThread(
  452. context_proxy, v8_context, service_worker_version_id,
  453. service_worker_scope, script_url, service_worker_token);
  454. #endif
  455. }
  456. void RendererClientBase::DidStartServiceWorkerContextOnWorkerThread(
  457. int64_t service_worker_version_id,
  458. const GURL& service_worker_scope,
  459. const GURL& script_url) {
  460. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  461. extensions_renderer_client_->dispatcher()
  462. ->DidStartServiceWorkerContextOnWorkerThread(
  463. service_worker_version_id, service_worker_scope, script_url);
  464. #endif
  465. }
  466. void RendererClientBase::WillDestroyServiceWorkerContextOnWorkerThread(
  467. v8::Local<v8::Context> context,
  468. int64_t service_worker_version_id,
  469. const GURL& service_worker_scope,
  470. const GURL& script_url) {
  471. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  472. extensions_renderer_client_->dispatcher()
  473. ->WillDestroyServiceWorkerContextOnWorkerThread(
  474. context, service_worker_version_id, service_worker_scope, script_url);
  475. #endif
  476. }
  477. void RendererClientBase::WebViewCreated(blink::WebView* web_view,
  478. bool was_created_by_renderer,
  479. const url::Origin* outermost_origin) {
  480. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  481. new extensions::ExtensionWebViewHelper(web_view, outermost_origin);
  482. #endif
  483. }
  484. v8::Local<v8::Context> RendererClientBase::GetContext(
  485. blink::WebLocalFrame* frame,
  486. v8::Isolate* isolate) const {
  487. auto* render_frame = content::RenderFrame::FromWebFrame(frame);
  488. DCHECK(render_frame);
  489. if (render_frame && render_frame->GetBlinkPreferences().context_isolation)
  490. return frame->GetScriptContextFromWorldId(isolate,
  491. WorldIDs::ISOLATED_WORLD_ID);
  492. else
  493. return frame->MainWorldScriptContext();
  494. }
  495. bool RendererClientBase::IsWebViewFrame(
  496. v8::Local<v8::Context> context,
  497. content::RenderFrame* render_frame) const {
  498. auto* isolate = context->GetIsolate();
  499. if (render_frame->IsMainFrame())
  500. return false;
  501. gin::Dictionary window_dict(
  502. isolate, GetContext(render_frame->GetWebFrame(), isolate)->Global());
  503. v8::Local<v8::Object> frame_element;
  504. if (!window_dict.Get("frameElement", &frame_element))
  505. return false;
  506. gin_helper::Dictionary frame_element_dict(isolate, frame_element);
  507. bool is_webview = false;
  508. return frame_element_dict.GetHidden("isWebView", &is_webview) && is_webview;
  509. }
  510. void RendererClientBase::SetupMainWorldOverrides(
  511. v8::Local<v8::Context> context,
  512. content::RenderFrame* render_frame) {
  513. auto prefs = render_frame->GetBlinkPreferences();
  514. // We only need to run the isolated bundle if webview is enabled
  515. if (!prefs.webview_tag)
  516. return;
  517. // Setup window overrides in the main world context
  518. // Wrap the bundle into a function that receives the isolatedApi as
  519. // an argument.
  520. auto* isolate = context->GetIsolate();
  521. v8::HandleScope handle_scope(isolate);
  522. v8::Context::Scope context_scope(context);
  523. auto isolated_api = gin_helper::Dictionary::CreateEmpty(isolate);
  524. isolated_api.SetMethod("allowGuestViewElementDefinition",
  525. &AllowGuestViewElementDefinition);
  526. isolated_api.SetMethod("setIsWebView", &SetIsWebView);
  527. auto source_context = GetContext(render_frame->GetWebFrame(), isolate);
  528. gin_helper::Dictionary global(isolate, source_context->Global());
  529. v8::Local<v8::Value> guest_view_internal;
  530. if (global.GetHidden("guestViewInternal", &guest_view_internal)) {
  531. auto result = api::PassValueToOtherContext(
  532. source_context, context, guest_view_internal, source_context->Global(),
  533. false, api::BridgeErrorTarget::kSource);
  534. if (!result.IsEmpty()) {
  535. isolated_api.Set("guestViewInternal", result.ToLocalChecked());
  536. }
  537. }
  538. std::vector<v8::Local<v8::String>> isolated_bundle_params = {
  539. node::FIXED_ONE_BYTE_STRING(isolate, "isolatedApi")};
  540. std::vector<v8::Local<v8::Value>> isolated_bundle_args = {
  541. isolated_api.GetHandle()};
  542. util::CompileAndCall(context, "electron/js2c/isolated_bundle",
  543. &isolated_bundle_params, &isolated_bundle_args);
  544. }
  545. // static
  546. void RendererClientBase::AllowGuestViewElementDefinition(
  547. v8::Isolate* isolate,
  548. v8::Local<v8::Object> context,
  549. v8::Local<v8::Function> register_cb) {
  550. v8::HandleScope handle_scope(isolate);
  551. v8::Context::Scope context_scope(context->GetCreationContextChecked());
  552. blink::WebCustomElement::EmbedderNamesAllowedScope embedder_names_scope;
  553. content::RenderFrame* render_frame = GetRenderFrame(context);
  554. if (!render_frame)
  555. return;
  556. render_frame->GetWebFrame()->RequestExecuteV8Function(
  557. context->GetCreationContextChecked(), register_cb, v8::Null(isolate), 0,
  558. nullptr, base::NullCallback());
  559. }
  560. } // namespace electron