/***************************************************************************** * Open MCT Web, Copyright (c) 2014-2015, United States Government * as represented by the Administrator of the National Aeronautics and Space * Administration. All rights reserved. * * Open MCT Web 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 Web 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/controllers/HistoricalTableController" ], function (TableController) { describe('The Table Controller', function () { var mockScope, mockTelemetryHandler, mockTelemetryHandle, mockTelemetryFormatter, mockDomainObject, mockTable, mockConfiguration, mockAngularTimeout, mockTimeoutHandle, watches, controller; function promise(value) { return { then: function (callback) { return promise(callback(value)); } }; } beforeEach(function () { watches = {}; mockScope = jasmine.createSpyObj('scope', [ '$on', '$watch', '$watchCollection' ]); mockScope.$on.andCallFake(function (expression, callback) { watches[expression] = callback; }); mockScope.$watch.andCallFake(function (expression, callback) { watches[expression] = callback; }); mockScope.$watchCollection.andCallFake(function (expression, callback) { watches[expression] = callback; }); mockTimeoutHandle = jasmine.createSpy("timeoutHandle"); mockAngularTimeout = jasmine.createSpy("$timeout"); mockAngularTimeout.andReturn(mockTimeoutHandle); mockAngularTimeout.cancel = jasmine.createSpy("cancelTimeout"); mockConfiguration = { 'range1': true, 'range2': true, 'domain1': true }; mockTable = jasmine.createSpyObj('table', [ 'populateColumns', 'buildColumnConfiguration', 'getRowValues', 'saveColumnConfiguration' ] ); mockTable.columns = []; mockTable.buildColumnConfiguration.andReturn(mockConfiguration); mockDomainObject = jasmine.createSpyObj('domainObject', [ 'getCapability', 'useCapability', 'getModel' ]); mockDomainObject.getModel.andReturn({}); mockScope.domainObject = mockDomainObject; mockTelemetryHandle = jasmine.createSpyObj('telemetryHandle', [ 'request', 'promiseTelemetryObjects', 'getTelemetryObjects', 'getMetadata', 'getSeries', 'unsubscribe', 'makeDatum' ]); mockTelemetryHandle.promiseTelemetryObjects.andReturn(promise(undefined)); mockTelemetryHandle.request.andReturn(promise(undefined)); mockTelemetryHandle.getTelemetryObjects.andReturn([]); mockTelemetryHandler = jasmine.createSpyObj('telemetryHandler', [ 'handle' ]); mockTelemetryHandler.handle.andReturn(mockTelemetryHandle); controller = new TableController(mockScope, mockTelemetryHandler, mockTelemetryFormatter, mockAngularTimeout); controller.table = mockTable; controller.handle = mockTelemetryHandle; }); it('subscribes to telemetry handler for telemetry updates', function () { controller.subscribe(); expect(mockTelemetryHandler.handle).toHaveBeenCalled(); expect(mockTelemetryHandle.request).toHaveBeenCalled(); }); it('Unsubscribes from telemetry when scope is destroyed', function () { controller.handle = mockTelemetryHandle; watches.$destroy(); expect(mockTelemetryHandle.unsubscribe).toHaveBeenCalled(); }); describe('makes use of the table', function () { it('to create column definitions from telemetry' + ' metadata', function () { controller.setup(); expect(mockTable.populateColumns).toHaveBeenCalled(); }); it('to create column configuration, which is written to the' + ' object model', function () { controller.setup(); expect(mockTable.buildColumnConfiguration).toHaveBeenCalled(); }); }); it('updates the rows on scope when historical telemetry is received', function () { var mockSeries = { getPointCount: function () { return 5; }, getDomainValue: function () { return 'Domain Value'; }, getRangeValue: function () { return 'Range Value'; } }, mockRow = {'domain': 'Domain Value', 'range': 'Range' + ' Value'}; mockTelemetryHandle.makeDatum.andCallFake(function () { return mockRow; }); mockTable.getRowValues.andReturn(mockRow); mockTelemetryHandle.getTelemetryObjects.andReturn([mockDomainObject]); mockTelemetryHandle.getSeries.andReturn(mockSeries); controller.addHistoricalData(mockDomainObject, mockSeries); // Angular timeout is called a minumum of twice, regardless // of batch size used. expect(mockAngularTimeout).toHaveBeenCalled(); mockAngularTimeout.mostRecentCall.args[0](); expect(mockAngularTimeout.calls.length).toEqual(2); mockAngularTimeout.mostRecentCall.args[0](); expect(controller.$scope.rows.length).toBe(5); expect(controller.$scope.rows[0]).toBe(mockRow); }); it('filters the visible columns based on configuration', function () { controller.filterColumns(); expect(controller.$scope.headers.length).toBe(3); expect(controller.$scope.headers[2]).toEqual('domain1'); mockConfiguration.domain1 = false; controller.filterColumns(); expect(controller.$scope.headers.length).toBe(2); expect(controller.$scope.headers[2]).toBeUndefined(); }); describe('creates event listeners', function () { beforeEach(function () { spyOn(controller, 'subscribe'); spyOn(controller, 'filterColumns'); }); it('triggers telemetry subscription update when domain' + ' object changes', function () { controller.registerChangeListeners(); //'watches' object is populated by fake scope watch and // watchCollection functions defined above expect(watches.domainObject).toBeDefined(); watches.domainObject(mockDomainObject); expect(controller.subscribe).toHaveBeenCalled(); }); it('triggers telemetry subscription update when domain' + ' object composition changes', function () { controller.registerChangeListeners(); expect(watches['domainObject.getModel().composition']).toBeDefined(); watches['domainObject.getModel().composition']([], []); expect(controller.subscribe).toHaveBeenCalled(); }); it('triggers telemetry subscription update when time' + ' conductor bounds change', function () { controller.registerChangeListeners(); expect(watches['telemetry:display:bounds']).toBeDefined(); watches['telemetry:display:bounds'](); expect(controller.subscribe).toHaveBeenCalled(); }); it('triggers refiltering of the columns when configuration' + ' changes', function () { controller.setup(); expect(watches['domainObject.getModel().configuration.table.columns']).toBeDefined(); watches['domainObject.getModel().configuration.table.columns'](); expect(controller.filterColumns).toHaveBeenCalled(); }); }); describe('Yields thread', function () { var mockSeries, mockRow; beforeEach(function () { mockSeries = { getPointCount: function () { return 5; }, getDomainValue: function () { return 'Domain Value'; }, getRangeValue: function () { return 'Range Value'; } }; mockRow = {'domain': 'Domain Value', 'range': 'Range Value'}; mockTelemetryHandle.makeDatum.andCallFake(function () { return mockRow; }); mockTable.getRowValues.andReturn(mockRow); mockTelemetryHandle.getTelemetryObjects.andReturn([mockDomainObject]); mockTelemetryHandle.getSeries.andReturn(mockSeries); }); it('when row count exceeds batch size', function () { controller.batchSize = 3; controller.addHistoricalData(mockDomainObject, mockSeries); //Timeout is called a minimum of two times expect(mockAngularTimeout).toHaveBeenCalled(); mockAngularTimeout.mostRecentCall.args[0](); expect(mockAngularTimeout.calls.length).toEqual(2); mockAngularTimeout.mostRecentCall.args[0](); //Because it yields, timeout will have been called a // third time for the batch. expect(mockAngularTimeout.calls.length).toEqual(3); mockAngularTimeout.mostRecentCall.args[0](); expect(controller.$scope.rows.length).toBe(5); expect(controller.$scope.rows[0]).toBe(mockRow); }); it('cancelling any outstanding timeouts', function () { controller.batchSize = 3; controller.addHistoricalData(mockDomainObject, mockSeries); expect(mockAngularTimeout).toHaveBeenCalled(); mockAngularTimeout.mostRecentCall.args[0](); controller.addHistoricalData(mockDomainObject, mockSeries); expect(mockAngularTimeout.cancel).toHaveBeenCalledWith(mockTimeoutHandle); }); it('cancels timeout on scope destruction', function () { controller.batchSize = 3; controller.addHistoricalData(mockDomainObject, mockSeries); //Destroy is used by parent class as well, so multiple // calls are made to scope.$on var destroyCalls = mockScope.$on.calls.filter(function (call) { return call.args[0] === '$destroy'; }); //Call destroy function expect(destroyCalls.length).toEqual(2); destroyCalls[0].args[1](); expect(mockAngularTimeout.cancel).toHaveBeenCalledWith(mockTimeoutHandle); }); }); }); } );