From 1dbbf1ce79f8fa3f48c12df3a9803820ecae982c Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Micha=C3=ABl=20Zasso?=
Date: Tue, 18 Aug 2020 13:23:01 +0200
Subject: [PATCH] test: update WPT harness and tests
---
test/fixtures/wpt/README.md | 15 +-
test/fixtures/wpt/common/META.yml | 3 +
.../wpt/common/PrefixedLocalStorage.js | 116 +
.../common/PrefixedLocalStorage.js.headers | 1 +
.../wpt/common/PrefixedPostMessage.js | 100 +
.../wpt/common/PrefixedPostMessage.js.headers | 1 +
test/fixtures/wpt/common/README.md | 10 +
test/fixtures/wpt/common/arrays.js | 31 +
test/fixtures/wpt/common/blank.html | 0
.../wpt/common/domain-setter.sub.html | 8 +
test/fixtures/wpt/common/dummy.xhtml | 2 +
test/fixtures/wpt/common/dummy.xml | 1 +
test/fixtures/wpt/common/get-host-info.sub.js | 58 +
.../wpt/common/get-host-info.sub.js.headers | 1 +
test/fixtures/wpt/common/media.js | 55 +
test/fixtures/wpt/common/media.js.headers | 1 +
.../fixtures/wpt/common/object-association.js | 68 +
.../wpt/common/object-association.js.headers | 1 +
.../wpt/common/performance-timeline-utils.js | 56 +
.../performance-timeline-utils.js.headers | 1 +
test/fixtures/wpt/common/reftest-wait.js | 20 +
.../wpt/common/reftest-wait.js.headers | 1 +
test/fixtures/wpt/common/rendering-utils.js | 19 +
test/fixtures/wpt/common/sab.js | 13 +
.../wpt/common/security-features/README.md | 460 ++
.../security-features/resources/common.sub.js | 1281 ++++++
.../resources/common.sub.js.headers | 1 +
.../scope/template/document.html.template | 30 +
.../scope/template/worker.js.template | 29 +
.../template/document.html.template | 16 +
.../subresource/template/font.css.template | 9 +
.../subresource/template/image.css.template | 3 +
.../subresource/template/script.js.template | 3 +
.../template/shared-worker.js.template | 5 +
.../template/static-import.js.template | 1 +
.../subresource/template/svg.css.template | 3 +
.../template/svg.embedded.template | 5 +
.../subresource/template/worker.js.template | 3 +
.../security-features/tools/spec.src.json | 546 +++
.../tools/template/disclaimer.template | 1 +
.../tools/template/spec_json.js.template | 1 +
.../tools/template/test.debug.html.template | 26 +
.../tools/template/test.release.html.template | 22 +
.../wpt/common/security-features/types.md | 62 +
test/fixtures/wpt/common/square.png | Bin 0 -> 20527 bytes
test/fixtures/wpt/common/stringifiers.js | 57 +
.../wpt/common/stringifiers.js.headers | 1 +
.../wpt/common/subset-tests-by-key.js | 83 +
test/fixtures/wpt/common/subset-tests.js | 60 +
.../test-setting-immutable-prototype.js | 67 +
...est-setting-immutable-prototype.js.headers | 1 +
test/fixtures/wpt/common/text-plain.txt | 4 +
test/fixtures/wpt/common/utils.js | 98 +
test/fixtures/wpt/common/utils.js.headers | 1 +
test/fixtures/wpt/common/worklet-reftest.js | 50 +
.../wpt/common/worklet-reftest.js.headers | 1 +
.../console/console-label-conversion.any.js | 2 +-
...nsole-namespace-object-class-string.any.js | 40 +
.../wpt/encoding/api-invalid-label.any.js | 9 +-
.../encoding/api-replacement-encodings.any.js | 3 +-
test/fixtures/wpt/encoding/bom-handling.html | 17 +
.../wpt/encoding/bom-handling.html.headers | 1 +
test/fixtures/wpt/encoding/encodeInto.any.js | 89 +-
.../eucjp-decode-cseucpkdfmtjapanese.html | 1 +
.../euc-jp/eucjp-decode-errors.html | 1 +
.../euc-jp/eucjp-decode-x-euc-jp.html | 1 +
.../euc-jp/eucjp-decode.html | 1 +
.../iso-2022-jp/iso2022jp-decode-errors.html | 1 +
.../shift_jis/sjis-encode-href.html | 1 +
.../euc-kr/euckr-decode-ksc_5601.html | 18 +
.../euc-kr/euckr-encode-form-windows-949.html | 1 +
.../big5/big5-encode-form-cn-big5.html | 1 +
.../big5/big5-encode-href-errors-han.html | 1 +
.../big5/big5-encode-href-errors-hangul.html | 1 +
.../wpt/encoding/resources/encodings.js | 16 +-
.../wpt/encoding/streams/backpressure.any.js | 14 +-
.../encoding/streams/decode-attributes.any.js | 24 +-
.../encoding/streams/decode-bad-chunks.any.js | 20 +-
.../encoding/streams/decode-ignore-bom.any.js | 2 +-
.../streams/decode-incomplete-input.any.js | 4 +-
.../encoding/streams/decode-non-utf8.any.js | 6 +-
.../streams/decode-split-character.any.js | 2 +-
.../wpt/encoding/streams/decode-utf8.any.js | 72 +-
.../encoding/streams/encode-bad-chunks.any.js | 10 +-
.../wpt/encoding/streams/encode-utf8.any.js | 2 +-
.../readable-writable-properties.any.js | 2 +-
.../wpt/encoding/streams/realms.window.js | 117 +-
.../wpt/encoding/textdecoder-copy.any.js | 35 +-
.../textdecoder-fatal-single-byte.any.js | 15 +-
.../textdecoder-fatal-streaming.any.js | 6 +-
.../wpt/encoding/textdecoder-fatal.any.js | 8 +-
.../wpt/encoding/textdecoder-ignorebom.any.js | 12 +
.../wpt/encoding/textdecoder-streaming.any.js | 40 +-
.../textdecoder-utf16-surrogates.any.js | 2 +-
.../microtask-queuing/queue-microtask.any.js | 12 +-
.../timers/cleartimeout-clearinterval.any.js | 29 +
.../timers/negative-setinterval.html | 1 +
.../timers/negative-settimeout.html | 1 +
.../timers/type-long-setinterval.html | 1 +
.../timers/type-long-settimeout.html | 1 +
test/fixtures/wpt/interfaces/console.idl | 42 +-
test/fixtures/wpt/interfaces/dom.idl | 94 +-
test/fixtures/wpt/interfaces/encoding.idl | 4 +-
test/fixtures/wpt/interfaces/hr-time.idl | 4 +-
test/fixtures/wpt/interfaces/html.idl | 467 +-
test/fixtures/wpt/interfaces/url.idl | 12 +-
test/fixtures/wpt/resources/META.yml | 1 -
.../fixtures/wpt/resources/check-layout-th.js | 48 +-
test/fixtures/wpt/resources/idlharness.js | 674 ++-
test/fixtures/wpt/resources/readme.md | 22 +-
test/fixtures/wpt/resources/sriharness.js | 108 +
test/fixtures/wpt/resources/test-only-api.js | 84 +
.../wpt/resources/test-only-api.js.headers | 2 +
.../wpt/resources/testdriver-actions.js | 44 +-
test/fixtures/wpt/resources/testdriver.js | 304 +-
test/fixtures/wpt/resources/testharness.js | 742 ++-
.../wpt/resources/webidl2/lib/webidl2.js | 3999 +++++++++++++----
test/fixtures/wpt/url/failure.html | 12 +-
test/fixtures/wpt/url/historical.any.js | 2 +-
.../wpt/url/resources/setters_tests.json | 43 +-
test/fixtures/wpt/url/resources/toascii.json | 22 +
.../wpt/url/resources/urltestdata.json | 307 +-
test/fixtures/wpt/url/toascii.window.js | 2 +-
test/fixtures/wpt/url/url-constructor.html | 2 +-
test/fixtures/wpt/url/url-searchparams.any.js | 2 +-
.../wpt/url/url-setters-stripping.any.js | 125 +
.../fixtures/wpt/url/urlencoded-parser.any.js | 6 +-
.../url/urlsearchparams-constructor.any.js | 45 +-
.../url/urlsearchparams-stringifier.any.js | 12 +
test/fixtures/wpt/versions.json | 18 +-
test/wpt/status/console.json | 5 +-
test/wpt/status/encoding.json | 3 +
test/wpt/status/url.json | 9 +
133 files changed, 9646 insertions(+), 1775 deletions(-)
create mode 100644 test/fixtures/wpt/common/META.yml
create mode 100644 test/fixtures/wpt/common/PrefixedLocalStorage.js
create mode 100644 test/fixtures/wpt/common/PrefixedLocalStorage.js.headers
create mode 100644 test/fixtures/wpt/common/PrefixedPostMessage.js
create mode 100644 test/fixtures/wpt/common/PrefixedPostMessage.js.headers
create mode 100644 test/fixtures/wpt/common/README.md
create mode 100644 test/fixtures/wpt/common/arrays.js
create mode 100644 test/fixtures/wpt/common/blank.html
create mode 100644 test/fixtures/wpt/common/domain-setter.sub.html
create mode 100644 test/fixtures/wpt/common/dummy.xhtml
create mode 100644 test/fixtures/wpt/common/dummy.xml
create mode 100644 test/fixtures/wpt/common/get-host-info.sub.js
create mode 100644 test/fixtures/wpt/common/get-host-info.sub.js.headers
create mode 100644 test/fixtures/wpt/common/media.js
create mode 100644 test/fixtures/wpt/common/media.js.headers
create mode 100644 test/fixtures/wpt/common/object-association.js
create mode 100644 test/fixtures/wpt/common/object-association.js.headers
create mode 100644 test/fixtures/wpt/common/performance-timeline-utils.js
create mode 100644 test/fixtures/wpt/common/performance-timeline-utils.js.headers
create mode 100644 test/fixtures/wpt/common/reftest-wait.js
create mode 100644 test/fixtures/wpt/common/reftest-wait.js.headers
create mode 100644 test/fixtures/wpt/common/rendering-utils.js
create mode 100644 test/fixtures/wpt/common/sab.js
create mode 100644 test/fixtures/wpt/common/security-features/README.md
create mode 100644 test/fixtures/wpt/common/security-features/resources/common.sub.js
create mode 100644 test/fixtures/wpt/common/security-features/resources/common.sub.js.headers
create mode 100644 test/fixtures/wpt/common/security-features/scope/template/document.html.template
create mode 100644 test/fixtures/wpt/common/security-features/scope/template/worker.js.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/document.html.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/font.css.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/image.css.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/script.js.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/shared-worker.js.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/static-import.js.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/svg.css.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/svg.embedded.template
create mode 100644 test/fixtures/wpt/common/security-features/subresource/template/worker.js.template
create mode 100644 test/fixtures/wpt/common/security-features/tools/spec.src.json
create mode 100644 test/fixtures/wpt/common/security-features/tools/template/disclaimer.template
create mode 100644 test/fixtures/wpt/common/security-features/tools/template/spec_json.js.template
create mode 100644 test/fixtures/wpt/common/security-features/tools/template/test.debug.html.template
create mode 100644 test/fixtures/wpt/common/security-features/tools/template/test.release.html.template
create mode 100644 test/fixtures/wpt/common/security-features/types.md
create mode 100644 test/fixtures/wpt/common/square.png
create mode 100644 test/fixtures/wpt/common/stringifiers.js
create mode 100644 test/fixtures/wpt/common/stringifiers.js.headers
create mode 100644 test/fixtures/wpt/common/subset-tests-by-key.js
create mode 100644 test/fixtures/wpt/common/subset-tests.js
create mode 100644 test/fixtures/wpt/common/test-setting-immutable-prototype.js
create mode 100644 test/fixtures/wpt/common/test-setting-immutable-prototype.js.headers
create mode 100644 test/fixtures/wpt/common/text-plain.txt
create mode 100644 test/fixtures/wpt/common/utils.js
create mode 100644 test/fixtures/wpt/common/utils.js.headers
create mode 100644 test/fixtures/wpt/common/worklet-reftest.js
create mode 100644 test/fixtures/wpt/common/worklet-reftest.js.headers
create mode 100644 test/fixtures/wpt/console/console-namespace-object-class-string.any.js
create mode 100644 test/fixtures/wpt/encoding/bom-handling.html
create mode 100644 test/fixtures/wpt/encoding/bom-handling.html.headers
create mode 100644 test/fixtures/wpt/html/webappapis/timers/cleartimeout-clearinterval.any.js
create mode 100644 test/fixtures/wpt/resources/test-only-api.js
create mode 100644 test/fixtures/wpt/resources/test-only-api.js.headers
create mode 100644 test/fixtures/wpt/url/url-setters-stripping.any.js
diff --git a/test/fixtures/wpt/README.md b/test/fixtures/wpt/README.md
index db10cdaf87fa86..b62220fd1b653c 100644
--- a/test/fixtures/wpt/README.md
+++ b/test/fixtures/wpt/README.md
@@ -10,14 +10,15 @@ See [test/wpt](../../wpt/README.md) for information on how these tests are run.
Last update:
-- console: https://github.com/web-platform-tests/wpt/tree/9786a4b131/console
-- encoding: https://github.com/web-platform-tests/wpt/tree/5059d2c777/encoding
-- url: https://github.com/web-platform-tests/wpt/tree/43feb7f612/url
-- resources: https://github.com/web-platform-tests/wpt/tree/e1fddfbf80/resources
-- interfaces: https://github.com/web-platform-tests/wpt/tree/8ada332aea/interfaces
-- html/webappapis/microtask-queuing: https://github.com/web-platform-tests/wpt/tree/0c3bed38df/html/webappapis/microtask-queuing
-- html/webappapis/timers: https://github.com/web-platform-tests/wpt/tree/ddfe9c089b/html/webappapis/timers
+- console: https://github.com/web-platform-tests/wpt/tree/3b1f72e99a/console
+- encoding: https://github.com/web-platform-tests/wpt/tree/11e6941923/encoding
+- url: https://github.com/web-platform-tests/wpt/tree/551c9d604f/url
+- resources: https://github.com/web-platform-tests/wpt/tree/55e9dc7f5e/resources
+- interfaces: https://github.com/web-platform-tests/wpt/tree/4471cda31b/interfaces
+- html/webappapis/microtask-queuing: https://github.com/web-platform-tests/wpt/tree/2c5c3c4c27/html/webappapis/microtask-queuing
+- html/webappapis/timers: https://github.com/web-platform-tests/wpt/tree/264f12bc7b/html/webappapis/timers
- hr-time: https://github.com/web-platform-tests/wpt/tree/a5d1774ecf/hr-time
+- common: https://github.com/web-platform-tests/wpt/tree/4dacb6e2ff/common
[Web Platform Tests]: https://github.com/web-platform-tests/wpt
[`git node wpt`]: https://github.com/nodejs/node-core-utils/blob/master/docs/git-node.md#git-node-wpt
diff --git a/test/fixtures/wpt/common/META.yml b/test/fixtures/wpt/common/META.yml
new file mode 100644
index 00000000000000..ca4d2e523c0866
--- /dev/null
+++ b/test/fixtures/wpt/common/META.yml
@@ -0,0 +1,3 @@
+suggested_reviewers:
+ - zqzhang
+ - deniak
diff --git a/test/fixtures/wpt/common/PrefixedLocalStorage.js b/test/fixtures/wpt/common/PrefixedLocalStorage.js
new file mode 100644
index 00000000000000..2f4e7b6a055caa
--- /dev/null
+++ b/test/fixtures/wpt/common/PrefixedLocalStorage.js
@@ -0,0 +1,116 @@
+/**
+ * Supports pseudo-"namespacing" localStorage for a given test
+ * by generating and using a unique prefix for keys. Why trounce on other
+ * tests' localStorage items when you can keep it "separated"?
+ *
+ * PrefixedLocalStorageTest: Instantiate in testharness.js tests to generate
+ * a new unique-ish prefix
+ * PrefixedLocalStorageResource: Instantiate in supporting test resource
+ * files to use/share a prefix generated by a test.
+ */
+var PrefixedLocalStorage = function () {
+ this.prefix = ''; // Prefix for localStorage keys
+ this.param = 'prefixedLocalStorage'; // Param to use in querystrings
+};
+
+PrefixedLocalStorage.prototype.clear = function () {
+ if (this.prefix === '') { return; }
+ Object.keys(localStorage).forEach(sKey => {
+ if (sKey.indexOf(this.prefix) === 0) {
+ localStorage.removeItem(sKey);
+ }
+ });
+};
+
+/**
+ * Append/replace prefix parameter and value in URI querystring
+ * Use to generate URLs to resource files that will share the prefix.
+ */
+PrefixedLocalStorage.prototype.url = function (uri) {
+ function updateUrlParameter (uri, key, value) {
+ var i = uri.indexOf('#');
+ var hash = (i === -1) ? '' : uri.substr(i);
+ uri = (i === -1) ? uri : uri.substr(0, i);
+ var re = new RegExp(`([?&])${key}=.*?(&|$)`, 'i');
+ var separator = uri.indexOf('?') !== -1 ? '&' : '?';
+ uri = (uri.match(re)) ? uri.replace(re, `$1${key}=${value}$2`) :
+ `${uri}${separator}${key}=${value}`;
+ return uri + hash;
+ }
+ return updateUrlParameter(uri, this.param, this.prefix);
+};
+
+PrefixedLocalStorage.prototype.prefixedKey = function (baseKey) {
+ return `${this.prefix}${baseKey}`;
+};
+
+PrefixedLocalStorage.prototype.setItem = function (baseKey, value) {
+ localStorage.setItem(this.prefixedKey(baseKey), value);
+};
+
+/**
+ * Listen for `storage` events pertaining to a particular key,
+ * prefixed with this object's prefix. Ignore when value is being set to null
+ * (i.e. removeItem).
+ */
+PrefixedLocalStorage.prototype.onSet = function (baseKey, fn) {
+ window.addEventListener('storage', e => {
+ var match = this.prefixedKey(baseKey);
+ if (e.newValue !== null && e.key.indexOf(match) === 0) {
+ fn.call(this, e);
+ }
+ });
+};
+
+/*****************************************************************************
+ * Use in a testharnessjs test to generate a new key prefix.
+ * async_test(t => {
+ * var prefixedStorage = new PrefixedLocalStorageTest();
+ * t.add_cleanup(() => prefixedStorage.cleanup());
+ * /...
+ * });
+ */
+var PrefixedLocalStorageTest = function () {
+ PrefixedLocalStorage.call(this);
+ this.prefix = `${document.location.pathname}-${Math.random()}-${Date.now()}-`;
+};
+PrefixedLocalStorageTest.prototype = Object.create(PrefixedLocalStorage.prototype);
+PrefixedLocalStorageTest.prototype.constructor = PrefixedLocalStorageTest;
+
+/**
+ * Use in a cleanup function to clear out prefixed entries in localStorage
+ */
+PrefixedLocalStorageTest.prototype.cleanup = function () {
+ this.setItem('closeAll', 'true');
+ this.clear();
+};
+
+/*****************************************************************************
+ * Use in test resource files to share a prefix generated by a
+ * PrefixedLocalStorageTest. Will look in URL querystring for prefix.
+ * Setting `close_on_cleanup` opt truthy will make this script's window listen
+ * for storage `closeAll` event from controlling test and close itself.
+ *
+ * var PrefixedLocalStorageResource({ close_on_cleanup: true });
+ */
+var PrefixedLocalStorageResource = function (options) {
+ PrefixedLocalStorage.call(this);
+ this.options = Object.assign({}, {
+ close_on_cleanup: false
+ }, options || {});
+ // Check URL querystring for prefix to use
+ var regex = new RegExp(`[?&]${this.param}(=([^]*)|&|#|$)`),
+ results = regex.exec(document.location.href);
+ if (results && results[2]) {
+ this.prefix = results[2];
+ }
+ // Optionally have this window close itself when the PrefixedLocalStorageTest
+ // sets a `closeAll` item.
+ if (this.options.close_on_cleanup) {
+ this.onSet('closeAll', () => {
+ window.close();
+ });
+ }
+};
+PrefixedLocalStorageResource.prototype = Object.create(PrefixedLocalStorage.prototype);
+PrefixedLocalStorageResource.prototype.constructor = PrefixedLocalStorageResource;
diff --git a/test/fixtures/wpt/common/PrefixedLocalStorage.js.headers b/test/fixtures/wpt/common/PrefixedLocalStorage.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/PrefixedLocalStorage.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/PrefixedPostMessage.js b/test/fixtures/wpt/common/PrefixedPostMessage.js
new file mode 100644
index 00000000000000..674b52877c0d5d
--- /dev/null
+++ b/test/fixtures/wpt/common/PrefixedPostMessage.js
@@ -0,0 +1,100 @@
+/**
+ * Supports pseudo-"namespacing" for window-posted messages for a given test
+ * by generating and using a unique prefix that gets wrapped into message
+ * objects. This makes it more feasible to have multiple tests that use
+ * `window.postMessage` in a single test file. Basically, make it possible
+ * for the each test to listen for only the messages that are pertinent to it.
+ *
+ * 'Prefix' not an elegant term to use here but this models itself after
+ * PrefixedLocalStorage.
+ *
+ * PrefixedMessageTest: Instantiate in testharness.js tests to generate
+ * a new unique-ish prefix that can be used by other test support files
+ * PrefixedMessageResource: Instantiate in supporting test resource
+ * files to use/share a prefix generated by a test.
+ */
+var PrefixedMessage = function () {
+ this.prefix = '';
+ this.param = 'prefixedMessage'; // Param to use in querystrings
+};
+
+/**
+ * Generate a URL that adds/replaces param with this object's prefix
+ * Use to link to test support files that make use of
+ * PrefixedMessageResource.
+ */
+PrefixedMessage.prototype.url = function (uri) {
+ function updateUrlParameter (uri, key, value) {
+ var i = uri.indexOf('#');
+ var hash = (i === -1) ? '' : uri.substr(i);
+ uri = (i === -1) ? uri : uri.substr(0, i);
+ var re = new RegExp(`([?&])${key}=.*?(&|$)`, 'i');
+ var separator = uri.indexOf('?') !== -1 ? '&' : '?';
+ uri = (uri.match(re)) ? uri.replace(re, `$1${key}=${value}$2`) :
+ `${uri}${separator}${key}=${value}`;
+ return uri + hash;
+ }
+ return updateUrlParameter(uri, this.param, this.prefix);
+};
+
+/**
+ * Add an eventListener on `message` but only invoke the given callback
+ * for messages whose object contains this object's prefix. Remove the
+ * event listener once the anticipated message has been received.
+ */
+PrefixedMessage.prototype.onMessage = function (fn) {
+ window.addEventListener('message', e => {
+ if (typeof e.data === 'object' && e.data.hasOwnProperty('prefix')) {
+ if (e.data.prefix === this.prefix) {
+ // Only invoke callback when `data` is an object containing
+ // a `prefix` key with this object's prefix value
+ // Note fn is invoked with "unwrapped" data first, then the event `e`
+ // (which contains the full, wrapped e.data should it be needed)
+ fn.call(this, e.data.data, e);
+ window.removeEventListener('message', fn);
+ }
+ }
+ });
+};
+
+/**
+ * Instantiate in a test file (e.g. during `setup`) to create a unique-ish
+ * prefix that can be shared by support files
+ */
+var PrefixedMessageTest = function () {
+ PrefixedMessage.call(this);
+ this.prefix = `${document.location.pathname}-${Math.random()}-${Date.now()}-`;
+};
+PrefixedMessageTest.prototype = Object.create(PrefixedMessage.prototype);
+PrefixedMessageTest.prototype.constructor = PrefixedMessageTest;
+
+/**
+ * Instantiate in a test support script to use a "prefix" generated by a
+ * PrefixedMessageTest in a controlling test file. It will look for
+ * the prefix in a URL param (see also PrefixedMessage#url)
+ */
+var PrefixedMessageResource = function () {
+ PrefixedMessage.call(this);
+ // Check URL querystring for prefix to use
+ var regex = new RegExp(`[?&]${this.param}(=([^]*)|&|#|$)`),
+ results = regex.exec(document.location.href);
+ if (results && results[2]) {
+ this.prefix = results[2];
+ }
+};
+PrefixedMessageResource.prototype = Object.create(PrefixedMessage.prototype);
+PrefixedMessageResource.prototype.constructor = PrefixedMessageResource;
+
+/**
+ * This is how a test resource document can "send info" to its
+ * opener context. It will whatever message is being sent (`data`) in
+ * an object that injects the prefix.
+ */
+PrefixedMessageResource.prototype.postToOpener = function (data) {
+ if (window.opener) {
+ window.opener.postMessage({
+ prefix: this.prefix,
+ data: data
+ }, '*');
+ }
+};
diff --git a/test/fixtures/wpt/common/PrefixedPostMessage.js.headers b/test/fixtures/wpt/common/PrefixedPostMessage.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/PrefixedPostMessage.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/README.md b/test/fixtures/wpt/common/README.md
new file mode 100644
index 00000000000000..9aef19cb73fa10
--- /dev/null
+++ b/test/fixtures/wpt/common/README.md
@@ -0,0 +1,10 @@
+The files in this directory are non-infrastructure support files that can be used by tests.
+
+* `blank.html` - An empty HTML document.
+* `domain-setter.sub.html` - An HTML document that sets `document.domain`.
+* `dummy.xhtml` - An XHTML document.
+* `dummy.xml` - An XML document.
+* `text-plain.txt` - A text/plain document.
+* `*.js` - Utility scripts. These are documented in the source.
+* `*.py` - wptserve [Python Handlers](https://web-platform-tests.org/writing-tests/python-handlers/). These are documented in the source.
+* `security-features` - Documented in `security-features/README.md`.
diff --git a/test/fixtures/wpt/common/arrays.js b/test/fixtures/wpt/common/arrays.js
new file mode 100644
index 00000000000000..2b31bb4179c26d
--- /dev/null
+++ b/test/fixtures/wpt/common/arrays.js
@@ -0,0 +1,31 @@
+/**
+ * Callback for checking equality of c and d.
+ *
+ * @callback equalityCallback
+ * @param {*} c
+ * @param {*} d
+ * @returns {boolean}
+ */
+
+/**
+ * Returns true if the given arrays are equal. Optionally can pass an equality function.
+ * @param {Array} a
+ * @param {Array} b
+ * @param {equalityCallback} callbackFunction - defaults to `c === d`
+ * @returns {boolean}
+ */
+export function areArraysEqual(a, b, equalityFunction = (c, d) => { return c === d; }) {
+ try {
+ if (a.length !== b.length)
+ return false;
+
+ for (let i = 0; i < a.length; i++) {
+ if (!equalityFunction(a[i], b[i]))
+ return false;
+ }
+ } catch (ex) {
+ return false;
+ }
+
+ return true;
+}
diff --git a/test/fixtures/wpt/common/blank.html b/test/fixtures/wpt/common/blank.html
new file mode 100644
index 00000000000000..e69de29bb2d1d6
diff --git a/test/fixtures/wpt/common/domain-setter.sub.html b/test/fixtures/wpt/common/domain-setter.sub.html
new file mode 100644
index 00000000000000..ad3b9f8b80847f
--- /dev/null
+++ b/test/fixtures/wpt/common/domain-setter.sub.html
@@ -0,0 +1,8 @@
+
+
+A page that will likely be same-origin-domain but not same-origin
+
+
diff --git a/test/fixtures/wpt/common/dummy.xhtml b/test/fixtures/wpt/common/dummy.xhtml
new file mode 100644
index 00000000000000..dba6945f4baf55
--- /dev/null
+++ b/test/fixtures/wpt/common/dummy.xhtml
@@ -0,0 +1,2 @@
+
+Dummy XHTML document
diff --git a/test/fixtures/wpt/common/dummy.xml b/test/fixtures/wpt/common/dummy.xml
new file mode 100644
index 00000000000000..4a60c3035fcc36
--- /dev/null
+++ b/test/fixtures/wpt/common/dummy.xml
@@ -0,0 +1 @@
+Dummy XML document
diff --git a/test/fixtures/wpt/common/get-host-info.sub.js b/test/fixtures/wpt/common/get-host-info.sub.js
new file mode 100644
index 00000000000000..8f37d557583b99
--- /dev/null
+++ b/test/fixtures/wpt/common/get-host-info.sub.js
@@ -0,0 +1,58 @@
+/**
+ * Host information for cross-origin tests.
+ * @returns {Object} with properties for different host information.
+ */
+function get_host_info() {
+
+ var HTTP_PORT = '{{ports[http][0]}}';
+ var HTTP_PORT2 = '{{ports[http][1]}}';
+ var HTTPS_PORT = '{{ports[https][0]}}';
+ var HTTPS_PORT2 = '{{ports[https][1]}}';
+ var PROTOCOL = self.location.protocol;
+ var IS_HTTPS = (PROTOCOL == "https:");
+ var HTTP_PORT_ELIDED = HTTP_PORT == "80" ? "" : (":" + HTTP_PORT);
+ var HTTP_PORT2_ELIDED = HTTP_PORT2 == "80" ? "" : (":" + HTTP_PORT2);
+ var HTTPS_PORT_ELIDED = HTTPS_PORT == "443" ? "" : (":" + HTTPS_PORT);
+ var PORT_ELIDED = IS_HTTPS ? HTTPS_PORT_ELIDED : HTTP_PORT_ELIDED;
+ var ORIGINAL_HOST = '{{host}}';
+ var REMOTE_HOST = (ORIGINAL_HOST === 'localhost') ? '127.0.0.1' : ('www1.' + ORIGINAL_HOST);
+ var OTHER_HOST = '{{domains[www2]}}';
+ var NOTSAMESITE_HOST = (ORIGINAL_HOST === 'localhost') ? '127.0.0.1' : ('{{hosts[alt][]}}');
+
+ return {
+ HTTP_PORT: HTTP_PORT,
+ HTTP_PORT2: HTTP_PORT2,
+ HTTPS_PORT: HTTPS_PORT,
+ HTTPS_PORT2: HTTPS_PORT2,
+ ORIGINAL_HOST: ORIGINAL_HOST,
+ REMOTE_HOST: REMOTE_HOST,
+
+ ORIGIN: PROTOCOL + "//" + ORIGINAL_HOST + PORT_ELIDED,
+ HTTP_ORIGIN: 'http://' + ORIGINAL_HOST + HTTP_PORT_ELIDED,
+ HTTPS_ORIGIN: 'https://' + ORIGINAL_HOST + HTTPS_PORT_ELIDED,
+ HTTPS_ORIGIN_WITH_CREDS: 'https://foo:bar@' + ORIGINAL_HOST + HTTPS_PORT_ELIDED,
+ HTTP_ORIGIN_WITH_DIFFERENT_PORT: 'http://' + ORIGINAL_HOST + HTTP_PORT2_ELIDED,
+ REMOTE_ORIGIN: PROTOCOL + "//" + REMOTE_HOST + PORT_ELIDED,
+ HTTP_REMOTE_ORIGIN: 'http://' + REMOTE_HOST + HTTP_PORT_ELIDED,
+ HTTP_NOTSAMESITE_ORIGIN: 'http://' + NOTSAMESITE_HOST + HTTP_PORT_ELIDED,
+ HTTP_REMOTE_ORIGIN_WITH_DIFFERENT_PORT: 'http://' + REMOTE_HOST + HTTP_PORT2_ELIDED,
+ HTTPS_REMOTE_ORIGIN: 'https://' + REMOTE_HOST + HTTPS_PORT_ELIDED,
+ HTTPS_REMOTE_ORIGIN_WITH_CREDS: 'https://foo:bar@' + REMOTE_HOST + HTTPS_PORT_ELIDED,
+ HTTPS_NOTSAMESITE_ORIGIN: 'https://' + NOTSAMESITE_HOST + HTTPS_PORT_ELIDED,
+ UNAUTHENTICATED_ORIGIN: 'http://' + OTHER_HOST + HTTP_PORT_ELIDED,
+ AUTHENTICATED_ORIGIN: 'https://' + OTHER_HOST + HTTPS_PORT_ELIDED
+ };
+}
+
+/**
+ * When a default port is used, location.port returns the empty string.
+ * This function attempts to provide an exact port, assuming we are running under wptserve.
+ * @param {*} loc - can be Location///URL, but assumes http/https only.
+ * @returns {string} The port number.
+ */
+function get_port(loc) {
+ if (loc.port) {
+ return loc.port;
+ }
+ return loc.protocol === 'https:' ? '443' : '80';
+}
diff --git a/test/fixtures/wpt/common/get-host-info.sub.js.headers b/test/fixtures/wpt/common/get-host-info.sub.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/get-host-info.sub.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/media.js b/test/fixtures/wpt/common/media.js
new file mode 100644
index 00000000000000..e9b1e6b0fbe6ae
--- /dev/null
+++ b/test/fixtures/wpt/common/media.js
@@ -0,0 +1,55 @@
+/**
+ * Returns the URL of a supported video source based on the user agent
+ * @param {string} base - media URL without file extension
+ * @returns {string}
+ */
+function getVideoURI(base)
+{
+ var extension = '.mp4';
+
+ var videotag = document.createElement("video");
+
+ if ( videotag.canPlayType &&
+ videotag.canPlayType('video/ogg; codecs="theora, vorbis"') )
+ {
+ extension = '.ogv';
+ }
+
+ return base + extension;
+}
+
+/**
+ * Returns the URL of a supported audio source based on the user agent
+ * @param {string} base - media URL without file extension
+ * @returns {string}
+ */
+function getAudioURI(base)
+{
+ var extension = '.mp3';
+
+ var audiotag = document.createElement("audio");
+
+ if ( audiotag.canPlayType &&
+ audiotag.canPlayType('audio/ogg') )
+ {
+ extension = '.oga';
+ }
+
+ return base + extension;
+}
+
+/**
+ * Returns the MIME type for a media URL based on the file extension.
+ * @param {string} url
+ * @returns {string}
+ */
+function getMediaContentType(url) {
+ var extension = new URL(url, location).pathname.split(".").pop();
+ var map = {
+ "mp4": "video/mp4",
+ "ogv": "video/ogg",
+ "mp3": "audio/mp3",
+ "oga": "audio/ogg",
+ };
+ return map[extension];
+}
diff --git a/test/fixtures/wpt/common/media.js.headers b/test/fixtures/wpt/common/media.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/media.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/object-association.js b/test/fixtures/wpt/common/object-association.js
new file mode 100644
index 00000000000000..458aae67db0cef
--- /dev/null
+++ b/test/fixtures/wpt/common/object-association.js
@@ -0,0 +1,68 @@
+"use strict";
+
+// For now this only has per-Window tests, but we could expand it to also test per-Document
+
+/**
+ * Run tests for window[propertyName] after discarding the browsing context, navigating, etc.
+ * @param {string} propertyName
+ */
+window.testIsPerWindow = propertyName => {
+ test(t => {
+ const iframe = document.createElement("iframe");
+ document.body.appendChild(iframe);
+ const frame = iframe.contentWindow;
+
+ const before = frame[propertyName];
+ assert_true(before !== undefined && before !== null, `window.${propertyName} must be implemented`);
+
+ iframe.remove();
+
+ const after = frame[propertyName];
+ assert_equals(after, before, `window.${propertyName} should not change after iframe.remove()`);
+ }, `Discarding the browsing context must not change window.${propertyName}`);
+
+ async_test(t => {
+ const iframe = document.createElement("iframe");
+ document.body.appendChild(iframe);
+ const frame = iframe.contentWindow;
+
+ const before = frame[propertyName];
+ assert_true(before !== undefined && before !== null, `window.${propertyName} must be implemented`);
+
+ // Note: cannot use step_func_done for this because it might be called twice, per the below comment.
+ iframe.onload = t.step_func(() => {
+ if (frame.location.href === "about:blank") {
+ // Browsers are not reliable on whether about:blank fires the load event; see
+ // https://github.com/whatwg/html/issues/490
+ return;
+ }
+
+ const after = frame[propertyName];
+ assert_equals(after, before);
+ t.done();
+ });
+
+ iframe.src = "/common/blank.html";
+ }, `Navigating from the initial about:blank must not replace window.${propertyName}`);
+
+ // Per spec, document.open() should not change any of the Window state.
+ async_test(t => {
+ const iframe = document.createElement("iframe");
+
+ iframe.onload = t.step_func_done(() => {
+ const frame = iframe.contentWindow;
+ const before = frame[propertyName];
+ assert_true(before !== undefined && before !== null, `window.${propertyName} must be implemented`);
+
+ frame.document.open();
+
+ const after = frame[propertyName];
+ assert_equals(after, before);
+
+ frame.document.close();
+ });
+
+ iframe.src = "/common/blank.html";
+ document.body.appendChild(iframe);
+ }, `document.open() must replace window.${propertyName}`);
+};
diff --git a/test/fixtures/wpt/common/object-association.js.headers b/test/fixtures/wpt/common/object-association.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/object-association.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/performance-timeline-utils.js b/test/fixtures/wpt/common/performance-timeline-utils.js
new file mode 100644
index 00000000000000..b20241cc6106f9
--- /dev/null
+++ b/test/fixtures/wpt/common/performance-timeline-utils.js
@@ -0,0 +1,56 @@
+/*
+author: W3C http://www.w3.org/
+help: http://www.w3.org/TR/navigation-timing/#sec-window.performance-attribute
+*/
+var performanceNamespace = window.performance;
+var namespace_check = false;
+function wp_test(func, msg, properties)
+{
+ // only run the namespace check once
+ if (!namespace_check)
+ {
+ namespace_check = true;
+
+ if (performanceNamespace === undefined || performanceNamespace == null)
+ {
+ // show a single error that window.performance is undefined
+ // The window.performance attribute provides a hosting area for performance related attributes.
+ test(function() { assert_true(performanceNamespace !== undefined && performanceNamespace != null, "window.performance is defined and not null"); }, "window.performance is defined and not null.");
+ }
+ }
+
+ test(func, msg, properties);
+}
+
+function test_true(value, msg, properties)
+{
+ wp_test(function () { assert_true(value, msg); }, msg, properties);
+}
+
+function test_equals(value, equals, msg, properties)
+{
+ wp_test(function () { assert_equals(value, equals, msg); }, msg, properties);
+}
+
+// assert for every entry in `expectedEntries`, there is a matching entry _somewhere_ in `actualEntries`
+function test_entries(actualEntries, expectedEntries) {
+ test_equals(actualEntries.length, expectedEntries.length)
+ expectedEntries.forEach(function (expectedEntry) {
+ var foundEntry = actualEntries.find(function (actualEntry) {
+ return typeof Object.keys(expectedEntry).find(function (key) {
+ return actualEntry[key] !== expectedEntry[key]
+ }) === 'undefined'
+ })
+ test_true(!!foundEntry, `Entry ${JSON.stringify(expectedEntry)} could not be found.`)
+ if (foundEntry) {
+ assert_object_equals(foundEntry.toJSON(), expectedEntry)
+ }
+ })
+}
+
+function delayedLoadListener(callback) {
+ window.addEventListener('load', function() {
+ // TODO(cvazac) Remove this setTimeout when spec enforces sync entries.
+ step_timeout(callback, 0)
+ })
+}
diff --git a/test/fixtures/wpt/common/performance-timeline-utils.js.headers b/test/fixtures/wpt/common/performance-timeline-utils.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/performance-timeline-utils.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/reftest-wait.js b/test/fixtures/wpt/common/reftest-wait.js
new file mode 100644
index 00000000000000..0a30a197f07f4d
--- /dev/null
+++ b/test/fixtures/wpt/common/reftest-wait.js
@@ -0,0 +1,20 @@
+/**
+ * Remove the `reftest-wait` class on the document element.
+ * The reftest runner will wait with taking a screenshot while
+ * this class is present.
+ *
+ * See https://web-platform-tests.org/writing-tests/reftests.html#controlling-when-comparison-occurs
+ */
+function takeScreenshot() {
+ document.documentElement.classList.remove("reftest-wait");
+}
+
+/**
+ * Call `takeScreenshot()` after a delay of at least |timeout| milliseconds.
+ * @param {number} timeout - milliseconds
+ */
+function takeScreenshotDelayed(timeout) {
+ setTimeout(function() {
+ takeScreenshot();
+ }, timeout);
+}
diff --git a/test/fixtures/wpt/common/reftest-wait.js.headers b/test/fixtures/wpt/common/reftest-wait.js.headers
new file mode 100644
index 00000000000000..6805c323df5a97
--- /dev/null
+++ b/test/fixtures/wpt/common/reftest-wait.js.headers
@@ -0,0 +1 @@
+Content-Type: text/javascript; charset=utf-8
diff --git a/test/fixtures/wpt/common/rendering-utils.js b/test/fixtures/wpt/common/rendering-utils.js
new file mode 100644
index 00000000000000..46283bd5d078a1
--- /dev/null
+++ b/test/fixtures/wpt/common/rendering-utils.js
@@ -0,0 +1,19 @@
+"use strict";
+
+/**
+ * Waits until we have at least one frame rendered, regardless of the engine.
+ *
+ * @returns {Promise}
+ */
+function waitForAtLeastOneFrame() {
+ return new Promise(resolve => {
+ // Different web engines work slightly different on this area but waiting
+ // for two requestAnimationFrames() to happen, one after another, should be
+ // sufficient to ensure at least one frame has been generated anywhere.
+ window.requestAnimationFrame(() => {
+ window.requestAnimationFrame(() => {
+ resolve();
+ });
+ });
+ });
+}
diff --git a/test/fixtures/wpt/common/sab.js b/test/fixtures/wpt/common/sab.js
new file mode 100644
index 00000000000000..c7fd1a742e64f6
--- /dev/null
+++ b/test/fixtures/wpt/common/sab.js
@@ -0,0 +1,13 @@
+const createBuffer = (() => {
+ // See https://github.com/whatwg/html/issues/5380 for why not `new SharedArrayBuffer()`
+ const sabConstructor = new WebAssembly.Memory({ shared:true, initial:0, maximum:0 }).buffer.constructor;
+ return (type, length) => {
+ if (type === "ArrayBuffer") {
+ return new ArrayBuffer(length);
+ } else if (type === "SharedArrayBuffer") {
+ return new sabConstructor(length);
+ } else {
+ throw new Error("type has to be ArrayBuffer or SharedArrayBuffer");
+ }
+ }
+})();
diff --git a/test/fixtures/wpt/common/security-features/README.md b/test/fixtures/wpt/common/security-features/README.md
new file mode 100644
index 00000000000000..fb9a2f108f9faf
--- /dev/null
+++ b/test/fixtures/wpt/common/security-features/README.md
@@ -0,0 +1,460 @@
+This directory contains the common infrastructure for the following tests (also referred below as projects).
+
+- referrer-policy/
+- mixed-content/
+- upgrade-insecure-requests/
+
+Subdirectories:
+
+- `resources`:
+ Serves JavaScript test helpers.
+- `subresource`:
+ Serves subresources, with support for redirects, stash, etc.
+ The subresource paths are managed by `subresourceMap` and
+ fetched in `requestVia*()` functions in `resources/common.js`.
+- `scope`:
+ Serves nested contexts, such as iframe documents or workers.
+ Used from `invokeFrom*()` functions in `resources/common.js`.
+- `tools`:
+ Scripts that generate test HTML files. Not used while running tests.
+- `/referrer-policy/generic/subresource-test`:
+ Sanity checking tests for subresource invocation
+ (This is still placed outside common/)
+
+# Test generator
+
+The test generator ([common/security-features/tools/generate.py](tools/generate.py)) generates test HTML files from templates and a seed (`spec.src.json`) that defines all the test scenarios.
+
+The project (i.e. a WPT subdirectory, for example `referrer-policy/`) that uses the generator should define per-project data and invoke the common generator logic in `common/security-features/tools`.
+
+This is the overview of the project structure:
+
+```
+common/security-features/
+└── tools/ - the common test generator logic
+ ├── spec.src.json
+ └── template/ - the test files templates
+project-directory/ (e.g. referrer-policy/)
+├── spec.src.json
+├── generic/
+│ ├── test-case.sub.js - Per-project test helper
+│ ├── sanity-checker.js (Used by debug target only)
+│ └── spec_json.js (Used by debug target only)
+└── gen/ - generated tests
+```
+
+## Generating the tests
+
+Note: When the repository already contains generated tests, [remove all generated tests](#removing-all-generated-tests) first.
+
+```bash
+# Install json5 module if needed.
+pip install --user json5
+
+# Generate the test files under gen/ (HTMLs and .headers files).
+path/to/common/security-features/tools/generate.py --spec path/to/project-directory/
+
+# Add all generated tests to the repo.
+git add path/to/project-directory/gen/ && git commit -m "Add generated tests"
+```
+
+This will parse the spec JSON5 files and determine which tests to generate (or skip) while using templates.
+
+- The default spec JSON5: `common/security-features/tools/spec.src.json`.
+ - Describes common configurations, such as subresource types, source context types, etc.
+- The per-project spec JSON5: `project-directory/spec.src.json`.
+ - Describes project-specific configurations, particularly those related to test generation patterns (`specification`), policy deliveries (e.g. `delivery_type`, `delivery_value`) and `expectation`.
+
+For how these two spec JSON5 files are merged, see [Sub projects](#sub-projects) section.
+
+Note: `spec.src.json` is transitioning to JSON5 [#21710](https://github.com/web-platform-tests/wpt/issues/21710).
+
+During the generation, the spec is validated by ```common/security-features/tools/spec_validator.py```. This is specially important when you're making changes to `spec.src.json`. Make sure it's a valid JSON (no comments or trailing commas). The validator reports specific errors (missing keys etc.), if any.
+
+### Removing all generated tests
+
+Simply remove all files under `project-directory/gen/`.
+
+```bash
+rm -r path/to/project-directory/gen/
+```
+
+### Options for generating tests
+
+Note: this section is currently obsolete. Only the release template is working.
+
+The generator script has two targets: ```release``` and ```debug```.
+
+* Using **release** for the target will produce tests using a template for optimizing size and performance. The release template is intended for the official web-platform-tests and possibly other test suites. No sanity checking is done in release mode. Use this option whenever you're checking into web-platform-tests.
+
+* When generating for ```debug```, the produced tests will contain more verbosity and sanity checks. Use this target to identify problems with the test suites when making changes locally. Make sure you don't check in tests generated with the debug target.
+
+Note that **release** is the default target when invoking ```generate.py```.
+
+
+## Sub projects
+
+Projects can be nested, for example to reuse a single `spec.src.json` across similar but slightly different sets of generated tests.
+The directory structure would look like:
+
+```
+project-directory/ (e.g. referrer-policy/)
+├── spec.src.json - Parent project's spec JSON
+├── generic/
+│ └── test-case.sub.js - Parent project's test helper
+├── gen/ - parent project's generated tests
+└── sub-project-directory/ (e.g. 4K)
+ ├── spec.src.json - Child project's spec JSON
+ ├── generic/
+ │ └── test-case.sub.js - Child project's test helper
+ └── gen/ - child project's generated tests
+```
+
+`generate.py --spec project-directory/sub-project-directory` generates test files under `project-directory/sub-project-directory/gen`, based on `project-directory/spec.src.json` and `project-directory/sub-project-directory/spec.src.json`.
+
+- The child project's `spec.src.json` is merged into parent project's `spec.src.json`.
+ - Two spec JSON objects are merged recursively.
+ - If a same key exists in both objects, the child's value overwrites the parent's value.
+ - If both (child's and parent's) values are arrays, then the child's value is concatenated to the parent's value.
+ - For debugging, `generate.py` dumps the merged spec JSON object as `generic/debug-output.spec.src.json`.
+- The child project's generated tests include both of the parent and child project's `test-case.sub.js`:
+ ```html
+
+
+
+ ```
+
+
+## Updating the tests
+
+The main test logic lives in ```project-directory/generic/test-case.sub.js``` with helper functions defined in ```/common/security-features/resources/common.js``` so you should probably start there.
+
+For updating the test suites you will most likely do **a subset** of the following:
+
+* Add a new subresource type:
+
+ * Add a new sub-resource python script to `/common/security-features/subresource/`.
+ * Add a sanity check test for a sub-resource to `referrer-policy/generic/subresource-test/`.
+ * Add a new entry to `subresourceMap` in `/common/security-features/resources/common.js`.
+ * Add a new entry to `valid_subresource_names` in `/common/security-features/tools/spec_validator.py`.
+ * Add a new entry to `subresource_schema` in `spec.src.json`.
+ * Update `source_context_schema` to specify in which source context the subresource can be used.
+
+* Add a new subresource redirection type
+
+ * TODO: to be documented. Example: [https://github.com/web-platform-tests/wpt/pull/18939](https://github.com/web-platform-tests/wpt/pull/18939)
+
+* Add a new subresource origin type
+
+ * TODO: to be documented. Example: [https://github.com/web-platform-tests/wpt/pull/18940](https://github.com/web-platform-tests/wpt/pull/18940)
+
+* Add a new source context (e.g. "module sharedworker global scope")
+
+ * TODO: to be documented. Example: [https://github.com/web-platform-tests/wpt/pull/18904](https://github.com/web-platform-tests/wpt/pull/18904)
+
+* Add a new source context list (e.g. "subresource request from a dedicated worker in a `