75b824d032
js2-mode is no longer developed and we recommend js-mode these days, so switch the modelines to specify that, and make them consistent across all files. https://bugzilla.gnome.org/show_bug.cgi?id=660358
371 lines
12 KiB
JavaScript
371 lines
12 KiB
JavaScript
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
|
|
|
const DBus = imports.dbus;
|
|
const Gio = imports.gi.Gio;
|
|
const Mainloop = imports.mainloop;
|
|
const Meta = imports.gi.Meta;
|
|
const Shell = imports.gi.Shell;
|
|
|
|
const Main = imports.ui.main;
|
|
|
|
// This module provides functionality for driving the shell user interface
|
|
// in an automated fashion. The primary current use case for this is
|
|
// automated performance testing (see runPerfScript()), but it could
|
|
// be applied to other forms of automation, such as testing for
|
|
// correctness as well.
|
|
//
|
|
// When scripting an automated test we want to make a series of calls
|
|
// in a linear fashion, but we also want to be able to let the main
|
|
// loop run so actions can finish. For this reason we write the script
|
|
// as a generator function that yields when it want to let the main
|
|
// loop run.
|
|
//
|
|
// yield Scripting.sleep(1000);
|
|
// main.overview.show();
|
|
// yield Scripting.waitLeisure();
|
|
//
|
|
// While it isn't important to the person writing the script, the actual
|
|
// yielded result is a function that the caller uses to provide the
|
|
// callback for resuming the script.
|
|
|
|
/**
|
|
* sleep:
|
|
* @milliseconds: number of milliseconds to wait
|
|
*
|
|
* Used within an automation script to pause the the execution of the
|
|
* current script for the specified amount of time. Use as
|
|
* 'yield Scripting.sleep(500);'
|
|
*/
|
|
function sleep(milliseconds) {
|
|
let cb;
|
|
|
|
Mainloop.timeout_add(milliseconds, function() {
|
|
if (cb)
|
|
cb();
|
|
return false;
|
|
});
|
|
|
|
return function(callback) {
|
|
cb = callback;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* waitLeisure:
|
|
*
|
|
* Used within an automation script to pause the the execution of the
|
|
* current script until the shell is completely idle. Use as
|
|
* 'yield Scripting.waitLeisure();'
|
|
*/
|
|
function waitLeisure() {
|
|
let cb;
|
|
|
|
global.run_at_leisure(function() {
|
|
if (cb)
|
|
cb();
|
|
});
|
|
|
|
return function(callback) {
|
|
cb = callback;
|
|
};
|
|
}
|
|
|
|
const PerfHelperIface = {
|
|
name: 'org.gnome.Shell.PerfHelper',
|
|
methods: [{ name: 'CreateWindow', inSignature: 'iibb', outSignature: '' },
|
|
{ name: 'WaitWindows', inSignature: '', outSignature: '' },
|
|
{ name: 'DestroyWindows', inSignature: '', outSignature: ''}]
|
|
};
|
|
|
|
const PerfHelper = function () {
|
|
this._init();
|
|
};
|
|
|
|
PerfHelper.prototype = {
|
|
_init: function() {
|
|
DBus.session.proxifyObject(this, 'org.gnome.Shell.PerfHelper', '/org/gnome/Shell/PerfHelper');
|
|
}
|
|
};
|
|
|
|
DBus.proxifyPrototype(PerfHelper.prototype, PerfHelperIface);
|
|
|
|
let _perfHelper = null;
|
|
function _getPerfHelper() {
|
|
if (_perfHelper == null)
|
|
_perfHelper = new PerfHelper();
|
|
|
|
return _perfHelper;
|
|
}
|
|
|
|
/**
|
|
* createTestWindow:
|
|
* @width: width of window, in pixels
|
|
* @height: height of window, in pixels
|
|
* @alpha: whether the window should be alpha transparent
|
|
* @maximized: whethe the window should be created maximized
|
|
*
|
|
* Creates a window using gnome-shell-perf-helper for testing purposes.
|
|
* While this function can be used with yield in an automation
|
|
* script to pause until the D-Bus call to the helper process returns,
|
|
* because of the normal X asynchronous mapping process, to actually wait
|
|
* until the window has been mapped and exposed, use waitTestWindows().
|
|
*/
|
|
function createTestWindow(width, height, alpha, maximized) {
|
|
let cb;
|
|
let perfHelper = _getPerfHelper();
|
|
|
|
perfHelper.CreateWindowRemote(width, height, alpha, maximized,
|
|
function(result, excp) {
|
|
if (cb)
|
|
cb();
|
|
});
|
|
|
|
return function(callback) {
|
|
cb = callback;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* waitTestWindows:
|
|
*
|
|
* Used within an automation script to pause until all windows previously
|
|
* created with createTestWindow have been mapped and exposed.
|
|
*/
|
|
function waitTestWindows() {
|
|
let cb;
|
|
let perfHelper = _getPerfHelper();
|
|
|
|
perfHelper.WaitWindowsRemote(function(result, excp) {
|
|
if (cb)
|
|
cb();
|
|
});
|
|
|
|
return function(callback) {
|
|
cb = callback;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* destroyTestWindows:
|
|
*
|
|
* Destroys all windows previously created with createTestWindow().
|
|
* While this function can be used with yield in an automation
|
|
* script to pause until the D-Bus call to the helper process returns,
|
|
* this doesn't guarantee that Mutter has actually finished the destroy
|
|
* process because of normal X asynchronicity.
|
|
*/
|
|
function destroyTestWindows() {
|
|
let cb;
|
|
let perfHelper = _getPerfHelper();
|
|
|
|
perfHelper.DestroyWindowsRemote(function(result, excp) {
|
|
if (cb)
|
|
cb();
|
|
});
|
|
|
|
return function(callback) {
|
|
cb = callback;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* defineScriptEvent
|
|
* @name: The event will be called script.<name>
|
|
* @description: Short human-readable description of the event
|
|
*
|
|
* Convenience function to define a zero-argument performance event
|
|
* within the 'script' namespace that is reserved for events defined locally
|
|
* within a performance automation script
|
|
*/
|
|
function defineScriptEvent(name, description) {
|
|
Shell.PerfLog.get_default().define_event("script." + name,
|
|
description,
|
|
"");
|
|
}
|
|
|
|
/**
|
|
* scriptEvent
|
|
* @name: Name registered with defineScriptEvent()
|
|
*
|
|
* Convenience function to record a script-local performance event
|
|
* previously defined with defineScriptEvent
|
|
*/
|
|
function scriptEvent(name) {
|
|
Shell.PerfLog.get_default().event("script." + name);
|
|
}
|
|
|
|
/**
|
|
* collectStatistics
|
|
*
|
|
* Convenience function to trigger statistics collection
|
|
*/
|
|
function collectStatistics() {
|
|
Shell.PerfLog.get_default().collect_statistics();
|
|
}
|
|
|
|
function _step(g, finish, onError) {
|
|
try {
|
|
let waitFunction = g.next();
|
|
waitFunction(function() {
|
|
_step(g, finish, onError);
|
|
});
|
|
} catch (err if err instanceof StopIteration) {
|
|
if (finish)
|
|
finish();
|
|
} catch (err) {
|
|
if (onError)
|
|
onError(err);
|
|
}
|
|
}
|
|
|
|
function _collect(scriptModule, outputFile) {
|
|
let eventHandlers = {};
|
|
|
|
for (let f in scriptModule) {
|
|
let m = /([A-Za-z]+)_([A-Za-z]+)/.exec(f);
|
|
if (m)
|
|
eventHandlers[m[1] + "." + m[2]] = scriptModule[f];
|
|
}
|
|
|
|
Shell.PerfLog.get_default().replay(
|
|
function(time, eventName, signature, arg) {
|
|
if (eventName in eventHandlers)
|
|
eventHandlers[eventName](time, arg);
|
|
});
|
|
|
|
if ('finish' in scriptModule)
|
|
scriptModule.finish();
|
|
|
|
if (outputFile) {
|
|
let f = Gio.file_new_for_path(outputFile);
|
|
let raw = f.replace(null, false,
|
|
Gio.FileCreateFlags.NONE,
|
|
null);
|
|
let out = Gio.BufferedOutputStream.new_sized (raw, 4096);
|
|
Shell.write_string_to_stream (out, "{\n");
|
|
|
|
Shell.write_string_to_stream(out, '"events":\n');
|
|
Shell.PerfLog.get_default().dump_events(out);
|
|
|
|
let monitors = Main.layoutManager.monitors;
|
|
let primary = Main.layoutManager.primaryIndex;
|
|
Shell.write_string_to_stream(out, ',\n"monitors":\n[');
|
|
for (let i = 0; i < monitors.length; i++) {
|
|
let monitor = monitors[i];
|
|
if (i != 0)
|
|
Shell.write_string_to_stream(out, ', ');
|
|
Shell.write_string_to_stream(out, '"%s%dx%d+%d+%d"'.format(i == primary ? "*" : "",
|
|
monitor.width, monitor.height,
|
|
monitor.x, monitor.y));
|
|
}
|
|
Shell.write_string_to_stream(out, ' ]');
|
|
|
|
Shell.write_string_to_stream(out, ',\n"metrics":\n[ ');
|
|
let first = true;
|
|
for (let name in scriptModule.METRICS) {
|
|
let metric = scriptModule.METRICS[name];
|
|
// Extra checks here because JSON.stringify generates
|
|
// invalid JSON for undefined values
|
|
if (metric.description == null) {
|
|
log("Error: No description found for metric " + name);
|
|
continue;
|
|
}
|
|
if (metric.units == null) {
|
|
log("Error: No units found for metric " + name);
|
|
continue;
|
|
}
|
|
if (metric.value == null) {
|
|
log("Error: No value found for metric " + name);
|
|
continue;
|
|
}
|
|
|
|
if (!first)
|
|
Shell.write_string_to_stream(out, ',\n ');
|
|
first = false;
|
|
|
|
Shell.write_string_to_stream(out,
|
|
'{ "name": ' + JSON.stringify(name) + ',\n' +
|
|
' "description": ' + JSON.stringify(metric.description) + ',\n' +
|
|
' "units": ' + JSON.stringify(metric.units) + ',\n' +
|
|
' "value": ' + JSON.stringify(metric.value) + ' }');
|
|
}
|
|
Shell.write_string_to_stream(out, ' ]');
|
|
|
|
Shell.write_string_to_stream (out, ',\n"log":\n');
|
|
Shell.PerfLog.get_default().dump_log(out);
|
|
|
|
Shell.write_string_to_stream (out, '\n}\n');
|
|
out.close(null);
|
|
} else {
|
|
let metrics = [];
|
|
for (let metric in scriptModule.METRICS)
|
|
metrics.push(metric);
|
|
|
|
metrics.sort();
|
|
|
|
print ('------------------------------------------------------------');
|
|
for (let i = 0; i < metrics.length; i++) {
|
|
let metric = metrics[i];
|
|
print ('# ' + scriptModule.METRIC_DESCRIPTIONS[metric]);
|
|
print (metric + ': ' + scriptModule.METRICS[metric]);
|
|
}
|
|
print ('------------------------------------------------------------');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* runPerfScript
|
|
* @scriptModule: module object with run and finish functions
|
|
* and event handlers
|
|
*
|
|
* Runs a script for automated collection of performance data. The
|
|
* script is defined as a Javascript module with specified contents.
|
|
*
|
|
* First the run() function within the module will be called as a
|
|
* generator to automate a series of actions. These actions will
|
|
* trigger performance events and the script can also record its
|
|
* own performance events.
|
|
*
|
|
* Then the recorded event log is replayed using handler functions
|
|
* within the module. The handler for the event 'foo.bar' is called
|
|
* foo_bar().
|
|
*
|
|
* Finally if the module has a function called finish(), that will
|
|
* be called.
|
|
*
|
|
* The event handler and finish functions are expected to fill in
|
|
* metrics to an object within the module called METRICS. Each
|
|
* property of this object represents an individual metric. The
|
|
* name of the property is the name of the metric, the value
|
|
* of the property is an object with the following properties:
|
|
*
|
|
* description: human readable description of the metric
|
|
* units: a string representing the units of the metric. It has
|
|
* the form '<unit> <unit> ... / <unit> / <unit> ...'. Certain
|
|
* unit values are recognized: s, ms, us, B, KiB, MiB. Other
|
|
* values can appear but are uninterpreted. Examples 's',
|
|
* '/ s', 'frames', 'frames / s', 'MiB / s / frame'
|
|
* value: computed value of the metric
|
|
*
|
|
* The resulting metrics will be written to @outputFile as JSON, or,
|
|
* if @outputFile is not provided, logged.
|
|
*
|
|
* After running the script and collecting statistics from the
|
|
* event log, GNOME Shell will exit.
|
|
**/
|
|
function runPerfScript(scriptModule, outputFile) {
|
|
Shell.PerfLog.get_default().set_enabled(true);
|
|
|
|
let g = scriptModule.run();
|
|
|
|
_step(g,
|
|
function() {
|
|
_collect(scriptModule, outputFile);
|
|
Meta.exit(Meta.ExitCode.SUCCESS);
|
|
},
|
|
function(err) {
|
|
log("Script failed: " + err + "\n" + err.stack);
|
|
Meta.exit(Meta.ExitCode.ERROR);
|
|
});
|
|
}
|