renderer_specs.ts 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. import { describe, beforeEach, it, sinon, expect, angularMocks } from '../../../../../test/lib/common';
  2. import '../module';
  3. import angular from 'angular';
  4. import $ from 'jquery';
  5. import helpers from 'test/specs/helpers';
  6. import TimeSeries from 'app/core/time_series2';
  7. import moment from 'moment';
  8. import { Emitter } from 'app/core/core';
  9. import rendering from '../rendering';
  10. import { convertToHeatMap, convertToCards, histogramToHeatmap, calculateBucketSize } from '../heatmap_data_converter';
  11. describe('grafanaHeatmap', function() {
  12. beforeEach(angularMocks.module('grafana.core'));
  13. function heatmapScenario(desc, func, elementWidth = 500) {
  14. describe(desc, function() {
  15. var ctx: any = {};
  16. ctx.setup = function(setupFunc) {
  17. beforeEach(
  18. angularMocks.module(function($provide) {
  19. $provide.value('timeSrv', new helpers.TimeSrvStub());
  20. })
  21. );
  22. beforeEach(
  23. angularMocks.inject(function($rootScope, $compile) {
  24. var ctrl: any = {
  25. colorSchemes: [
  26. {
  27. name: 'Oranges',
  28. value: 'interpolateOranges',
  29. invert: 'dark',
  30. },
  31. { name: 'Reds', value: 'interpolateReds', invert: 'dark' },
  32. ],
  33. events: new Emitter(),
  34. height: 200,
  35. panel: {
  36. heatmap: {},
  37. cards: {
  38. cardPadding: null,
  39. cardRound: null,
  40. },
  41. color: {
  42. mode: 'spectrum',
  43. cardColor: '#b4ff00',
  44. colorScale: 'linear',
  45. exponent: 0.5,
  46. colorScheme: 'interpolateOranges',
  47. fillBackground: false,
  48. },
  49. legend: {
  50. show: false,
  51. },
  52. xBucketSize: 1000,
  53. xBucketNumber: null,
  54. yBucketSize: 1,
  55. yBucketNumber: null,
  56. xAxis: {
  57. show: true,
  58. },
  59. yAxis: {
  60. show: true,
  61. format: 'short',
  62. decimals: null,
  63. logBase: 1,
  64. splitFactor: null,
  65. min: null,
  66. max: null,
  67. removeZeroValues: false,
  68. },
  69. tooltip: {
  70. show: true,
  71. seriesStat: false,
  72. showHistogram: false,
  73. },
  74. highlightCards: true,
  75. },
  76. renderingCompleted: sinon.spy(),
  77. hiddenSeries: {},
  78. dashboard: {
  79. getTimezone: sinon.stub().returns('utc'),
  80. },
  81. range: {
  82. from: moment.utc('01 Mar 2017 10:00:00', 'DD MMM YYYY HH:mm:ss'),
  83. to: moment.utc('01 Mar 2017 11:00:00', 'DD MMM YYYY HH:mm:ss'),
  84. },
  85. };
  86. var scope = $rootScope.$new();
  87. scope.ctrl = ctrl;
  88. ctx.series = [];
  89. ctx.series.push(
  90. new TimeSeries({
  91. datapoints: [[1, 1422774000000], [2, 1422774060000]],
  92. alias: 'series1',
  93. })
  94. );
  95. ctx.series.push(
  96. new TimeSeries({
  97. datapoints: [[2, 1422774000000], [3, 1422774060000]],
  98. alias: 'series2',
  99. })
  100. );
  101. ctx.data = {
  102. heatmapStats: {
  103. min: 1,
  104. max: 3,
  105. minLog: 1,
  106. },
  107. xBucketSize: ctrl.panel.xBucketSize,
  108. yBucketSize: ctrl.panel.yBucketSize,
  109. };
  110. setupFunc(ctrl, ctx);
  111. let logBase = ctrl.panel.yAxis.logBase;
  112. let bucketsData;
  113. if (ctrl.panel.dataFormat === 'tsbuckets') {
  114. bucketsData = histogramToHeatmap(ctx.series);
  115. } else {
  116. bucketsData = convertToHeatMap(ctx.series, ctx.data.yBucketSize, ctx.data.xBucketSize, logBase);
  117. }
  118. ctx.data.buckets = bucketsData;
  119. let { cards, cardStats } = convertToCards(bucketsData);
  120. ctx.data.cards = cards;
  121. ctx.data.cardStats = cardStats;
  122. let elemHtml = `
  123. <div class="heatmap-wrapper">
  124. <div class="heatmap-canvas-wrapper">
  125. <div class="heatmap-panel" style='width:${elementWidth}px'></div>
  126. </div>
  127. </div>`;
  128. var element = angular.element(elemHtml);
  129. $compile(element)(scope);
  130. scope.$digest();
  131. ctrl.data = ctx.data;
  132. ctx.element = element;
  133. rendering(scope, $(element), [], ctrl);
  134. ctrl.events.emit('render');
  135. })
  136. );
  137. };
  138. func(ctx);
  139. });
  140. }
  141. heatmapScenario('default options', function(ctx) {
  142. ctx.setup(function(ctrl) {
  143. ctrl.panel.yAxis.logBase = 1;
  144. });
  145. it('should draw correct Y axis', function() {
  146. var yTicks = getTicks(ctx.element, '.axis-y');
  147. expect(yTicks).to.eql(['1', '2', '3']);
  148. });
  149. it('should draw correct X axis', function() {
  150. var xTicks = getTicks(ctx.element, '.axis-x');
  151. let expectedTicks = [
  152. formatTime('01 Mar 2017 10:00:00'),
  153. formatTime('01 Mar 2017 10:15:00'),
  154. formatTime('01 Mar 2017 10:30:00'),
  155. formatTime('01 Mar 2017 10:45:00'),
  156. formatTime('01 Mar 2017 11:00:00'),
  157. ];
  158. expect(xTicks).to.eql(expectedTicks);
  159. });
  160. });
  161. heatmapScenario('when logBase is 2', function(ctx) {
  162. ctx.setup(function(ctrl) {
  163. ctrl.panel.yAxis.logBase = 2;
  164. });
  165. it('should draw correct Y axis', function() {
  166. var yTicks = getTicks(ctx.element, '.axis-y');
  167. expect(yTicks).to.eql(['1', '2', '4']);
  168. });
  169. });
  170. heatmapScenario('when logBase is 10', function(ctx) {
  171. ctx.setup(function(ctrl, ctx) {
  172. ctrl.panel.yAxis.logBase = 10;
  173. ctx.series.push(
  174. new TimeSeries({
  175. datapoints: [[10, 1422774000000], [20, 1422774060000]],
  176. alias: 'series3',
  177. })
  178. );
  179. ctx.data.heatmapStats.max = 20;
  180. });
  181. it('should draw correct Y axis', function() {
  182. var yTicks = getTicks(ctx.element, '.axis-y');
  183. expect(yTicks).to.eql(['1', '10', '100']);
  184. });
  185. });
  186. heatmapScenario('when logBase is 32', function(ctx) {
  187. ctx.setup(function(ctrl) {
  188. ctrl.panel.yAxis.logBase = 32;
  189. ctx.series.push(
  190. new TimeSeries({
  191. datapoints: [[10, 1422774000000], [100, 1422774060000]],
  192. alias: 'series3',
  193. })
  194. );
  195. ctx.data.heatmapStats.max = 100;
  196. });
  197. it('should draw correct Y axis', function() {
  198. var yTicks = getTicks(ctx.element, '.axis-y');
  199. expect(yTicks).to.eql(['1', '32', '1.0 K']);
  200. });
  201. });
  202. heatmapScenario('when logBase is 1024', function(ctx) {
  203. ctx.setup(function(ctrl) {
  204. ctrl.panel.yAxis.logBase = 1024;
  205. ctx.series.push(
  206. new TimeSeries({
  207. datapoints: [[2000, 1422774000000], [300000, 1422774060000]],
  208. alias: 'series3',
  209. })
  210. );
  211. ctx.data.heatmapStats.max = 300000;
  212. });
  213. it('should draw correct Y axis', function() {
  214. var yTicks = getTicks(ctx.element, '.axis-y');
  215. expect(yTicks).to.eql(['1', '1 K', '1.0 Mil']);
  216. });
  217. });
  218. heatmapScenario('when Y axis format set to "none"', function(ctx) {
  219. ctx.setup(function(ctrl) {
  220. ctrl.panel.yAxis.logBase = 1;
  221. ctrl.panel.yAxis.format = 'none';
  222. ctx.data.heatmapStats.max = 10000;
  223. });
  224. it('should draw correct Y axis', function() {
  225. var yTicks = getTicks(ctx.element, '.axis-y');
  226. expect(yTicks).to.eql(['0', '2000', '4000', '6000', '8000', '10000', '12000']);
  227. });
  228. });
  229. heatmapScenario('when Y axis format set to "second"', function(ctx) {
  230. ctx.setup(function(ctrl) {
  231. ctrl.panel.yAxis.logBase = 1;
  232. ctrl.panel.yAxis.format = 's';
  233. ctx.data.heatmapStats.max = 3600;
  234. });
  235. it('should draw correct Y axis', function() {
  236. var yTicks = getTicks(ctx.element, '.axis-y');
  237. expect(yTicks).to.eql(['0 ns', '17 min', '33 min', '50 min', '1.11 hour']);
  238. });
  239. });
  240. heatmapScenario('when data format is Time series buckets', function(ctx) {
  241. ctx.setup(function(ctrl, ctx) {
  242. ctrl.panel.dataFormat = 'tsbuckets';
  243. const series = [
  244. {
  245. alias: '1',
  246. datapoints: [[1000, 1422774000000], [200000, 1422774060000]],
  247. },
  248. {
  249. alias: '2',
  250. datapoints: [[3000, 1422774000000], [400000, 1422774060000]],
  251. },
  252. {
  253. alias: '3',
  254. datapoints: [[2000, 1422774000000], [300000, 1422774060000]],
  255. },
  256. ];
  257. ctx.series = series.map(s => new TimeSeries(s));
  258. ctx.data.tsBuckets = series.map(s => s.alias).concat('');
  259. ctx.data.yBucketSize = 1;
  260. let xBucketBoundSet = series[0].datapoints.map(dp => dp[1]);
  261. ctx.data.xBucketSize = calculateBucketSize(xBucketBoundSet);
  262. });
  263. it('should draw correct Y axis', function() {
  264. var yTicks = getTicks(ctx.element, '.axis-y');
  265. expect(yTicks).to.eql(['1', '2', '3', '']);
  266. });
  267. });
  268. });
  269. function getTicks(element, axisSelector) {
  270. return element
  271. .find(axisSelector)
  272. .find('text')
  273. .map(function() {
  274. return this.textContent;
  275. })
  276. .get();
  277. }
  278. function formatTime(timeStr) {
  279. let format = 'HH:mm';
  280. return moment.utc(timeStr, 'DD MMM YYYY HH:mm:ss').format(format);
  281. }