electron_api_sw_ipc_handler_impl.cc 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. // Copyright (c) 2025 Salesforce, 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/browser/electron_api_sw_ipc_handler_impl.h"
  5. #include <utility>
  6. #include "base/containers/unique_ptr_adapters.h"
  7. #include "content/public/browser/render_frame_host.h"
  8. #include "content/public/browser/render_process_host.h"
  9. #include "mojo/public/cpp/bindings/self_owned_receiver.h"
  10. #include "shell/browser/api/electron_api_session.h"
  11. #include "shell/browser/electron_browser_context.h"
  12. #include "shell/browser/javascript_environment.h"
  13. #include "shell/common/gin_helper/dictionary.h"
  14. namespace electron {
  15. namespace {
  16. const void* const kUserDataKey = &kUserDataKey;
  17. class ServiceWorkerIPCList : public base::SupportsUserData::Data {
  18. public:
  19. std::vector<std::unique_ptr<ElectronApiSWIPCHandlerImpl>> list;
  20. static ServiceWorkerIPCList* Get(
  21. content::RenderProcessHost* render_process_host,
  22. bool create_if_not_exists) {
  23. auto* service_worker_ipc_list = static_cast<ServiceWorkerIPCList*>(
  24. render_process_host->GetUserData(kUserDataKey));
  25. if (!service_worker_ipc_list && !create_if_not_exists) {
  26. return nullptr;
  27. }
  28. if (!service_worker_ipc_list) {
  29. auto new_ipc_list = std::make_unique<ServiceWorkerIPCList>();
  30. service_worker_ipc_list = new_ipc_list.get();
  31. render_process_host->SetUserData(kUserDataKey, std::move(new_ipc_list));
  32. }
  33. return service_worker_ipc_list;
  34. }
  35. };
  36. } // namespace
  37. ElectronApiSWIPCHandlerImpl::ElectronApiSWIPCHandlerImpl(
  38. content::RenderProcessHost* render_process_host,
  39. int64_t version_id,
  40. mojo::PendingAssociatedReceiver<mojom::ElectronApiIPC> receiver)
  41. : render_process_host_(render_process_host), version_id_(version_id) {
  42. DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
  43. receiver_.Bind(std::move(receiver));
  44. receiver_.set_disconnect_handler(
  45. base::BindOnce(&ElectronApiSWIPCHandlerImpl::RemoteDisconnected,
  46. base::Unretained(this)));
  47. render_process_host_->AddObserver(this);
  48. }
  49. ElectronApiSWIPCHandlerImpl::~ElectronApiSWIPCHandlerImpl() {
  50. render_process_host_->RemoveObserver(this);
  51. }
  52. void ElectronApiSWIPCHandlerImpl::RemoteDisconnected() {
  53. receiver_.reset();
  54. Destroy();
  55. }
  56. void ElectronApiSWIPCHandlerImpl::Message(bool internal,
  57. const std::string& channel,
  58. blink::CloneableMessage arguments) {
  59. auto* session = GetSession();
  60. v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
  61. v8::HandleScope handle_scope(isolate);
  62. auto event = MakeIPCEvent(isolate, session, internal);
  63. if (event.IsEmpty())
  64. return;
  65. session->Message(event, channel, std::move(arguments));
  66. }
  67. void ElectronApiSWIPCHandlerImpl::Invoke(bool internal,
  68. const std::string& channel,
  69. blink::CloneableMessage arguments,
  70. InvokeCallback callback) {
  71. auto* session = GetSession();
  72. v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
  73. v8::HandleScope handle_scope(isolate);
  74. auto event = MakeIPCEvent(isolate, session, internal, std::move(callback));
  75. if (event.IsEmpty())
  76. return;
  77. session->Invoke(event, channel, std::move(arguments));
  78. }
  79. void ElectronApiSWIPCHandlerImpl::ReceivePostMessage(
  80. const std::string& channel,
  81. blink::TransferableMessage message) {
  82. auto* session = GetSession();
  83. v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
  84. v8::HandleScope handle_scope(isolate);
  85. auto event = MakeIPCEvent(isolate, session, false);
  86. if (event.IsEmpty())
  87. return;
  88. session->ReceivePostMessage(event, channel, std::move(message));
  89. }
  90. void ElectronApiSWIPCHandlerImpl::MessageSync(bool internal,
  91. const std::string& channel,
  92. blink::CloneableMessage arguments,
  93. MessageSyncCallback callback) {
  94. auto* session = GetSession();
  95. v8::Isolate* isolate = electron::JavascriptEnvironment::GetIsolate();
  96. v8::HandleScope handle_scope(isolate);
  97. auto event = MakeIPCEvent(isolate, session, internal, std::move(callback));
  98. if (event.IsEmpty())
  99. return;
  100. session->MessageSync(event, channel, std::move(arguments));
  101. }
  102. void ElectronApiSWIPCHandlerImpl::MessageHost(
  103. const std::string& channel,
  104. blink::CloneableMessage arguments) {
  105. NOTIMPLEMENTED(); // Service workers have no <webview>
  106. }
  107. ElectronBrowserContext* ElectronApiSWIPCHandlerImpl::GetBrowserContext() {
  108. auto* browser_context = static_cast<ElectronBrowserContext*>(
  109. render_process_host_->GetBrowserContext());
  110. return browser_context;
  111. }
  112. api::Session* ElectronApiSWIPCHandlerImpl::GetSession() {
  113. return api::Session::FromBrowserContext(GetBrowserContext());
  114. }
  115. gin::Handle<gin_helper::internal::Event>
  116. ElectronApiSWIPCHandlerImpl::MakeIPCEvent(
  117. v8::Isolate* isolate,
  118. api::Session* session,
  119. bool internal,
  120. electron::mojom::ElectronApiIPC::InvokeCallback callback) {
  121. if (!session) {
  122. if (callback) {
  123. // We must always invoke the callback if present.
  124. gin_helper::internal::ReplyChannel::Create(isolate, std::move(callback))
  125. ->SendError("Session does not exist");
  126. }
  127. return {};
  128. }
  129. gin::Handle<gin_helper::internal::Event> event =
  130. gin_helper::internal::Event::New(isolate);
  131. v8::Local<v8::Object> event_object = event.ToV8().As<v8::Object>();
  132. gin_helper::Dictionary dict(isolate, event_object);
  133. dict.Set("type", "service-worker");
  134. dict.Set("versionId", version_id_);
  135. dict.Set("processId", render_process_host_->GetID().GetUnsafeValue());
  136. // Set session to provide context for getting preloads
  137. dict.Set("session", session);
  138. if (callback)
  139. dict.Set("_replyChannel", gin_helper::internal::ReplyChannel::Create(
  140. isolate, std::move(callback)));
  141. if (internal)
  142. dict.SetHidden("internal", internal);
  143. return event;
  144. }
  145. void ElectronApiSWIPCHandlerImpl::Destroy() {
  146. DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
  147. auto* service_worker_ipc_list = ServiceWorkerIPCList::Get(
  148. render_process_host_, /*create_if_not_exists=*/false);
  149. CHECK(service_worker_ipc_list);
  150. // std::erase_if will lead to a call to the destructor for this object.
  151. std::erase_if(service_worker_ipc_list->list, base::MatchesUniquePtr(this));
  152. }
  153. void ElectronApiSWIPCHandlerImpl::RenderProcessExited(
  154. content::RenderProcessHost* host,
  155. const content::ChildProcessTerminationInfo& info) {
  156. CHECK_EQ(host, render_process_host_);
  157. // TODO(crbug.com/1407197): Investigate clearing the user data from
  158. // RenderProcessHostImpl::Cleanup.
  159. Destroy();
  160. // This instance has now been deleted.
  161. }
  162. // static
  163. void ElectronApiSWIPCHandlerImpl::BindReceiver(
  164. int render_process_id,
  165. int64_t version_id,
  166. mojo::PendingAssociatedReceiver<mojom::ElectronApiIPC> receiver) {
  167. DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
  168. auto* render_process_host =
  169. content::RenderProcessHost::FromID(render_process_id);
  170. if (!render_process_host) {
  171. return;
  172. }
  173. auto* service_worker_ipc_list = ServiceWorkerIPCList::Get(
  174. render_process_host, /*create_if_not_exists=*/true);
  175. service_worker_ipc_list->list.push_back(
  176. std::make_unique<ElectronApiSWIPCHandlerImpl>(
  177. render_process_host, version_id, std::move(receiver)));
  178. }
  179. } // namespace electron