openmct/platform/telemetry/test/TelemetryCapabilitySpec.js

378 lines
15 KiB
JavaScript
Raw Normal View History

/*****************************************************************************
* Open MCT, Copyright (c) 2014-2021, 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.
*****************************************************************************/
define(
["../src/TelemetryCapability"],
function (TelemetryCapability) {
describe("The telemetry capability", function () {
var mockInjector,
mockQ,
mockLog,
mockDomainObject,
mockTelemetryService,
mockReject,
mockUnsubscribe,
2017-02-22 03:04:28 +00:00
telemetry,
2017-02-22 03:53:56 +00:00
mockTelemetryAPI,
mockMetadata,
2017-02-22 03:53:56 +00:00
mockAPI;
function mockPromise(value) {
return {
then: function (callback) {
return mockPromise(callback(value));
}
};
}
2017-02-22 03:53:56 +00:00
function noop() {
}
beforeEach(function () {
mockInjector = jasmine.createSpyObj("$injector", ["get"]);
mockQ = jasmine.createSpyObj("$q", ["when", "reject"]);
mockLog = jasmine.createSpyObj("$log", ["warn", "info", "debug"]);
mockDomainObject = jasmine.createSpyObj(
"domainObject",
["getId", "getCapability", "getModel"]
);
mockTelemetryService = jasmine.createSpyObj(
"telemetryService",
["requestTelemetry", "subscribe"]
);
mockReject = jasmine.createSpyObj("reject", ["then"]);
mockUnsubscribe = jasmine.createSpy("unsubscribe");
mockInjector.get.and.returnValue(mockTelemetryService);
mockQ.when.and.callFake(mockPromise);
mockQ.reject.and.returnValue(mockReject);
mockDomainObject.getId.and.returnValue("testId");
mockDomainObject.getModel.and.returnValue({
telemetry: {
source: "testSource",
key: "testKey"
}
});
mockTelemetryService.requestTelemetry
.and.returnValue(mockPromise({}));
mockTelemetryService.subscribe
.and.returnValue(mockUnsubscribe);
// Bubble up...
mockReject.then.and.returnValue(mockReject);
2017-02-22 03:53:56 +00:00
mockTelemetryAPI = jasmine.createSpyObj("telemetryAPI", [
"getMetadata",
"subscribe",
"request",
"findRequestProvider",
"findSubscriptionProvider"
]);
mockMetadata = jasmine.createSpyObj('telemetryMetadata', [
'valuesForHints',
'values'
]);
mockMetadata.valuesForHints.and.callFake(function (hints) {
var hint = hints[0];
var metadatum = {
key: 'default' + hint
};
metadatum[hint] = "foo";
return [metadatum];
2017-02-22 03:53:56 +00:00
});
2017-02-22 03:04:28 +00:00
mockTelemetryAPI.getMetadata.and.returnValue(mockMetadata);
2017-02-22 03:04:28 +00:00
mockAPI = {
telemetry: mockTelemetryAPI,
time: {
bounds: function () {
return {
start: 0,
end: 1
};
},
timeSystem: function () {
return {
key: 'mockTimeSystem'
};
}
}
2017-02-22 03:04:28 +00:00
};
telemetry = new TelemetryCapability(
2017-02-22 03:04:28 +00:00
mockAPI,
mockInjector,
mockQ,
mockLog,
mockDomainObject
);
});
it("applies only to objects with telemetry sources", function () {
expect(TelemetryCapability.appliesTo({
telemetry: { source: "testSource" }
})).toBeTruthy();
expect(TelemetryCapability.appliesTo({
xtelemetry: { source: "testSource" }
})).toBeFalsy();
expect(TelemetryCapability.appliesTo({})).toBeFalsy();
expect(TelemetryCapability.appliesTo()).toBeFalsy();
});
it("gets a telemetry service from the injector", function () {
telemetry.requestData();
expect(mockInjector.get)
.toHaveBeenCalledWith("telemetryService");
});
it("applies request arguments", function () {
telemetry.requestData({ start: 42 });
expect(mockTelemetryService.requestTelemetry)
.toHaveBeenCalledWith([{
id: "testId", // from domain object
source: "testSource", // from model
key: "testKey", // from model
start: 42, // from argument
domain: 'mockTimeSystem',
domains: [{
domain: "foo",
key: 'defaultdomain'
}],
ranges: [{
range: "foo",
key: 'defaultrange'
}]
}]);
});
it("provides an empty series when telemetry is missing", function () {
var series;
mockTelemetryService.requestTelemetry.and.returnValue(mockPromise({}));
telemetry.requestData({}).then(function (s) {
series = s;
});
expect(series.getPointCount()).toEqual(0);
});
it("provides telemetry metadata", function () {
expect(telemetry.getMetadata()).toEqual({
id: "testId", // from domain object
source: "testSource",
key: "testKey",
start: 0,
end: 1,
domain: 'mockTimeSystem',
domains: [{
domain: "foo",
key: 'defaultdomain'
}],
ranges: [{
range: "foo",
key: 'defaultrange'
}]
});
});
it("uses domain object as a key if needed", function () {
// Don't include key in telemetry
mockDomainObject.getModel.and.returnValue({
telemetry: { source: "testSource" }
});
// Should have used the domain object's ID
expect(telemetry.getMetadata()).toEqual({
id: "testId", // from domain object
source: "testSource", // from model
key: "testId", // from domain object
start: 0,
end: 1,
domain: 'mockTimeSystem',
domains: [{
domain: "foo",
key: 'defaultdomain'
}],
ranges: [{
range: "foo",
key: 'defaultrange'
}]
});
});
it("warns if no telemetry service can be injected", function () {
mockInjector.get.and.callFake(function () {
throw "";
});
// Verify precondition
expect(mockLog.warn).not.toHaveBeenCalled();
telemetry.requestData();
expect(mockLog.info).toHaveBeenCalled();
});
2017-02-22 03:53:56 +00:00
it("if a new style telemetry source is available, use it", function () {
var mockProvider = {};
mockTelemetryAPI.findSubscriptionProvider.and.returnValue(mockProvider);
2017-02-22 03:53:56 +00:00
telemetry.subscribe(noop, {});
expect(mockTelemetryService.subscribe).not.toHaveBeenCalled();
expect(mockTelemetryAPI.subscribe).toHaveBeenCalled();
});
it("if a new style telemetry source is not available, revert to old API", function () {
mockTelemetryAPI.findSubscriptionProvider.and.returnValue(undefined);
2017-02-22 03:53:56 +00:00
telemetry.subscribe(noop, {});
expect(mockTelemetryAPI.subscribe).not.toHaveBeenCalled();
expect(mockTelemetryService.subscribe).toHaveBeenCalled();
});
it("Wraps telemetry returned from the new API as a telemetry series", function () {
var returnedTelemetry;
var mockTelemetry = [{
prop1: "val1",
prop2: "val2",
prop3: "val3"
},
{
prop1: "val4",
prop2: "val5",
prop3: "val6"
}];
var mockProvider = {};
mockMetadata.values.and.returnValue([
{
key: 'defaultrange',
source: 'prop1'
},
{
key: 'defaultdomain',
source: 'prop2'
},
{
key: 'prop3',
source: 'prop3'
}
]);
mockTelemetryAPI.findRequestProvider.and.returnValue(mockProvider);
mockTelemetryAPI.request.and.returnValue(Promise.resolve(mockTelemetry));
2017-02-22 03:53:56 +00:00
return telemetry.requestData({}).then(function (data) {
2017-02-22 03:53:56 +00:00
returnedTelemetry = data;
expect(returnedTelemetry.getPointCount).toBeDefined();
expect(returnedTelemetry.getDomainValue).toBeDefined();
expect(returnedTelemetry.getRangeValue).toBeDefined();
expect(returnedTelemetry.getPointCount()).toBe(2);
// Default domain + remap should work.
expect(returnedTelemetry.getDomainValue(0)).toBe('val2');
expect(returnedTelemetry.getDomainValue(1)).toBe('val5');
// explicit domain should work
expect(returnedTelemetry.getDomainValue(0, 'prop3')).toBe('val3');
expect(returnedTelemetry.getDomainValue(1, 'prop3')).toBe('val6');
// default range + remap should work
expect(returnedTelemetry.getRangeValue(0)).toBe('val1');
expect(returnedTelemetry.getRangeValue(1)).toBe('val4');
// explicit range should work
expect(returnedTelemetry.getRangeValue(0, 'prop3')).toBe('val3');
expect(returnedTelemetry.getRangeValue(1, 'prop3')).toBe('val6');
2017-02-22 03:53:56 +00:00
});
});
it("allows subscriptions to updates", function () {
var mockCallback = jasmine.createSpy("callback"),
subscription = telemetry.subscribe(mockCallback);
// Verify subscription to the appropriate object
expect(mockTelemetryService.subscribe).toHaveBeenCalledWith(
jasmine.any(Function),
[{
id: "testId", // from domain object
source: "testSource",
key: "testKey",
start: 0,
end: 1,
domain: 'mockTimeSystem',
domains: [{
domain: "foo",
key: "defaultdomain"
}],
ranges: [{
range: "foo",
key: "defaultrange"
}]
}]
);
// Check that the callback gets invoked
expect(mockCallback).not.toHaveBeenCalled();
mockTelemetryService.subscribe.calls.mostRecent().args[0]({
testSource: { testKey: { someKey: "some value" } }
});
expect(mockCallback).toHaveBeenCalledWith(
{ someKey: "some value" }
);
// Finally, unsubscribe
expect(mockUnsubscribe).not.toHaveBeenCalled();
subscription(); // should be an unsubscribe function
expect(mockUnsubscribe).toHaveBeenCalled();
});
it("applies time conductor bounds if request bounds not defined", function () {
var fullRequest = telemetry.buildRequest({});
var mockBounds = mockAPI.time.bounds();
expect(fullRequest.start).toBe(mockBounds.start);
expect(fullRequest.end).toBe(mockBounds.end);
fullRequest = telemetry.buildRequest({
start: 10,
end: 20
});
expect(fullRequest.start).toBe(10);
expect(fullRequest.end).toBe(20);
});
it("applies domain from time system if none defined", function () {
var fullRequest = telemetry.buildRequest({});
var mockTimeSystem = mockAPI.time.timeSystem();
expect(fullRequest.domain).toBe(mockTimeSystem.key);
fullRequest = telemetry.buildRequest({domain: 'someOtherDomain'});
expect(fullRequest.domain).toBe('someOtherDomain');
});
});
}
);