* initial telemetry api updates for staleness support * modifying staleness to a subsription style * fixing variable name * debuggin * put the subscribe in the wrong place * stale class for object views * temp cyan border for testing * added staleness to swg, working on stacked plot staleness * WIP: stacked plot staleness * reverting, going a different route * staleness on stacked plots * plot legend staleness * remove debug code * staleness for alphanumerics * lad table and table set staleness * overlay plot staleness * remove debug code * hardened lad staleness functionality fixed plots without composition bug * adding staleness to gauges * renaming telemetry age check functionality so it does not conflict with new staleness functionality * couple one-off fixes here and there, and WIP for condition sets, moving to telemetry tables to facilitate styling of completed components * small fix on lad tables, added staleness functionality to tables * finishing up condition sets * some cleaning up * adding border to condition sets when an item is stale * fixing dub sub * addressing PR comment, moving repeated code to a function * robustified the SWG stalenes provider, little fixes here and there as far as cleaning up listeners and... whatnot * removing debug code * typo fixes * cleanin up debug code * created a simple stalenes mixin for more basic usage in components * more robustification, if a new staleness subscription happens, will now send the current staleness value if we have it, beefed up example stalenes swg provider * beefed up staleness mixin a bit to give it more use * copyright * cleanin up ladtable code * cleanin up ladtable code * cleaning up lad table sets * some minor updates * Closes #6109 - New staleness glyph and font CSS added. * Closes #6109 - Normalized staleness colors as theme constants. - New mixins for staleness application to view elements. - Applied staleness styling to all relevant view elements. - TODO: smoke-test in Show theme. * adding staleness utils helper, mixin and isStale functionalirty for telemtry api * Closes #6109 - Refined style for Snow theme. * need to have one domainObject per stalenes utility * making sure we handle domains correctly while dealing with staleness * couple fixes * moving abort controller logic to a spot where it makes more sense * added some more info for the StalenesProvider interface docs * returning undefinded for ifStale requests with no provider * debuggin * debuggin * missed "isStale" call in condtioncollections * removing debug code and using mixin unsubscribe in gauge * fixing tests * more targeted tree item click Co-authored-by: Charles Hacskaylo <charlesh88@gmail.com> Co-authored-by: Andrew Henry <akhenry@gmail.com> Co-authored-by: Scott Bell <scott@traclabs.com>
335 lines
9.9 KiB
JavaScript
335 lines
9.9 KiB
JavaScript
/*****************************************************************************
|
|
* Open MCT, Copyright (c) 2014-2022, United States Government
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
* Administration. All rights reserved.
|
|
*
|
|
* Open MCT is licensed under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
* http://www.apache.org/licenses/LICENSE-2.0.
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
* Open MCT includes source code licensed under additional open source
|
|
* licenses. See the Open Source Licenses file (LICENSES.md) included with
|
|
* this source code distribution or the Licensing information page available
|
|
* at runtime from the About dialog for additional information.
|
|
*****************************************************************************/
|
|
|
|
import { IS_OLD_KEY, IS_STALE_KEY } from "./constants";
|
|
|
|
function convertToNumbers(input) {
|
|
let numberInputs = [];
|
|
input.forEach(inputValue => numberInputs.push(Number(inputValue)));
|
|
|
|
return numberInputs;
|
|
}
|
|
|
|
function convertToStrings(input) {
|
|
let stringInputs = [];
|
|
input.forEach(inputValue => stringInputs.push(inputValue !== undefined ? inputValue.toString() : ''));
|
|
|
|
return stringInputs;
|
|
}
|
|
|
|
function joinValues(values, length) {
|
|
return values.slice(0, length).join(', ');
|
|
}
|
|
|
|
export const OPERATIONS = [
|
|
{
|
|
name: 'equalTo',
|
|
operation: function (input) {
|
|
return Number(input[0]) === Number(input[1]);
|
|
},
|
|
text: 'is equal to',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'notEqualTo',
|
|
operation: function (input) {
|
|
return Number(input[0]) !== Number(input[1]);
|
|
},
|
|
text: 'is not equal to',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is not ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'greaterThan',
|
|
operation: function (input) {
|
|
return Number(input[0]) > Number(input[1]);
|
|
},
|
|
text: 'is greater than',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' > ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'lessThan',
|
|
operation: function (input) {
|
|
return Number(input[0]) < Number(input[1]);
|
|
},
|
|
text: 'is less than',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' < ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'greaterThanOrEq',
|
|
operation: function (input) {
|
|
return Number(input[0]) >= Number(input[1]);
|
|
},
|
|
text: 'is greater than or equal to',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' >= ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'lessThanOrEq',
|
|
operation: function (input) {
|
|
return Number(input[0]) <= Number(input[1]);
|
|
},
|
|
text: 'is less than or equal to',
|
|
appliesTo: ['number'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' <= ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'between',
|
|
operation: function (input) {
|
|
let numberInputs = convertToNumbers(input);
|
|
let larger = Math.max(...numberInputs.slice(1, 3));
|
|
let smaller = Math.min(...numberInputs.slice(1, 3));
|
|
|
|
return (numberInputs[0] > smaller) && (numberInputs[0] < larger);
|
|
},
|
|
text: 'is between',
|
|
appliesTo: ['number'],
|
|
inputCount: 2,
|
|
getDescription: function (values) {
|
|
return ' is between ' + values[0] + ' and ' + values[1];
|
|
}
|
|
},
|
|
{
|
|
name: 'notBetween',
|
|
operation: function (input) {
|
|
let numberInputs = convertToNumbers(input);
|
|
let larger = Math.max(...numberInputs.slice(1, 3));
|
|
let smaller = Math.min(...numberInputs.slice(1, 3));
|
|
|
|
return (numberInputs[0] < smaller) || (numberInputs[0] > larger);
|
|
},
|
|
text: 'is not between',
|
|
appliesTo: ['number'],
|
|
inputCount: 2,
|
|
getDescription: function (values) {
|
|
return ' is not between ' + values[0] + ' and ' + values[1];
|
|
}
|
|
},
|
|
{
|
|
name: 'textContains',
|
|
operation: function (input) {
|
|
return input[0] && input[1] && input[0].includes(input[1]);
|
|
},
|
|
text: 'text contains',
|
|
appliesTo: ['string'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' contains ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'textDoesNotContain',
|
|
operation: function (input) {
|
|
return input[0] && input[1] && !input[0].includes(input[1]);
|
|
},
|
|
text: 'text does not contain',
|
|
appliesTo: ['string'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' does not contain ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'textStartsWith',
|
|
operation: function (input) {
|
|
return input[0].startsWith(input[1]);
|
|
},
|
|
text: 'text starts with',
|
|
appliesTo: ['string'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' starts with ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'textEndsWith',
|
|
operation: function (input) {
|
|
return input[0].endsWith(input[1]);
|
|
},
|
|
text: 'text ends with',
|
|
appliesTo: ['string'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' ends with ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'textIsExactly',
|
|
operation: function (input) {
|
|
return input[0] === input[1];
|
|
},
|
|
text: 'text is exactly',
|
|
appliesTo: ['string'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is exactly ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'isUndefined',
|
|
operation: function (input) {
|
|
return typeof input[0] === 'undefined';
|
|
},
|
|
text: 'is undefined',
|
|
appliesTo: ['string', 'number', 'enum'],
|
|
inputCount: 0,
|
|
getDescription: function () {
|
|
return ' is undefined';
|
|
}
|
|
},
|
|
{
|
|
name: 'isDefined',
|
|
operation: function (input) {
|
|
return typeof input[0] !== 'undefined';
|
|
},
|
|
text: 'is defined',
|
|
appliesTo: ['string', 'number', 'enum'],
|
|
inputCount: 0,
|
|
getDescription: function () {
|
|
return ' is defined';
|
|
}
|
|
},
|
|
{
|
|
name: 'enumValueIs',
|
|
operation: function (input) {
|
|
let stringInputs = convertToStrings(input);
|
|
|
|
return stringInputs[0] === stringInputs[1];
|
|
},
|
|
text: 'is',
|
|
appliesTo: ['enum'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'enumValueIsNot',
|
|
operation: function (input) {
|
|
let stringInputs = convertToStrings(input);
|
|
|
|
return stringInputs[0] !== stringInputs[1];
|
|
},
|
|
text: 'is not',
|
|
appliesTo: ['enum'],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is not ' + joinValues(values, 1);
|
|
}
|
|
},
|
|
{
|
|
name: 'isOneOf',
|
|
operation: function (input) {
|
|
const lhsValue = input[0] !== undefined ? input[0].toString() : '';
|
|
if (input[1]) {
|
|
const values = input[1].split(',');
|
|
|
|
return values.some((value) => lhsValue === value.toString().trim());
|
|
}
|
|
|
|
return false;
|
|
},
|
|
text: 'is one of',
|
|
appliesTo: ["string", "number"],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is one of ' + values[0];
|
|
}
|
|
},
|
|
{
|
|
name: 'isNotOneOf',
|
|
operation: function (input) {
|
|
const lhsValue = input[0] !== undefined ? input[0].toString() : '';
|
|
if (input[1]) {
|
|
const values = input[1].split(',');
|
|
const found = values.some((value) => lhsValue === value.toString().trim());
|
|
|
|
return !found;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
text: 'is not one of',
|
|
appliesTo: ["string", "number"],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ' is not one of ' + values[0];
|
|
}
|
|
},
|
|
{
|
|
name: IS_OLD_KEY,
|
|
operation: function () {
|
|
return false;
|
|
},
|
|
text: 'is older than',
|
|
appliesTo: ["number"],
|
|
inputCount: 1,
|
|
getDescription: function (values) {
|
|
return ` is older than ${values[0] || ''} seconds`;
|
|
}
|
|
},
|
|
{
|
|
name: IS_STALE_KEY,
|
|
operation: function () {
|
|
return false;
|
|
},
|
|
text: 'is stale',
|
|
appliesTo: ["number"],
|
|
inputCount: 0,
|
|
getDescription: function () {
|
|
return ' is stale';
|
|
}
|
|
}
|
|
];
|
|
|
|
export const INPUT_TYPES = {
|
|
'string': 'text',
|
|
'number': 'number'
|
|
};
|
|
|
|
export function getOperatorText(operationName, values) {
|
|
const found = OPERATIONS.find((operation) => operation.name === operationName);
|
|
|
|
return found?.getDescription(values) ?? '';
|
|
}
|