<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=1.0">
<title>SECoP web client</title>
<link rel="shortcut icon" href="">
<link rel="stylesheet" href="https://cdn.datatables.net/1.13.7/css/jquery.dataTables.min.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bulma@0.9.4/css/bulma.min.css">
<style type="text/css">
table.accessibles td { border: 0; padding-left: 0 !important; min-width: 7em; }
div.notif-cont { display: none; position: absolute; top: 2em; z-index: 100;
width: 100%; max-width: 100% !important; }
div.notif-cont div.notification { display: inline-block; }
</style>
<script src="https://cdn.jsdelivr.net/npm/jquery@3.7.1/dist/jquery.min.js"></script>
<script src="https://cdn.datatables.net/1.13.7/js/jquery.dataTables.min.js"></script>
<script src="https://cdn.plot.ly/plotly-2.27.0.min.js" charset="utf-8"></script>
<script src="https://use.fontawesome.com/releases/v6.4.0/js/all.js"></script>
<script type="text/javascript">
const msg_rx = /^([a-z_]+)(?: ([\w.]+)(?::(\w+))?(?: (.*))?)?/;
var ws = null;
var ws2 = null;
var ws2_initmsg = null;
var dtable = null;
var description = null;
var plots = [];
var enums = {};
function do_connect() {
let host = $('#connect-host')[0].value;
let port = $('#connect-port')[0].value;
ws = null;
ws2 = null;
dtable = null;
description = null;
plots = [];
enums = {};
$('#connect').slideUp();
$('#disconnect').show();
$('#node-address').text('?');
$('#node-id').text('?');
$('#full-desc').text('');
$('#main').show();
$('#error').hide();
$('#console').val('');
$('#console-activator').show();
$('#console-block').hide();
let addr = `ws://${host}:${port}`;
ws = new WebSocket(addr);
ws.addEventListener('open', on_connect);
ws.addEventListener('close', on_disconnect);
ws.addEventListener('message', on_message);
$('#node-address').text(addr);
}
function do_disconnect() {
$('#main').hide();
ws.close(3124); // special code signals that we requested the closure
}
function on_connect(event) {
ws.send('*IDN?');
}
function on_disconnect(event) {
if (event.code != 3124) {
msg = {
1000: 'normal closure',
1001: 'going away',
1002: 'protocol error',
1003: 'unsupported data',
1004: '(reserved)',
1005: 'no status',
1006: 'connection failed or broken',
1007: 'invalid payload',
1008: 'policy violation',
1009: 'message too big',
1010: 'extension required',
1011: 'internal server error',
1012: 'service restart',
1013: 'try again later',
1014: 'bad gateway',
1015: 'TLS handshake',
}[event.code] || 'unknown';
show_error(`disconnected (${msg})`);
}
$('#main').hide();
$('#connect').slideDown();
$('#disconnect').hide();
if (dtable) {
dtable.destroy();
$('#tablebody tr').remove();
}
if (plots) {
let plot_el = $('#plot')[0];
Plotly.purge(plot_el);
}
}
function on_message(event) {
let msg = event.data;
if (msg.startsWith('ISSE')) {
ws.send('describe');
return;
}
let parts = msg_rx.exec(msg);
if (parts === null) return;
if (parts[1].startsWith('error_')) {
let [typ, args, _] = JSON.parse(parts[4]);
show_error(typ + ': ' + args);
return;
}
switch (parts[1]) {
case 'describing':
description = JSON.parse(parts[4]);
handle_desc(description);
ws.send('activate');
break;
case 'update':
let data = JSON.parse(parts[4]);
update_param(parts[2], parts[3], data[0], data[1]['t']);
break;
case 'done':
let result = JSON.parse(parts[4]);
if (result[0] !== null) {
$('#resultmsg').text(result[0]);
$('#result').show();
}
break;
case 'active':
case 'changed':
break;
default:
console.log(parts);
console.error('unhandled: ' + msg);
}
}
function show_error(err) {
$('#errormsg').text(err);
$('#error').show();
}
function handle_desc(desc) {
document.title = desc.description.split('\n')[0];
$('#full-desc').html(desc.description.replaceAll('\n', '<br>'));
$('#node-id').text(desc.equipment_id);
for (const mod in desc.modules) {
let moddesc = desc.modules[mod];
let accdesc = desc.modules[mod].accessibles;
let mainunit = accdesc.value ? (accdesc.value.datainfo.unit || '') : '';
let descr = moddesc.description;
if ((moddesc.interface_classes || []).length > 0) {
descr += '\nInterface: ' + moddesc.interface_classes.join(', ');
}
if (moddesc.implementation) {
descr += '\nImpl: ' + moddesc.implementation;
}
let symbol = 'plug';
if (moddesc.interface_classes.includes('Drivable')) {
symbol = 'car-side';
} else if (moddesc.interface_classes.includes('Readable')) {
symbol = 'book';
}
var row = $('<tr>');
row.append($('<td>').append(
$(`<i class="fas fa-${symbol} fa-fw has-text-grey">`),
' ',
$('<b>').attr('title', moddesc.description).text(mod),
$('<span class="is-pulled-right"><i class="fas fa-circle-info has-text-grey"></span>').attr('title', descr),
));
// handle status
let sts = $('<td>').attr('id', mod + '-s');
if (accdesc.status !== undefined) {
sts.append(
$(`<i class="fas fa-check" style="display: none" id="${mod}-simg-idle">`),
$(`<i class="fas fa-gears" style="display: none" id="${mod}-simg-busy">`),
$(`<i class="fas fa-triangle-exclamation" style="display: none" id="${mod}-simg-err">`),
' ',
$('<span>').attr('id', mod + '-sval')
);
}
row.append(sts);
// handle current value for Readables
let val = $('<td>');
if (accdesc.value !== undefined) {
val.append(
$('<span>').attr('id', mod + '-v'),
' ',
$('<span class="has-text-grey">').text(mainunit),
$('<a class="is-pulled-right"><i class="fas fa-chart-line"></a>').on('click', () => {
add_plot(mod);
}),
);
}
row.append(val);
// handle target for Drivables
let tgt = $('<td>');
if (accdesc.target !== undefined) {
tgt.append(
$('<span>').attr('id', mod + '-t'),
' ',
$('<span class="has-text-grey">').text(
(accdesc.target.datainfo.unit || '').replace('$', mainunit)),
);
}
row.append(tgt);
// add input field and "go" button for the new value
let move = $('<td>');
if (accdesc.target !== undefined) {
move.append(
$(`<input type=text size=15 id="${mod}-tset">`).on('keypress', (ev) => {
if (ev.key == 'Enter') {
$(`#${mod}-tset-btn`).click();
}
}),
' ',
$(`<a id="${mod}-tset-btn"><i class="fas fa-circle-play ` +
'has-text-success-dark"></a>').on('click', () => {
make_change('change', mod, 'target', $(`#${mod}-tset`)[0].value)
}),
);
}
// add the stop button here
if (accdesc.stop !== undefined) {
move.append(
' ',
$('<a><i class="fas fa-circle-stop has-text-danger"></a>')
.on('click', () => { ws.send(`do ${mod}:stop`); }),
);
}
row.append(move);
// handle accessibles
let accs = $('<td>');
let num_accs = 0;
let acc_table = $(`<table class="accessibles" id="${mod}-acc-tbl" style="display: none">`);
for (const acc in accdesc) {
let full = mod + '-' + acc;
if (accdesc[acc].datainfo.type == 'enum') {
enums[full] = [accdesc[acc].datainfo.members, {}];
for (const [name, val] of Object.entries(accdesc[acc].datainfo.members)) {
enums[full][1][val] = name;
}
}
if (acc == 'value' || acc == 'status' || acc == 'target' || acc == 'stop') {
continue;
}
num_accs += 1;
let dispname = acc.startsWith('_') ? acc.substr(1) : acc;
// for commands, show a button and an input field for the argument
if (accdesc[acc].datainfo.type == 'command') {
acc_table.append($('<tr>').append(
$('<td>').append(
$(`<button id="${full}-do-btn">${dispname}</button>`).on('click', () => {
let field = $(`#${full}-do-arg`);
let arg = field.length ? field[0].value : '';
make_change('do', mod, acc, arg);
}),
),
$('<td>'),
accdesc[acc].datainfo.argument ?
$('<td>').append(
$(`<input type=text size=20 id="${full}-do-arg">`).on('keypress', (ev) => {
if (ev.key == 'Enter') {
$(`#${full}-do-btn`).click();
}
})
)
: $('<td>')
));
continue;
}
// for parameters, show the name, current value, and an input field to set it
acc_table.append($('<tr>').append(
$('<td>').text(dispname),
$('<td>').append(
$('<span>').attr('id', full + '-v'),
' ',
$('<span class="has-text-grey">').text(
(accdesc[acc].datainfo.unit || '').replace('$', mainunit))
),
accdesc[acc].readonly ? $('<td>') :
$('<td>').append(
$(`<input type=text size=20 id="${full}-vset">`).on('keypress', (ev) => {
if (ev.key == 'Enter') {
$(`#${full}-vset-btn`).click();
}
}),
' ',
$(`<a id="${full}-vset-btn"><i class="fas fa-circle-right ` +
'has-text-success-dark"></a>').on('click', () => {
make_change('change', mod, acc, $(`#${full}-vset`)[0].value);
}),
)
));
}
if (num_accs > 2) {
accs.append(
$(`<a id="${mod}-show-btn">Show <i class="fas fa-chevron-down ` +
'has-text-link-dark"></a>').on('click', () => {
$(`#${mod}-hide-btn`).show();
$(`#${mod}-show-btn`).hide();
$(`#${mod}-acc-tbl`).show();
}),
$(`<a id="${mod}-hide-btn" style="display: none">Hide <i class="fas fa-chevron-up ` +
'has-text-link-dark"></a>').on('click', () => {
$(`#${mod}-hide-btn`).hide();
$(`#${mod}-show-btn`).show();
$(`#${mod}-acc-tbl`).hide();
}),
acc_table
);
} else if (num_accs > 0) {
accs.append(acc_table.attr('style', ''));
}
row.append(accs);
$('#tablebody').append(row);
}
dtable = $('#table').DataTable({
'pageLength': 25,
'order': [[0, 'asc']],
'autoWidth': false,
'columns': [
{'width': '15%'},
{'width': '13%'},
{'width': '15%'},
{'width': '10%'},
{'width': '10%'},
{'width': '33%'},
]
});
}
function make_change(verb, mod, acc, val) {
try {
val = JSON.parse(val);
} catch (err) {
show_error('Please enter valid JSON');
return;
}
ws.send(`${verb} ${mod}:${acc} ${JSON.stringify(val)}`);
}
function add_plot(mod) {
let plot_el = $('#plot')[0];
if (plots.length == 0) {
// initialize plot div
Plotly.newPlot(plot_el, [], {
margin: {t: 0},
xaxis: {title: 'Time', zeroline: false},
yaxis: {title: 'Value'},
showlegend: true,
});
}
if (!plots.includes(mod)) {
Plotly.addTraces(plot_el, {x: [], y: [], name: mod});
plots.push(mod);
}
}
function update_param(mod, par, val, t) {
// handle enum conversion
let full = mod + '-' + par;
if (Object.hasOwn(enums, full)) {
let name = enums[full][1][val];
if (name !== undefined) val = name;
}
switch (par) {
case 'value':
if (plots.includes(mod)) {
let plot_el = $('#plot')[0];
let date = new Date(1000 * t);
let n = plots.indexOf(mod);
Plotly.extendTraces(plot_el, {x: [[date]], y: [[val]]}, [n], 1000);
}
if (typeof val === 'number') {
val = val.toLocaleString();
}
$(`#${mod}-v`).text(val);
break;
case 'target':
$(`#${mod}-t`).text(val);
break;
case 'status':
let [cst, text] = val;
let color = 'is-danger';
let image = 'err';
if (100 <= cst && cst < 200) {
color = 'is-success';
image = 'idle';
} else if (200 <= cst && cst < 400) {
color = 'is-warning';
image = 'busy';
}
$(`#${mod}-sval`).text(text);
$(`#${mod}-simg-idle`).hide();
$(`#${mod}-simg-busy`).hide();
$(`#${mod}-simg-err`).hide();
$(`#${mod}-simg-${image}`).show();
$(`#${mod}-s`).attr('class', color);
break;
default:
if (typeof val === 'number') {
val = val.toLocaleString();
}
$(`#${full}-v`).text(val);
}
}
function show_console() {
$('#console-activator').hide();
$('#console-block').show();
$('#console-input').focus();
}
function console_send() {
let msg = $('#console-input')[0].value;
if (!ws2) {
ws2 = new WebSocket(ws.url);
ws2_initmsg = msg;
ws2.addEventListener('open', on_console_init);
ws2.addEventListener('message', on_console_message);
} else {
ws2.send(msg);
}
add_console_out('> ' + msg);
$('#console-input')[0].value = '';
}
function on_console_init() {
ws2.send(ws2_initmsg);
}
function on_console_message(event) {
add_console_out('< ' + event.data);
}
function add_console_out(text) {
let console = $('#console')[0];
console.value = console.value + text + '\n';
console.scrollTop = console.scrollHeight;
}
$(document).ready(() => {
$('#connect-host')[0].value = document.location.hostname;
$('#connect-port')[0].value = document.location.port;
});
</script>
</head>
<body>
<nav class="navbar has-shadow has-background-primary-light" role="navigation">
<div class="container">
<div class="navbar-brand">
<span class="navbar-item">
<img src="" />
node <em id="node-id">?</em> at <em id="node-address">?</em>
</span>
</div>
<div class="navbar-end">
<div class="navbar-item">provided by Frappy</div>
</div>
</div>
</nav>
<main>
<section id="connect" class="section">
<div class="columns">
<div class="column is-one-quarter"></div>
<div class="column has-text-centered">
<h2 class="subtitle is-medium">Connect to SECoP node</h2>
<div class="field is-horizontal">
<div class="field-label is-normal"><label class="label">Host:</label></div>
<div class="field-body">
<div class="field">
<input class="input" type="text" id="connect-host" size="20" value="localhost"
onkeypress="if (event.key == 'Enter') do_connect()">
</div>
</div>
</div>
<div class="field is-horizontal">
<div class="field-label is-normal"><label class="label">Port:</label></div>
<div class="field-body">
<div class="field">
<input class="input" type="text" id="connect-port" size="10" value="10767"
onkeypress="if (event.key == 'Enter') do_connect()">
</div>
</div>
</div>
<div class="field">
<button class="button is-link" onclick="do_connect()">Connect</button>
</div>
</div>
<div class="column is-one-quarter"></div>
</div>
</section>
<section id="main" class="section" style="display: none">
<div class="container is-fluid">
<p class="block"><span id="full-desc"></span></p>
<table class="table is-striped is-narrow is-hoverable is-fullwidth" id="table">
<thead>
<tr>
<th>Module</th>
<th>Status</th>
<th>Value</th>
<th>Target</th>
<th> </th>
<th>Accessibles</th>
</tr>
</thead>
<tbody id="tablebody">
</tbody>
</table>
</div>
<div class="container" id="plot">
</div>
<div id="console-activator" class="block has-text-centered">
<a onclick="show_console()">Show console</a>
</div>
<div id="console-block" class="block has-text-centered" style="display: none">
<textarea id="console" cols="100" rows="20"></textarea>
<br>
<input id="console-input" type="text" size="100"
onkeypress="if (event.key == 'Enter') console_send()">
<button onclick="console_send()">Send</button>
</div>
</section>
<section id="disconnect" style="display: none">
<div class="container has-text-centered">
<div class="field">
<button class="button is-warning" onclick="do_disconnect()">Disconnect</button>
</div>
</div>
</section>
</main>
<div class="container has-text-centered notif-cont" id="error">
<div class="notification is-danger">
<button class="delete" onclick="$('#error').hide()"></button>
Error: <span id="errormsg"></span>
</div>
</div>
<div class="container has-text-centered notif-cont" id="result">
<div class="notification is-success">
<button class="delete" onclick="$('#result').hide()"></button>
Result: <span id="resultmsg"></span>
</div>
</div>
</body>
</html>