Mercurial > hg > thermostat-ng > web-client
view src/app/components/jvm-info/jvm-gc/jvm-gc.controller.spec.js @ 245:2eb2ae0f3b3f
Transition project to Angular 4 Hybrid
Reviewed-by: jkang
Review-thread: http://icedtea.classpath.org/pipermail/thermostat/2017-October/025266.html
author | Andrew Azores <aazores@redhat.com> |
---|---|
date | Fri, 13 Oct 2017 14:05:09 -0400 |
parents | 07e0f128e67c |
children | 9f3ee8157ac6 |
line wrap: on
line source
/** * Copyright 2012-2017 Red Hat, Inc. * * Thermostat is distributed under the GNU General Public License, * version 2 or any later version (with a special exception described * below, commonly known as the "Classpath Exception"). * * A copy of GNU General Public License (GPL) is included in this * distribution, in the file COPYING. * * Linking Thermostat code with other modules is making a combined work * based on Thermostat. Thus, the terms and conditions of the GPL * cover the whole combination. * * As a special exception, the copyright holders of Thermostat give you * permission to link this code with independent modules to produce an * executable, regardless of the license terms of these independent * modules, and to copy and distribute the resulting executable under * terms of your choice, provided that you also meet, for each linked * independent module, the terms and conditions of the license of that * module. An independent module is a module which is not derived from * or based on Thermostat code. If you modify Thermostat, you may * extend this exception to your version of the software, but you are * not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. */ import controllerModule from './jvm-gc.controller.js'; import filtersModule from 'shared/filters/filters.module.js'; import { default as servicesModule, init as initServices } from 'shared/services/services.module.js'; describe('JvmGcController', () => { let interval, dateFilterStub, dateFormatSpy, svc, promise, ctrl, translate, sanitizeService; beforeEach(() => { angular.mock.module(filtersModule); angular.mock.module(servicesModule); initServices(); angular.mock.module(controllerModule); angular.mock.inject(($controller) => { 'ngInject'; dateFilterStub = sinon.stub().returns('mockDate'); dateFormatSpy = { time: { medium: sinon.spy() } }; interval = sinon.stub().returns('interval-sentinel'); interval.cancel = sinon.spy(); promise = { then: sinon.spy() }; svc = { getJvmGcData: sinon.stub().returns(promise) }; sanitizeService = { sanitize: sinon.spy() }; translate = sinon.stub().returns({ then: sinon.stub().yields({ 'jvmGc.chart.UNITS': 'microseconds', 'jvmGc.chart.X_AXIS_LABEL': 'timestamp', 'jvmGc.chart.Y_AXIS_LABEL': 'elapsed' }) }); ctrl = $controller('JvmGcController', { $stateParams: { jvmId: 'foo-jvmId' }, $interval: interval, dateFilter: dateFilterStub, DATE_FORMAT: dateFormatSpy, jvmGcService: svc, sanitizeService: sanitizeService, $translate: translate }); ctrl.$onInit(); }); }); it('should exist', () => { should.exist(ctrl); }); it('should update on init', () => { svc.getJvmGcData.should.be.calledWith('foo-jvmId'); }); it('should call to service on update', () => { svc.getJvmGcData.should.be.calledOnce(); promise.then.should.be.calledOnce(); ctrl.update(); svc.getJvmGcData.should.be.calledTwice(); promise.then.should.be.calledTwice(); promise.then.secondCall.should.be.calledWith(sinon.match.func, sinon.match.func); ctrl.collectors.should.deepEqual([]); let successHandler = promise.then.args[1][0]; successHandler({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: '100' }, wallTimeInMicros: { $numberLong: '50' } } ] } }); ctrl.collectors.should.deepEqual(['fooCollector']); let errorHandler = promise.then.args[1][1]; errorHandler.should.equal(angular.noop); errorHandler(); }); it('should reset interval on refreshRate change', () => { ctrl.should.have.ownProperty('_refresh'); ctrl.refreshRate = '1'; interval.should.be.calledWith(sinon.match.func, sinon.match(1)); ctrl.should.have.ownProperty('_refresh'); ctrl._refresh.should.equal('interval-sentinel'); ctrl.refreshRate.should.equal('1'); }); it('should trim data on dataAgeLimit change', () => { sinon.spy(ctrl, 'trimData'); ctrl.trimData.should.not.be.called(); ctrl.dataAgeLimit = 10000; ctrl.trimData.should.be.calledOnce(); ctrl.trimData.restore(); ctrl.dataAgeLimit.should.equal('10000'); }); it('should trim old data', () => { let oldSample = { micros: 100, timestamp: 1 }; let futureSample = { micros: 200, timestamp: Date.now() + 600000 }; ctrl.collectorData.set('fooCollector', [oldSample, futureSample]); ctrl.dataAgeLimit = '30000'; let expected = new Map(); expected.set('fooCollector', [futureSample]); ctrl.collectorData.should.deepEqual(expected); }); it('should set interval on start', () => { interval.should.be.calledOnce(); interval.should.be.calledWith(sinon.match.func, '1000'); interval.cancel.should.not.be.called(); }); it('should disable when set refreshRate is called with a non-positive value', () => { interval.cancel.should.not.be.called(); ctrl.update.should.not.be.called(); ctrl.refreshRate = '1'; interval.cancel.should.be.calledOnce(); ctrl.should.have.ownProperty('_refresh'); ctrl.refreshRate = '-1'; interval.cancel.should.be.calledTwice(); ctrl.should.not.have.ownProperty('_refresh'); }); it('should call controller#update() on refresh', () => { ctrl.refreshRate = 1; let func = interval.args[0][0]; let callCount = svc.getJvmGcData.callCount; func(); svc.getJvmGcData.callCount.should.equal(callCount + 1); }); describe('makeConfig', () => { let cfg; beforeEach(() => { ctrl.makeConfig('fooCollector'); cfg = ctrl.chartConfigs.fooCollector; }); it('should cache', () => { ctrl.collectors.should.deepEqual(['fooCollector']); ctrl.makeConfig('fooCollector'); ctrl.collectors.should.deepEqual(['fooCollector']); }); it('should sort', () => { ctrl.makeConfig('gooCollector'); ctrl.makeConfig('booCollector'); ctrl.collectors.should.deepEqual(['booCollector', 'fooCollector', 'gooCollector']); }); it('should set chartId', () => { cfg.chartId.should.equal('chart-fooCollector'); }); it('should use dateFilter with DATE_FORMAT.time.medium to format x ticks', () => { let fn = cfg.axis.x.tick.format; fn.should.be.a.Function(); fn('fooTimestamp').should.equal('mockDate'); dateFilterStub.should.be.calledWith('fooTimestamp', dateFormatSpy.time.medium); }); it('should format y ticks directly', () => { let fn = cfg.axis.y.tick.format; fn.should.be.a.Function(); fn(100).should.equal(100); }); it('should set tooltip', () => { let fmt = cfg.tooltip.format; fmt.should.have.ownProperty('title'); fmt.title.should.be.a.Function(); fmt.should.have.ownProperty('value'); fmt.value.should.be.a.Function(); fmt.title(100).should.equal(100); fmt.value(200).should.equal(200); }); it('should stop on mouseover', () => { ctrl.start(); let fn = cfg.onmouseover; fn.should.be.a.Function(); interval.cancel.should.be.calledOnce(); fn(); interval.cancel.should.be.calledTwice(); }); it('should start on mouseout', () => { ctrl.stop(); let fn = cfg.onmouseout; fn.should.be.a.Function(); interval.should.be.calledOnce(); fn(); interval.should.be.calledTwice(); }); }); describe('constructChartData', () => { it('should leave chartData empty if no collectorData present', () => { ctrl.collectorData = new Map(); ctrl.constructChartData; ctrl.chartData.should.deepEqual({}); }); it('should construct chartData according to collectorData', () => { let map = new Map(); map.set('fooCollector', [{ timestamp: 100, micros: 50 }, { timestamp: 101, micros: 60 }]); ctrl.collectorData = map; ctrl.constructChartData(); ctrl.chartData.should.deepEqual({ fooCollector: { xData: ['timestamps', 101], yData: ['fooCollector', 10] } }); }); }); describe('processData', () => { it('should process singleton service results', () => { ctrl.collectors.should.deepEqual([]); ctrl.chartConfigs.should.deepEqual({}); ctrl.collectorData.has('fooCollector').should.be.false(); let timestamp = Date.now().toString(); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestamp }, wallTimeInMicros: { $numberLong: '50' } } ] } }); ctrl.collectorData.has('fooCollector').should.be.true(); ctrl.collectorData.get('fooCollector').should.be.an.Array(); ctrl.collectorData.get('fooCollector').length.should.equal(1); ctrl.collectorData.get('fooCollector')[0].should.deepEqual({ timestamp: parseInt(timestamp), micros: 50 }); }); it('should process multiple service results', () => { ctrl.collectors.should.deepEqual([]); ctrl.chartConfigs.should.deepEqual({}); ctrl.collectorData.has('fooCollector').should.be.false(); let timestampA = Date.now().toString(); let timestampB = (Date.now() - 10).toString(); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestampA }, wallTimeInMicros: { $numberLong: '50' } }, { collectorName: 'fooCollector', timeStamp: { $numberLong: timestampB }, wallTimeInMicros: { $numberLong: '25' } } ] } }); ctrl.collectorData.has('fooCollector').should.be.true(); ctrl.collectorData.get('fooCollector').should.be.an.Array(); ctrl.collectorData.get('fooCollector').length.should.equal(2); ctrl.collectorData.get('fooCollector')[0].should.deepEqual({ timestamp: parseInt(timestampB), micros: 25 }); ctrl.collectorData.get('fooCollector')[1].should.deepEqual({ timestamp: parseInt(timestampA), micros: 50 }); }); it('should append new data', () => { let timestampA = Date.now().toString(); let timestampB = (Date.now() + 5000).toString(); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestampA }, wallTimeInMicros: { $numberLong: '50' } } ] } }); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestampB }, wallTimeInMicros: { $numberLong: '100' } } ] } }); ctrl.collectorData.has('fooCollector').should.be.true(); ctrl.collectorData.get('fooCollector').should.be.an.Array(); ctrl.collectorData.get('fooCollector').length.should.equal(2); ctrl.collectorData.get('fooCollector')[0].should.deepEqual({ timestamp: parseInt(timestampA), micros: 50 }); ctrl.collectorData.get('fooCollector')[1].should.deepEqual({ timestamp: parseInt(timestampB), micros: 100 }); }); it('should append a sample with duplicate elapsed time if no sample received for a collector', () => { let timestampA = Date.now().toString(); let timestampB = (Date.now() + 10).toString(); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestampA }, wallTimeInMicros: { $numberLong: '100' } } ] } }); ctrl.processData({ data: { response: [ { collectorName: 'barCollector', timeStamp: { $numberLong: timestampB }, wallTimeInMicros: { $numberLong: '200' } } ] } }); ctrl.collectorData.get('fooCollector').length.should.equal(2); ctrl.collectorData.get('fooCollector')[0].should.deepEqual({ timestamp: parseInt(timestampA), micros: 100 }); ctrl.collectorData.get('fooCollector')[1].should.deepEqual({ timestamp: parseInt(timestampB), micros: 100 }); ctrl.collectorData.get('barCollector').length.should.equal(1); ctrl.collectorData.get('barCollector')[0].should.deepEqual({ timestamp: parseInt(timestampB), micros: 200 }); }); it('should ignore duplicate timestamps', () => { let timestamp = Date.now().toString(); ctrl.processData({ data: { response: [ { collectorName: 'fooCollector', timeStamp: { $numberLong: timestamp }, wallTimeInMicros: { $numberLong: 200 } }, { collectorName: 'fooCollector', timeStamp: { $numberLong: timestamp }, wallTimeInMicros: { $numberLong: 100 } }, { collectorName: 'fooCollector', timeStamp: { $numberLong: timestamp }, wallTimeInMicros: { $numberLong: 100 } } ] } }); ctrl.collectorData.get('fooCollector').length.should.equal(1); // note: response is processed in reverse order, so 100 is seen first ctrl.collectorData.get('fooCollector')[0].should.deepEqual({ timestamp: parseInt(timestamp), micros: 100 }); }); }); describe('ondestroy handler', () => { it('should cancel refresh', () => { ctrl.$onDestroy(); interval.cancel.should.be.calledWith('interval-sentinel'); }); it('should do nothing if refresh is undefined', () => { delete ctrl._refresh; ctrl.$onDestroy(); interval.cancel.should.not.be.called(); }); }); describe('sanitize()', () => { it('should delegate to sanitizeService', () => { sanitizeService.sanitize.should.not.be.called(); ctrl.sanitize('foo'); sanitizeService.sanitize.should.be.calledOnce(); sanitizeService.sanitize.should.be.calledWith('foo'); }); }); describe('multichartFn', () => { it('should return a promise', () => { let res = ctrl.multichartFn(); res.should.be.a.Promise(); }); it('should resolve jvm-gc stat', done => { promise.then.should.be.calledOnce(); let res = ctrl.multichartFn(); res.then(v => { v.should.equal(400); done(); }); promise.then.should.be.calledTwice(); let prom = promise.then.secondCall.args[0]; prom({ data: { response: [ { wallTimeInMicros: { $numberLong: '400' } } ] } }); }); }); });