2015-05-13 23:42:35 +00:00
|
|
|
/*****************************************************************************
|
2020-09-14 18:17:31 +00:00
|
|
|
* Open MCT, Copyright (c) 2014-2020, United States Government
|
2015-05-13 23:42:35 +00:00
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
2016-07-12 23:21:58 +00:00
|
|
|
* Open MCT is licensed under the Apache License, Version 2.0 (the
|
2015-05-13 23:42:35 +00:00
|
|
|
* "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.
|
|
|
|
*
|
2016-07-12 23:21:58 +00:00
|
|
|
* Open MCT includes source code licensed under additional open source
|
2015-05-13 23:42:35 +00:00
|
|
|
* 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.
|
|
|
|
*****************************************************************************/
|
2014-11-29 04:29:30 +00:00
|
|
|
|
|
|
|
define(
|
|
|
|
["../src/TelemetryCapability"],
|
|
|
|
function (TelemetryCapability) {
|
|
|
|
|
|
|
|
describe("The telemetry capability", function () {
|
|
|
|
var mockInjector,
|
|
|
|
mockQ,
|
|
|
|
mockLog,
|
|
|
|
mockDomainObject,
|
|
|
|
mockTelemetryService,
|
|
|
|
mockReject,
|
2014-12-30 22:00:20 +00:00
|
|
|
mockUnsubscribe,
|
2017-02-22 03:04:28 +00:00
|
|
|
telemetry,
|
2017-02-22 03:53:56 +00:00
|
|
|
mockTelemetryAPI,
|
2017-09-21 17:26:26 +00:00
|
|
|
mockMetadata,
|
2017-02-22 03:53:56 +00:00
|
|
|
mockAPI;
|
2014-11-29 04:29:30 +00:00
|
|
|
|
|
|
|
function mockPromise(value) {
|
|
|
|
return {
|
|
|
|
then: function (callback) {
|
|
|
|
return mockPromise(callback(value));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-02-22 03:53:56 +00:00
|
|
|
function noop() {
|
|
|
|
}
|
|
|
|
|
2014-11-29 04:29:30 +00:00
|
|
|
beforeEach(function () {
|
|
|
|
mockInjector = jasmine.createSpyObj("$injector", ["get"]);
|
|
|
|
mockQ = jasmine.createSpyObj("$q", ["when", "reject"]);
|
|
|
|
mockLog = jasmine.createSpyObj("$log", ["warn", "info", "debug"]);
|
|
|
|
mockDomainObject = jasmine.createSpyObj(
|
|
|
|
"domainObject",
|
2016-05-19 18:29:13 +00:00
|
|
|
["getId", "getCapability", "getModel"]
|
2014-11-29 04:29:30 +00:00
|
|
|
);
|
|
|
|
mockTelemetryService = jasmine.createSpyObj(
|
|
|
|
"telemetryService",
|
2016-05-19 18:29:13 +00:00
|
|
|
["requestTelemetry", "subscribe"]
|
2014-11-29 04:29:30 +00:00
|
|
|
);
|
|
|
|
mockReject = jasmine.createSpyObj("reject", ["then"]);
|
2014-12-30 22:00:20 +00:00
|
|
|
mockUnsubscribe = jasmine.createSpy("unsubscribe");
|
2014-11-29 04:29:30 +00:00
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockInjector.get.and.returnValue(mockTelemetryService);
|
2014-11-29 04:29:30 +00:00
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockQ.when.and.callFake(mockPromise);
|
|
|
|
mockQ.reject.and.returnValue(mockReject);
|
2014-11-29 04:29:30 +00:00
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockDomainObject.getId.and.returnValue("testId");
|
|
|
|
mockDomainObject.getModel.and.returnValue({
|
2014-11-29 04:29:30 +00:00
|
|
|
telemetry: {
|
|
|
|
source: "testSource",
|
|
|
|
key: "testKey"
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-12-30 21:45:42 +00:00
|
|
|
mockTelemetryService.requestTelemetry
|
2018-06-30 00:32:59 +00:00
|
|
|
.and.returnValue(mockPromise({}));
|
2014-12-30 22:00:20 +00:00
|
|
|
mockTelemetryService.subscribe
|
2018-06-30 00:32:59 +00:00
|
|
|
.and.returnValue(mockUnsubscribe);
|
2014-12-30 21:45:42 +00:00
|
|
|
|
2014-11-29 04:29:30 +00:00
|
|
|
// Bubble up...
|
2018-06-30 00:32:59 +00:00
|
|
|
mockReject.then.and.returnValue(mockReject);
|
2014-11-29 04:29:30 +00:00
|
|
|
|
2017-02-22 03:53:56 +00:00
|
|
|
mockTelemetryAPI = jasmine.createSpyObj("telemetryAPI", [
|
|
|
|
"getMetadata",
|
|
|
|
"subscribe",
|
|
|
|
"request",
|
|
|
|
"findRequestProvider",
|
|
|
|
"findSubscriptionProvider"
|
|
|
|
]);
|
2017-09-21 17:26:26 +00:00
|
|
|
|
|
|
|
mockMetadata = jasmine.createSpyObj('telemetryMetadata', [
|
|
|
|
'valuesForHints',
|
|
|
|
'values'
|
|
|
|
]);
|
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockMetadata.valuesForHints.and.callFake(function (hints) {
|
2017-09-21 17:26:26 +00:00
|
|
|
var hint = hints[0];
|
|
|
|
var metadatum = {
|
|
|
|
key: 'default' + hint
|
|
|
|
};
|
|
|
|
metadatum[hint] = "foo";
|
2020-07-31 19:11:03 +00:00
|
|
|
|
2017-09-21 17:26:26 +00:00
|
|
|
return [metadatum];
|
2017-02-22 03:53:56 +00:00
|
|
|
});
|
2017-02-22 03:04:28 +00:00
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockTelemetryAPI.getMetadata.and.returnValue(mockMetadata);
|
2017-09-21 17:26:26 +00:00
|
|
|
|
2017-02-22 03:04:28 +00:00
|
|
|
mockAPI = {
|
2017-02-28 00:07:56 +00:00
|
|
|
telemetry: mockTelemetryAPI,
|
2017-05-01 23:19:11 +00:00
|
|
|
time: {
|
2017-02-28 00:07:56 +00:00
|
|
|
bounds: function () {
|
|
|
|
return {
|
|
|
|
start: 0,
|
|
|
|
end: 1
|
|
|
|
};
|
2017-03-27 21:53:41 +00:00
|
|
|
},
|
|
|
|
timeSystem: function () {
|
|
|
|
return {
|
2017-05-01 23:19:11 +00:00
|
|
|
key: 'mockTimeSystem'
|
2017-03-27 21:53:41 +00:00
|
|
|
};
|
2017-02-28 00:07:56 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-22 03:04:28 +00:00
|
|
|
};
|
|
|
|
|
2014-11-29 04:29:30 +00:00
|
|
|
telemetry = new TelemetryCapability(
|
2017-02-22 03:04:28 +00:00
|
|
|
mockAPI,
|
2014-11-29 04:29:30 +00:00
|
|
|
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
|
2017-03-27 21:53:41 +00:00
|
|
|
start: 42, // from argument
|
2017-08-30 16:44:35 +00:00
|
|
|
domain: 'mockTimeSystem',
|
2020-07-31 19:11:03 +00:00
|
|
|
domains: [{
|
|
|
|
domain: "foo",
|
|
|
|
key: 'defaultdomain'
|
|
|
|
}],
|
|
|
|
ranges: [{
|
|
|
|
range: "foo",
|
|
|
|
key: 'defaultrange'
|
|
|
|
}]
|
2014-11-29 04:29:30 +00:00
|
|
|
}]);
|
|
|
|
});
|
|
|
|
|
2015-07-14 22:51:13 +00:00
|
|
|
it("provides an empty series when telemetry is missing", function () {
|
|
|
|
var series;
|
2018-06-30 00:32:59 +00:00
|
|
|
mockTelemetryService.requestTelemetry.and.returnValue(mockPromise({}));
|
2016-05-19 18:29:13 +00:00
|
|
|
telemetry.requestData({}).then(function (s) {
|
|
|
|
series = s;
|
|
|
|
});
|
2015-07-14 22:51:13 +00:00
|
|
|
expect(series.getPointCount()).toEqual(0);
|
|
|
|
});
|
|
|
|
|
2014-11-29 04:29:30 +00:00
|
|
|
it("provides telemetry metadata", function () {
|
|
|
|
expect(telemetry.getMetadata()).toEqual({
|
|
|
|
id: "testId", // from domain object
|
|
|
|
source: "testSource",
|
2017-02-28 00:07:56 +00:00
|
|
|
key: "testKey",
|
|
|
|
start: 0,
|
2017-03-27 21:53:41 +00:00
|
|
|
end: 1,
|
2017-08-30 16:44:35 +00:00
|
|
|
domain: 'mockTimeSystem',
|
2020-07-31 19:11:03 +00:00
|
|
|
domains: [{
|
|
|
|
domain: "foo",
|
|
|
|
key: 'defaultdomain'
|
|
|
|
}],
|
|
|
|
ranges: [{
|
|
|
|
range: "foo",
|
|
|
|
key: 'defaultrange'
|
|
|
|
}]
|
2014-11-29 04:29:30 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("uses domain object as a key if needed", function () {
|
|
|
|
// Don't include key in telemetry
|
2018-06-30 00:32:59 +00:00
|
|
|
mockDomainObject.getModel.and.returnValue({
|
2014-11-29 04:29:30 +00:00
|
|
|
telemetry: { source: "testSource" }
|
|
|
|
});
|
|
|
|
|
|
|
|
// Should have used the domain object's ID
|
|
|
|
expect(telemetry.getMetadata()).toEqual({
|
|
|
|
id: "testId", // from domain object
|
|
|
|
source: "testSource", // from model
|
2017-02-28 00:07:56 +00:00
|
|
|
key: "testId", // from domain object
|
|
|
|
start: 0,
|
2017-03-27 21:53:41 +00:00
|
|
|
end: 1,
|
2017-08-30 16:44:35 +00:00
|
|
|
domain: 'mockTimeSystem',
|
2020-07-31 19:11:03 +00:00
|
|
|
domains: [{
|
|
|
|
domain: "foo",
|
|
|
|
key: 'defaultdomain'
|
|
|
|
}],
|
|
|
|
ranges: [{
|
|
|
|
range: "foo",
|
|
|
|
key: 'defaultrange'
|
|
|
|
}]
|
2014-11-29 04:29:30 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("warns if no telemetry service can be injected", function () {
|
2018-06-30 00:32:59 +00:00
|
|
|
mockInjector.get.and.callFake(function () {
|
2016-05-19 18:29:13 +00:00
|
|
|
throw "";
|
|
|
|
});
|
2014-11-29 04:29:30 +00:00
|
|
|
|
|
|
|
// Verify precondition
|
|
|
|
expect(mockLog.warn).not.toHaveBeenCalled();
|
|
|
|
|
|
|
|
telemetry.requestData();
|
|
|
|
|
2017-07-10 16:46:51 +00:00
|
|
|
expect(mockLog.info).toHaveBeenCalled();
|
2014-11-29 04:29:30 +00:00
|
|
|
});
|
|
|
|
|
2017-02-22 03:53:56 +00:00
|
|
|
it("if a new style telemetry source is available, use it", function () {
|
|
|
|
var mockProvider = {};
|
2018-06-30 00:32:59 +00:00
|
|
|
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 () {
|
2018-06-30 00:32:59 +00:00
|
|
|
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 = {};
|
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockMetadata.values.and.returnValue([
|
2017-09-21 17:26:26 +00:00
|
|
|
{
|
|
|
|
key: 'defaultrange',
|
|
|
|
source: 'prop1'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'defaultdomain',
|
|
|
|
source: 'prop2'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'prop3',
|
|
|
|
source: 'prop3'
|
|
|
|
}
|
|
|
|
]);
|
|
|
|
|
2018-06-30 00:32:59 +00:00
|
|
|
mockTelemetryAPI.findRequestProvider.and.returnValue(mockProvider);
|
|
|
|
mockTelemetryAPI.request.and.returnValue(Promise.resolve(mockTelemetry));
|
2017-02-22 03:53:56 +00:00
|
|
|
|
2018-06-30 00:32:59 +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);
|
2017-09-21 17:26:26 +00:00
|
|
|
// 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
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2014-12-30 22:00:20 +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",
|
2017-02-28 00:07:56 +00:00
|
|
|
key: "testKey",
|
|
|
|
start: 0,
|
2017-03-27 21:53:41 +00:00
|
|
|
end: 1,
|
2017-08-30 16:44:35 +00:00
|
|
|
domain: 'mockTimeSystem',
|
2020-07-31 19:11:03 +00:00
|
|
|
domains: [{
|
|
|
|
domain: "foo",
|
|
|
|
key: "defaultdomain"
|
|
|
|
}],
|
|
|
|
ranges: [{
|
|
|
|
range: "foo",
|
|
|
|
key: "defaultrange"
|
|
|
|
}]
|
2014-12-30 22:00:20 +00:00
|
|
|
}]
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check that the callback gets invoked
|
|
|
|
expect(mockCallback).not.toHaveBeenCalled();
|
2018-06-30 00:32:59 +00:00
|
|
|
mockTelemetryService.subscribe.calls.mostRecent().args[0]({
|
2014-12-30 22:00:20 +00:00
|
|
|
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();
|
2017-03-27 21:53:41 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("applies time conductor bounds if request bounds not defined", function () {
|
|
|
|
var fullRequest = telemetry.buildRequest({});
|
2017-05-01 23:19:11 +00:00
|
|
|
var mockBounds = mockAPI.time.bounds();
|
2017-03-27 21:53:41 +00:00
|
|
|
|
|
|
|
expect(fullRequest.start).toBe(mockBounds.start);
|
|
|
|
expect(fullRequest.end).toBe(mockBounds.end);
|
|
|
|
|
2020-07-31 19:11:03 +00:00
|
|
|
fullRequest = telemetry.buildRequest({
|
|
|
|
start: 10,
|
|
|
|
end: 20
|
|
|
|
});
|
2017-03-27 21:53:41 +00:00
|
|
|
|
|
|
|
expect(fullRequest.start).toBe(10);
|
|
|
|
expect(fullRequest.end).toBe(20);
|
|
|
|
});
|
2014-12-30 22:00:20 +00:00
|
|
|
|
2017-03-27 21:53:41 +00:00
|
|
|
it("applies domain from time system if none defined", function () {
|
|
|
|
var fullRequest = telemetry.buildRequest({});
|
2017-05-01 23:19:11 +00:00
|
|
|
var mockTimeSystem = mockAPI.time.timeSystem();
|
|
|
|
expect(fullRequest.domain).toBe(mockTimeSystem.key);
|
2014-12-30 22:00:20 +00:00
|
|
|
|
2017-03-27 21:53:41 +00:00
|
|
|
fullRequest = telemetry.buildRequest({domain: 'someOtherDomain'});
|
|
|
|
expect(fullRequest.domain).toBe('someOtherDomain');
|
2014-12-30 22:00:20 +00:00
|
|
|
});
|
2014-11-29 04:29:30 +00:00
|
|
|
});
|
|
|
|
}
|
2015-07-14 22:51:13 +00:00
|
|
|
);
|