Chromium Code Reviews| Index: extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.cc |
| diff --git a/extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.cc b/extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.cc |
| index 78b76340f07594e71f6b6e7d9303300e5cf9cdd8..76e77681ee336d8a66db4fce75fc2c1e551af5f4 100644 |
| --- a/extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.cc |
| +++ b/extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.cc |
| @@ -4,12 +4,16 @@ |
| #include "extensions/browser/guest_view/mime_handler_view/mime_handler_stream_manager.h" |
| +#include <map> |
| #include "base/memory/ptr_util.h" |
| #include "base/memory/singleton.h" |
| #include "components/keyed_service/content/browser_context_dependency_manager.h" |
| #include "components/keyed_service/content/browser_context_keyed_service_factory.h" |
| +#include "content/public/browser/navigation_handle.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/render_process_host.h" |
| +#include "content/public/browser/web_contents.h" |
| +#include "content/public/common/browser_side_navigation_policy.h" |
| #include "extensions/browser/extension_registry.h" |
| #include "extensions/browser/extensions_browser_client.h" |
| #include "extensions/browser/guest_view/mime_handler_view/mime_handler_view_guest.h" |
| @@ -72,9 +76,10 @@ class MimeHandlerStreamManager::EmbedderObserver |
| : public content::WebContentsObserver { |
| public: |
| EmbedderObserver(MimeHandlerStreamManager* stream_manager, |
| + const std::string& view_id, |
| + int frame_tree_node_id, |
| int render_process_id, |
| - int render_frame_id, |
| - const std::string& view_id); |
| + int render_frame_id); |
| private: |
| // WebContentsObserver overrides. |
| @@ -86,15 +91,29 @@ class MimeHandlerStreamManager::EmbedderObserver |
| bool is_error_page, |
| bool is_iframe_srcdoc) override; |
| void WebContentsDestroyed() override; |
| + void DidStartNavigation( |
| + content::NavigationHandle* navigation_handle) override; |
| + void RenderFrameHostChanged(content::RenderFrameHost* old_host, |
| + content::RenderFrameHost* new_host) override; |
| void AbortStream(); |
| bool IsTrackedRenderFrameHost(content::RenderFrameHost* render_frame_host); |
| MimeHandlerStreamManager* const stream_manager_; |
| + const std::string view_id_; |
| + const int frame_tree_node_id_; |
| const int render_process_id_; |
| const int render_frame_id_; |
| - const std::string view_id_; |
| + // With PlzNavigate we get an initial provisional load notification for the |
| + // URL the mime handler is serving. We don't want to clean up the stream |
| + // here. This field helps us track the first load notification. Defaults |
| + // to true. |
| + bool initial_load_for_frame_; |
| + // If a RFH is swapped with another RFH, this is set to the new RFH. This |
| + // ensures that we don't inadvarently clean up the stream when the old RFH |
| + // dies. |
| + content::RenderFrameHost* new_host_; |
| }; |
| MimeHandlerStreamManager::MimeHandlerStreamManager() |
| @@ -113,13 +132,14 @@ MimeHandlerStreamManager* MimeHandlerStreamManager::Get( |
| void MimeHandlerStreamManager::AddStream( |
| const std::string& view_id, |
| std::unique_ptr<StreamContainer> stream, |
| + int frame_tree_node_id, |
| int render_process_id, |
| int render_frame_id) { |
| streams_by_extension_id_[stream->extension_id()].insert(view_id); |
| auto result = streams_.insert(std::make_pair(view_id, std::move(stream))); |
| DCHECK(result.second); |
| embedder_observers_[view_id] = base::MakeUnique<EmbedderObserver>( |
| - this, render_process_id, render_frame_id, view_id); |
| + this, view_id, frame_tree_node_id, render_process_id, render_frame_id); |
| } |
| std::unique_ptr<StreamContainer> MimeHandlerStreamManager::ReleaseStream( |
| @@ -153,23 +173,32 @@ void MimeHandlerStreamManager::OnExtensionUnloaded( |
| MimeHandlerStreamManager::EmbedderObserver::EmbedderObserver( |
| MimeHandlerStreamManager* stream_manager, |
| + const std::string& view_id, |
| + int frame_tree_node_id, |
| int render_process_id, |
| - int render_frame_id, |
| - const std::string& view_id) |
| - : WebContentsObserver(content::WebContents::FromRenderFrameHost( |
| - content::RenderFrameHost::FromID(render_process_id, |
| - render_frame_id))), |
| - stream_manager_(stream_manager), |
| + int render_frame_id) |
| + : stream_manager_(stream_manager), |
| + view_id_(view_id), |
| + frame_tree_node_id_(frame_tree_node_id), |
| render_process_id_(render_process_id), |
| render_frame_id_(render_frame_id), |
| - view_id_(view_id) { |
| + initial_load_for_frame_(true), |
| + new_host_(nullptr) { |
| + content::WebContents* web_contents = nullptr; |
| + if (frame_tree_node_id != -1) { |
| + web_contents = |
| + content::WebContents::FromFrameTreeNodeId(frame_tree_node_id); |
| + } else { |
| + web_contents = content::WebContents::FromRenderFrameHost( |
| + content::RenderFrameHost::FromID(render_process_id, render_frame_id)); |
| + } |
| + content::WebContentsObserver::Observe(web_contents); |
| } |
| void MimeHandlerStreamManager::EmbedderObserver::RenderFrameDeleted( |
| content::RenderFrameHost* render_frame_host) { |
| if (!IsTrackedRenderFrameHost(render_frame_host)) |
| return; |
| - |
| AbortStream(); |
| } |
| @@ -177,6 +206,7 @@ void MimeHandlerStreamManager::EmbedderObserver::RenderProcessGone( |
| base::TerminationStatus status) { |
| AbortStream(); |
| } |
| + |
| void MimeHandlerStreamManager::EmbedderObserver:: |
| DidStartProvisionalLoadForFrame(content::RenderFrameHost* render_frame_host, |
| const GURL& validated_url, |
| @@ -184,10 +214,35 @@ void MimeHandlerStreamManager::EmbedderObserver:: |
| bool is_iframe_srcdoc) { |
| if (!IsTrackedRenderFrameHost(render_frame_host)) |
| return; |
| - |
| + // With PlzNavigate we get a provisional load notification for the URL we are |
| + // serving. We don't want to clean up the stream here. |
| + if (initial_load_for_frame_ && content::IsBrowserSideNavigationEnabled()) { |
| + initial_load_for_frame_ = false; |
| + return; |
| + } |
| AbortStream(); |
| } |
| +void MimeHandlerStreamManager::EmbedderObserver::DidStartNavigation( |
| + content::NavigationHandle* navigation_handle) { |
| + // If the top level frame is navigating away, clean up the stream. |
| + if (navigation_handle->IsInMainFrame()) |
| + AbortStream(); |
| +} |
| + |
| +void MimeHandlerStreamManager::EmbedderObserver::RenderFrameHostChanged( |
| + content::RenderFrameHost* old_host, |
| + content::RenderFrameHost* new_host) { |
| + new_host_ = new_host; |
| + // Update the RFNID, RFID, RPIDs to those of the new RFH. This ensures |
| + // that if the new RFH gets deleted before loading the stream, we will |
| + // abort it. |
| + frame_tree_node_id_ = new_host_->GetFrameTreeNodeId(); |
|
lazyboy
2016/09/22 22:23:45
Thanks.
Doesn't FTN id stay the same?
I'd DCHECK_
ananta
2016/09/22 23:03:18
Yes. It does. Added a DCHECK
|
| + render_frame_id_ = new_host_->GetRoutingID(); |
| + render_process_id_ = new_host_->GetProcess()->GetID(); |
| + |
| +} |
| + |
| void MimeHandlerStreamManager::EmbedderObserver::WebContentsDestroyed() { |
| AbortStream(); |
| } |
| @@ -200,8 +255,18 @@ void MimeHandlerStreamManager::EmbedderObserver::AbortStream() { |
| bool MimeHandlerStreamManager::EmbedderObserver::IsTrackedRenderFrameHost( |
| content::RenderFrameHost* render_frame_host) { |
| - return render_frame_host->GetRoutingID() == render_frame_id_ && |
| - render_frame_host->GetProcess()->GetID() == render_process_id_; |
| + // We don't want to abort the stream if the frame we were tracking changed to |
| + // new_host_. |
| + if (new_host_ && (render_frame_host != new_host_)) |
| + return false; |
| + |
| + if (frame_tree_node_id_ != -1) { |
| + return render_frame_host->GetFrameTreeNodeId() == frame_tree_node_id_; |
| + } else { |
| + DCHECK((render_frame_id_ != -1) && (render_process_id_ != -1)); |
| + return render_frame_host->GetRoutingID() == render_frame_id_ && |
| + render_frame_host->GetProcess()->GetID() == render_process_id_; |
| + } |
| } |
| } // namespace extensions |