mirror of
https://github.com/nasa/openmct.git
synced 2025-01-03 11:54:10 +00:00
433dee0314
* Updated Karma and Jasmine versions * Added DOMObserver class. Supports promise-based testing of DOM changes Update asynchronous test specs to use promises or done() instead of waitsFor/runs * Modified ActionCapability to duplicate context object properties as own properties for better object equality comparisons * Global find + replace to fix syntax issues * Fixed various issues caused by non-deterministic runtime order of tests in Jasmine 3. Fixed issues caused by changes to determination of object equality * Addressed review comments * Resolved merge conflicts with master * Fixed style errors * Use spy.calls.count() instead of manually tracking
267 lines
11 KiB
JavaScript
267 lines
11 KiB
JavaScript
/*****************************************************************************
|
|
* Open MCT, Copyright (c) 2014-2018, 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);
|
|
});
|
|
});
|
|
}
|
|
);
|