mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
557 lines
14 KiB
JavaScript
557 lines
14 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
const TEST_BASE_HTTP = "http://example.com/browser/browser/devtools/commandline/test/";
|
|
const TEST_BASE_HTTPS = "https://example.com/browser/browser/devtools/commandline/test/";
|
|
|
|
let console = (function() {
|
|
let tempScope = {};
|
|
Components.utils.import("resource:///modules/devtools/Console.jsm", tempScope);
|
|
return tempScope.console;
|
|
})();
|
|
|
|
/**
|
|
* Open a new tab at a URL and call a callback on load
|
|
*/
|
|
function addTab(aURL, aCallback)
|
|
{
|
|
waitForExplicitFinish();
|
|
|
|
gBrowser.selectedTab = gBrowser.addTab();
|
|
content.location = aURL;
|
|
|
|
let tab = gBrowser.selectedTab;
|
|
let browser = gBrowser.getBrowserForTab(tab);
|
|
|
|
function onTabLoad() {
|
|
browser.removeEventListener("load", onTabLoad, true);
|
|
aCallback(browser, tab, browser.contentDocument);
|
|
}
|
|
|
|
browser.addEventListener("load", onTabLoad, true);
|
|
}
|
|
|
|
registerCleanupFunction(function tearDown() {
|
|
while (gBrowser.tabs.length > 1) {
|
|
gBrowser.removeCurrentTab();
|
|
}
|
|
|
|
console = undefined;
|
|
});
|
|
|
|
/**
|
|
* Various functions for testing DeveloperToolbar.
|
|
* Parts of this code exist in:
|
|
* - browser/devtools/commandline/test/head.js
|
|
* - browser/devtools/shared/test/head.js
|
|
*/
|
|
let DeveloperToolbarTest = { };
|
|
|
|
/**
|
|
* Paranoid DeveloperToolbar.show();
|
|
*/
|
|
DeveloperToolbarTest.show = function DTT_show(aCallback) {
|
|
if (DeveloperToolbar.visible) {
|
|
ok(false, "DeveloperToolbar.visible at start of openDeveloperToolbar");
|
|
}
|
|
else {
|
|
DeveloperToolbar.show(true, aCallback);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Paranoid DeveloperToolbar.hide();
|
|
*/
|
|
DeveloperToolbarTest.hide = function DTT_hide() {
|
|
if (!DeveloperToolbar.visible) {
|
|
ok(false, "!DeveloperToolbar.visible at start of closeDeveloperToolbar");
|
|
}
|
|
else {
|
|
DeveloperToolbar.display.inputter.setInput("");
|
|
DeveloperToolbar.hide();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* check() is the new status. Similar API except that it doesn't attempt to
|
|
* alter the display/requisition at all, and it makes extra checks.
|
|
* Test inputs
|
|
* typed: The text to type at the input
|
|
* Available checks:
|
|
* input: The text displayed in the input field
|
|
* cursor: The position of the start of the cursor
|
|
* status: One of "VALID", "ERROR", "INCOMPLETE"
|
|
* emptyParameters: Array of parameters still to type. e.g. [ "<message>" ]
|
|
* directTabText: Simple completion text
|
|
* arrowTabText: When the completion is not an extension (without arrow)
|
|
* markup: What state should the error markup be in. e.g. "VVVIIIEEE"
|
|
* args: Maps of checks to make against the arguments:
|
|
* value: i.e. assignment.value (which ignores defaultValue)
|
|
* type: Argument/BlankArgument/MergedArgument/etc i.e. what's assigned
|
|
* Care should be taken with this since it's something of an
|
|
* implementation detail
|
|
* arg: The toString value of the argument
|
|
* status: i.e. assignment.getStatus
|
|
* message: i.e. assignment.getMessage
|
|
* name: For commands - checks assignment.value.name
|
|
*/
|
|
DeveloperToolbarTest.checkInputStatus = function DTT_checkInputStatus(checks) {
|
|
if (!checks.emptyParameters) {
|
|
checks.emptyParameters = [];
|
|
}
|
|
if (!checks.directTabText) {
|
|
checks.directTabText = '';
|
|
}
|
|
if (!checks.arrowTabText) {
|
|
checks.arrowTabText = '';
|
|
}
|
|
|
|
var display = DeveloperToolbar.display;
|
|
|
|
if (checks.typed) {
|
|
display.inputter.setInput(checks.typed);
|
|
}
|
|
else {
|
|
ok(false, "Missing typed for " + JSON.stringify(checks));
|
|
return;
|
|
}
|
|
|
|
if (checks.cursor) {
|
|
display.inputter.setCursor(checks.cursor)
|
|
}
|
|
|
|
var cursor = checks.cursor ? checks.cursor.start : checks.typed.length;
|
|
|
|
var requisition = display.requisition;
|
|
var completer = display.completer;
|
|
var actual = completer._getCompleterTemplateData();
|
|
|
|
/*
|
|
if (checks.input) {
|
|
is(display.inputter.element.value,
|
|
checks.input,
|
|
'input');
|
|
}
|
|
|
|
if (checks.cursor) {
|
|
is(display.inputter.element.selectionStart,
|
|
checks.cursor,
|
|
'cursor');
|
|
}
|
|
*/
|
|
|
|
if (checks.status) {
|
|
is(requisition.getStatus().toString(),
|
|
checks.status,
|
|
'status');
|
|
}
|
|
|
|
if (checks.markup) {
|
|
var statusMarkup = requisition.getInputStatusMarkup(cursor);
|
|
var actualMarkup = statusMarkup.map(function(s) {
|
|
return Array(s.string.length + 1).join(s.status.toString()[0]);
|
|
}).join('');
|
|
|
|
is(checks.markup,
|
|
actualMarkup,
|
|
'markup');
|
|
}
|
|
|
|
if (checks.emptyParameters) {
|
|
var actualParams = actual.emptyParameters;
|
|
is(actualParams.length,
|
|
checks.emptyParameters.length,
|
|
'emptyParameters.length');
|
|
|
|
if (actualParams.length === checks.emptyParameters.length) {
|
|
for (var i = 0; i < actualParams.length; i++) {
|
|
is(actualParams[i].replace(/\u00a0/g, ' '),
|
|
checks.emptyParameters[i],
|
|
'emptyParameters[' + i + ']');
|
|
}
|
|
}
|
|
}
|
|
|
|
if (checks.directTabText) {
|
|
is(actual.directTabText,
|
|
checks.directTabText,
|
|
'directTabText');
|
|
}
|
|
|
|
if (checks.arrowTabText) {
|
|
is(actual.arrowTabText,
|
|
' \u00a0\u21E5 ' + checks.arrowTabText,
|
|
'arrowTabText');
|
|
}
|
|
|
|
if (checks.args) {
|
|
Object.keys(checks.args).forEach(function(paramName) {
|
|
var check = checks.args[paramName];
|
|
|
|
var assignment;
|
|
if (paramName === 'command') {
|
|
assignment = requisition.commandAssignment;
|
|
}
|
|
else {
|
|
assignment = requisition.getAssignment(paramName);
|
|
}
|
|
|
|
if (assignment == null) {
|
|
ok(false, 'Unknown parameter: ' + paramName);
|
|
return;
|
|
}
|
|
|
|
if (check.value) {
|
|
is(assignment.value,
|
|
check.value,
|
|
'checkStatus value for ' + paramName);
|
|
}
|
|
|
|
if (check.name) {
|
|
is(assignment.value.name,
|
|
check.name,
|
|
'checkStatus name for ' + paramName);
|
|
}
|
|
|
|
if (check.type) {
|
|
is(assignment.arg.type,
|
|
check.type,
|
|
'checkStatus type for ' + paramName);
|
|
}
|
|
|
|
if (check.arg) {
|
|
is(assignment.arg.toString(),
|
|
check.arg,
|
|
'checkStatus arg for ' + paramName);
|
|
}
|
|
|
|
if (check.status) {
|
|
is(assignment.getStatus().toString(),
|
|
check.status,
|
|
'checkStatus status for ' + paramName);
|
|
}
|
|
|
|
if (check.message) {
|
|
is(assignment.getMessage(),
|
|
check.message,
|
|
'checkStatus message for ' + paramName);
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Execute a command:
|
|
*
|
|
* DeveloperToolbarTest.exec({
|
|
* // Test inputs
|
|
* typed: "echo hi", // Optional, uses existing if undefined
|
|
*
|
|
* // Thing to check
|
|
* args: { message: "hi" }, // Check that the args were understood properly
|
|
* outputMatch: /^hi$/, // RegExp to test against textContent of output
|
|
* // (can also be array of RegExps)
|
|
* blankOutput: true, // Special checks when there is no output
|
|
* });
|
|
*/
|
|
DeveloperToolbarTest.exec = function DTT_exec(tests) {
|
|
tests = tests || {};
|
|
|
|
if (tests.typed) {
|
|
DeveloperToolbar.display.inputter.setInput(tests.typed);
|
|
}
|
|
|
|
let typed = DeveloperToolbar.display.inputter.getInputState().typed;
|
|
let output = DeveloperToolbar.display.requisition.exec();
|
|
|
|
is(typed, output.typed, 'output.command for: ' + typed);
|
|
|
|
if (tests.completed !== false) {
|
|
ok(output.completed, 'output.completed false for: ' + typed);
|
|
}
|
|
else {
|
|
// It is actually an error if we say something is async and it turns
|
|
// out not to be? For now we're saying 'no'
|
|
// ok(!output.completed, 'output.completed true for: ' + typed);
|
|
}
|
|
|
|
if (tests.args != null) {
|
|
is(Object.keys(tests.args).length, Object.keys(output.args).length,
|
|
'arg count for ' + typed);
|
|
|
|
Object.keys(output.args).forEach(function(arg) {
|
|
let expectedArg = tests.args[arg];
|
|
let actualArg = output.args[arg];
|
|
|
|
if (typeof expectedArg === 'function') {
|
|
ok(expectedArg(actualArg), 'failed test func. ' + typed + '/' + arg);
|
|
}
|
|
else {
|
|
if (Array.isArray(expectedArg)) {
|
|
if (!Array.isArray(actualArg)) {
|
|
ok(false, 'actual is not an array. ' + typed + '/' + arg);
|
|
return;
|
|
}
|
|
|
|
is(expectedArg.length, actualArg.length,
|
|
'array length: ' + typed + '/' + arg);
|
|
for (let i = 0; i < expectedArg.length; i++) {
|
|
is(expectedArg[i], actualArg[i],
|
|
'member: "' + typed + '/' + arg + '/' + i);
|
|
}
|
|
}
|
|
else {
|
|
is(expectedArg, actualArg, 'typed: "' + typed + '" arg: ' + arg);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
let displayed = DeveloperToolbar.outputPanel._div.textContent;
|
|
|
|
if (tests.outputMatch) {
|
|
function doTest(match, against) {
|
|
if (!match.test(against)) {
|
|
ok(false, "html output for " + typed + " against " + match.source +
|
|
" (textContent sent to info)");
|
|
info("Actual textContent");
|
|
info(against);
|
|
}
|
|
}
|
|
if (Array.isArray(tests.outputMatch)) {
|
|
tests.outputMatch.forEach(function(match) {
|
|
doTest(match, displayed);
|
|
});
|
|
}
|
|
else {
|
|
doTest(tests.outputMatch, displayed);
|
|
}
|
|
}
|
|
|
|
if (tests.blankOutput != null) {
|
|
if (!/^$/.test(displayed)) {
|
|
ok(false, "html output for " + typed + " (textContent sent to info)");
|
|
info("Actual textContent");
|
|
info(displayed);
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Quick wrapper around the things you need to do to run DeveloperToolbar
|
|
* command tests:
|
|
* - Set the pref 'devtools.toolbar.enabled' to true
|
|
* - Add a tab pointing at |uri|
|
|
* - Open the DeveloperToolbar
|
|
* - Register a cleanup function to undo the above
|
|
* - Run the tests
|
|
*
|
|
* @param uri The uri of a page to load. Can be 'about:blank' or 'data:...'
|
|
* @param testFunc A function containing the tests to run. This should
|
|
* arrange for 'finish()' to be called on completion.
|
|
*/
|
|
DeveloperToolbarTest.test = function DTT_test(uri, testFunc) {
|
|
let menuItem = document.getElementById("menu_devToolbar");
|
|
let command = document.getElementById("Tools:DevToolbar");
|
|
let appMenuItem = document.getElementById("appmenu_devToolbar");
|
|
|
|
registerCleanupFunction(function() {
|
|
DeveloperToolbarTest.hide();
|
|
|
|
// a.k.a Services.prefs.clearUserPref("devtools.toolbar.enabled");
|
|
if (menuItem) {
|
|
menuItem.hidden = true;
|
|
}
|
|
if (command) {
|
|
command.setAttribute("disabled", "true");
|
|
}
|
|
if (appMenuItem) {
|
|
appMenuItem.hidden = true;
|
|
}
|
|
|
|
// leakHunt({ DeveloperToolbar: DeveloperToolbar });
|
|
});
|
|
|
|
// a.k.a: Services.prefs.setBoolPref("devtools.toolbar.enabled", true);
|
|
if (menuItem) {
|
|
menuItem.hidden = false;
|
|
}
|
|
if (command) {
|
|
command.removeAttribute("disabled");
|
|
}
|
|
if (appMenuItem) {
|
|
appMenuItem.hidden = false;
|
|
}
|
|
|
|
addTab(uri, function(browser, tab) {
|
|
DeveloperToolbarTest.show(function() {
|
|
|
|
try {
|
|
testFunc(browser, tab);
|
|
}
|
|
catch (ex) {
|
|
ok(false, "" + ex);
|
|
console.error(ex);
|
|
finish();
|
|
throw ex;
|
|
}
|
|
});
|
|
});
|
|
};
|
|
|
|
|
|
/**
|
|
* Memory leak hunter. Walks a tree of objects looking for DOM nodes.
|
|
* Usage:
|
|
* leakHunt({
|
|
* thing: thing,
|
|
* otherthing: otherthing
|
|
* });
|
|
*/
|
|
|
|
var noRecurse = [
|
|
/^string$/, /^number$/, /^boolean$/, /^null/, /^undefined/,
|
|
/^Window$/, /^Document$/,
|
|
/^XULDocument$/, /^XULElement$/,
|
|
/^DOMWindow$/, /^HTMLDocument$/, /^HTML.*Element$/
|
|
];
|
|
|
|
var hide = [ /^string$/, /^number$/, /^boolean$/, /^null/, /^undefined/ ];
|
|
|
|
function leakHunt(root, path, seen) {
|
|
path = path || [];
|
|
seen = seen || [];
|
|
|
|
try {
|
|
var output = leakHuntInner(root, path, seen);
|
|
output.forEach(function(line) {
|
|
dump(line + '\n');
|
|
});
|
|
}
|
|
catch (ex) {
|
|
dump(ex + '\n');
|
|
}
|
|
}
|
|
|
|
function leakHuntInner(root, path, seen) {
|
|
var prefix = new Array(path.length).join(' ');
|
|
|
|
var reply = [];
|
|
function log(msg) {
|
|
reply.push(msg);
|
|
}
|
|
|
|
var direct
|
|
try {
|
|
direct = Object.keys(root);
|
|
}
|
|
catch (ex) {
|
|
log(prefix + ' Error enumerating: ' + ex);
|
|
return reply;
|
|
}
|
|
|
|
for (var prop in root) {
|
|
var newPath = path.slice();
|
|
newPath.push(prop);
|
|
prefix = new Array(newPath.length).join(' ');
|
|
|
|
var data;
|
|
try {
|
|
data = root[prop];
|
|
}
|
|
catch (ex) {
|
|
log(prefix + prop + ' Error reading: ' + ex);
|
|
continue;
|
|
}
|
|
|
|
var recurse = true;
|
|
var message = getType(data);
|
|
|
|
if (matchesAnyPattern(message, hide)) {
|
|
continue;
|
|
}
|
|
|
|
if (message === 'function' && direct.indexOf(prop) == -1) {
|
|
continue;
|
|
}
|
|
|
|
if (message === 'string') {
|
|
var extra = data.length > 10 ? data.substring(0, 9) + '_' : data;
|
|
message += ' "' + extra.replace(/\n/g, "|") + '"';
|
|
recurse = false;
|
|
}
|
|
else if (matchesAnyPattern(message, noRecurse)) {
|
|
message += ' (no recurse)'
|
|
recurse = false;
|
|
}
|
|
else if (seen.indexOf(data) !== -1) {
|
|
message += ' (already seen)';
|
|
recurse = false;
|
|
}
|
|
|
|
if (recurse) {
|
|
seen.push(data);
|
|
var lines = leakHuntInner(data, newPath, seen);
|
|
if (lines.length == 0) {
|
|
if (message !== 'function') {
|
|
log(prefix + prop + ' = ' + message + ' { }');
|
|
}
|
|
}
|
|
else {
|
|
log(prefix + prop + ' = ' + message + ' {');
|
|
lines.forEach(function(line) {
|
|
reply.push(line);
|
|
});
|
|
log(prefix + '}');
|
|
}
|
|
}
|
|
else {
|
|
log(prefix + prop + ' = ' + message);
|
|
}
|
|
}
|
|
|
|
return reply;
|
|
}
|
|
|
|
function matchesAnyPattern(str, patterns) {
|
|
var match = false;
|
|
patterns.forEach(function(pattern) {
|
|
if (str.match(pattern)) {
|
|
match = true;
|
|
}
|
|
});
|
|
return match;
|
|
}
|
|
|
|
function getType(data) {
|
|
if (data === null) {
|
|
return 'null';
|
|
}
|
|
if (data === undefined) {
|
|
return 'undefined';
|
|
}
|
|
|
|
var type = typeof data;
|
|
if (type === 'object' || type === 'Object') {
|
|
type = getCtorName(data);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
function getCtorName(aObj) {
|
|
try {
|
|
if (aObj.constructor && aObj.constructor.name) {
|
|
return aObj.constructor.name;
|
|
}
|
|
}
|
|
catch (ex) {
|
|
return 'UnknownObject';
|
|
}
|
|
|
|
// If that fails, use Objects toString which sometimes gives something
|
|
// better than 'Object', and at least defaults to Object if nothing better
|
|
return Object.prototype.toString.call(aObj).slice(8, -1);
|
|
}
|