openmct/platform/telemetry/test/TelemetrySubscriptionSpec.js
2020-09-14 11:17:31 -07:00

272 lines
11 KiB
JavaScript

/*****************************************************************************
* Open MCT, Copyright (c) 2014-2020, 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/TelemetrySubscription"],
function (TelemetrySubscription) {
describe("A telemetry subscription", function () {
var mockQ,
mockTimeout,
mockDomainObject,
mockCallback,
mockTelemetry,
mockMutation,
mockUnsubscribe,
mockUnlisten,
mockSeries,
testMetadata,
subscription;
function mockPromise(value) {
return (value && value.then) ? value : {
then: function (callback) {
return mockPromise(callback(value));
}
};
}
beforeEach(function () {
testMetadata = { someKey: "some value" };
mockQ = jasmine.createSpyObj("$q", ["when", "all"]);
mockTimeout = jasmine.createSpy("$timeout");
mockDomainObject = jasmine.createSpyObj(
"domainObject",
["getCapability", "useCapability", "hasCapability", "getId"]
);
mockCallback = jasmine.createSpy("callback");
mockTelemetry = jasmine.createSpyObj(
"telemetry",
["subscribe", "getMetadata"]
);
mockMutation = jasmine.createSpyObj(
"mutation",
["mutate", "listen"]
);
mockUnsubscribe = jasmine.createSpy("unsubscribe");
mockUnlisten = jasmine.createSpy("unlisten");
mockSeries = jasmine.createSpyObj(
"series",
["getPointCount", "getDomainValue", "getRangeValue"]
);
mockQ.when.and.callFake(mockPromise);
mockDomainObject.hasCapability.and.returnValue(true);
mockDomainObject.getCapability.and.callFake(function (c) {
return {
telemetry: mockTelemetry,
mutation: mockMutation
}[c];
});
mockDomainObject.getId.and.returnValue('test-id');
mockTelemetry.subscribe.and.returnValue(mockUnsubscribe);
mockTelemetry.getMetadata.and.returnValue(testMetadata);
mockMutation.listen.and.returnValue(mockUnlisten);
mockSeries.getPointCount.and.returnValue(42);
mockSeries.getDomainValue.and.returnValue(123456);
mockSeries.getRangeValue.and.returnValue(789);
subscription = new TelemetrySubscription(
mockQ,
mockTimeout,
mockDomainObject,
mockCallback
);
});
it("subscribes to the provided object", function () {
expect(mockTelemetry.subscribe).toHaveBeenCalled();
});
it("unsubscribes on request", function () {
expect(mockUnsubscribe).not.toHaveBeenCalled();
subscription.unsubscribe();
expect(mockUnsubscribe).toHaveBeenCalled();
});
it("fires callbacks when subscriptions update", function () {
// Callback fires when telemetry objects become available,
// so track initial call count instead of verifying that
// it hasn't been called at all.
var initialCalls = mockCallback.calls.count();
mockTelemetry.subscribe.calls.mostRecent().args[0](mockSeries);
// This gets fired via a timeout, so trigger that
expect(mockTimeout).toHaveBeenCalledWith(
jasmine.any(Function),
0
);
mockTimeout.calls.mostRecent().args[0]();
// Should have triggered the callback to alert that
// new data was available
expect(mockCallback.calls.count()).toEqual(initialCalls + 1);
});
it("fires subscription callbacks once per cycle", function () {
var i;
// Verify precondition - one call for telemetryObjects
expect(mockCallback.calls.count()).toEqual(1);
for (i = 0; i < 100; i += 1) {
mockTelemetry.subscribe.calls.mostRecent().args[0](mockSeries);
}
// This gets fired via a timeout, so trigger any of those
mockTimeout.calls.all().forEach(function (call) {
call.args[0]();
});
// Should have only triggered the
expect(mockCallback.calls.count()).toEqual(2);
});
it("reports its latest observed data values", function () {
mockTelemetry.subscribe.calls.mostRecent().args[0](mockSeries);
// This gets fired via a timeout, so trigger that
mockTimeout.calls.mostRecent().args[0]();
// Verify that the last sample was looked at
expect(mockSeries.getDomainValue).toHaveBeenCalledWith(41);
expect(mockSeries.getRangeValue).toHaveBeenCalledWith(41);
// Domain and range values should now be available
expect(subscription.getDomainValue(mockDomainObject))
.toEqual(123456);
expect(subscription.getRangeValue(mockDomainObject))
.toEqual(789);
});
it("provides no objects if no domain object is provided", function () {
// omit last arguments
subscription = new TelemetrySubscription(mockQ, mockTimeout);
// Should have no objects
expect(subscription.getTelemetryObjects()).toEqual([]);
});
// This test case corresponds to plot usage of
// telemetrySubscription, where failure to callback
// once-per-update results in loss of data, WTD-784
it("fires one event per update if requested", function () {
var i, domains = [], ranges = [], lastCall, initialCalls;
// Clear out the subscription from beforeEach
subscription.unsubscribe();
// Create a subscription which does not drop events
subscription = new TelemetrySubscription(
mockQ,
mockTimeout,
mockDomainObject,
mockCallback,
true // Don't drop updates!
);
// Track calls at this point
initialCalls = mockCallback.calls.count();
// Snapshot getDomainValue, getRangeValue at time of callback
mockCallback.and.callFake(function () {
domains.push(subscription.getDomainValue(mockDomainObject));
ranges.push(subscription.getRangeValue(mockDomainObject));
});
// Send 100 updates
for (i = 0; i < 100; i += 1) {
// Return different values to verify later
mockSeries.getDomainValue.and.returnValue(i);
mockSeries.getRangeValue.and.returnValue(i * 2);
mockTelemetry.subscribe.calls.mostRecent().args[0](mockSeries);
}
// Fire all timeouts that get scheduled
while (mockTimeout.calls.mostRecent() !== lastCall) {
lastCall = mockTimeout.calls.mostRecent();
lastCall.args[0]();
}
// Should have only triggered the
expect(mockCallback.calls.count()).toEqual(100 + initialCalls);
});
it("provides domain object metadata", function () {
expect(subscription.getMetadata()[0])
.toEqual(testMetadata);
});
it("fires callback when telemetry objects are available", function () {
expect(mockCallback.calls.count()).toEqual(1);
});
it("exposes a promise for telemetry objects", function () {
var mockCallback2 = jasmine.createSpy('callback');
subscription.promiseTelemetryObjects().then(mockCallback2);
expect(mockCallback2)
.toHaveBeenCalledWith([mockDomainObject]);
});
it("reinitializes on mutation", function () {
expect(mockTelemetry.subscribe.calls.count()).toEqual(1);
// Notify of a mutation which appears to change composition
mockMutation.listen.calls.mostRecent().args[0]({
composition: ['Z']
});
// Use subscribe call as an indication of reinitialization
expect(mockTelemetry.subscribe.calls.count()).toEqual(2);
});
it("stops listening for mutation on unsubscribe", function () {
expect(mockUnlisten).not.toHaveBeenCalled();
subscription.unsubscribe();
expect(mockUnlisten).toHaveBeenCalled();
});
it("provides telemetry as datum objects", function () {
var testDatum = {
a: 1,
b: 13,
c: 42,
d: -1977
};
function lookup(index, key) {
return testDatum[key];
}
mockSeries.getDomainValue.and.callFake(lookup);
mockSeries.getRangeValue.and.callFake(lookup);
testMetadata.domains = [{ key: 'a' }, { key: 'b'}];
testMetadata.ranges = [{ key: 'c' }, { key: 'd'}];
mockTelemetry.subscribe.calls.mostRecent().args[0](mockSeries);
mockTimeout.calls.mostRecent().args[0]();
expect(subscription.getDatum(mockDomainObject))
.toEqual(testDatum);
});
});
}
);