// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.android_webview.test;
import static org.junit.Assert.assertNotEquals;
import static org.chromium.android_webview.test.AwActivityTestRule.SCALED_WAIT_TIMEOUT_MS;
import static org.chromium.android_webview.test.AwActivityTestRule.WAIT_TIMEOUT_MS;
import android.webkit.WebSettings;
import androidx.test.filters.SmallTest;
import org.junit.After;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.UseParametersRunnerFactory;
import org.chromium.android_webview.AwContents;
import org.chromium.android_webview.AwContentsClient.AwWebResourceError;
import org.chromium.android_webview.AwContentsClient.AwWebResourceRequest;
import org.chromium.android_webview.WebviewErrorCode;
import org.chromium.android_webview.test.util.AwTestTouchUtils;
import org.chromium.android_webview.test.util.CommonResources;
import org.chromium.base.test.util.Feature;
import org.chromium.content_public.browser.test.util.TestCallbackHelperContainer;
import org.chromium.content_public.common.ContentUrlConstants;
import org.chromium.net.test.util.TestWebServer;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
/**
* Tests for the ContentViewClient.onReceivedError() method. Historically, this test suite focused
* on the new features added in the 2nd iteration of the callback added in M. Now chromium only
* supports one version of the callback, so the distinction between this and
* ClientOnReceivedErrorTest.java is no longer as significant.
*/
@RunWith(Parameterized.class)
@UseParametersRunnerFactory(AwJUnit4ClassRunnerWithParameters.Factory.class)
public class ClientOnReceivedError2Test extends AwParameterizedTest {
@Rule public AwActivityTestRule mActivityTestRule;
private VerifyOnReceivedErrorCallClient mContentsClient;
private AwTestContainerView mTestContainerView;
private AwContents mAwContents;
private TestWebServer mWebServer;
// URLs which do not exist on the public internet (because they use the ".test" TLD).
private static final String BAD_HTML_URL = "http://fake.domain.test/a.html";
private static final String BAD_IMAGE_URL = "http://fake.domain.test/a.png";
public ClientOnReceivedError2Test(AwSettingsMutation param) {
this.mActivityTestRule = new AwActivityTestRule(param.getMutation());
}
@Before
public void setUp() {
mContentsClient = new VerifyOnReceivedErrorCallClient();
mTestContainerView = mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
mAwContents = mTestContainerView.getAwContents();
}
@After
public void tearDown() {
if (mWebServer != null) mWebServer.shutdown();
}
private void startWebServer() throws Exception {
mWebServer = TestWebServer.start();
}
private void useDefaultTestAwContentsClient() {
mContentsClient.enableBypass();
}
private static class VerifyOnReceivedErrorCallClient extends TestAwContentsClient {
private boolean mBypass;
private boolean mIsOnPageFinishedCalled;
private boolean mIsOnReceivedErrorCalled;
void enableBypass() {
mBypass = true;
}
@Override
public void onPageFinished(String url) {
if (!mBypass) {
Assert.assertEquals(
"onPageFinished called twice for " + url, false, mIsOnPageFinishedCalled);
mIsOnPageFinishedCalled = true;
Assert.assertEquals(
"onReceivedError not called before onPageFinished for " + url,
true,
mIsOnReceivedErrorCalled);
}
super.onPageFinished(url);
}
@Override
public void onReceivedError(AwWebResourceRequest request, AwWebResourceError error) {
if (!mBypass) {
Assert.assertEquals(
"onReceivedError called twice for " + request.url,
false,
mIsOnReceivedErrorCalled);
mIsOnReceivedErrorCalled = true;
}
super.onReceivedError(request, error);
}
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testMainFrame() throws Throwable {
mActivityTestRule.loadUrlSync(
mAwContents, mContentsClient.getOnPageFinishedHelper(), BAD_HTML_URL);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(BAD_HTML_URL, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
// request headers may or may not be empty, this is an implementation detail,
// in the network service code path they may e.g. contain user agent, crbug.com/893573.
Assert.assertTrue(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
// The particular error code that is returned depends on the configuration of the device
// (such as existence of a proxy) so we don't test for it.
assertNotEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testUserGesture() throws Throwable {
useDefaultTestAwContentsClient();
final String pageHtml = CommonResources.makeHtmlPageWithSimpleLinkTo(BAD_HTML_URL);
mActivityTestRule.loadDataAsync(mAwContents, pageHtml, "text/html", false);
mActivityTestRule.waitForPixelColorAtCenterOfView(
mAwContents, mTestContainerView, CommonResources.LINK_COLOR);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
int onReceivedErrorCount = onReceivedErrorHelper.getCallCount();
AwTestTouchUtils.simulateTouchCenterOfView(mTestContainerView);
onReceivedErrorHelper.waitForCallback(
onReceivedErrorCount,
/* numberOfCallsToWaitFor= */ 1,
WAIT_TIMEOUT_MS,
TimeUnit.MILLISECONDS);
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(BAD_HTML_URL, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
// request headers may or may not be empty, this is an implementation detail,
// in the network service code path they may e.g. contain user agent, crbug.com/893573.
Assert.assertTrue(request.isOutermostMainFrame);
Assert.assertTrue(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
// The particular error code that is returned depends on the configuration of the device
// (such as existence of a proxy) so we don't test for it.
assertNotEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testIframeSubresource() throws Throwable {
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<iframe src='" + BAD_HTML_URL + "' />");
mActivityTestRule.loadDataSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(BAD_HTML_URL, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
// request headers may or may not be empty, this is an implementation detail,
// in the network service code path they may e.g. contain user agent, crbug.com/893573.
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
// The particular error code that is returned depends on the configuration of the device
// (such as existence of a proxy) so we don't test for it.
assertNotEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testUserGestureForIframeSubresource() throws Throwable {
useDefaultTestAwContentsClient();
startWebServer();
final String iframeHtml = CommonResources.makeHtmlPageWithSimpleLinkTo(BAD_HTML_URL);
final String iframeUrl = mWebServer.setResponse("/iframe.html", iframeHtml, null);
final String pageHtml =
CommonResources.makeHtmlPageFrom(
"", "<iframe style='width:100%;height:100%;' src='" + iframeUrl + "' />");
mActivityTestRule.loadDataAsync(mAwContents, pageHtml, "text/html", false);
mActivityTestRule.waitForPixelColorAtCenterOfView(
mAwContents, mTestContainerView, CommonResources.LINK_COLOR);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
int onReceivedErrorCount = onReceivedErrorHelper.getCallCount();
AwTestTouchUtils.simulateTouchCenterOfView(mTestContainerView);
onReceivedErrorHelper.waitForCallback(
onReceivedErrorCount,
/* numberOfCallsToWaitFor= */ 1,
WAIT_TIMEOUT_MS,
TimeUnit.MILLISECONDS);
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(BAD_HTML_URL, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
// request headers may or may not be empty, this is an implementation detail,
// in the network service code path they may e.g. contain user agent, crbug.com/893573.
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertTrue(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
// The particular error code that is returned depends on the configuration of the device
// (such as existence of a proxy) so we don't test for it.
assertNotEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
@SkipMutations(reason = "This test depends on AwSettings.setImagesEnabled(true)")
public void testImageSubresource() throws Throwable {
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<img src='" + BAD_IMAGE_URL + "' />");
mActivityTestRule.loadDataSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(BAD_IMAGE_URL, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
// request headers may or may not be empty, this is an implementation detail,
// in the network service code path they may e.g. contain user agent, crbug.com/893573.
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
// The particular error code that is returned depends on the configuration of the device
// (such as existence of a proxy) so we don't test for it.
assertNotEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testOnInvalidScheme() throws Throwable {
final String iframeUrl = "foo://some/resource";
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<iframe src='" + iframeUrl + "' />");
mActivityTestRule.loadDataSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(iframeUrl, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
Assert.assertFalse(request.requestHeaders.isEmpty());
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
Assert.assertEquals(WebviewErrorCode.ERROR_UNSUPPORTED_SCHEME, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testOnNonExistentAssetUrl() throws Throwable {
final String baseUrl = "file:///android_asset/";
final String iframeUrl = baseUrl + "does_not_exist.html";
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<iframe src='" + iframeUrl + "' />");
mActivityTestRule.loadDataWithBaseUrlSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false,
baseUrl,
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(iframeUrl, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
Assert.assertFalse(request.requestHeaders.isEmpty());
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
Assert.assertEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testOnNonExistentResourceUrl() throws Throwable {
final String baseUrl = "file:///android_res/raw/";
final String iframeUrl = baseUrl + "does_not_exist.html";
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<iframe src='" + iframeUrl + "' />");
mActivityTestRule.loadDataWithBaseUrlSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false,
baseUrl,
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(iframeUrl, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
Assert.assertFalse(request.requestHeaders.isEmpty());
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
Assert.assertEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testOnCacheMiss() throws Throwable {
final String iframeUrl = "http://example.com/index.html";
final String pageHtml =
CommonResources.makeHtmlPageFrom("", "<iframe src='" + iframeUrl + "' />");
mActivityTestRule
.getAwSettingsOnUiThread(mAwContents)
.setCacheMode(WebSettings.LOAD_CACHE_ONLY);
mActivityTestRule.loadDataSync(
mAwContents,
mContentsClient.getOnPageFinishedHelper(),
pageHtml,
"text/html",
false);
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
AwWebResourceRequest request = onReceivedErrorHelper.getRequest();
Assert.assertNotNull(request);
Assert.assertEquals(iframeUrl, request.url);
Assert.assertEquals("GET", request.method);
Assert.assertNotNull(request.requestHeaders);
Assert.assertFalse(request.requestHeaders.isEmpty());
Assert.assertFalse(request.isOutermostMainFrame);
Assert.assertFalse(request.hasUserGesture);
AwWebResourceError error = onReceivedErrorHelper.getError();
Assert.assertEquals(WebviewErrorCode.ERROR_UNKNOWN, error.errorCode);
Assert.assertNotNull(error.description);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testNotCalledOnStopLoading() throws Throwable {
useDefaultTestAwContentsClient();
final CountDownLatch latch = new CountDownLatch(1);
startWebServer();
final String url =
mWebServer.setResponseWithRunnableAction(
"/about.html",
CommonResources.ABOUT_HTML,
null,
() -> {
try {
latch.await(SCALED_WAIT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
Assert.fail("Caught InterruptedException " + e);
}
});
TestCallbackHelperContainer.OnPageFinishedHelper onPageFinishedHelper =
mContentsClient.getOnPageFinishedHelper();
final int onPageFinishedCallCount = onPageFinishedHelper.getCallCount();
mActivityTestRule.loadUrlAsync(mAwContents, url);
mActivityTestRule.stopLoading(mAwContents);
onPageFinishedHelper.waitForCallback(
onPageFinishedCallCount,
/* numberOfCallsToWaitFor= */ 1,
WAIT_TIMEOUT_MS,
TimeUnit.MILLISECONDS);
latch.countDown(); // Release the server.
// Instead of waiting for OnReceivedError not to be called, we schedule
// a load that will result in a error, and check that we have only got one callback,
// originating from the last attempt.
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
final int onReceivedErrorCount = onReceivedErrorHelper.getCallCount();
mActivityTestRule.loadUrlAsync(mAwContents, BAD_HTML_URL);
onReceivedErrorHelper.waitForCallback(
onReceivedErrorCount,
/* numberOfCallsToWaitFor= */ 1,
WAIT_TIMEOUT_MS,
TimeUnit.MILLISECONDS);
Assert.assertEquals(onReceivedErrorCount + 1, onReceivedErrorHelper.getCallCount());
Assert.assertEquals(BAD_HTML_URL, onReceivedErrorHelper.getRequest().url);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testUnsafeRedirect_FileUrl() throws Throwable {
startWebServer();
final String redirectUrl = mWebServer.setRedirect("/302.html", "file:///foo");
TestAwContentsClient.OnReceivedErrorHelper onReceivedErrorHelper =
mContentsClient.getOnReceivedErrorHelper();
final int onReceivedErrorCount = onReceivedErrorHelper.getCallCount();
mActivityTestRule.loadUrlSync(
mAwContents, mContentsClient.getOnPageFinishedHelper(), redirectUrl);
onReceivedErrorHelper.waitForCallback(
onReceivedErrorCount,
/* numberOfCallsToWaitFor= */ 1,
WAIT_TIMEOUT_MS,
TimeUnit.MILLISECONDS);
Assert.assertEquals(onReceivedErrorCount + 1, onReceivedErrorHelper.getCallCount());
AwWebResourceError error = onReceivedErrorHelper.getError();
Assert.assertEquals("net::ERR_UNSAFE_REDIRECT", error.description);
}
}