forked from mirrors/gecko-dev
Bug 1918386 - browser.search.get() should be returning data URIs for application provided search engines. a=diannaS
Original Revision: https://phabricator.services.mozilla.com/D222264 Differential Revision: https://phabricator.services.mozilla.com/D222466
This commit is contained in:
parent
718f786f39
commit
0ad3c9da9b
2 changed files with 86 additions and 8 deletions
|
|
@ -41,14 +41,18 @@ this.search = class extends ExtensionAPI {
|
|||
return Promise.all(
|
||||
visibleEngines.map(async engine => {
|
||||
let favIconUrl = await engine.getIconURL();
|
||||
// Convert moz-extension:-URLs to data:-URLs to make sure that
|
||||
// Convert blob:-URLs to data:-URLs since they can't be shared
|
||||
// across processes. blob:-URLs originate from application provided
|
||||
// search engines.
|
||||
// Also convert moz-extension:-URLs to data:-URLs to make sure that
|
||||
// extensions can see icons from other extensions, even if they
|
||||
// are not web-accessible.
|
||||
// Also prevents leakage of extension UUIDs to other extensions..
|
||||
if (
|
||||
favIconUrl &&
|
||||
favIconUrl.startsWith("moz-extension:") &&
|
||||
!favIconUrl.startsWith(context.extension.baseURL)
|
||||
(favIconUrl.startsWith("blob:") ||
|
||||
(favIconUrl.startsWith("moz-extension:") &&
|
||||
!favIconUrl.startsWith(context.extension.baseURL)))
|
||||
) {
|
||||
favIconUrl = await ExtensionUtils.makeDataURI(favIconUrl);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -8,9 +8,19 @@ const { AddonTestUtils } = ChromeUtils.importESModule(
|
|||
const { XPCShellContentUtils } = ChromeUtils.importESModule(
|
||||
"resource://testing-common/XPCShellContentUtils.sys.mjs"
|
||||
);
|
||||
const { SearchTestUtils } = ChromeUtils.importESModule(
|
||||
"resource://testing-common/SearchTestUtils.sys.mjs"
|
||||
);
|
||||
const { sinon } = ChromeUtils.importESModule(
|
||||
"resource://testing-common/Sinon.sys.mjs"
|
||||
);
|
||||
const { AppProvidedSearchEngine } = ChromeUtils.importESModule(
|
||||
"resource://gre/modules/AppProvidedSearchEngine.sys.mjs"
|
||||
);
|
||||
|
||||
AddonTestUtils.initMochitest(this);
|
||||
XPCShellContentUtils.initMochitest(this);
|
||||
SearchTestUtils.init(this);
|
||||
|
||||
// Base64-encoded "Fake icon data".
|
||||
const FAKE_ICON_DATA = "RmFrZSBpY29uIGRhdGE=";
|
||||
|
|
@ -18,11 +28,40 @@ const FAKE_ICON_DATA = "RmFrZSBpY29uIGRhdGE=";
|
|||
// Base64-encoded "HTTP icon data".
|
||||
const HTTP_ICON_DATA = "SFRUUCBpY29uIGRhdGE=";
|
||||
const HTTP_ICON_URL = "http://example.org/ico.png";
|
||||
const server = XPCShellContentUtils.createHttpServer({
|
||||
hosts: ["example.org"],
|
||||
});
|
||||
server.registerPathHandler("/ico.png", (request, response) => {
|
||||
response.write(atob(HTTP_ICON_DATA));
|
||||
|
||||
const IMAGE_DATA_URI =
|
||||
"";
|
||||
|
||||
add_setup(async () => {
|
||||
const server = XPCShellContentUtils.createHttpServer({
|
||||
hosts: ["example.org"],
|
||||
});
|
||||
server.registerPathHandler("/ico.png", (request, response) => {
|
||||
response.write(atob(HTTP_ICON_DATA));
|
||||
});
|
||||
|
||||
await SearchTestUtils.updateRemoteSettingsConfig([
|
||||
{ identifier: "appEngineWithIcon" },
|
||||
]);
|
||||
|
||||
let createdBlobURLs = [];
|
||||
|
||||
sinon
|
||||
.stub(AppProvidedSearchEngine.prototype, "getIconURL")
|
||||
.callsFake(async () => {
|
||||
let response = await fetch(IMAGE_DATA_URI);
|
||||
|
||||
let blobURL = URL.createObjectURL(await response.blob());
|
||||
createdBlobURLs.push(blobURL);
|
||||
return blobURL;
|
||||
});
|
||||
|
||||
registerCleanupFunction(async () => {
|
||||
sinon.restore();
|
||||
for (let blobURL of createdBlobURLs) {
|
||||
URL.revokeObjectURL(blobURL);
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
async function promiseEngineIconLoaded(engineName) {
|
||||
|
|
@ -182,3 +221,38 @@ add_task(async function test_search_favicon() {
|
|||
await searchExtWithBadIcon.unload();
|
||||
await searchExtWithHttpIcon.unload();
|
||||
});
|
||||
|
||||
add_task(async function test_app_provided_return_data_uris() {
|
||||
let extension = ExtensionTestUtils.loadExtension({
|
||||
manifest: {
|
||||
permissions: ["search"],
|
||||
},
|
||||
files: {
|
||||
"myFavicon.png": imageBuffer,
|
||||
},
|
||||
useAddonManager: "temporary",
|
||||
async background() {
|
||||
let engines = await browser.search.get();
|
||||
browser.test.sendMessage(
|
||||
"appEngine",
|
||||
engines.find(engine => engine.name == "appEngineWithIcon")
|
||||
);
|
||||
},
|
||||
});
|
||||
|
||||
await extension.startup();
|
||||
await AddonTestUtils.waitForSearchProviderStartup(extension);
|
||||
|
||||
Assert.deepEqual(
|
||||
await extension.awaitMessage("appEngine"),
|
||||
{
|
||||
name: "appEngineWithIcon",
|
||||
isDefault: true,
|
||||
alias: undefined,
|
||||
favIconUrl: IMAGE_DATA_URI,
|
||||
},
|
||||
"browser.search.get result for app provided engine should have a data URL for the icon"
|
||||
);
|
||||
|
||||
await extension.unload();
|
||||
});
|
||||
|
|
|
|||
Loading…
Reference in a new issue