datasource.jest.ts 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. import '../datasource';
  2. import { TemplateSrvStub } from 'test/specs/helpers';
  3. import CloudWatchDatasource from '../datasource';
  4. import 'app/features/dashboard/time_srv';
  5. import * as dateMath from 'app/core/utils/datemath';
  6. describe('CloudWatchDatasource', function() {
  7. let instanceSettings = {
  8. jsonData: { defaultRegion: 'us-east-1', access: 'proxy' },
  9. };
  10. let templateSrv = new TemplateSrvStub();
  11. let timeSrv = {
  12. time: { from: 'now-1h', to: 'now' },
  13. timeRange: jest.fn(() => {
  14. return {
  15. from: dateMath.parse(timeSrv.time.from, false),
  16. to: dateMath.parse(timeSrv.time.to, true),
  17. };
  18. }),
  19. };
  20. let backendSrv = {};
  21. let ctx = <any>{
  22. backendSrv,
  23. templateSrv,
  24. };
  25. beforeEach(() => {
  26. ctx.ds = new CloudWatchDatasource(instanceSettings, {}, backendSrv, templateSrv, timeSrv);
  27. });
  28. describe('When performing CloudWatch query', function() {
  29. var requestParams;
  30. var query = {
  31. range: { from: 'now-1h', to: 'now' },
  32. rangeRaw: { from: 1483228800, to: 1483232400 },
  33. targets: [
  34. {
  35. region: 'us-east-1',
  36. namespace: 'AWS/EC2',
  37. metricName: 'CPUUtilization',
  38. dimensions: {
  39. InstanceId: 'i-12345678',
  40. },
  41. statistics: ['Average'],
  42. period: '300',
  43. },
  44. ],
  45. };
  46. var response = {
  47. timings: [null],
  48. results: {
  49. A: {
  50. error: '',
  51. refId: 'A',
  52. series: [
  53. {
  54. name: 'CPUUtilization_Average',
  55. points: [[1, 1483228800000], [2, 1483229100000], [5, 1483229700000]],
  56. tags: {
  57. InstanceId: 'i-12345678',
  58. },
  59. },
  60. ],
  61. },
  62. },
  63. };
  64. beforeEach(async () => {
  65. ctx.backendSrv.datasourceRequest = await jest.fn(params => {
  66. requestParams = params.data;
  67. return Promise.resolve({ data: response });
  68. });
  69. });
  70. it('should generate the correct query', function(done) {
  71. ctx.ds.query(query).then(function() {
  72. var params = requestParams.queries[0];
  73. expect(params.namespace).toBe(query.targets[0].namespace);
  74. expect(params.metricName).toBe(query.targets[0].metricName);
  75. expect(params.dimensions['InstanceId']).toBe('i-12345678');
  76. expect(params.statistics).toEqual(query.targets[0].statistics);
  77. expect(params.period).toBe(query.targets[0].period);
  78. done();
  79. });
  80. });
  81. it('should generate the correct query with interval variable', function(done) {
  82. ctx.templateSrv.data = {
  83. period: '10m',
  84. };
  85. var query = {
  86. range: { from: 'now-1h', to: 'now' },
  87. rangeRaw: { from: 1483228800, to: 1483232400 },
  88. targets: [
  89. {
  90. region: 'us-east-1',
  91. namespace: 'AWS/EC2',
  92. metricName: 'CPUUtilization',
  93. dimensions: {
  94. InstanceId: 'i-12345678',
  95. },
  96. statistics: ['Average'],
  97. period: '[[period]]',
  98. },
  99. ],
  100. };
  101. ctx.ds.query(query).then(function() {
  102. var params = requestParams.queries[0];
  103. expect(params.period).toBe('600');
  104. done();
  105. });
  106. });
  107. it('should return series list', function(done) {
  108. ctx.ds.query(query).then(function(result) {
  109. expect(result.data[0].target).toBe(response.results.A.series[0].name);
  110. expect(result.data[0].datapoints[0][0]).toBe(response.results.A.series[0].points[0][0]);
  111. done();
  112. });
  113. });
  114. it('should generate the correct targets by expanding template variables', function() {
  115. var templateSrv = {
  116. variables: [
  117. {
  118. name: 'instance_id',
  119. options: [
  120. { text: 'i-23456789', value: 'i-23456789', selected: false },
  121. { text: 'i-34567890', value: 'i-34567890', selected: true },
  122. ],
  123. current: {
  124. text: 'i-34567890',
  125. value: 'i-34567890',
  126. },
  127. },
  128. ],
  129. replace: function(target, scopedVars) {
  130. if (target === '$instance_id' && scopedVars['instance_id']['text'] === 'i-34567890') {
  131. return 'i-34567890';
  132. } else {
  133. return '';
  134. }
  135. },
  136. getVariableName: function(e) {
  137. return 'instance_id';
  138. },
  139. variableExists: function(e) {
  140. return true;
  141. },
  142. containsVariable: function(str, variableName) {
  143. return str.indexOf('$' + variableName) !== -1;
  144. },
  145. };
  146. var targets = [
  147. {
  148. region: 'us-east-1',
  149. namespace: 'AWS/EC2',
  150. metricName: 'CPUUtilization',
  151. dimensions: {
  152. InstanceId: '$instance_id',
  153. },
  154. statistics: ['Average'],
  155. period: 300,
  156. },
  157. ];
  158. var result = ctx.ds.expandTemplateVariable(targets, {}, templateSrv);
  159. expect(result[0].dimensions.InstanceId).toBe('i-34567890');
  160. });
  161. it('should generate the correct targets by expanding template variables from url', function() {
  162. var templateSrv = {
  163. variables: [
  164. {
  165. name: 'instance_id',
  166. options: [
  167. { text: 'i-23456789', value: 'i-23456789', selected: false },
  168. { text: 'i-34567890', value: 'i-34567890', selected: false },
  169. ],
  170. current: 'i-45678901',
  171. },
  172. ],
  173. replace: function(target, scopedVars) {
  174. if (target === '$instance_id') {
  175. return 'i-45678901';
  176. } else {
  177. return '';
  178. }
  179. },
  180. getVariableName: function(e) {
  181. return 'instance_id';
  182. },
  183. variableExists: function(e) {
  184. return true;
  185. },
  186. containsVariable: function(str, variableName) {
  187. return str.indexOf('$' + variableName) !== -1;
  188. },
  189. };
  190. var targets = [
  191. {
  192. region: 'us-east-1',
  193. namespace: 'AWS/EC2',
  194. metricName: 'CPUUtilization',
  195. dimensions: {
  196. InstanceId: '$instance_id',
  197. },
  198. statistics: ['Average'],
  199. period: 300,
  200. },
  201. ];
  202. var result = ctx.ds.expandTemplateVariable(targets, {}, templateSrv);
  203. expect(result[0].dimensions.InstanceId).toBe('i-45678901');
  204. });
  205. });
  206. describe('When query region is "default"', function() {
  207. it('should return the datasource region if empty or "default"', function() {
  208. var defaultRegion = instanceSettings.jsonData.defaultRegion;
  209. expect(ctx.ds.getActualRegion()).toBe(defaultRegion);
  210. expect(ctx.ds.getActualRegion('')).toBe(defaultRegion);
  211. expect(ctx.ds.getActualRegion('default')).toBe(defaultRegion);
  212. });
  213. it('should return the specified region if specified', function() {
  214. expect(ctx.ds.getActualRegion('some-fake-region-1')).toBe('some-fake-region-1');
  215. });
  216. var requestParams;
  217. beforeEach(function() {
  218. ctx.ds.performTimeSeriesQuery = jest.fn(request => {
  219. requestParams = request;
  220. return Promise.resolve({ data: {} });
  221. });
  222. });
  223. it('should query for the datasource region if empty or "default"', function(done) {
  224. var query = {
  225. range: { from: 'now-1h', to: 'now' },
  226. rangeRaw: { from: 1483228800, to: 1483232400 },
  227. targets: [
  228. {
  229. region: 'default',
  230. namespace: 'AWS/EC2',
  231. metricName: 'CPUUtilization',
  232. dimensions: {
  233. InstanceId: 'i-12345678',
  234. },
  235. statistics: ['Average'],
  236. period: 300,
  237. },
  238. ],
  239. };
  240. ctx.ds.query(query).then(function(result) {
  241. expect(requestParams.queries[0].region).toBe(instanceSettings.jsonData.defaultRegion);
  242. done();
  243. });
  244. });
  245. });
  246. describe('When performing CloudWatch query for extended statistics', function() {
  247. var query = {
  248. range: { from: 'now-1h', to: 'now' },
  249. rangeRaw: { from: 1483228800, to: 1483232400 },
  250. targets: [
  251. {
  252. region: 'us-east-1',
  253. namespace: 'AWS/ApplicationELB',
  254. metricName: 'TargetResponseTime',
  255. dimensions: {
  256. LoadBalancer: 'lb',
  257. TargetGroup: 'tg',
  258. },
  259. statistics: ['p90.00'],
  260. period: 300,
  261. },
  262. ],
  263. };
  264. var response = {
  265. timings: [null],
  266. results: {
  267. A: {
  268. error: '',
  269. refId: 'A',
  270. series: [
  271. {
  272. name: 'TargetResponseTime_p90.00',
  273. points: [[1, 1483228800000], [2, 1483229100000], [5, 1483229700000]],
  274. tags: {
  275. LoadBalancer: 'lb',
  276. TargetGroup: 'tg',
  277. },
  278. },
  279. ],
  280. },
  281. },
  282. };
  283. beforeEach(function() {
  284. ctx.backendSrv.datasourceRequest = jest.fn(params => {
  285. return Promise.resolve({ data: response });
  286. });
  287. });
  288. it('should return series list', function(done) {
  289. ctx.ds.query(query).then(function(result) {
  290. expect(result.data[0].target).toBe(response.results.A.series[0].name);
  291. expect(result.data[0].datapoints[0][0]).toBe(response.results.A.series[0].points[0][0]);
  292. done();
  293. });
  294. });
  295. });
  296. function describeMetricFindQuery(query, func) {
  297. describe('metricFindQuery ' + query, () => {
  298. let scenario: any = {};
  299. scenario.setup = setupCallback => {
  300. beforeEach(() => {
  301. setupCallback();
  302. ctx.backendSrv.datasourceRequest = jest.fn(args => {
  303. scenario.request = args.data;
  304. return Promise.resolve({ data: scenario.requestResponse });
  305. });
  306. ctx.ds.metricFindQuery(query).then(args => {
  307. scenario.result = args;
  308. });
  309. });
  310. };
  311. func(scenario);
  312. });
  313. }
  314. describeMetricFindQuery('regions()', scenario => {
  315. scenario.setup(() => {
  316. scenario.requestResponse = {
  317. results: {
  318. metricFindQuery: {
  319. tables: [{ rows: [['us-east-1', 'us-east-1']] }],
  320. },
  321. },
  322. };
  323. });
  324. it('should call __GetRegions and return result', () => {
  325. expect(scenario.result[0].text).toContain('us-east-1');
  326. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  327. expect(scenario.request.queries[0].subtype).toBe('regions');
  328. });
  329. });
  330. describeMetricFindQuery('namespaces()', scenario => {
  331. scenario.setup(() => {
  332. scenario.requestResponse = {
  333. results: {
  334. metricFindQuery: {
  335. tables: [{ rows: [['AWS/EC2', 'AWS/EC2']] }],
  336. },
  337. },
  338. };
  339. });
  340. it('should call __GetNamespaces and return result', () => {
  341. expect(scenario.result[0].text).toContain('AWS/EC2');
  342. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  343. expect(scenario.request.queries[0].subtype).toBe('namespaces');
  344. });
  345. });
  346. describeMetricFindQuery('metrics(AWS/EC2)', scenario => {
  347. scenario.setup(() => {
  348. scenario.requestResponse = {
  349. results: {
  350. metricFindQuery: {
  351. tables: [{ rows: [['CPUUtilization', 'CPUUtilization']] }],
  352. },
  353. },
  354. };
  355. });
  356. it('should call __GetMetrics and return result', () => {
  357. expect(scenario.result[0].text).toBe('CPUUtilization');
  358. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  359. expect(scenario.request.queries[0].subtype).toBe('metrics');
  360. });
  361. });
  362. describeMetricFindQuery('dimension_keys(AWS/EC2)', scenario => {
  363. scenario.setup(() => {
  364. scenario.requestResponse = {
  365. results: {
  366. metricFindQuery: {
  367. tables: [{ rows: [['InstanceId', 'InstanceId']] }],
  368. },
  369. },
  370. };
  371. });
  372. it('should call __GetDimensions and return result', () => {
  373. expect(scenario.result[0].text).toBe('InstanceId');
  374. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  375. expect(scenario.request.queries[0].subtype).toBe('dimension_keys');
  376. });
  377. });
  378. describeMetricFindQuery('dimension_values(us-east-1,AWS/EC2,CPUUtilization,InstanceId)', scenario => {
  379. scenario.setup(() => {
  380. scenario.requestResponse = {
  381. results: {
  382. metricFindQuery: {
  383. tables: [{ rows: [['i-12345678', 'i-12345678']] }],
  384. },
  385. },
  386. };
  387. });
  388. it('should call __ListMetrics and return result', () => {
  389. expect(scenario.result[0].text).toContain('i-12345678');
  390. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  391. expect(scenario.request.queries[0].subtype).toBe('dimension_values');
  392. });
  393. });
  394. describeMetricFindQuery('dimension_values(default,AWS/EC2,CPUUtilization,InstanceId)', scenario => {
  395. scenario.setup(() => {
  396. scenario.requestResponse = {
  397. results: {
  398. metricFindQuery: {
  399. tables: [{ rows: [['i-12345678', 'i-12345678']] }],
  400. },
  401. },
  402. };
  403. });
  404. it('should call __ListMetrics and return result', () => {
  405. expect(scenario.result[0].text).toContain('i-12345678');
  406. expect(scenario.request.queries[0].type).toBe('metricFindQuery');
  407. expect(scenario.request.queries[0].subtype).toBe('dimension_values');
  408. });
  409. });
  410. it('should caclculate the correct period', function() {
  411. var hourSec = 60 * 60;
  412. var daySec = hourSec * 24;
  413. var start = 1483196400 * 1000;
  414. var testData: any[] = [
  415. [
  416. { period: 60, namespace: 'AWS/EC2' },
  417. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  418. hourSec * 3,
  419. 60,
  420. ],
  421. [
  422. { period: null, namespace: 'AWS/EC2' },
  423. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  424. hourSec * 3,
  425. 300,
  426. ],
  427. [
  428. { period: 60, namespace: 'AWS/ELB' },
  429. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  430. hourSec * 3,
  431. 60,
  432. ],
  433. [
  434. { period: null, namespace: 'AWS/ELB' },
  435. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  436. hourSec * 3,
  437. 60,
  438. ],
  439. [
  440. { period: 1, namespace: 'CustomMetricsNamespace' },
  441. {
  442. range: {
  443. from: new Date(start),
  444. to: new Date(start + (1440 - 1) * 1000),
  445. },
  446. },
  447. hourSec * 3 - 1,
  448. 1,
  449. ],
  450. [
  451. { period: 1, namespace: 'CustomMetricsNamespace' },
  452. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  453. hourSec * 3 - 1,
  454. 60,
  455. ],
  456. [
  457. { period: 60, namespace: 'CustomMetricsNamespace' },
  458. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  459. hourSec * 3,
  460. 60,
  461. ],
  462. [
  463. { period: null, namespace: 'CustomMetricsNamespace' },
  464. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  465. hourSec * 3 - 1,
  466. 60,
  467. ],
  468. [
  469. { period: null, namespace: 'CustomMetricsNamespace' },
  470. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  471. hourSec * 3,
  472. 60,
  473. ],
  474. [
  475. { period: null, namespace: 'CustomMetricsNamespace' },
  476. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  477. daySec * 15,
  478. 60,
  479. ],
  480. [
  481. { period: null, namespace: 'CustomMetricsNamespace' },
  482. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  483. daySec * 63,
  484. 300,
  485. ],
  486. [
  487. { period: null, namespace: 'CustomMetricsNamespace' },
  488. { range: { from: new Date(start), to: new Date(start + 3600 * 1000) } },
  489. daySec * 455,
  490. 3600,
  491. ],
  492. ];
  493. for (let t of testData) {
  494. let target = t[0];
  495. let options = t[1];
  496. let now = new Date(options.range.from.valueOf() + t[2] * 1000);
  497. let expected = t[3];
  498. let actual = ctx.ds.getPeriod(target, options, now);
  499. expect(actual).toBe(expected);
  500. }
  501. });
  502. });