blob: a354c27d2e9725bf7151fad3bad1ee0a50506401 [file] [log] [blame]
// Copyright 2012 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.
package org.chromium.android_webview.test;
import static org.chromium.android_webview.test.AwActivityTestRule.WAIT_TIMEOUT_MS;
import static org.chromium.android_webview.test.OnlyRunIn.ProcessMode.MULTI_PROCESS;
import static org.chromium.android_webview.test.OnlyRunIn.ProcessMode.SINGLE_PROCESS;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.support.test.InstrumentationRegistry;
import android.support.test.filters.LargeTest;
import android.support.test.filters.SmallTest;
import android.util.Pair;
import android.view.KeyEvent;
import android.view.View;
import android.webkit.JavascriptInterface;
import org.junit.Assert;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.chromium.android_webview.AwContents;
import org.chromium.android_webview.AwRenderProcess;
import org.chromium.android_webview.AwSettings;
import org.chromium.android_webview.renderer_priority.RendererPriority;
import org.chromium.android_webview.test.TestAwContentsClient.OnDownloadStartHelper;
import org.chromium.android_webview.test.util.CommonResources;
import org.chromium.base.BuildInfo;
import org.chromium.base.metrics.RecordHistogram;
import org.chromium.base.test.util.CallbackHelper;
import org.chromium.base.test.util.CommandLineFlags;
import org.chromium.base.test.util.Feature;
import org.chromium.content_public.browser.test.util.TestThreadUtils;
import org.chromium.content_public.common.ContentSwitches;
import org.chromium.content_public.common.ContentUrlConstants;
import org.chromium.net.test.EmbeddedTestServer;
import org.chromium.net.test.util.TestWebServer;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
/**
* AwContents tests.
*/
@RunWith(AwJUnit4ClassRunner.class)
public class AwContentsTest {
@Rule
public AwActivityTestRule mActivityTestRule = new AwActivityTestRule();
private TestAwContentsClient mContentsClient = new TestAwContentsClient();
private volatile Integer mHistogramTotalCount = 0;
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testCreateDestroy() throws Throwable {
// NOTE this test runs on UI thread, so we cannot call any async methods.
mActivityTestRule.runOnUiThread(() -> mActivityTestRule.createAwTestContainerView(
mContentsClient)
.getAwContents()
.destroy());
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testCreateLoadPageDestroy() throws Throwable {
AwTestContainerView awTestContainerView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
mActivityTestRule.loadDataSync(awTestContainerView.getAwContents(),
mContentsClient.getOnPageFinishedHelper(), CommonResources.ABOUT_HTML, "text/html",
false);
mActivityTestRule.destroyAwContentsOnMainSync(awTestContainerView.getAwContents());
// It should be safe to call destroy multiple times.
mActivityTestRule.destroyAwContentsOnMainSync(awTestContainerView.getAwContents());
}
@Test
@LargeTest
@Feature({"AndroidWebView"})
public void testCreateLoadDestroyManyTimes() throws Throwable {
for (int i = 0; i < 10; ++i) {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
mActivityTestRule.destroyAwContentsOnMainSync(awContents);
}
}
@Test
@LargeTest
@Feature({"AndroidWebView"})
public void testCreateLoadDestroyManyAtOnce() throws Throwable {
AwTestContainerView views[] = new AwTestContainerView[10];
for (int i = 0; i < views.length; ++i) {
views[i] = mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
mActivityTestRule.loadUrlSync(views[i].getAwContents(),
mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
}
for (int i = 0; i < views.length; ++i) {
mActivityTestRule.destroyAwContentsOnMainSync(views[i].getAwContents());
views[i] = null;
}
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testWebViewApisFailGracefullyAfterDestruction() throws Throwable {
mActivityTestRule.runOnUiThread(() -> {
AwContents awContents = mActivityTestRule.createAwTestContainerView(mContentsClient)
.getAwContents();
awContents.destroy();
// The documentation for WebView#destroy() reads "This method should be called
// after this WebView has been removed from the view system. No other methods
// may be called on this WebView after destroy".
// However, some apps do not respect that restriction so we need to ensure that
// we fail gracefully and do not crash when APIs are invoked after destruction.
// Due to the large number of APIs we only test a representative selection here.
awContents.clearHistory();
Assert.assertNull(awContents.getOriginalUrl());
Assert.assertNull(awContents.getNavigationHistory());
awContents.loadUrl("http://www.google.com");
awContents.findAllAsync("search");
Assert.assertNull(awContents.getUrl());
Assert.assertFalse(awContents.canGoBack());
awContents.disableJavascriptInterfacesInspection();
awContents.invokeZoomPicker();
awContents.onResume();
awContents.stopLoading();
awContents.onWindowVisibilityChanged(View.VISIBLE);
awContents.requestFocus();
awContents.isMultiTouchZoomSupported();
awContents.setOverScrollMode(View.OVER_SCROLL_NEVER);
awContents.pauseTimers();
awContents.onContainerViewScrollChanged(200, 200, 100, 100);
awContents.computeScroll();
awContents.onMeasure(100, 100);
awContents.onDraw(new Canvas());
awContents.getMostRecentProgress();
Assert.assertEquals(0, awContents.computeHorizontalScrollOffset());
Assert.assertEquals(0, awContents.getContentWidthCss());
awContents.onKeyUp(KeyEvent.KEYCODE_BACK,
new KeyEvent(KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MENU));
});
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testUseAwSettingsAfterDestroy() throws Throwable {
AwTestContainerView awTestContainerView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwSettings awSettings =
mActivityTestRule.getAwSettingsOnUiThread(awTestContainerView.getAwContents());
mActivityTestRule.loadDataSync(awTestContainerView.getAwContents(),
mContentsClient.getOnPageFinishedHelper(), CommonResources.ABOUT_HTML, "text/html",
false);
mActivityTestRule.destroyAwContentsOnMainSync(awTestContainerView.getAwContents());
// AwSettings should still be usable even after native side is destroyed.
String newFontFamily = "serif";
awSettings.setStandardFontFamily(newFontFamily);
Assert.assertEquals(newFontFamily, awSettings.getStandardFontFamily());
boolean newBlockNetworkLoads = !awSettings.getBlockNetworkLoads();
awSettings.setBlockNetworkLoads(newBlockNetworkLoads);
Assert.assertEquals(newBlockNetworkLoads, awSettings.getBlockNetworkLoads());
}
private int callDocumentHasImagesSync(final AwContents awContents)
throws Throwable, InterruptedException {
// Set up a container to hold the result object and a semaphore to
// make the test wait for the result.
final AtomicInteger val = new AtomicInteger();
final Semaphore s = new Semaphore(0);
final Message msg = Message.obtain(new Handler(Looper.getMainLooper()) {
@Override
public void handleMessage(Message msg) {
val.set(msg.arg1);
s.release();
}
});
InstrumentationRegistry.getInstrumentation().runOnMainSync(
() -> awContents.documentHasImages(msg));
Assert.assertTrue(s.tryAcquire(AwActivityTestRule.WAIT_TIMEOUT_MS, TimeUnit.MILLISECONDS));
int result = val.get();
return result;
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testDocumentHasImages() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
final CallbackHelper loadHelper = mContentsClient.getOnPageFinishedHelper();
final String mime = "text/html";
final String emptyDoc = "<head/><body/>";
final String imageDoc = "<head/><body><img/><img/></body>";
// Make sure a document that does not have images returns 0
mActivityTestRule.loadDataSync(awContents, loadHelper, emptyDoc, mime, false);
int result = callDocumentHasImagesSync(awContents);
Assert.assertEquals(0, result);
// Make sure a document that does have images returns 1
mActivityTestRule.loadDataSync(awContents, loadHelper, imageDoc, mime, false);
result = callDocumentHasImagesSync(awContents);
Assert.assertEquals(1, result);
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testClearCacheMemoryAndDisk() throws Throwable {
final AwTestContainerView testContainer =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testContainer.getAwContents();
TestWebServer webServer = TestWebServer.start();
try {
final String pagePath = "/clear_cache_test.html";
List<Pair<String, String>> headers = new ArrayList<Pair<String, String>>();
// Set Cache-Control headers to cache this request. One century should be long enough.
headers.add(Pair.create("Cache-Control", "max-age=3153600000"));
headers.add(Pair.create("Last-Modified", "Wed, 3 Oct 2012 00:00:00 GMT"));
final String pageUrl = webServer.setResponse(
pagePath, "<html><body>foo</body></html>", headers);
// First load to populate cache.
mActivityTestRule.clearCacheOnUiThread(awContents, true);
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), pageUrl);
Assert.assertEquals(1, webServer.getRequestCount(pagePath));
// Load about:blank so next load is not treated as reload by webkit and force
// revalidate with the server.
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
// No clearCache call, so should be loaded from cache.
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), pageUrl);
Assert.assertEquals(1, webServer.getRequestCount(pagePath));
// Same as above.
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
// Clear cache, so should hit server again.
mActivityTestRule.clearCacheOnUiThread(awContents, true);
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), pageUrl);
Assert.assertEquals(2, webServer.getRequestCount(pagePath));
} finally {
webServer.shutdown();
}
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testClearCacheInQuickSuccession() throws Throwable {
final AwTestContainerView testContainer =
mActivityTestRule.createAwTestContainerViewOnMainSync(new TestAwContentsClient());
final AwContents awContents = testContainer.getAwContents();
InstrumentationRegistry.getInstrumentation().runOnMainSync(() -> {
for (int i = 0; i < 10; ++i) {
awContents.clearCache(true);
}
});
}
@Test
@SmallTest
@Feature({"AndroidWebView"})
public void testGetFavicon() throws Throwable {
AwContents.setShouldDownloadFavicons();
final AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
TestWebServer webServer = TestWebServer.start();
try {
final String faviconUrl = webServer.setResponseBase64(
"/" + CommonResources.FAVICON_FILENAME, CommonResources.FAVICON_DATA_BASE64,
CommonResources.getImagePngHeaders(false));
final String pageUrl = webServer.setResponse("/favicon.html",
CommonResources.FAVICON_STATIC_HTML, null);
// The getFavicon will return the right icon a certain time after
// the page load completes which makes it slightly hard to test.
final Bitmap defaultFavicon = awContents.getFavicon();
mActivityTestRule.getAwSettingsOnUiThread(awContents).setImagesEnabled(true);
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), pageUrl);
mActivityTestRule.pollUiThread(() -> awContents.getFavicon() != null
&& !awContents.getFavicon().sameAs(defaultFavicon));
final Object originalFaviconSource = (new URL(faviconUrl)).getContent();
final Bitmap originalFavicon =
BitmapFactory.decodeStream((InputStream) originalFaviconSource);
Assert.assertNotNull(originalFavicon);
Assert.assertTrue(awContents.getFavicon().sameAs(originalFavicon));
} finally {
webServer.shutdown();
}
}
@Test
@Feature({"AndroidWebView", "Downloads"})
@SmallTest
public void testDownload() throws Throwable {
downloadAndCheck(null);
}
@Test
@Feature({"AndroidWebView", "Downloads"})
@SmallTest
public void testDownloadWithCustomUserAgent() throws Throwable {
downloadAndCheck("Custom User Agent");
}
private void downloadAndCheck(String customUserAgent) throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
if (customUserAgent != null) {
AwSettings awSettings = mActivityTestRule.getAwSettingsOnUiThread(awContents);
awSettings.setUserAgentString(customUserAgent);
}
final String data = "download data";
final String contentDisposition = "attachment;filename=\"download.txt\"";
final String mimeType = "text/plain";
List<Pair<String, String>> downloadHeaders = new ArrayList<Pair<String, String>>();
downloadHeaders.add(Pair.create("Content-Disposition", contentDisposition));
downloadHeaders.add(Pair.create("Content-Type", mimeType));
downloadHeaders.add(Pair.create("Content-Length", Integer.toString(data.length())));
TestWebServer webServer = TestWebServer.start();
try {
final String pageUrl = webServer.setResponse(
"/download.txt", data, downloadHeaders);
final OnDownloadStartHelper downloadStartHelper =
mContentsClient.getOnDownloadStartHelper();
final int callCount = downloadStartHelper.getCallCount();
mActivityTestRule.loadUrlAsync(awContents, pageUrl);
downloadStartHelper.waitForCallback(callCount);
Assert.assertEquals(pageUrl, downloadStartHelper.getUrl());
Assert.assertEquals(contentDisposition, downloadStartHelper.getContentDisposition());
Assert.assertEquals(mimeType, downloadStartHelper.getMimeType());
Assert.assertEquals(data.length(), downloadStartHelper.getContentLength());
Assert.assertFalse(downloadStartHelper.getUserAgent().isEmpty());
if (customUserAgent != null) {
Assert.assertEquals(customUserAgent, downloadStartHelper.getUserAgent());
} else {
Assert.assertEquals(
downloadStartHelper.getUserAgent(), AwSettings.getDefaultUserAgent());
}
} finally {
webServer.shutdown();
}
}
@Test
@Feature({"AndroidWebView", "setNetworkAvailable"})
@SmallTest
public void testSetNetworkAvailable() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
String script = "navigator.onLine";
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
// Default to "online".
Assert.assertEquals("true",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
// Forcing "offline".
AwActivityTestRule.setNetworkAvailableOnUiThread(awContents, false);
Assert.assertEquals("false",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
// Forcing "online".
AwActivityTestRule.setNetworkAvailableOnUiThread(awContents, true);
Assert.assertEquals("true",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
}
static class JavaScriptObject {
private CallbackHelper mCallbackHelper;
public JavaScriptObject(CallbackHelper callbackHelper) {
mCallbackHelper = callbackHelper;
}
@JavascriptInterface
public void run() {
mCallbackHelper.notifyCalled();
}
}
@Test
@Feature({"AndroidWebView", "Android-JavaBridge"})
@SmallTest
public void testJavaBridge() throws Throwable {
final AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final CallbackHelper callback = new CallbackHelper();
AwContents awContents = testView.getAwContents();
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
AwActivityTestRule.addJavascriptInterfaceOnUiThread(
awContents, new JavaScriptObject(callback), "bridge");
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, "window.bridge.run();");
callback.waitForCallback(0, 1, WAIT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testEscapingOfErrorPage() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
String script = "window.failed == true";
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
CallbackHelper onPageFinishedHelper = mContentsClient.getOnPageFinishedHelper();
int currentCallCount = onPageFinishedHelper.getCallCount();
mActivityTestRule.loadUrlAsync(awContents,
"file:///file-that-does-not-exist#<script>window.failed = true;</script>");
onPageFinishedHelper.waitForCallback(
currentCallCount, 1, WAIT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
Assert.assertEquals("false",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testCanInjectHeaders() throws Throwable {
final AwTestContainerView testContainer =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testContainer.getAwContents();
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
EmbeddedTestServer testServer = EmbeddedTestServer.createAndStartServer(
InstrumentationRegistry.getInstrumentation().getContext());
try {
String url = testServer.getURL("/echoheader?X-foo");
final Map<String, String> extraHeaders = new HashMap<String, String>();
extraHeaders.put("X-foo", "bar");
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), url, extraHeaders);
String xfoo = mActivityTestRule.getJavaScriptResultBodyTextContent(
awContents, mContentsClient);
Assert.assertEquals("bar", xfoo);
url = testServer.getURL("/echoheader?Referer");
extraHeaders.clear();
extraHeaders.put("Referer", "http://www.example.com/");
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), url, extraHeaders);
String referer = mActivityTestRule.getJavaScriptResultBodyTextContent(
awContents, mContentsClient);
Assert.assertEquals("http://www.example.com/", referer);
} finally {
testServer.stopAndDestroyServer();
}
}
// This is a meta test that we don't accidentally turn off hardware
// acceleration in instrumentation tests without notice. Do not add the
// @DisableHardwareAccelerationForTest annotation for this test.
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testHardwareModeWorks() throws Throwable {
AwTestContainerView testContainer =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
Assert.assertTrue(testContainer.isHardwareAccelerated());
Assert.assertTrue(testContainer.isBackedByHardwareView());
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testBasicCookieFunctionality() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
TestWebServer webServer = TestWebServer.start();
try {
List<Pair<String, String>> responseHeaders = CommonResources.getTextHtmlHeaders(true);
final String cookie = "key=value";
responseHeaders.add(Pair.create("Set-Cookie", cookie));
final String url = webServer.setResponse("/" + CommonResources.ABOUT_FILENAME,
CommonResources.ABOUT_HTML, responseHeaders);
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
mActivityTestRule.loadUrlSync(
awContents, mContentsClient.getOnPageFinishedHelper(), url);
final String script = "document.cookie";
Assert.assertEquals("\"key=value\"",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
} finally {
webServer.shutdown();
}
}
/**
* Verifies that Web Notifications and the Push API are not exposed in WebView.
*/
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testPushAndNotificationsDisabled() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
AwContents awContents = testView.getAwContents();
String script = "window.Notification || window.PushManager";
AwActivityTestRule.enableJavaScriptOnUiThread(awContents);
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
Assert.assertEquals("null",
mActivityTestRule.executeJavaScriptAndWaitForResult(
awContents, mContentsClient, script));
}
private @RendererPriority int getRendererPriorityOnUiThread(final AwContents awContents)
throws Exception {
return TestThreadUtils.runOnUiThreadBlocking(
() -> awContents.getEffectivePriorityForTesting());
}
private void setRendererPriorityOnUiThread(final AwContents awContents,
final @RendererPriority int priority, final boolean waivedWhenNotVisible)
throws Throwable {
mActivityTestRule.runOnUiThread(
() -> awContents.setRendererPriorityPolicy(priority, waivedWhenNotVisible));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(MULTI_PROCESS)
@CommandLineFlags.Add(ContentSwitches.RENDER_PROCESS_LIMIT + "=1")
public void testForegroundPriorityOneProcess() throws Throwable {
final AwTestContainerView view1 =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents contents1 = view1.getAwContents();
final AwTestContainerView view2 =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents contents2 = view2.getAwContents();
mActivityTestRule.loadUrlSync(contents1, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
mActivityTestRule.loadUrlSync(contents2, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
// Process should start out high.
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents2));
// Set one to low. Process should take max priority of contents, so still high.
setRendererPriorityOnUiThread(contents1, RendererPriority.LOW, false);
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents2));
// Set both to low and check.
setRendererPriorityOnUiThread(contents2, RendererPriority.LOW, false);
Assert.assertEquals(RendererPriority.LOW, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.LOW, getRendererPriorityOnUiThread(contents2));
// Set both to waive and check.
setRendererPriorityOnUiThread(contents1, RendererPriority.WAIVED, false);
setRendererPriorityOnUiThread(contents2, RendererPriority.WAIVED, false);
Assert.assertEquals(RendererPriority.WAIVED, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.WAIVED, getRendererPriorityOnUiThread(contents2));
// Set one to high and check.
setRendererPriorityOnUiThread(contents1, RendererPriority.HIGH, false);
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents2));
// Destroy contents with high priority, and process should fall back to low.
// Destroy posts on UI, but getRendererPriorityOnUiThread posts after, so there should
// be no flakiness and no need for polling.
mActivityTestRule.destroyAwContentsOnMainSync(contents1);
Assert.assertEquals(RendererPriority.WAIVED, getRendererPriorityOnUiThread(contents2));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(MULTI_PROCESS)
@CommandLineFlags.Add(ContentSwitches.RENDER_PROCESS_LIMIT + "=2")
public void testForegroundPriorityTwoProcesses() throws Throwable {
final AwTestContainerView view1 =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents contents1 = view1.getAwContents();
final AwTestContainerView view2 =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents contents2 = view2.getAwContents();
mActivityTestRule.loadUrlSync(contents1, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
mActivityTestRule.loadUrlSync(contents2, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
// Process should start out high.
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents2));
// Set one to low. Other should not be affected.
setRendererPriorityOnUiThread(contents1, RendererPriority.LOW, false);
Assert.assertEquals(RendererPriority.LOW, getRendererPriorityOnUiThread(contents1));
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(contents2));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(MULTI_PROCESS)
public void testBackgroundPriority() throws Throwable {
final AwContents awContents =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient)
.getAwContents();
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(awContents));
mActivityTestRule.runOnUiThread(() -> awContents.onPause());
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(awContents));
setRendererPriorityOnUiThread(
awContents, RendererPriority.HIGH, true /* waivedWhenNotVisible */);
Assert.assertEquals(RendererPriority.WAIVED, getRendererPriorityOnUiThread(awContents));
mActivityTestRule.runOnUiThread(() -> awContents.onResume());
Assert.assertEquals(RendererPriority.HIGH, getRendererPriorityOnUiThread(awContents));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(MULTI_PROCESS)
public void testPauseDestroyResume() throws Throwable {
mActivityTestRule.runOnUiThread(() -> {
AwContents awContents;
awContents = mActivityTestRule.createAwTestContainerView(mContentsClient)
.getAwContents();
awContents.pauseTimers();
awContents.pauseTimers();
awContents.destroy();
awContents = mActivityTestRule.createAwTestContainerView(mContentsClient)
.getAwContents();
awContents.resumeTimers();
});
}
private AwRenderProcess getRenderProcessOnUiThread(final AwContents awContents)
throws Exception {
return TestThreadUtils.runOnUiThreadBlocking(() -> awContents.getRenderProcess());
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(MULTI_PROCESS)
public void testRenderProcessInMultiProcessMode() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
final AwRenderProcess preLoadRenderProcess = getRenderProcessOnUiThread(awContents);
Assert.assertNotNull(preLoadRenderProcess);
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
final AwRenderProcess postLoadRenderProcess = getRenderProcessOnUiThread(awContents);
Assert.assertEquals(preLoadRenderProcess, postLoadRenderProcess);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
@OnlyRunIn(SINGLE_PROCESS)
public void testNoRenderProcessInSingleProcessMode() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
mActivityTestRule.loadUrlSync(awContents, mContentsClient.getOnPageFinishedHelper(),
ContentUrlConstants.ABOUT_BLANK_DISPLAY_URL);
final AwRenderProcess renderProcess = getRenderProcessOnUiThread(awContents);
Assert.assertEquals(renderProcess, null);
}
/** Regression test for https://crbug.com/732976. Load a data URL, then immediately
* after that load a javascript URL. The data URL navigation shouldn't be blocked.
*/
@Test
@LargeTest
@Feature({"AndroidWebView"})
public void testJavaScriptUrlAfterLoadData() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
mActivityTestRule.runOnUiThread(() -> {
// Run javascript navigation immediately, without waiting for the completion of data
// URL.
awContents.loadData("<html>test</html>", "text/html", "utf-8");
awContents.loadUrl("javascript: void(0)");
});
mContentsClient.getOnPageFinishedHelper().waitForCallback(
0, 1, WAIT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
Assert.assertEquals("data:text/html,<html>test</html>", awContents.getLastCommittedUrl());
TestAwContentsClient.AddMessageToConsoleHelper consoleHelper =
mContentsClient.getAddMessageToConsoleHelper();
Assert.assertEquals(0, consoleHelper.getMessages().size());
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testHardwareRenderingSmokeTest() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
String html = "<html>"
+ " <body style=\""
+ " padding: 0;"
+ " margin: 0;"
+ " display: grid;"
+ " display: grid;"
+ " grid-template-columns: 50% 50%;"
+ " grid-template-rows: 50% 50%;\">"
+ " <div style=\"background-color: rgb(255, 0, 0);\"></div>"
+ " <div style=\"background-color: rgb(0, 255, 0);\"></div>"
+ " <div style=\"background-color: rgb(0, 0, 255);\"></div>"
+ " <div style=\"background-color: rgb(128, 128, 128);\"></div>"
+ " </body>"
+ "</html>";
mActivityTestRule.loadDataSync(testView.getAwContents(),
mContentsClient.getOnPageFinishedHelper(), html, "text/html", false);
mActivityTestRule.waitForVisualStateCallback(testView.getAwContents());
// Poll for 10s in case raster is slow.
final Object lock = new Object();
final Object[] resultHolder = new Object[1];
for (int i = 0; i < 100; ++i) {
mActivityTestRule.runOnUiThread(() -> {
testView.readbackQuadrantColors((int[] result) -> {
synchronized (lock) {
resultHolder[0] = result;
lock.notifyAll();
}
});
});
int[] quadrantColors;
synchronized (lock) {
while (resultHolder[0] == null) {
lock.wait();
}
quadrantColors = (int[]) resultHolder[0];
}
if (Color.rgb(255, 0, 0) == quadrantColors[0]
&& Color.rgb(0, 255, 0) == quadrantColors[1]
&& Color.rgb(0, 0, 255) == quadrantColors[2]
&& Color.rgb(128, 128, 128) == quadrantColors[3]) {
return;
}
Thread.sleep(100);
}
// If this test is failing for your CL, then chances are your change is breaking Android
// WebView hardware rendering. Please build the "real" webview and check if this is the
// case and if so, fix your CL.
int[] quadrantColors = (int[]) resultHolder[0];
Assert.assertEquals(Color.rgb(255, 0, 0), quadrantColors[0]);
Assert.assertEquals(Color.rgb(0, 255, 0), quadrantColors[1]);
Assert.assertEquals(Color.rgb(0, 0, 255), quadrantColors[2]);
Assert.assertEquals(Color.rgb(128, 128, 128), quadrantColors[3]);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testFixupOctothorpesInLoadDataContent() {
// If there are no octothorpes the function should have no effect.
final String noOctothorpeString = "<div id='foo1'>This content has no octothorpe</div>";
Assert.assertEquals(noOctothorpeString,
AwContents.fixupOctothorpesInLoadDataContent(noOctothorpeString));
// One '#' followed by a valid DOM id requires us to duplicate it into a real fragment.
Assert.assertEquals("abc%23A#A", AwContents.fixupOctothorpesInLoadDataContent("abc#A"));
Assert.assertEquals("abc%23a#a", AwContents.fixupOctothorpesInLoadDataContent("abc#a"));
Assert.assertEquals("abc%23Aa#Aa", AwContents.fixupOctothorpesInLoadDataContent("abc#Aa"));
Assert.assertEquals("abc%23aA#aA", AwContents.fixupOctothorpesInLoadDataContent("abc#aA"));
Assert.assertEquals(
"abc%23a1-_:.#a1-_:.", AwContents.fixupOctothorpesInLoadDataContent("abc#a1-_:."));
// One '#' followed by an invalid DOM id just means we encode the '#'.
Assert.assertEquals("abc%231", AwContents.fixupOctothorpesInLoadDataContent("abc#1"));
Assert.assertEquals("abc%231a", AwContents.fixupOctothorpesInLoadDataContent("abc#1a"));
Assert.assertEquals(
"abc%23not valid", AwContents.fixupOctothorpesInLoadDataContent("abc#not valid"));
Assert.assertEquals("abc%23a@", AwContents.fixupOctothorpesInLoadDataContent("abc#a@"));
// Multiple '#', whether or not they have a valid DOM id afterwards, just means we encode
// the '#'.
Assert.assertEquals("abc%23%23a", AwContents.fixupOctothorpesInLoadDataContent("abc##a"));
Assert.assertEquals("abc%23a%23b", AwContents.fixupOctothorpesInLoadDataContent("abc#a#b"));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadDataOctothorpeHandling() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
// Before Android Q, the loadData API is expected to handle the encoding for users.
boolean encodeOctothorpes = !BuildInfo.targetsAtLeastQ();
// A URL with no '#' character.
mActivityTestRule.loadDataSync(awContents, mContentsClient.getOnPageFinishedHelper(),
"<html>test</html>", "text/html", false);
Assert.assertEquals("data:text/html,<html>test</html>", awContents.getLastCommittedUrl());
// A URL with one '#' character.
mActivityTestRule.loadDataSync(awContents, mContentsClient.getOnPageFinishedHelper(),
"<html>test#foo</html>", "text/html", false);
String expectedUrl = encodeOctothorpes ? "data:text/html,<html>test%23foo</html>"
: "data:text/html,<html>test#foo</html>";
Assert.assertEquals(expectedUrl, awContents.getLastCommittedUrl());
// A URL with many '#' characters.
mActivityTestRule.loadDataSync(awContents, mContentsClient.getOnPageFinishedHelper(),
"<html>test#foo#bar#</html>", "text/html", false);
expectedUrl = encodeOctothorpes ? "data:text/html,<html>test%23foo%23bar%23</html>"
: "data:text/html,<html>test#foo#bar#</html>";
Assert.assertEquals(expectedUrl, awContents.getLastCommittedUrl());
// An already encoded '#' character.
mActivityTestRule.loadDataSync(awContents, mContentsClient.getOnPageFinishedHelper(),
"<html>test%23foo</html>", "text/html", false);
Assert.assertEquals(
"data:text/html,<html>test%23foo</html>", awContents.getLastCommittedUrl());
// A URL with a valid fragment. Before Q, this must be manipulated so that it renders the
// same and still scrolls to the fragment location.
if (encodeOctothorpes) {
String contents = "<div style='height: 5000px'></div><a id='target'>Target</a>#target";
mActivityTestRule.loadDataSync(awContents, mContentsClient.getOnPageFinishedHelper(),
contents, "text/html", false);
Assert.assertEquals(
"data:text/html,<div style='height: 5000px'></div><a id='target'>Target</a>"
+ "%23target#target",
awContents.getLastCommittedUrl());
// TODO(smcgruer): I can physically see that this has scrolled on the test page, and
// have traced scrolling through PaintLayerScrollableArea, but I don't know how to check
// it.
}
}
private int getHistogramSampleCount(String name) throws Throwable {
TestThreadUtils.runOnUiThreadBlocking(() -> {
mHistogramTotalCount = RecordHistogram.getHistogramTotalCountForTesting(name);
});
return mHistogramTotalCount;
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadDataRecordsOctothorpeHistogram() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
// AwContents.DATA_URI_HISTOGRAM_NAME is a boolean histogram, but as it only records
// positive samples we can just use the total count directly.
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with no '#' character.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadData("<html>test</html>", "text/html", null); });
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with a '#' character.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadData("<html>test#foo</html>", "text/html", null); });
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// An encoded '#' should not cause the histogram to increment.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadData("<html>test%23foo</html>", "text/html", null); });
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Finally, check null values are handled correctly.
mActivityTestRule.runOnUiThread(() -> { awContents.loadData(null, "text/html", "utf-8"); });
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadDataWithBaseURLRecordsOctothorpeHistogram() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
// AwContents.DATA_URI_HISTOGRAM_NAME is a boolean histogram, but as it only records
// positive samples we can just use the total count directly.
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with no '#' character.
mActivityTestRule.runOnUiThread(() -> {
awContents.loadDataWithBaseURL(
"http://www.example.com", "<html>test</html>", "text/html", null, null);
});
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// '#' is legal if the baseUrl is not data scheme, because loadDataWithBaseURL accepts
// unencoded content.
mActivityTestRule.runOnUiThread(() -> {
awContents.loadDataWithBaseURL(
"http://www.example.com", "<html>test#foo</html>", "text/html", null, null);
});
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with a '#' character, with data-scheme baseUrl.
mActivityTestRule.runOnUiThread(() -> {
awContents.loadDataWithBaseURL(
"data:text/html", "<html>test#foo</html>", "text/html", null, null);
});
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// An encoded '#' should not cause the histogram to increment.
mActivityTestRule.runOnUiThread(() -> {
awContents.loadDataWithBaseURL(
"http://www.example.com", "<html>test%23foo</html>", "text/html", null, null);
});
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Finally, check null values are handled correctly.
mActivityTestRule.runOnUiThread(() -> {
awContents.loadDataWithBaseURL("http://www.example.com", null, "text/html", null, null);
});
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsOctothorpeHistogram() throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
// AwContents.DATA_URI_HISTOGRAM_NAME is a boolean histogram, but as it only records
// positive samples we can just use the total count directly.
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with no '#' character.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadUrl("data:text/html,<html>test</html>"); });
Assert.assertEquals(0, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// Check a URL with a '#' character.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadUrl("data:text/html,<html>test#foo</html>"); });
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// An encoded '#' should not cause the histogram to increment.
mActivityTestRule.runOnUiThread(
() -> { awContents.loadUrl("data:text/html,<html>test%23foo</html>"); });
Assert.assertEquals(1, getHistogramSampleCount(AwContents.DATA_URI_HISTOGRAM_NAME));
// |loadUrl| doesn't allow a null url, so it is not necessary to check that for this API.
// See http://crbug.com/864708.
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_http() throws Throwable {
// No need to spin up a web server, since we don't care if the load ever succeeds.
final String httpUrlWithNoRealPage = "http://some.origin/some/path.html";
loadUrlAndCheckScheme(httpUrlWithNoRealPage, AwContents.UrlScheme.HTTP_SCHEME);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_javascript() throws Throwable {
loadUrlAndCheckScheme(
"javascript:console.log('message')", AwContents.UrlScheme.JAVASCRIPT_SCHEME);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_fileAndroidAsset() throws Throwable {
loadUrlAndCheckScheme("file:///android_asset/some/asset/page.html",
AwContents.UrlScheme.FILE_ANDROID_ASSET_SCHEME);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_fileRegular() throws Throwable {
loadUrlAndCheckScheme("file:///some/path/on/disk.html", AwContents.UrlScheme.FILE_SCHEME);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_data() throws Throwable {
loadUrlAndCheckScheme(
"data:text/html,<html><body>foo</body></html>", AwContents.UrlScheme.DATA_SCHEME);
}
@Test
@Feature({"AndroidWebView"})
@SmallTest
public void testLoadUrlRecordsScheme_blank() throws Throwable {
loadUrlAndCheckScheme("about:blank", AwContents.UrlScheme.EMPTY);
}
private void loadUrlAndCheckScheme(String url, @AwContents.UrlScheme int expectedSchemeEnum)
throws Throwable {
AwTestContainerView testView =
mActivityTestRule.createAwTestContainerViewOnMainSync(mContentsClient);
final AwContents awContents = testView.getAwContents();
Assert.assertEquals(0,
RecordHistogram.getHistogramTotalCountForTesting(
AwContents.LOAD_URL_SCHEME_HISTOGRAM_NAME));
// Note: we use async because not all loads emit onPageFinished. This relies on the UMA
// metric being logged in the synchronous part of loadUrl().
mActivityTestRule.loadUrlAsync(awContents, url);
Assert.assertEquals(1,
RecordHistogram.getHistogramTotalCountForTesting(
AwContents.LOAD_URL_SCHEME_HISTOGRAM_NAME));
Assert.assertEquals(1,
RecordHistogram.getHistogramValueCountForTesting(
AwContents.LOAD_URL_SCHEME_HISTOGRAM_NAME, expectedSchemeEnum));
}
}