334 lines
11 KiB
JavaScript
334 lines
11 KiB
JavaScript
/*
|
|
* Copyright (C) 2018 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.
|
|
*/
|
|
|
|
WI.AuditTestGroup = class AuditTestGroup extends WI.AuditTestBase
|
|
{
|
|
constructor(name, tests, options = {})
|
|
{
|
|
console.assert(Array.isArray(tests), tests);
|
|
|
|
// Set disabled once `_tests` is set so that it propagates.
|
|
let disabled = options.disabled;
|
|
options.disabled = false;
|
|
|
|
super(name, options);
|
|
|
|
this._tests = [];
|
|
for (let test of tests)
|
|
this.addTest(test);
|
|
|
|
if (disabled)
|
|
this.updateDisabled(true);
|
|
}
|
|
|
|
// Static
|
|
|
|
static async fromPayload(payload)
|
|
{
|
|
if (typeof payload !== "object" || payload === null)
|
|
return null;
|
|
|
|
if (payload.type !== WI.AuditTestGroup.TypeIdentifier)
|
|
return null;
|
|
|
|
if (typeof payload.name !== "string") {
|
|
WI.AuditManager.synthesizeError(WI.UIString("\u0022%s\u0022 has a non-string \u0022%s\u0022 value").format(payload.name, WI.unlocalizedString("name")));
|
|
return null;
|
|
}
|
|
|
|
if (!Array.isArray(payload.tests)) {
|
|
WI.AuditManager.synthesizeError(WI.UIString("\u0022%s\u0022 has a non-array \u0022%s\u0022 value").format(payload.name, WI.unlocalizedString("tests")));
|
|
return null;
|
|
}
|
|
|
|
let tests = await Promise.all(payload.tests.map(async (test) => {
|
|
let testCase = await WI.AuditTestCase.fromPayload(test);
|
|
if (testCase)
|
|
return testCase;
|
|
|
|
let testGroup = await WI.AuditTestGroup.fromPayload(test);
|
|
if (testGroup)
|
|
return testGroup;
|
|
|
|
return null;
|
|
}));
|
|
tests = tests.filter((test) => !!test);
|
|
if (!tests.length)
|
|
return null;
|
|
|
|
let options = {};
|
|
|
|
if (typeof payload.description === "string")
|
|
options.description = payload.description;
|
|
else if ("description" in payload)
|
|
WI.AuditManager.synthesizeWarning(WI.UIString("\u0022%s\u0022 has a non-string \u0022%s\u0022 value").format(payload.name, WI.unlocalizedString("description")));
|
|
|
|
if (typeof payload.supports === "number")
|
|
options.supports = payload.supports;
|
|
else if ("supports" in payload)
|
|
WI.AuditManager.synthesizeWarning(WI.UIString("\u0022%s\u0022 has a non-number \u0022%s\u0022 value").format(payload.name, WI.unlocalizedString("supports")));
|
|
|
|
if (typeof payload.setup === "string")
|
|
options.setup = payload.setup;
|
|
else if ("setup" in payload)
|
|
WI.AuditManager.synthesizeWarning(WI.UIString("\u0022%s\u0022 has a non-string \u0022%s\u0022 value").format(payload.name, WI.unlocalizedString("setup")));
|
|
|
|
if (typeof payload.disabled === "boolean")
|
|
options.disabled = payload.disabled;
|
|
|
|
return new WI.AuditTestGroup(payload.name, tests, options);
|
|
}
|
|
|
|
// Public
|
|
|
|
get tests() { return this._tests; }
|
|
|
|
addTest(test)
|
|
{
|
|
console.assert(test instanceof WI.AuditTestBase, test);
|
|
console.assert(!this._tests.includes(test), test);
|
|
console.assert(!test._parent, test);
|
|
|
|
this._tests.push(test);
|
|
test._parent = this;
|
|
|
|
test.addEventListener(WI.AuditTestBase.Event.Completed, this._handleTestCompleted, this);
|
|
test.addEventListener(WI.AuditTestBase.Event.DisabledChanged, this._handleTestDisabledChanged, this);
|
|
test.addEventListener(WI.AuditTestBase.Event.Progress, this._handleTestProgress, this);
|
|
if (this.editable) {
|
|
test.addEventListener(WI.AuditTestBase.Event.SupportedChanged, this._handleTestSupportedChanged, this);
|
|
test.addEventListener(WI.AuditTestBase.Event.TestChanged, this._handleTestChanged, this);
|
|
}
|
|
|
|
this.dispatchEventToListeners(WI.AuditTestGroup.Event.TestAdded, {test});
|
|
|
|
this.determineIfSupported();
|
|
|
|
if (this._checkDisabled(test))
|
|
test.updateDisabled(true, {silent: true});
|
|
}
|
|
|
|
removeTest(test)
|
|
{
|
|
console.assert(this.editable);
|
|
console.assert(WI.auditManager.editing);
|
|
console.assert(test instanceof WI.AuditTestBase, test);
|
|
console.assert(test.editable, test);
|
|
console.assert(this._tests.includes(test), test);
|
|
console.assert(test._parent === this, test);
|
|
|
|
test.removeEventListener(WI.AuditTestBase.Event.Completed, this._handleTestCompleted, this);
|
|
test.removeEventListener(WI.AuditTestBase.Event.DisabledChanged, this._handleTestDisabledChanged, this);
|
|
test.removeEventListener(WI.AuditTestBase.Event.Progress, this._handleTestProgress, this);
|
|
test.removeEventListener(WI.AuditTestBase.Event.SupportedChanged, this._handleTestSupportedChanged, this);
|
|
test.removeEventListener(WI.AuditTestBase.Event.TestChanged, this._handleTestChanged, this);
|
|
|
|
this._tests.remove(test);
|
|
test._parent = null;
|
|
|
|
this.dispatchEventToListeners(WI.AuditTestGroup.Event.TestRemoved, {test});
|
|
|
|
this.determineIfSupported();
|
|
|
|
this._checkDisabled();
|
|
}
|
|
|
|
stop()
|
|
{
|
|
// Called from WI.AuditManager.
|
|
|
|
for (let test of this._tests)
|
|
test.stop();
|
|
|
|
super.stop();
|
|
}
|
|
|
|
clearResult(options = {})
|
|
{
|
|
let cleared = !!this.result;
|
|
|
|
if (!options.excludeTests && this._tests) {
|
|
for (let test of this._tests) {
|
|
if (test.clearResult(options))
|
|
cleared = true;
|
|
}
|
|
}
|
|
|
|
return super.clearResult({
|
|
...options,
|
|
suppressResultChangedEvent: !cleared,
|
|
});
|
|
}
|
|
|
|
toJSON(key)
|
|
{
|
|
let json = super.toJSON(key);
|
|
json.tests = this._tests.map((testCase) => testCase.toJSON(key));
|
|
return json;
|
|
}
|
|
|
|
// Protected
|
|
|
|
async run()
|
|
{
|
|
let count = this._tests.length;
|
|
for (let index = 0; index < count && this._runningState === WI.AuditManager.RunningState.Active; ++index) {
|
|
let test = this._tests[index];
|
|
if (test.disabled || !test.supported)
|
|
continue;
|
|
|
|
await test.start();
|
|
|
|
if (test instanceof WI.AuditTestCase)
|
|
this.dispatchEventToListeners(WI.AuditTestBase.Event.Progress, {index, count});
|
|
}
|
|
|
|
this.updateResult();
|
|
}
|
|
|
|
determineIfSupported(options = {})
|
|
{
|
|
if (this._tests) {
|
|
for (let test of this._tests)
|
|
test.determineIfSupported({...options, warn: false, silent: true});
|
|
}
|
|
|
|
return super.determineIfSupported(options);
|
|
}
|
|
|
|
updateSupported(supported, options = {})
|
|
{
|
|
if (this._tests && (!supported || this._tests.every((test) => !test.supported))) {
|
|
supported = false;
|
|
|
|
for (let test of this._tests)
|
|
test.updateSupported(supported, {silent: true});
|
|
}
|
|
|
|
super.updateSupported(supported, options);
|
|
}
|
|
|
|
updateDisabled(disabled, options = {})
|
|
{
|
|
if (!options.excludeTests && this._tests) {
|
|
for (let test of this._tests)
|
|
test.updateDisabled(disabled, options);
|
|
}
|
|
|
|
super.updateDisabled(disabled, options);
|
|
}
|
|
|
|
updateResult()
|
|
{
|
|
let results = this._tests.map((test) => test.result).filter((result) => !!result);
|
|
if (!results.length)
|
|
return;
|
|
|
|
super.updateResult(new WI.AuditTestGroupResult(this.name, results, {
|
|
description: this.description,
|
|
}));
|
|
}
|
|
|
|
// Private
|
|
|
|
_checkDisabled(test)
|
|
{
|
|
let testDisabled = !test || !test.supported || test.disabled;
|
|
let enabledTestCount = this._tests.filter((existing) => existing.supported && !existing.disabled).length;
|
|
|
|
if (testDisabled && !enabledTestCount)
|
|
this.updateDisabled(true);
|
|
else if (!testDisabled && enabledTestCount === 1)
|
|
this.updateDisabled(false, {excludeTests: true});
|
|
else {
|
|
// Don't change `disabled`, as we're currently in an "indeterminate" state.
|
|
this.dispatchEventToListeners(WI.AuditTestBase.Event.DisabledChanged);
|
|
}
|
|
|
|
return this.disabled;
|
|
}
|
|
|
|
_handleTestCompleted(event)
|
|
{
|
|
if (this._runningState === WI.AuditManager.RunningState.Active)
|
|
return;
|
|
|
|
this.updateResult();
|
|
|
|
this.dispatchEventToListeners(WI.AuditTestBase.Event.Completed);
|
|
}
|
|
|
|
_handleTestDisabledChanged(event)
|
|
{
|
|
this._checkDisabled(event.target);
|
|
}
|
|
|
|
_handleTestProgress(event)
|
|
{
|
|
if (this._runningState !== WI.AuditManager.RunningState.Active)
|
|
return;
|
|
|
|
let walk = (tests) => {
|
|
let count = 0;
|
|
for (let test of tests) {
|
|
if (test.disabled || !test.supported)
|
|
continue;
|
|
|
|
if (test instanceof WI.AuditTestCase)
|
|
++count;
|
|
else if (test instanceof WI.AuditTestGroup)
|
|
count += walk(test.tests);
|
|
}
|
|
return count;
|
|
};
|
|
|
|
this.dispatchEventToListeners(WI.AuditTestBase.Event.Progress, {
|
|
index: event.data.index + walk(this._tests.slice(0, this._tests.indexOf(event.target))),
|
|
count: walk(this._tests),
|
|
});
|
|
}
|
|
|
|
_handleTestSupportedChanged(event)
|
|
{
|
|
this.determineIfSupported();
|
|
}
|
|
|
|
_handleTestChanged(event)
|
|
{
|
|
console.assert(WI.auditManager.editing);
|
|
|
|
this.clearResult({excludeTests: true});
|
|
|
|
this.dispatchEventToListeners(WI.AuditTestBase.Event.TestChanged);
|
|
}
|
|
};
|
|
|
|
WI.AuditTestGroup.TypeIdentifier = "test-group";
|
|
|
|
WI.AuditTestGroup.Event = {
|
|
TestAdded: "audit-test-group-test-added",
|
|
TestRemoved: "audit-test-group-test-removed",
|
|
};
|