blob: 22a3017b92ef8012b0a93266a581f83d8bb6cef3 [file] [log] [blame]
/*
* 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",
};