| // Copyright 2019 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/js_injection/renderer/js_communication.h" |
| |
| #include "components/js_injection/common/origin_matcher.h" |
| #include "components/js_injection/renderer/js_binding.h" |
| #include "content/public/common/isolated_world_ids.h" |
| #include "content/public/renderer/render_frame.h" |
| #include "third_party/blink/public/common/associated_interfaces/associated_interface_registry.h" |
| #include "third_party/blink/public/platform/task_type.h" |
| #include "third_party/blink/public/web/web_local_frame.h" |
| #include "third_party/blink/public/web/web_script_source.h" |
| #include "url/gurl.h" |
| #include "url/origin.h" |
| |
| namespace js_injection { |
| |
| struct JsCommunication::JsObjectInfo { |
| OriginMatcher origin_matcher; |
| mojo::AssociatedRemote<mojom::JsToBrowserMessaging> js_to_java_messaging; |
| }; |
| |
| struct JsCommunication::DocumentStartJavaScript { |
| OriginMatcher origin_matcher; |
| blink::WebString script; |
| int32_t script_id; |
| }; |
| |
| JsCommunication::JsCommunication(content::RenderFrame* render_frame) |
| : RenderFrameObserver(render_frame), |
| RenderFrameObserverTracker<JsCommunication>(render_frame) { |
| render_frame->GetAssociatedInterfaceRegistry()->AddInterface( |
| base::BindRepeating(&JsCommunication::BindPendingReceiver, |
| base::Unretained(this))); |
| } |
| |
| JsCommunication::~JsCommunication() = default; |
| |
| void JsCommunication::SetJsObjects( |
| std::vector<mojom::JsObjectPtr> js_object_ptrs) { |
| JsObjectMap js_objects; |
| for (const auto& js_object : js_object_ptrs) { |
| const auto& js_object_info_pair = js_objects.insert( |
| {js_object->js_object_name, std::make_unique<JsObjectInfo>()}); |
| JsObjectInfo* js_object_info = js_object_info_pair.first->second.get(); |
| js_object_info->origin_matcher = js_object->origin_matcher; |
| js_object_info->js_to_java_messaging = |
| mojo::AssociatedRemote<mojom::JsToBrowserMessaging>( |
| std::move(js_object->js_to_browser_messaging)); |
| } |
| js_objects_.swap(js_objects); |
| } |
| |
| void JsCommunication::AddDocumentStartScript( |
| mojom::DocumentStartJavaScriptPtr script_ptr) { |
| DocumentStartJavaScript* script = new DocumentStartJavaScript{ |
| script_ptr->origin_matcher, |
| blink::WebString::FromUTF16(script_ptr->script), script_ptr->script_id}; |
| scripts_.push_back(std::unique_ptr<DocumentStartJavaScript>(script)); |
| } |
| |
| void JsCommunication::RemoveDocumentStartScript(int32_t script_id) { |
| for (auto it = scripts_.begin(); it != scripts_.end(); ++it) { |
| if ((*it)->script_id == script_id) { |
| scripts_.erase(it); |
| break; |
| } |
| } |
| } |
| |
| void JsCommunication::DidClearWindowObject() { |
| if (inside_did_clear_window_object_) |
| return; |
| |
| base::AutoReset<bool> flag_entry(&inside_did_clear_window_object_, true); |
| |
| url::Origin frame_origin = |
| url::Origin(render_frame()->GetWebFrame()->GetSecurityOrigin()); |
| std::vector<std::unique_ptr<JsBinding>> js_bindings; |
| js_bindings.reserve(js_objects_.size()); |
| for (const auto& js_object : js_objects_) { |
| if (!js_object.second->origin_matcher.Matches(frame_origin)) |
| continue; |
| js_bindings.push_back( |
| JsBinding::Install(render_frame(), js_object.first, this)); |
| } |
| js_bindings_.swap(js_bindings); |
| } |
| |
| void JsCommunication::WillReleaseScriptContext(v8::Local<v8::Context> context, |
| int32_t world_id) { |
| // We created v8 global objects only in the main world, should clear them only |
| // when this is for main world. |
| if (world_id != content::ISOLATED_WORLD_ID_GLOBAL) |
| return; |
| |
| for (const auto& js_binding : js_bindings_) |
| js_binding->ReleaseV8GlobalObjects(); |
| } |
| |
| void JsCommunication::OnDestruct() { |
| delete this; |
| } |
| |
| void JsCommunication::RunScriptsAtDocumentStart() { |
| url::Origin frame_origin = |
| url::Origin(render_frame()->GetWebFrame()->GetSecurityOrigin()); |
| for (const auto& script : scripts_) { |
| if (!script->origin_matcher.Matches(frame_origin)) |
| continue; |
| render_frame()->GetWebFrame()->ExecuteScript( |
| blink::WebScriptSource(script->script)); |
| } |
| } |
| |
| void JsCommunication::BindPendingReceiver( |
| mojo::PendingAssociatedReceiver<mojom::JsCommunication> pending_receiver) { |
| receiver_.reset(); |
| receiver_.Bind(std::move(pending_receiver), |
| render_frame()->GetTaskRunner( |
| blink::TaskType::kInternalNavigationAssociated)); |
| } |
| |
| mojom::JsToBrowserMessaging* JsCommunication::GetJsToJavaMessage( |
| const std::u16string& js_object_name) { |
| auto iterator = js_objects_.find(js_object_name); |
| if (iterator == js_objects_.end()) |
| return nullptr; |
| return iterator->second->js_to_java_messaging.get(); |
| } |
| |
| } // namespace js_injection |