Skip to content

Commit

Permalink
Bug 1731940 - Web Manifest: implement id processing r=saschanaz
Browse files Browse the repository at this point in the history
  • Loading branch information
marcoscaceres committed Oct 14, 2021
1 parent 2cc714c commit 7e3ff69
Show file tree
Hide file tree
Showing 6 changed files with 213 additions and 11 deletions.
2 changes: 2 additions & 0 deletions dom/locales/en-US/chrome/dom/dom.properties
Original file line number Diff line number Diff line change
Expand Up @@ -212,6 +212,8 @@ ServiceWorkerGraceTimeoutTermination=Terminating ServiceWorker for scope ‘%1$S
# LOCALIZATION NOTE (ServiceWorkerNoFetchHandler): Do not translate "Fetch".
ServiceWorkerNoFetchHandler=Fetch event handlers must be added during the worker script’s initial evaluation.
ExecCommandCutCopyDeniedNotInputDriven=document.execCommand(‘cut’/‘copy’) was denied because it was not called from inside a short running user-generated event handler.
ManifestIdIsInvalid=The id member did not resolve to a valid URL.
ManifestIdNotSameOrigin=The id member must have the same origin as the start_url member.
ManifestShouldBeObject=Manifest should be an object.
ManifestScopeURLInvalid=The scope URL is invalid.
ManifestScopeNotSameOrigin=The scope URL must be same origin as document.
Expand Down
48 changes: 42 additions & 6 deletions dom/manifest/ManifestProcessor.jsm
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,7 @@ var ManifestProcessor = {
background_color: processBackgroundColorMember(),
};
processedManifest.scope = processScopeMember();
processedManifest.id = processIdMember();
if (checkConformance) {
processedManifest.moz_validation = errors;
processedManifest.moz_manifest_url = manifestURL.href;
Expand Down Expand Up @@ -258,28 +259,27 @@ var ManifestProcessor = {
expectedType: "string",
trim: false,
};
let result = new URL(docURL).href;
const defaultStartURL = new URL(docURL).href;
const value = extractor.extractValue(spec);
if (value === undefined || value === "") {
return result;
return defaultStartURL;
}
let potentialResult;
try {
potentialResult = new URL(value, manifestURL);
} catch (e) {
const warn = domBundle.GetStringFromName("ManifestStartURLInvalid");
errors.push({ warn });
return result;
return defaultStartURL;
}
if (potentialResult.origin !== docURL.origin) {
const warn = domBundle.GetStringFromName(
"ManifestStartURLShouldBeSameOrigin"
);
errors.push({ warn });
} else {
result = potentialResult.href;
return defaultStartURL;
}
return result;
return potentialResult.href;
}

function processThemeColorMember() {
Expand Down Expand Up @@ -314,6 +314,42 @@ var ManifestProcessor = {
};
return extractor.extractLanguageValue(spec);
}

function processIdMember() {
// the start_url serves as the fallback, in case the id is not specified
// or in error. A start_url is assured.
const startURL = new URL(processedManifest.start_url);

const spec = {
objectName: "manifest",
object: rawManifest,
property: "id",
expectedType: "string",
trim: false,
};
const extractedValue = extractor.extractValue(spec);

if (typeof extractedValue !== "string" || extractedValue === "") {
return startURL.href;
}

let appId;
try {
appId = new URL(extractedValue, startURL.origin);
} catch {
const warn = domBundle.GetStringFromName("ManifestIdIsInvalid");
errors.push({ warn });
return startURL.href;
}

if (appId.origin !== startURL.origin) {
const warn = domBundle.GetStringFromName("ManifestIdNotSameOrigin");
errors.push({ warn });
return startURL.href;
}

return appId.href;
}
},
};
var EXPORTED_SYMBOLS = ["ManifestProcessor"];
1 change: 1 addition & 0 deletions dom/manifest/test/mochitest.ini
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ support-files =
[test_ManifestProcessor_dir.html]
[test_ManifestProcessor_display.html]
[test_ManifestProcessor_icons.html]
[test_ManifestProcessor_id.html]
[test_ManifestProcessor_JSON.html]
[test_ManifestProcessor_lang.html]
[test_ManifestProcessor_name_and_short_name.html]
Expand Down
123 changes: 123 additions & 0 deletions dom/manifest/test/test_ManifestProcessor_id.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
<!DOCTYPE HTML>
<html>
<!--
https://bugzilla.mozilla.org/show_bug.cgi?id=1731940
-->
<head>
<meta charset="utf-8">
<title>Test for Bug 1731940 - implement id member</title>
<script src="/tests/SimpleTest/SimpleTest.js"></script>
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
<script src="common.js"></script>
<script>
/**
* Manifest id member
* https://w3c.github.io/manifest/#id-member
**/
for (const type of typeTests) {
data.jsonText = JSON.stringify({
id: type,
});
const result = processor.process(data);
is(
result.id.toString(),
result.start_url.toString(),
`Expect non-string id to fall back to start_url: ${typeof type}.`
);
}

// Invalid URLs
const invalidURLs = [
"https://foo:65536",
"https://foo\u0000/",
"//invalid:65555",
"file:///passwords",
"about:blank",
"data:text/html,<html><script>alert('lol')<\/script></html>",
];

for (const url of invalidURLs) {
data.jsonText = JSON.stringify({
id: url,
});
const result = processor.process(data);
is(
result.id.toString(),
result.start_url.toString(),
"Expect invalid id URL to fall back to start_url."
);
}

// Not same origin
data.jsonText = JSON.stringify({
id: "http://not-same-origin",
});
var result = processor.process(data);
is(
result.id.toString(),
result.start_url,
"Expect different origin id to fall back to start_url."
);

// Empty string test
data.jsonText = JSON.stringify({
id: "",
});
result = processor.process(data);
is(
result.id.toString(),
result.start_url.toString(),
`Expect empty string for id to use start_url.`
);

// Resolve URLs relative to the start_url's origin
const URLs = [
"path",
"/path",
"../../path",
"./path",
`${whiteSpace}path${whiteSpace}`,
`${whiteSpace}/path`,
`${whiteSpace}../../path`,
`${whiteSpace}./path`,
];

for (const url of URLs) {
data.jsonText = JSON.stringify({
id: url,
start_url: "/path/some.html",
});
result = processor.process(data);
const baseOrigin = new URL(result.start_url.toString()).origin;
const expectedUrl = new URL(url, baseOrigin).toString();
is(
result.id.toString(),
expectedUrl,
"Expected id to be resolved relative to start_url's origin."
);
}

// Handles unicode encoded URLs
const specialCases = [
["😀", "%F0%9F%98%80"],
[
"this/is/ok?query_is_ok=😀#keep_hash",
"this/is/ok?query_is_ok=%F0%9F%98%80#keep_hash",
],
];
for (const [id, expected] of specialCases) {
data.jsonText = JSON.stringify({
id,
start_url: "/my-app/",
});
result = processor.process(data);
const baseOrigin = new URL(result.start_url.toString()).origin;
const expectedUrl = new URL(expected, baseOrigin).toString();
is(
result.id.toString(),
expectedUrl,
`Expect id to be encoded/decoded per URL spec.`
);
}
</script>
</head>
31 changes: 26 additions & 5 deletions dom/manifest/test/test_ManifestProcessor_start_url.html
Original file line number Diff line number Diff line change
Expand Up @@ -40,9 +40,11 @@
result = processor.process(data);
is(result.start_url.toString(), docURL.toString(), expected);


// Resolve URLs relative to manfiest
var URLs = ["path", "/path", "../../path",
// Resolve URLs relative to manifest
var URLs = [
"path",
"/path",
"../../path",
`${whiteSpace}path${whiteSpace}`,
`${whiteSpace}/path`,
`${whiteSpace}../../path`,
Expand All @@ -58,5 +60,24 @@
is(result.start_url.toString(), absURL, expected);
});

</script>
</head>
// It retains the fragment
var startURL = "./path?query=123#fragment";
data.jsonText = JSON.stringify({
start_url: startURL,
});
var absURL = new URL(startURL, manifestURL).href;
result = processor.process(data);
is(result.start_url.toString(), absURL, "Retains fragment");

// It retains the fragment on the document's location too.
window.location = "#here";
data.jsonText = JSON.stringify({});
result = processor.process(data);
is(
window.location.href,
result.start_url.toString(),
`Retains the fragment of document's location`
);
</script>
</head>
</html>
19 changes: 19 additions & 0 deletions dom/manifest/test/test_ManifestProcessor_warnings.html
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,25 @@
},
warn: "icons item at index 1 includes repeated purpose(s): any maskable.",
},
// testing dom.properties: ManifestIdIsInvalid
{
func() {
return (options.jsonText = JSON.stringify({
id: "http://test:65536/foo",
}));
},
warn: "The id member did not resolve to a valid URL.",
},
// testing dom.properties ManifestIdNotSameOrigin
{
func() {
return (options.jsonText = JSON.stringify({
id: "https://other.com",
start_url: "/this/place"
}));
},
warn: "The id member must have the same origin as the start_url member.",
}
].forEach((test, index) => {
test.func();
const result = processor.process(options);
Expand Down

0 comments on commit 7e3ff69

Please sign in to comment.