Skip to content

Commit

Permalink
Web Dashboard: added a page for displaying query progress at Czar
Browse files Browse the repository at this point in the history
Added a link (an additional column on the active queries table) to
the newely added Query progression plot.
Fixed minor layout bugs on the active and past query tables.
Loading the Highcharts.js library on demand.
  • Loading branch information
iagaponenko committed Sep 21, 2023
1 parent 37198d9 commit dc3b3b3
Show file tree
Hide file tree
Showing 6 changed files with 388 additions and 9 deletions.
2 changes: 2 additions & 0 deletions src/www/dashboard.html
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,9 @@
<title>Qserv monitoring dashboard</title>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">

<script data-main="qserv/js/QservMonitoringDashboard.js?bust=69" src="https://requirejs.org/docs/release/2.3.6/minified/require.js"></script>

</head>
<body></body>
</html>
60 changes: 60 additions & 0 deletions src/www/qserv/css/QservCzarQueryProgress.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
#fwk-qserv-czar-query-prog-controls label {
font-weight: bold;
}
table#fwk-qserv-czar-query-prog-status {
margin:0;
}
table#fwk-qserv-czar-query-prog-status caption {
caption-side: top;
text-align: right;
padding-top: 0;
}
table#fwk-qserv-czar-query-prog-status caption.updating {
background-color: #ffeeba;
}

#fwk-qserv-czar-query-prog-status-queries {
height: 712px;
}
.highcharts-figure,
.highcharts-data-table table {
min-width: 310px;
max-width: 800px;
margin: 1em auto;
}

.highcharts-data-table table {
font-family: Verdana, sans-serif;
border-collapse: collapse;
border: 1px solid #ebebeb;
margin: 10px auto;
text-align: center;
width: 100%;
max-width: 500px;
}

.highcharts-data-table caption {
padding: 1em 0;
font-size: 1.2em;
color: #555;
}

.highcharts-data-table th {
font-weight: 600;
padding: 0.5em;
}

.highcharts-data-table td,
.highcharts-data-table th,
.highcharts-data-table caption {
padding: 0.5em;
}

.highcharts-data-table thead tr,
.highcharts-data-table tr:nth-child(even) {
background: #f8f8f8;
}

.highcharts-data-table tr:hover {
background: #f1f7ff;
}
301 changes: 301 additions & 0 deletions src/www/qserv/js/QservCzarQueryProgress.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,301 @@
define([
'webfwk/CSSLoader',
'webfwk/Fwk',
'webfwk/FwkApplication',
'qserv/Common',
'underscore',
'highcharts',
'highcharts/modules/exporting',
'highcharts/modules/accessibility'],

function(CSSLoader,
Fwk,
FwkApplication,
Common,
_,
Highcharts) {

CSSLoader.load('qserv/css/QservCzarQueryProgress.css');

class QservCzarQueryProgress extends FwkApplication {

constructor(name) {
super(name);
this._data = undefined;
this._queries_chart = undefined;
}
fwk_app_on_show() {
console.log('show: ' + this.fwk_app_name);
this.fwk_app_on_update();
}
fwk_app_on_hide() {
console.log('hide: ' + this.fwk_app_name);
}
fwk_app_on_update() {
if (this.fwk_app_visible) {
this._init();
if (this._prev_update_sec === undefined) {
this._prev_update_sec = 0;
}
let now_sec = Fwk.now().sec;
if (now_sec - this._prev_update_sec > this._update_interval_sec()) {
this._prev_update_sec = now_sec;
this._init();
this._load();
}
}
}
/// Set the identifier and begin loading the query info in the background.
set_query_id(query_id) {
this._init();
this._set_query_ids([query_id]); // To get the minimally-polulated selector
this._set_query_id(query_id);
this._set_last_seconds(24 * 3600); // Track the known history (if any) of the query
this._load();
}
_init() {
if (this._initialized === undefined) this._initialized = false;
if (this._initialized) return;
this._initialized = true;
const lastMinutes = [1, 3, 5, 15, 30, 45];
const lastHours = [1, 2, 4, 8, 12, 16, 20, 24];
let html = `
<div class="row" id="fwk-qserv-czar-query-prog-controls">
<div class="col">
<div class="form-row">
<div class="form-group col-md">
<label for="query-id">Query Id:</label>
<select id="query-id" class="form-control form-control-selector">
<option value="0" selected>&lt;any&gt;</option>
</select>
</div>
<div class="form-group col-md">
<label for="last-seconds">Track last:</label>
<select id="last-seconds" class="form-control form-control-selector">` + _.reduce(lastMinutes, function (html, m) { return html + `
<option value="${m * 60}">${m} min</option>`; }, '') + _.reduce(lastHours, function (html, hr) { return html + `
<option value="${hr * 3600}">${hr} hr</option>`; }, '') + `
</select>
</div>
<div class="form-group col-md"
title="Zero values are not plotted in the logarithmic scale"
data-toggle="tooltip"
data-placement="top">
<label for="vertical-scale">Vertical scale:</label>
<select id="vertical-scale" class="form-control form-control-viewer">
<option selected value="logarithmic">logarithmic</option>
<option value="linear">linear</option>
</select>
</div>
<div class="form-group col-md"
title="Enabling auto-zoom would expand the plot all the way to the right"
data-toggle="tooltip"
data-placement="top">
<label for="horizontal-scale">Horizontal scale:</label>
<select id="horizontal-scale" class="form-control form-control-viewer">
<option selected value="">&nbsp;</option>
<option value="auto-zoom-in">auto-zoom-in</option>
</select>
</div>
<div class="form-group col-md">
${Common.html_update_ival('update-interval', 10)}
</div>
<div class="form-group col-md">
<label for="reset-form">&nbsp;</label>
<button id="reset-form" class="btn btn-primary form-control">Reset</button>
</div>
</div>
</div>
</div>
<div class="row">
<div class="col">
<table class="table table-sm" id="fwk-qserv-czar-query-prog-status">
<caption class="updating">Loading...</caption>
</table>
</div>
</div>
<div class="row">
<div class="col">
<div id="fwk-qserv-czar-query-prog-status-queries"></div>
</div>
</div>
`;
let cont = this.fwk_app_container.html(html);
cont.find('[data-toggle="tooltip"]').tooltip();
this._set_last_seconds(900);
cont.find(".form-control-selector").change(() => {
this._load();
});
cont.find(".form-control-viewer").change(() => {
if (_.isUndefined(this._data)) this._load();
else this._display(this._data.queries);
});
cont.find("button#reset-form").click(() => {
this._set_update_interval_sec(10);
this._set_query_id(0);
this._set_last_seconds(15 * 60);
this._set_vertical_scale('logarithmic');
this._set_horizontal_scale('');
this._load();
});
}
_form_control(elem_type, id) {
if (this._form_control_obj === undefined) this._form_control_obj = {};
if (!_.has(this._form_control_obj, id)) {
this._form_control_obj[id] = this.fwk_app_container.find(elem_type + '#' + id);
}
return this._form_control_obj[id];
}
_update_interval_sec() { return this._form_control('select', 'update-interval').val(); }
_set_update_interval_sec(val) { this._form_control('select', 'update-interval').val(val); }
_query_id() { return this._form_control('select', 'query-id').val(); }
_set_query_id(val) {
this._form_control('select', 'query-id').val(val);
}
_set_query_ids(queries) {
const prev_query = this._query_id();
let html = '<option value="0" selected>&lt;any&gt;</option>';
for (let i in queries) {
const query = queries[i];
const selected = (_.isEmpty(prev_query) && (i === 0)) ||
(!_.isEmpty(prev_query) && (prev_query === query));
html += `
<option value="${query}" ${selected ? "selected" : ""}>${query}</option>`;
}
this._form_control('select', 'query-id').html(html);
}
_vertical_scale() { return this._form_control('select', 'vertical-scale').val(); }
_set_vertical_scale(val) { this._form_control('select', 'vertical-scale').val(val); }
_horizontal_scale() { return this._form_control('select', 'horizontal-scale').val(); }
_set_horizontal_scale(val) { this._form_control('select', 'horizontal-scale').val(val); }
_last_seconds() { return this._form_control('select', 'last-seconds').val(); }
_set_last_seconds(val) { this._form_control('select', 'last-seconds').val(val); }
_table(name) {
if (_.isUndefined(this._table_obj)) this._table_obj = {};
if (!_.has(this._table_obj, name)) {
this._table_obj[name] = this.fwk_app_container.find('table#fwk-qserv-czar-query-prog-' + name);
}
return this._table_obj[name];
}
_status() {
if (_.isUndefined(this._status_obj)) {
this._status_obj = this._table('status').children('caption');
}
return this._status_obj;
}
_queries() {
if (_.isUndefined(this._queries_obj)) {
this._queries_obj = this.fwk_app_container.find('canvas#queries');
}
return this._queries_obj;
}
_load() {
if (this._loading === undefined) this._loading = false;
if (this._loading) return;
this._loading = true;
this._status().addClass('updating');
Fwk.web_service_GET(
"replication/qserv/master/queries/active/progress",
{ version: Common.RestAPIVersion,
query_id: this._query_id(),
last_seconds: this._last_seconds()
},
(data) => {
if (data.success) {
this._data = data;
this._display(data.queries);
Fwk.setLastUpdate(this._status());
} else {
console.log('request failed', this.fwk_app_name, data.error);
this._status().html('<span style="color:maroon">' + data.error + '</span>');
}
this._status().removeClass('updating');
this._loading = false;
},
(msg) => {
console.log('request failed', this.fwk_app_name, msg);
this._status().html('<span style="color:maroon">No Response</span>');
this._status().removeClass('updating');
this._loading = false;
}
);
}
_display(queries) {
const query_ids = _.keys(queries);
query_ids.sort();
query_ids.reverse();
this._set_query_ids(query_ids); // Update a collection of queries in the selector.
// Add a small delta to the points to allow seeing zeroes on the log scale,
// in case the one was requested.
const valueDeltaForLogScale = this._vertical_scale() === 'linear' ? 0 : 0.1;
let series = [];
for (let qid in queries) {
let points = [];
let query_data = queries[qid];
for (let i in query_data) {
const point = query_data[i];
// +1 hr is needed for correcting timestamp mismatch between UNIX and JS timing
const timestampSec = point[0] / 1000 + 3600;
let x = new Date(0);
x.setSeconds(timestampSec);
points.push([x.getTime(), point[1] + valueDeltaForLogScale]);
}
series.push({
name: qid,
data: points,
animation: {
enabled: false
}
});
}
if (!_.isUndefined(this._queries_chart)) {
this._queries_chart.destroy();
}
this._queries_chart = Highcharts.chart('fwk-qserv-czar-query-prog-status-queries', {
chart: {
type: 'line'
},
title: {
text: '# Unfinished Jobs'
},
subtitle: {
text: '< 24 hours'
},
xAxis: {
type: 'datetime',
title: {
text: 'Time'
},
// If auto-zoom is not enabled the plot will go all the way through
// the (viewer's) current time on the right.
max: this._horizontal_scale() === 'auto-zoom-in' ? undefined : new Date().setSeconds(0)
},
yAxis: {
type: this._vertical_scale(),
title: {
text: '# jobs'
}
},
tooltip: {
headerFormat: '<b>{series.name}</b><br>',
pointFormat: '{point.x:%e. %b}: {point.y:.2f} jobs'
},
time: {
// To ensure the time stamps are displaye din the (viewer's) local timezone.
timezoneOffset: new Date().getTimezoneOffset()
},
plotOptions: {
series: {
marker: {
fillColor: '#dddddd',
lineWidth: 2,
lineColor: null
}
}
},
colors: ['#6CF', '#39F', '#06C', '#036', '#000'],
series: series
});
}
}
return QservCzarQueryProgress;
});
Loading

0 comments on commit dc3b3b3

Please sign in to comment.