| /* |
| * Copyright (C) 2022-2024 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import "config.h" |
| |
| #if ENABLE(WK_WEB_EXTENSIONS) |
| |
| #import "HTTPServer.h" |
| #import "WebExtensionUtilities.h" |
| |
| namespace TestWebKitAPI { |
| |
| static auto *extensionManifest = @{ |
| @"manifest_version": @3, |
| |
| @"name": @"Extension Test", |
| @"description": @"Extension Test", |
| @"version": @"1", |
| |
| @"background": @{ |
| @"scripts": @[ @"background.js" ], |
| @"type": @"module", |
| @"persistent": @NO, |
| }, |
| |
| @"action": @{ |
| @"default_title": @"Test Action", |
| @"default_popup": @"popup.html", |
| @"default_icon": @{ |
| @"16": @"toolbar-16.png", |
| @"32": @"toolbar-32.png", |
| }, |
| }, |
| |
| @"content_scripts": @[ @{ |
| @"js": @[ @"content.js" ], |
| @"matches": @[ @"*://localhost/*" ], |
| } ], |
| }; |
| |
| static auto *extensionServiceWorkerManifest = @{ |
| @"manifest_version": @3, |
| |
| @"name": @"Extension Test", |
| @"description": @"Extension Test", |
| @"version": @"1", |
| |
| @"background": @{ |
| @"service_worker": @"background.js", |
| }, |
| }; |
| |
| TEST(WKWebExtensionAPIExtension, GetURL) |
| { |
| auto *baseURLString = @"test-extension://76C788B8-3374-400D-8259-40E5B9DF79D3"; |
| |
| auto *backgroundScript = Util::constructScript(@[ |
| // Variable Setup |
| [NSString stringWithFormat:@"const baseURL = '%@'", baseURLString], |
| |
| // Error Cases |
| @"browser.test.assertThrows(() => browser.extension.getURL(), /required argument is missing/i)", |
| @"browser.test.assertThrows(() => browser.extension.getURL(null), /'resourcePath' value is invalid, because a string is expected/i)", |
| @"browser.test.assertThrows(() => browser.extension.getURL(undefined), /'resourcePath' value is invalid, because a string is expected/i)", |
| @"browser.test.assertThrows(() => browser.extension.getURL(42), /'resourcePath' value is invalid, because a string is expected/i)", |
| @"browser.test.assertThrows(() => browser.extension.getURL(/test/), /'resourcePath' value is invalid, because a string is expected/i)", |
| |
| // Normal Cases |
| @"browser.test.assertEq(browser.extension.getURL(''), `${baseURL}/`)", |
| @"browser.test.assertEq(browser.extension.getURL('test.js'), `${baseURL}/test.js`)", |
| @"browser.test.assertEq(browser.extension.getURL('/test.js'), `${baseURL}/test.js`)", |
| @"browser.test.assertEq(browser.extension.getURL('../../test.js'), `${baseURL}/test.js`)", |
| @"browser.test.assertEq(browser.extension.getURL('./test.js'), `${baseURL}/test.js`)", |
| @"browser.test.assertEq(browser.extension.getURL('././/example'), `${baseURL}//example`)", |
| @"browser.test.assertEq(browser.extension.getURL('../../example/..//test/'), `${baseURL}//test/`)", |
| @"browser.test.assertEq(browser.extension.getURL('.'), `${baseURL}/`)", |
| @"browser.test.assertEq(browser.extension.getURL('..//../'), `${baseURL}/`)", |
| @"browser.test.assertEq(browser.extension.getURL('.././..'), `${baseURL}/`)", |
| @"browser.test.assertEq(browser.extension.getURL('/.././.'), `${baseURL}/`)", |
| |
| // Unexpected Cases |
| // FIXME: <https://webkit.org/b/248154> browser.extension.getURL() has some edge cases that should be failures or return different results |
| @"browser.test.assertEq(browser.extension.getURL('//'), 'test-extension://')", |
| @"browser.test.assertEq(browser.extension.getURL('//example'), `test-extension://example`)", |
| @"browser.test.assertEq(browser.extension.getURL('///'), 'test-extension:///')", |
| |
| // Finish |
| @"browser.test.notifyPass()" |
| ]); |
| |
| auto *manifest = @{ @"manifest_version": @2, @"background": @{ @"scripts": @[ @"background.js" ], @"type": @"module", @"persistent": @NO } }; |
| |
| auto manager = Util::parseExtension(manifest, @{ @"background.js": backgroundScript }); |
| |
| // Set a base URL so it is a known value and not the default random one. |
| [WKWebExtensionMatchPattern registerCustomURLScheme:@"test-extension"]; |
| manager.get().context.baseURL = [NSURL URLWithString:baseURLString]; |
| |
| [manager loadAndRun]; |
| |
| // Manifest v3 deprecates getURL(), so it should be an udefined property. |
| |
| backgroundScript = Util::constructScript(@[ |
| @"browser.test.assertEq(typeof browser.extension.getURL, 'undefined')", |
| @"browser.test.assertEq(browser.extension.getURL, undefined)", |
| |
| // Finish |
| @"browser.test.notifyPass()" |
| ]); |
| |
| manifest = @{ @"manifest_version": @3, @"background": @{ @"scripts": @[ @"background.js" ], @"type": @"module", @"persistent": @NO } }; |
| |
| Util::loadAndRunExtension(manifest, @{ @"background.js": backgroundScript }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetBackgroundPageFromBackground) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"const backgroundPage = browser.extension.getBackgroundPage()", |
| |
| @"browser.test.assertTrue(backgroundPage === window, 'Should be able to get the background window from itself')", |
| @"browser.test.assertSafe(() => backgroundPage.notifyTestPass())" |
| ]); |
| |
| Util::loadAndRunExtension(extensionManifest, @{ @"background.js": backgroundScript }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetBackgroundPageFromTab) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.tabs.create({ url: 'test.html' })" |
| ]); |
| |
| auto *testScript = Util::constructScript(@[ |
| @"const backgroundPage = browser.extension.getBackgroundPage()", |
| @"browser.test.assertSafe(() => backgroundPage.notifyTestPass())", |
| ]); |
| |
| auto *testHTML = @"<script type='module' src='test.js'></script>"; |
| |
| Util::loadAndRunExtension(extensionManifest, @{ |
| @"background.js": backgroundScript, |
| @"test.html": testHTML, |
| @"test.js": testScript |
| }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetBackgroundPageFromPopup) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.action.openPopup()" |
| ]); |
| |
| auto *popupScript = Util::constructScript(@[ |
| @"const backgroundPage = browser.extension.getBackgroundPage()", |
| @"browser.test.assertSafe(() => backgroundPage.notifyTestPass())" |
| ]); |
| |
| auto *popupHTML = @"<script type='module' src='popup.js'></script>"; |
| |
| auto resources = @{ |
| @"background.js": backgroundScript, |
| @"popup.html": popupHTML, |
| @"popup.js": popupScript |
| }; |
| |
| auto manager = Util::loadExtension(extensionManifest, resources); |
| |
| manager.get().internalDelegate.presentPopupForAction = ^(WKWebExtensionAction *action) { |
| // Do nothing so the popup web view will stay loaded. |
| }; |
| |
| [manager run]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetBackgroundPageForServiceWorker) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"browser.tabs.create({ url: 'test.html' })" |
| ]); |
| |
| auto *testScript = Util::constructScript(@[ |
| @"const backgroundPage = browser.extension.getBackgroundPage()", |
| @"browser.test.assertEq(backgroundPage, null, 'Background page should be null for service workers')", |
| @"browser.test.notifyPass()" |
| ]); |
| |
| auto *testHTML = @"<script type='module' src='test.js'></script>"; |
| |
| Util::loadAndRunExtension(extensionServiceWorkerManifest, @{ |
| @"background.js": backgroundScript, |
| @"test.html": testHTML, |
| @"test.js": testScript |
| }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsFromBackgroundPage) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"const views = browser.extension.getViews()", |
| |
| @"browser.test.assertEq(views.length, 1, 'Background view should be included')", |
| @"browser.test.assertSafe(() => views[0].notifyTestPass())" |
| ]); |
| |
| Util::loadAndRunExtension(extensionManifest, @{ @"background.js": backgroundScript }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsForBackground) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.tabs.create({ url: 'test.html' })" |
| ]); |
| |
| auto *testScript = Util::constructScript(@[ |
| @"const views = browser.extension.getViews()", |
| @"browser.test.assertEq(views.length, 2)", |
| |
| @"for (const view of views) {", |
| @" if (view.notifyTestPass) {", |
| @" browser.test.assertSafe(() => view.notifyTestPass())", |
| @" break", |
| @" }", |
| @"}" |
| ]); |
| |
| auto *testHTML = @"<script type='module' src='test.js'></script>"; |
| |
| Util::loadAndRunExtension(extensionManifest, @{ |
| @"background.js": backgroundScript, |
| @"test.html": testHTML, |
| @"test.js": testScript |
| }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsForTab) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"browser.runtime.onMessage.addListener((message, sender, sendResponse) => {", |
| @" browser.test.assertEq(message, 'ready')", |
| |
| @" const views = browser.extension.getViews({ type: 'tab' })", |
| @" browser.test.assertEq(views.length, 1)", |
| |
| @" browser.test.assertSafe(() => views[0].notifyTestPass())", |
| @"})", |
| |
| @"browser.tabs.create({ url: 'test.html' })" |
| ]); |
| |
| auto *testScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.runtime.sendMessage('ready')" |
| ]); |
| |
| auto *testHTML = @"<script type='module' src='test.js'></script>"; |
| |
| Util::loadAndRunExtension(extensionManifest, @{ |
| @"background.js": backgroundScript, |
| @"test.html": testHTML, |
| @"test.js": testScript |
| }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsForMultipleTabs) |
| { |
| TestWebKitAPI::HTTPServer server({ |
| { "/"_s, { { { "Content-Type"_s, "text/html"_s } }, ""_s } }, |
| }, TestWebKitAPI::HTTPServer::Protocol::Http); |
| |
| auto *backgroundScript = Util::constructScript(@[ |
| @"let tabCount = 0", |
| |
| @"browser.runtime.onMessage.addListener((message, sender, sendResponse) => {", |
| @" browser.test.assertEq(message, 'ready')", |
| |
| @" if (++tabCount === 2) {", |
| @" const views = browser.extension.getViews({ type: 'tab' })", |
| @" browser.test.assertEq(views.length, 2)", |
| |
| @" for (let i = 0; i < views.length; ++i) {", |
| @" const view = views[i]", |
| @" if (view.notifyTestPass)", |
| @" browser.test.assertSafe(() => view.notifyTestPass(i))", |
| @" }", |
| @" }", |
| @"})", |
| |
| @"browser.tabs.create({ url: 'test.html' })", |
| @"browser.tabs.create({ url: 'test.html' })" |
| ]); |
| |
| auto *testScript = Util::constructScript(@[ |
| @"window.notifyTestPass = (index) => {", |
| @" if (index === 1)", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.runtime.sendMessage('ready')" |
| ]); |
| |
| auto *testHTML = @"<script type='module' src='test.js'></script>"; |
| |
| auto *resources = @{ |
| @"background.js": backgroundScript, |
| @"test.html": testHTML, |
| @"test.js": testScript |
| }; |
| |
| auto manager = Util::loadExtension(extensionManifest, resources); |
| |
| [manager.get().defaultTab.webView loadRequest:server.requestWithLocalhost()]; |
| |
| [manager run]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsForPopup) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"browser.runtime.onMessage.addListener((message, sender, sendResponse) => {", |
| @" browser.test.assertEq(message, 'ready')", |
| |
| @" const popupViews = browser.extension.getViews({ type: 'popup' })", |
| @" browser.test.assertEq(popupViews.length, 1)", |
| |
| @" browser.test.assertSafe(() => popupViews[0].notifyTestPass())", |
| @"})", |
| |
| @"browser.action.openPopup()" |
| ]); |
| |
| auto *popupScript = Util::constructScript(@[ |
| @"window.notifyTestPass = () => {", |
| @" browser.test.notifyPass()", |
| @"}", |
| |
| @"browser.runtime.sendMessage('ready')" |
| ]); |
| |
| auto *popupHTML = @"<script type='module' src='popup.js'></script>"; |
| |
| auto resources = @{ |
| @"background.js": backgroundScript, |
| @"popup.html": popupHTML, |
| @"popup.js": popupScript |
| }; |
| |
| auto manager = Util::loadExtension(extensionManifest, resources); |
| |
| manager.get().internalDelegate.presentPopupForAction = ^(WKWebExtensionAction *action) { |
| // Do nothing so the popup web view will stay loaded. |
| }; |
| |
| [manager run]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, GetViewsExcludesServiceWorkerBackground) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"const views = browser.extension.getViews()", |
| @"browser.test.assertEq(views.length, 0, 'Background view should not be included for service workers')", |
| @"browser.test.notifyPass()" |
| ]); |
| |
| Util::loadAndRunExtension(extensionServiceWorkerManifest, @{ |
| @"background.js": backgroundScript, |
| }); |
| } |
| |
| TEST(WKWebExtensionAPIExtension, InIncognitoContext) |
| { |
| TestWebKitAPI::HTTPServer server({ |
| { "/"_s, { { { "Content-Type"_s, "text/html"_s } }, ""_s } }, |
| }, TestWebKitAPI::HTTPServer::Protocol::Http); |
| |
| auto *backgroundScript = Util::constructScript(@[ |
| @"browser.test.assertFalse(browser.extension.inIncognitoContext, 'Should not be in incognito in background script')" |
| ]); |
| |
| auto *contentScript = Util::constructScript(@[ |
| @"if (browser.extension.inIncognitoContext)", |
| @" browser.test.sendMessage('Content Script is Incognito')", |
| @"else", |
| @" browser.test.sendMessage('Content Script is Not Incognito')" |
| ]); |
| |
| auto manager = Util::loadExtension(extensionManifest, @{ @"background.js": backgroundScript, @"content.js": contentScript }); |
| |
| auto *urlRequest = server.requestWithLocalhost(); |
| [manager.get().context setPermissionStatus:WKWebExtensionContextPermissionStatusGrantedExplicitly forURL:urlRequest.URL]; |
| manager.get().context.hasAccessToPrivateData = YES; |
| |
| [manager.get().defaultTab.webView loadRequest:urlRequest]; |
| |
| [manager runUntilTestMessage:@"Content Script is Not Incognito"]; |
| |
| [manager closeWindow:manager.get().defaultWindow]; |
| |
| auto *privateWindow = [manager openNewWindowUsingPrivateBrowsing:YES]; |
| auto *privateTab = privateWindow.tabs.firstObject; |
| |
| [privateTab.webView loadRequest:urlRequest]; |
| |
| [manager runUntilTestMessage:@"Content Script is Incognito"]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, InIncognitoContextWithoutPrivateAccess) |
| { |
| TestWebKitAPI::HTTPServer server({ |
| { "/"_s, { { { "Content-Type"_s, "text/html"_s } }, ""_s } }, |
| }, TestWebKitAPI::HTTPServer::Protocol::Http); |
| |
| auto *backgroundScript = Util::constructScript(@[ |
| @"browser.test.assertFalse(browser.extension.inIncognitoContext, 'Should not be in incognito in background script')" |
| ]); |
| |
| auto *contentScript = Util::constructScript(@[ |
| @"if (browser.extension.inIncognitoContext)", |
| @" browser.test.sendMessage('Content Script is Incognito')", |
| @"else", |
| @" browser.test.sendMessage('Content Script is Not Incognito')" |
| ]); |
| |
| auto manager = Util::loadExtension(extensionManifest, @{ @"background.js": backgroundScript, @"content.js": contentScript }); |
| |
| auto *urlRequest = server.requestWithLocalhost(); |
| [manager.get().context setPermissionStatus:WKWebExtensionContextPermissionStatusGrantedExplicitly forURL:urlRequest.URL]; |
| |
| [manager.get().defaultTab.webView loadRequest:urlRequest]; |
| |
| [manager runUntilTestMessage:@"Content Script is Not Incognito"]; |
| |
| [manager closeWindow:manager.get().defaultWindow]; |
| |
| auto *privateWindow = [manager openNewWindowUsingPrivateBrowsing:YES]; |
| auto *privateTab = privateWindow.tabs.firstObject; |
| |
| [privateTab.webView loadRequest:urlRequest]; |
| |
| // The content script should not run in the private tab, so timeout after waiting for 3 seconds. |
| [manager runForTimeInterval:3]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, IsAllowedIncognitoAccess) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"const allowed = await browser.extension.isAllowedIncognitoAccess()", |
| @"browser.test.sendMessage(allowed ? 'Allowed Incognito Access' : 'Not Allowed Incognito Access')", |
| ]); |
| |
| auto manager = Util::loadExtension(extensionManifest, @{ @"background.js": backgroundScript }); |
| |
| manager.get().context.hasAccessToPrivateData = YES; |
| |
| [manager runUntilTestMessage:@"Allowed Incognito Access"]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, IsAllowedIncognitoAccessWithoutPrivateAccess) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"const allowed = await browser.extension.isAllowedIncognitoAccess()", |
| @"browser.test.sendMessage(allowed ? 'Allowed Incognito Access' : 'Not Allowed Incognito Access')", |
| ]); |
| |
| auto manager = Util::loadExtension(extensionManifest, @{ @"background.js": backgroundScript }); |
| |
| [manager runUntilTestMessage:@"Not Allowed Incognito Access"]; |
| } |
| |
| TEST(WKWebExtensionAPIExtension, IsAllowedFileSchemeAccess) |
| { |
| auto *backgroundScript = Util::constructScript(@[ |
| @"const allowed = await browser.extension.isAllowedFileSchemeAccess()", |
| @"browser.test.sendMessage(allowed ? 'Allowed File Access' : 'Not Allowed File Access')", |
| ]); |
| |
| auto manager = Util::loadExtension(extensionManifest, @{ @"background.js": backgroundScript }); |
| |
| [manager runUntilTestMessage:@"Not Allowed File Access"]; |
| } |
| |
| } // namespace TestWebKitAPI |
| |
| #endif // ENABLE(WK_WEB_EXTENSIONS) |