From c278e82252a35dd31989c5e0b658b3d10788163d Mon Sep 17 00:00:00 2001 From: Mikhail Naganov Date: Mon, 14 Mar 2016 16:45:44 -0700 Subject: [PATCH] Store and use last base URL between DidStart / DidStopLoading As it turns out that the committed entry may omit the base URL if there has been a javascript: URL navigation after LoadDataWithBaseURL. Thus we have to store the base url on DidStartLoading and use it in DidStopLoading / DidFinishLoad. We have to preserve the old hack with SetToBaseURLForDataURLIfNeeded because if the base URL is not valid, DidFinishLoad doesn't receive it from the renderer. This change is a short-term workaround. The correct solution is to pass a flag that the load was through LoadDataWithBaseURL via Blink, so base URL can be restored correctly within NavigationController. BUG=594001 Review URL: https://codereview.chromium.org/1779363004 Cr-Commit-Position: refs/heads/master@{#381108} (cherry picked from commit ce1f4d009316d8f5f4c49b18e1bd94687d80c59d) Review URL: https://codereview.chromium.org/1799973002 . Cr-Commit-Position: refs/branch-heads/2661@{#228} Cr-Branched-From: ef6f6ae5e4c96622286b563658d5cd62a6cf1197-refs/heads/master@{#378081} --- .../test/LoadDataWithBaseUrlTest.java | 33 +++++++++++++++++++ .../android/web_contents_observer_proxy.cc | 15 ++++++++- .../android/web_contents_observer_proxy.h | 1 + 3 files changed, 48 insertions(+), 1 deletion(-) diff --git a/android_webview/javatests/src/org/chromium/android_webview/test/LoadDataWithBaseUrlTest.java b/android_webview/javatests/src/org/chromium/android_webview/test/LoadDataWithBaseUrlTest.java index 177f8c73e89de..6fb0b3d6f36d4 100644 --- a/android_webview/javatests/src/org/chromium/android_webview/test/LoadDataWithBaseUrlTest.java +++ b/android_webview/javatests/src/org/chromium/android_webview/test/LoadDataWithBaseUrlTest.java @@ -373,4 +373,37 @@ public void testLoadLargeData() throws Throwable { assertEquals("true", executeJavaScriptAndWaitForResult(mAwContents, mContentsClient, "window.gotToEndOfBody")); } + + @SmallTest + @Feature({"AndroidWebView"}) + public void testOnPageFinishedWhenInterrupted() throws Throwable { + // See crbug.com/594001 -- when a javascript: URL is loaded, the pending entry + // gets discarded and the previous load goes through a different path + // inside NavigationController. + final String pageHtml = "Hello, world!"; + final String baseUrl = "http://example.com/"; + final TestCallbackHelperContainer.OnPageFinishedHelper onPageFinishedHelper = + mContentsClient.getOnPageFinishedHelper(); + final int callCount = onPageFinishedHelper.getCallCount(); + loadDataWithBaseUrlAsync(mAwContents, pageHtml, "text/html", false, baseUrl, null); + loadUrlAsync(mAwContents, "javascript:42"); + onPageFinishedHelper.waitForCallback(callCount); + assertEquals(baseUrl, onPageFinishedHelper.getUrl()); + } + + @SmallTest + @Feature({"AndroidWebView"}) + public void testOnPageFinishedWithInvalidBaseUrlWhenInterrupted() throws Throwable { + final String pageHtml = CommonResources.ABOUT_HTML; + final String invalidBaseUrl = "http://"; + final TestCallbackHelperContainer.OnPageFinishedHelper onPageFinishedHelper = + mContentsClient.getOnPageFinishedHelper(); + final int callCount = onPageFinishedHelper.getCallCount(); + getAwSettingsOnUiThread(mAwContents).setJavaScriptEnabled(true); + loadDataWithBaseUrlAsync(mAwContents, pageHtml, "text/html", false, invalidBaseUrl, null); + loadUrlAsync(mAwContents, "javascript:42"); + onPageFinishedHelper.waitForCallback(callCount); + // Verify that the load succeeds. The actual base url is undefined. + assertEquals(CommonResources.ABOUT_TITLE, getTitleOnUiThread(mAwContents)); + } } diff --git a/content/browser/android/web_contents_observer_proxy.cc b/content/browser/android/web_contents_observer_proxy.cc index 4b5b388ce16f7..0483fd867f004 100644 --- a/content/browser/android/web_contents_observer_proxy.cc +++ b/content/browser/android/web_contents_observer_proxy.cc @@ -93,6 +93,9 @@ void WebContentsObserverProxy::DidStartLoading() { ScopedJavaLocalRef obj(java_observer_); ScopedJavaLocalRef jstring_url( ConvertUTF8ToJavaString(env, web_contents()->GetVisibleURL().spec())); + if (auto entry = web_contents()->GetController().GetPendingEntry()) { + base_url_of_last_started_data_url_ = entry->GetBaseURLForDataURL(); + } Java_WebContentsObserverProxy_didStartLoading(env, obj.obj(), jstring_url.obj()); } @@ -102,6 +105,8 @@ void WebContentsObserverProxy::DidStopLoading() { ScopedJavaLocalRef obj(java_observer_); std::string url_string = web_contents()->GetLastCommittedURL().spec(); SetToBaseURLForDataURLIfNeeded(&url_string); + // DidStopLoading is the last event we should get. + base_url_of_last_started_data_url_ = GURL::EmptyGURL(); ScopedJavaLocalRef jstring_url(ConvertUTF8ToJavaString( env, url_string)); Java_WebContentsObserverProxy_didStopLoading(env, obj.obj(), @@ -317,8 +322,16 @@ void WebContentsObserverProxy::SetToBaseURLForDataURLIfNeeded( NavigationEntry* entry = web_contents()->GetController().GetLastCommittedEntry(); // Note that GetBaseURLForDataURL is only used by the Android WebView. - if (entry && !entry->GetBaseURLForDataURL().is_empty()) + // FIXME: Should we only return valid specs and "about:blank" for invalid + // ones? This may break apps. + if (entry && !entry->GetBaseURLForDataURL().is_empty()) { *url = entry->GetBaseURLForDataURL().possibly_invalid_spec(); + } else if (!base_url_of_last_started_data_url_.is_empty()) { + // NavigationController can lose the pending entry and recreate it without + // a base URL if there has been a loadUrl("javascript:...") after + // loadDataWithBaseUrl. + *url = base_url_of_last_started_data_url_.possibly_invalid_spec(); + } } bool RegisterWebContentsObserverProxy(JNIEnv* env) { diff --git a/content/browser/android/web_contents_observer_proxy.h b/content/browser/android/web_contents_observer_proxy.h index 1005a68497edf..d37f718c64d04 100644 --- a/content/browser/android/web_contents_observer_proxy.h +++ b/content/browser/android/web_contents_observer_proxy.h @@ -84,6 +84,7 @@ class WebContentsObserverProxy : public WebContentsObserver { bool was_ignored_by_handler); base::android::ScopedJavaGlobalRef java_observer_; + GURL base_url_of_last_started_data_url_; DISALLOW_COPY_AND_ASSIGN(WebContentsObserverProxy); };