datasource.test.ts 41 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210
  1. import _ from 'lodash';
  2. import moment from 'moment';
  3. import q from 'q';
  4. import {
  5. alignRange,
  6. determineQueryHints,
  7. extractRuleMappingFromGroups,
  8. PrometheusDatasource,
  9. prometheusSpecialRegexEscape,
  10. prometheusRegularEscape,
  11. } from '../datasource';
  12. jest.mock('../metric_find_query');
  13. const DEFAULT_TEMPLATE_SRV_MOCK = {
  14. getAdhocFilters: () => [],
  15. replace: a => a,
  16. };
  17. describe('PrometheusDatasource', () => {
  18. const ctx: any = {};
  19. const instanceSettings = {
  20. url: 'proxied',
  21. directUrl: 'direct',
  22. user: 'test',
  23. password: 'mupp',
  24. jsonData: {},
  25. };
  26. ctx.backendSrvMock = {};
  27. ctx.templateSrvMock = DEFAULT_TEMPLATE_SRV_MOCK;
  28. ctx.timeSrvMock = {
  29. timeRange: () => {
  30. return {
  31. from: moment(1531468681),
  32. to: moment(1531489712),
  33. };
  34. },
  35. };
  36. beforeEach(() => {
  37. ctx.ds = new PrometheusDatasource(instanceSettings, q, ctx.backendSrvMock, ctx.templateSrvMock, ctx.timeSrvMock);
  38. });
  39. describe('Datasource metadata requests', () => {
  40. it('should perform a GET request with the default config', () => {
  41. ctx.backendSrvMock.datasourceRequest = jest.fn();
  42. ctx.ds.metadataRequest('/foo');
  43. expect(ctx.backendSrvMock.datasourceRequest.mock.calls.length).toBe(1);
  44. expect(ctx.backendSrvMock.datasourceRequest.mock.calls[0][0].method).toBe('GET');
  45. });
  46. it('should still perform a GET request with the DS HTTP method set to POST', () => {
  47. ctx.backendSrvMock.datasourceRequest = jest.fn();
  48. const postSettings = _.cloneDeep(instanceSettings);
  49. postSettings.jsonData.httpMethod = 'POST';
  50. const ds = new PrometheusDatasource(postSettings, q, ctx.backendSrvMock, ctx.templateSrvMock, ctx.timeSrvMock);
  51. ds.metadataRequest('/foo');
  52. expect(ctx.backendSrvMock.datasourceRequest.mock.calls.length).toBe(1);
  53. expect(ctx.backendSrvMock.datasourceRequest.mock.calls[0][0].method).toBe('GET');
  54. });
  55. });
  56. describe('When using adhoc filters', () => {
  57. const DEFAULT_QUERY_EXPRESSION = 'metric{job="foo"} - metric';
  58. const target = { expr: DEFAULT_QUERY_EXPRESSION };
  59. afterEach(() => {
  60. ctx.templateSrvMock.getAdhocFilters = DEFAULT_TEMPLATE_SRV_MOCK.getAdhocFilters;
  61. });
  62. it('should not modify expression with no filters', () => {
  63. const result = ctx.ds.createQuery(target, { interval: '15s' });
  64. expect(result).toMatchObject({ expr: DEFAULT_QUERY_EXPRESSION });
  65. });
  66. it('should add filters to expression', () => {
  67. ctx.templateSrvMock.getAdhocFilters = () => [
  68. {
  69. key: 'k1',
  70. operator: '=',
  71. value: 'v1',
  72. },
  73. {
  74. key: 'k2',
  75. operator: '!=',
  76. value: 'v2',
  77. },
  78. ];
  79. const result = ctx.ds.createQuery(target, { interval: '15s' });
  80. expect(result).toMatchObject({ expr: 'metric{job="foo",k1="v1",k2!="v2"} - metric{k1="v1",k2!="v2"}' });
  81. });
  82. });
  83. describe('When performing performSuggestQuery', () => {
  84. it('should cache response', async () => {
  85. ctx.backendSrvMock.datasourceRequest.mockReturnValue(
  86. Promise.resolve({
  87. status: 'success',
  88. data: { data: ['value1', 'value2', 'value3'] },
  89. })
  90. );
  91. let results = await ctx.ds.performSuggestQuery('value', true);
  92. expect(results).toHaveLength(3);
  93. ctx.backendSrvMock.datasourceRequest.mockReset();
  94. results = await ctx.ds.performSuggestQuery('value', true);
  95. expect(results).toHaveLength(3);
  96. });
  97. });
  98. describe('When converting prometheus histogram to heatmap format', () => {
  99. beforeEach(() => {
  100. ctx.query = {
  101. range: { from: moment(1443454528000), to: moment(1443454528000) },
  102. targets: [{ expr: 'test{job="testjob"}', format: 'heatmap', legendFormat: '{{le}}' }],
  103. interval: '1s',
  104. };
  105. });
  106. it('should convert cumullative histogram to ordinary', () => {
  107. const resultMock = [
  108. {
  109. metric: { __name__: 'metric', job: 'testjob', le: '10' },
  110. values: [[1443454528.0, '10'], [1443454528.0, '10']],
  111. },
  112. {
  113. metric: { __name__: 'metric', job: 'testjob', le: '20' },
  114. values: [[1443454528.0, '20'], [1443454528.0, '10']],
  115. },
  116. {
  117. metric: { __name__: 'metric', job: 'testjob', le: '30' },
  118. values: [[1443454528.0, '25'], [1443454528.0, '10']],
  119. },
  120. ];
  121. const responseMock = { data: { data: { result: resultMock } } };
  122. const expected = [
  123. {
  124. target: '10',
  125. datapoints: [[10, 1443454528000], [10, 1443454528000]],
  126. },
  127. {
  128. target: '20',
  129. datapoints: [[10, 1443454528000], [0, 1443454528000]],
  130. },
  131. {
  132. target: '30',
  133. datapoints: [[5, 1443454528000], [0, 1443454528000]],
  134. },
  135. ];
  136. ctx.ds.performTimeSeriesQuery = jest.fn().mockReturnValue(responseMock);
  137. return ctx.ds.query(ctx.query).then(result => {
  138. const results = result.data;
  139. return expect(results).toMatchObject(expected);
  140. });
  141. });
  142. it('should sort series by label value', () => {
  143. const resultMock = [
  144. {
  145. metric: { __name__: 'metric', job: 'testjob', le: '2' },
  146. values: [[1443454528.0, '10'], [1443454528.0, '10']],
  147. },
  148. {
  149. metric: { __name__: 'metric', job: 'testjob', le: '4' },
  150. values: [[1443454528.0, '20'], [1443454528.0, '10']],
  151. },
  152. {
  153. metric: { __name__: 'metric', job: 'testjob', le: '+Inf' },
  154. values: [[1443454528.0, '25'], [1443454528.0, '10']],
  155. },
  156. {
  157. metric: { __name__: 'metric', job: 'testjob', le: '1' },
  158. values: [[1443454528.0, '25'], [1443454528.0, '10']],
  159. },
  160. ];
  161. const responseMock = { data: { data: { result: resultMock } } };
  162. const expected = ['1', '2', '4', '+Inf'];
  163. ctx.ds.performTimeSeriesQuery = jest.fn().mockReturnValue(responseMock);
  164. return ctx.ds.query(ctx.query).then(result => {
  165. const seriesLabels = _.map(result.data, 'target');
  166. return expect(seriesLabels).toEqual(expected);
  167. });
  168. });
  169. });
  170. describe('alignRange', () => {
  171. it('does not modify already aligned intervals with perfect step', () => {
  172. const range = alignRange(0, 3, 3);
  173. expect(range.start).toEqual(0);
  174. expect(range.end).toEqual(3);
  175. });
  176. it('does modify end-aligned intervals to reflect number of steps possible', () => {
  177. const range = alignRange(1, 6, 3);
  178. expect(range.start).toEqual(0);
  179. expect(range.end).toEqual(6);
  180. });
  181. it('does align intervals that are a multiple of steps', () => {
  182. const range = alignRange(1, 4, 3);
  183. expect(range.start).toEqual(0);
  184. expect(range.end).toEqual(6);
  185. });
  186. it('does align intervals that are not a multiple of steps', () => {
  187. const range = alignRange(1, 5, 3);
  188. expect(range.start).toEqual(0);
  189. expect(range.end).toEqual(6);
  190. });
  191. });
  192. describe('determineQueryHints()', () => {
  193. it('returns no hints for no series', () => {
  194. expect(determineQueryHints([])).toEqual([]);
  195. });
  196. it('returns no hints for empty series', () => {
  197. expect(determineQueryHints([{ datapoints: [], query: '' }])).toEqual([null]);
  198. });
  199. it('returns no hint for a monotonously decreasing series', () => {
  200. const series = [{ datapoints: [[23, 1000], [22, 1001]], query: 'metric', responseIndex: 0 }];
  201. const hints = determineQueryHints(series);
  202. expect(hints).toEqual([null]);
  203. });
  204. it('returns a rate hint for a monotonously increasing series', () => {
  205. const series = [{ datapoints: [[23, 1000], [24, 1001]], query: 'metric', responseIndex: 0 }];
  206. const hints = determineQueryHints(series);
  207. expect(hints.length).toBe(1);
  208. expect(hints[0]).toMatchObject({
  209. label: 'Time series is monotonously increasing.',
  210. index: 0,
  211. fix: {
  212. action: {
  213. type: 'ADD_RATE',
  214. query: 'metric',
  215. },
  216. },
  217. });
  218. });
  219. it('returns a rate hint w/o action for a complex monotonously increasing series', () => {
  220. const series = [{ datapoints: [[23, 1000], [24, 1001]], query: 'sum(metric)', responseIndex: 0 }];
  221. const hints = determineQueryHints(series);
  222. expect(hints.length).toBe(1);
  223. expect(hints[0].label).toContain('rate()');
  224. expect(hints[0].fix).toBeUndefined();
  225. });
  226. it('returns a rate hint for a monotonously increasing series with missing data', () => {
  227. const series = [{ datapoints: [[23, 1000], [null, 1001], [24, 1002]], query: 'metric', responseIndex: 0 }];
  228. const hints = determineQueryHints(series);
  229. expect(hints.length).toBe(1);
  230. expect(hints[0]).toMatchObject({
  231. label: 'Time series is monotonously increasing.',
  232. index: 0,
  233. fix: {
  234. action: {
  235. type: 'ADD_RATE',
  236. query: 'metric',
  237. },
  238. },
  239. });
  240. });
  241. it('returns a histogram hint for a bucket series', () => {
  242. const series = [{ datapoints: [[23, 1000]], query: 'metric_bucket', responseIndex: 0 }];
  243. const hints = determineQueryHints(series);
  244. expect(hints.length).toBe(1);
  245. expect(hints[0]).toMatchObject({
  246. label: 'Time series has buckets, you probably wanted a histogram.',
  247. index: 0,
  248. fix: {
  249. action: {
  250. type: 'ADD_HISTOGRAM_QUANTILE',
  251. query: 'metric_bucket',
  252. },
  253. },
  254. });
  255. });
  256. });
  257. describe('extractRuleMappingFromGroups()', () => {
  258. it('returns empty mapping for no rule groups', () => {
  259. expect(extractRuleMappingFromGroups([])).toEqual({});
  260. });
  261. it('returns a mapping for recording rules only', () => {
  262. const groups = [
  263. {
  264. rules: [
  265. {
  266. name: 'HighRequestLatency',
  267. query: 'job:request_latency_seconds:mean5m{job="myjob"} > 0.5',
  268. type: 'alerting',
  269. },
  270. {
  271. name: 'job:http_inprogress_requests:sum',
  272. query: 'sum(http_inprogress_requests) by (job)',
  273. type: 'recording',
  274. },
  275. ],
  276. file: '/rules.yaml',
  277. interval: 60,
  278. name: 'example',
  279. },
  280. ];
  281. const mapping = extractRuleMappingFromGroups(groups);
  282. expect(mapping).toEqual({ 'job:http_inprogress_requests:sum': 'sum(http_inprogress_requests) by (job)' });
  283. });
  284. });
  285. describe('Prometheus regular escaping', () => {
  286. it('should not escape non-string', () => {
  287. expect(prometheusRegularEscape(12)).toEqual(12);
  288. });
  289. it('should not escape simple string', () => {
  290. expect(prometheusRegularEscape('cryptodepression')).toEqual('cryptodepression');
  291. });
  292. it("should escape '", () => {
  293. expect(prometheusRegularEscape("looking'glass")).toEqual("looking\\\\'glass");
  294. });
  295. it('should escape multiple characters', () => {
  296. expect(prometheusRegularEscape("'looking'glass'")).toEqual("\\\\'looking\\\\'glass\\\\'");
  297. });
  298. });
  299. describe('Prometheus regexes escaping', () => {
  300. it('should not escape simple string', () => {
  301. expect(prometheusSpecialRegexEscape('cryptodepression')).toEqual('cryptodepression');
  302. });
  303. it('should escape $^*+?.()\\', () => {
  304. expect(prometheusSpecialRegexEscape("looking'glass")).toEqual("looking\\\\'glass");
  305. expect(prometheusSpecialRegexEscape('looking{glass')).toEqual('looking\\\\{glass');
  306. expect(prometheusSpecialRegexEscape('looking}glass')).toEqual('looking\\\\}glass');
  307. expect(prometheusSpecialRegexEscape('looking[glass')).toEqual('looking\\\\[glass');
  308. expect(prometheusSpecialRegexEscape('looking]glass')).toEqual('looking\\\\]glass');
  309. expect(prometheusSpecialRegexEscape('looking$glass')).toEqual('looking\\\\$glass');
  310. expect(prometheusSpecialRegexEscape('looking^glass')).toEqual('looking\\\\^glass');
  311. expect(prometheusSpecialRegexEscape('looking*glass')).toEqual('looking\\\\*glass');
  312. expect(prometheusSpecialRegexEscape('looking+glass')).toEqual('looking\\\\+glass');
  313. expect(prometheusSpecialRegexEscape('looking?glass')).toEqual('looking\\\\?glass');
  314. expect(prometheusSpecialRegexEscape('looking.glass')).toEqual('looking\\\\.glass');
  315. expect(prometheusSpecialRegexEscape('looking(glass')).toEqual('looking\\\\(glass');
  316. expect(prometheusSpecialRegexEscape('looking)glass')).toEqual('looking\\\\)glass');
  317. expect(prometheusSpecialRegexEscape('looking\\glass')).toEqual('looking\\\\\\\\glass');
  318. });
  319. it('should escape multiple special characters', () => {
  320. expect(prometheusSpecialRegexEscape('+looking$glass?')).toEqual('\\\\+looking\\\\$glass\\\\?');
  321. });
  322. });
  323. describe('metricFindQuery', () => {
  324. beforeEach(() => {
  325. const query = 'query_result(topk(5,rate(http_request_duration_microseconds_count[$__interval])))';
  326. ctx.templateSrvMock.replace = jest.fn();
  327. ctx.timeSrvMock.timeRange = () => {
  328. return {
  329. from: moment(1531468681),
  330. to: moment(1531489712),
  331. };
  332. };
  333. ctx.ds = new PrometheusDatasource(instanceSettings, q, ctx.backendSrvMock, ctx.templateSrvMock, ctx.timeSrvMock);
  334. ctx.ds.metricFindQuery(query);
  335. });
  336. it('should call templateSrv.replace with scopedVars', () => {
  337. expect(ctx.templateSrvMock.replace.mock.calls[0][1]).toBeDefined();
  338. });
  339. it('should have the correct range and range_ms', () => {
  340. const range = ctx.templateSrvMock.replace.mock.calls[0][1].__range;
  341. const rangeMs = ctx.templateSrvMock.replace.mock.calls[0][1].__range_ms;
  342. const rangeS = ctx.templateSrvMock.replace.mock.calls[0][1].__range_s;
  343. expect(range).toEqual({ text: '21s', value: '21s' });
  344. expect(rangeMs).toEqual({ text: 21031, value: 21031 });
  345. expect(rangeS).toEqual({ text: 21, value: 21 });
  346. });
  347. it('should pass the default interval value', () => {
  348. const interval = ctx.templateSrvMock.replace.mock.calls[0][1].__interval;
  349. const intervalMs = ctx.templateSrvMock.replace.mock.calls[0][1].__interval_ms;
  350. expect(interval).toEqual({ text: '15s', value: '15s' });
  351. expect(intervalMs).toEqual({ text: 15000, value: 15000 });
  352. });
  353. });
  354. });
  355. const SECOND = 1000;
  356. const MINUTE = 60 * SECOND;
  357. const HOUR = 60 * MINUTE;
  358. const time = ({ hours = 0, seconds = 0, minutes = 0 }) => moment(hours * HOUR + minutes * MINUTE + seconds * SECOND);
  359. const ctx = {} as any;
  360. const instanceSettings = {
  361. url: 'proxied',
  362. directUrl: 'direct',
  363. user: 'test',
  364. password: 'mupp',
  365. jsonData: { httpMethod: 'GET' },
  366. };
  367. const backendSrv = {
  368. datasourceRequest: jest.fn(),
  369. } as any;
  370. const templateSrv = {
  371. getAdhocFilters: () => [],
  372. replace: jest.fn(str => str),
  373. };
  374. const timeSrv = {
  375. timeRange: () => {
  376. return { to: { diff: () => 2000 }, from: '' };
  377. },
  378. };
  379. describe('PrometheusDatasource', () => {
  380. describe('When querying prometheus with one target using query editor target spec', async () => {
  381. let results;
  382. const query = {
  383. range: { from: time({ seconds: 63 }), to: time({ seconds: 183 }) },
  384. targets: [{ expr: 'test{job="testjob"}', format: 'time_series' }],
  385. interval: '60s',
  386. };
  387. // Interval alignment with step
  388. const urlExpected =
  389. 'proxied/api/v1/query_range?query=' + encodeURIComponent('test{job="testjob"}') + '&start=60&end=240&step=60';
  390. beforeEach(async () => {
  391. const response = {
  392. data: {
  393. status: 'success',
  394. data: {
  395. resultType: 'matrix',
  396. result: [
  397. {
  398. metric: { __name__: 'test', job: 'testjob' },
  399. values: [[60, '3846']],
  400. },
  401. ],
  402. },
  403. },
  404. };
  405. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  406. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  407. await ctx.ds.query(query).then(data => {
  408. results = data;
  409. });
  410. });
  411. it('should generate the correct query', () => {
  412. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  413. expect(res.method).toBe('GET');
  414. expect(res.url).toBe(urlExpected);
  415. });
  416. it('should return series list', async () => {
  417. expect(results.data.length).toBe(1);
  418. expect(results.data[0].target).toBe('test{job="testjob"}');
  419. });
  420. });
  421. describe('When querying prometheus with one target which return multiple series', () => {
  422. let results;
  423. const start = 60;
  424. const end = 360;
  425. const step = 60;
  426. const query = {
  427. range: { from: time({ seconds: start }), to: time({ seconds: end }) },
  428. targets: [{ expr: 'test{job="testjob"}', format: 'time_series' }],
  429. interval: '60s',
  430. };
  431. beforeEach(async () => {
  432. const response = {
  433. status: 'success',
  434. data: {
  435. data: {
  436. resultType: 'matrix',
  437. result: [
  438. {
  439. metric: { __name__: 'test', job: 'testjob', series: 'series 1' },
  440. values: [[start + step * 1, '3846'], [start + step * 3, '3847'], [end - step * 1, '3848']],
  441. },
  442. {
  443. metric: { __name__: 'test', job: 'testjob', series: 'series 2' },
  444. values: [[start + step * 2, '4846']],
  445. },
  446. ],
  447. },
  448. },
  449. };
  450. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  451. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  452. await ctx.ds.query(query).then(data => {
  453. results = data;
  454. });
  455. });
  456. it('should be same length', () => {
  457. expect(results.data.length).toBe(2);
  458. expect(results.data[0].datapoints.length).toBe((end - start) / step + 1);
  459. expect(results.data[1].datapoints.length).toBe((end - start) / step + 1);
  460. });
  461. it('should fill null until first datapoint in response', () => {
  462. expect(results.data[0].datapoints[0][1]).toBe(start * 1000);
  463. expect(results.data[0].datapoints[0][0]).toBe(null);
  464. expect(results.data[0].datapoints[1][1]).toBe((start + step * 1) * 1000);
  465. expect(results.data[0].datapoints[1][0]).toBe(3846);
  466. });
  467. it('should fill null after last datapoint in response', () => {
  468. const length = (end - start) / step + 1;
  469. expect(results.data[0].datapoints[length - 2][1]).toBe((end - step * 1) * 1000);
  470. expect(results.data[0].datapoints[length - 2][0]).toBe(3848);
  471. expect(results.data[0].datapoints[length - 1][1]).toBe(end * 1000);
  472. expect(results.data[0].datapoints[length - 1][0]).toBe(null);
  473. });
  474. it('should fill null at gap between series', () => {
  475. expect(results.data[0].datapoints[2][1]).toBe((start + step * 2) * 1000);
  476. expect(results.data[0].datapoints[2][0]).toBe(null);
  477. expect(results.data[1].datapoints[1][1]).toBe((start + step * 1) * 1000);
  478. expect(results.data[1].datapoints[1][0]).toBe(null);
  479. expect(results.data[1].datapoints[3][1]).toBe((start + step * 3) * 1000);
  480. expect(results.data[1].datapoints[3][0]).toBe(null);
  481. });
  482. });
  483. describe('When querying prometheus with one target and instant = true', () => {
  484. let results;
  485. const urlExpected = 'proxied/api/v1/query?query=' + encodeURIComponent('test{job="testjob"}') + '&time=123';
  486. const query = {
  487. range: { from: time({ seconds: 63 }), to: time({ seconds: 123 }) },
  488. targets: [{ expr: 'test{job="testjob"}', format: 'time_series', instant: true }],
  489. interval: '60s',
  490. };
  491. beforeEach(async () => {
  492. const response = {
  493. status: 'success',
  494. data: {
  495. data: {
  496. resultType: 'vector',
  497. result: [
  498. {
  499. metric: { __name__: 'test', job: 'testjob' },
  500. value: [123, '3846'],
  501. },
  502. ],
  503. },
  504. },
  505. };
  506. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  507. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  508. await ctx.ds.query(query).then(data => {
  509. results = data;
  510. });
  511. });
  512. it('should generate the correct query', () => {
  513. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  514. expect(res.method).toBe('GET');
  515. expect(res.url).toBe(urlExpected);
  516. });
  517. it('should return series list', () => {
  518. expect(results.data.length).toBe(1);
  519. expect(results.data[0].target).toBe('test{job="testjob"}');
  520. });
  521. });
  522. describe('When performing annotationQuery', () => {
  523. let results;
  524. const options: any = {
  525. annotation: {
  526. expr: 'ALERTS{alertstate="firing"}',
  527. tagKeys: 'job',
  528. titleFormat: '{{alertname}}',
  529. textFormat: '{{instance}}',
  530. },
  531. range: {
  532. from: time({ seconds: 63 }),
  533. to: time({ seconds: 123 }),
  534. },
  535. };
  536. const response = {
  537. status: 'success',
  538. data: {
  539. data: {
  540. resultType: 'matrix',
  541. result: [
  542. {
  543. metric: {
  544. __name__: 'ALERTS',
  545. alertname: 'InstanceDown',
  546. alertstate: 'firing',
  547. instance: 'testinstance',
  548. job: 'testjob',
  549. },
  550. values: [[123, '1']],
  551. },
  552. ],
  553. },
  554. },
  555. };
  556. describe('not use useValueForTime', () => {
  557. beforeEach(async () => {
  558. options.annotation.useValueForTime = false;
  559. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  560. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  561. await ctx.ds.annotationQuery(options).then(data => {
  562. results = data;
  563. });
  564. });
  565. it('should return annotation list', () => {
  566. expect(results.length).toBe(1);
  567. expect(results[0].tags).toContain('testjob');
  568. expect(results[0].title).toBe('InstanceDown');
  569. expect(results[0].text).toBe('testinstance');
  570. expect(results[0].time).toBe(123 * 1000);
  571. });
  572. });
  573. describe('use useValueForTime', () => {
  574. beforeEach(async () => {
  575. options.annotation.useValueForTime = true;
  576. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  577. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  578. await ctx.ds.annotationQuery(options).then(data => {
  579. results = data;
  580. });
  581. });
  582. it('should return annotation list', () => {
  583. expect(results[0].time).toEqual(1);
  584. });
  585. });
  586. });
  587. describe('When resultFormat is table and instant = true', () => {
  588. let results;
  589. const query = {
  590. range: { from: time({ seconds: 63 }), to: time({ seconds: 123 }) },
  591. targets: [{ expr: 'test{job="testjob"}', format: 'time_series', instant: true }],
  592. interval: '60s',
  593. };
  594. beforeEach(async () => {
  595. const response = {
  596. status: 'success',
  597. data: {
  598. data: {
  599. resultType: 'vector',
  600. result: [
  601. {
  602. metric: { __name__: 'test', job: 'testjob' },
  603. value: [123, '3846'],
  604. },
  605. ],
  606. },
  607. },
  608. };
  609. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  610. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  611. await ctx.ds.query(query).then(data => {
  612. results = data;
  613. });
  614. });
  615. it('should return result', () => {
  616. expect(results).not.toBe(null);
  617. });
  618. });
  619. describe('The "step" query parameter', () => {
  620. const response = {
  621. status: 'success',
  622. data: {
  623. data: {
  624. resultType: 'matrix',
  625. result: [],
  626. },
  627. },
  628. };
  629. it('should be min interval when greater than auto interval', async () => {
  630. const query = {
  631. // 6 minute range
  632. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  633. targets: [
  634. {
  635. expr: 'test',
  636. interval: '10s',
  637. },
  638. ],
  639. interval: '5s',
  640. };
  641. const urlExpected = 'proxied/api/v1/query_range?query=test&start=60&end=420&step=10';
  642. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  643. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  644. await ctx.ds.query(query);
  645. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  646. expect(res.method).toBe('GET');
  647. expect(res.url).toBe(urlExpected);
  648. });
  649. it('step should never go below 1', async () => {
  650. const query = {
  651. // 6 minute range
  652. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  653. targets: [{ expr: 'test' }],
  654. interval: '100ms',
  655. };
  656. const urlExpected = 'proxied/api/v1/query_range?query=test&start=60&end=420&step=1';
  657. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  658. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  659. await ctx.ds.query(query);
  660. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  661. expect(res.method).toBe('GET');
  662. expect(res.url).toBe(urlExpected);
  663. });
  664. it('should be auto interval when greater than min interval', async () => {
  665. const query = {
  666. // 6 minute range
  667. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  668. targets: [
  669. {
  670. expr: 'test',
  671. interval: '5s',
  672. },
  673. ],
  674. interval: '10s',
  675. };
  676. const urlExpected = 'proxied/api/v1/query_range?query=test&start=60&end=420&step=10';
  677. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  678. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  679. await ctx.ds.query(query);
  680. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  681. expect(res.method).toBe('GET');
  682. expect(res.url).toBe(urlExpected);
  683. });
  684. it('should result in querying fewer than 11000 data points', async () => {
  685. const query = {
  686. // 6 hour range
  687. range: { from: time({ hours: 1 }), to: time({ hours: 7 }) },
  688. targets: [{ expr: 'test' }],
  689. interval: '1s',
  690. };
  691. const end = 7 * 60 * 60;
  692. const start = 60 * 60;
  693. const urlExpected = 'proxied/api/v1/query_range?query=test&start=' + start + '&end=' + end + '&step=2';
  694. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  695. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  696. await ctx.ds.query(query);
  697. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  698. expect(res.method).toBe('GET');
  699. expect(res.url).toBe(urlExpected);
  700. });
  701. it('should not apply min interval when interval * intervalFactor greater', async () => {
  702. const query = {
  703. // 6 minute range
  704. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  705. targets: [
  706. {
  707. expr: 'test',
  708. interval: '10s',
  709. intervalFactor: 10,
  710. },
  711. ],
  712. interval: '5s',
  713. };
  714. // times get rounded up to interval
  715. const urlExpected = 'proxied/api/v1/query_range?query=test&start=50&end=450&step=50';
  716. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  717. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  718. await ctx.ds.query(query);
  719. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  720. expect(res.method).toBe('GET');
  721. expect(res.url).toBe(urlExpected);
  722. });
  723. it('should apply min interval when interval * intervalFactor smaller', async () => {
  724. const query = {
  725. // 6 minute range
  726. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  727. targets: [
  728. {
  729. expr: 'test',
  730. interval: '15s',
  731. intervalFactor: 2,
  732. },
  733. ],
  734. interval: '5s',
  735. };
  736. const urlExpected = 'proxied/api/v1/query_range?query=test' + '&start=60&end=420&step=15';
  737. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  738. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  739. await ctx.ds.query(query);
  740. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  741. expect(res.method).toBe('GET');
  742. expect(res.url).toBe(urlExpected);
  743. });
  744. it('should apply intervalFactor to auto interval when greater', async () => {
  745. const query = {
  746. // 6 minute range
  747. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  748. targets: [
  749. {
  750. expr: 'test',
  751. interval: '5s',
  752. intervalFactor: 10,
  753. },
  754. ],
  755. interval: '10s',
  756. };
  757. // times get aligned to interval
  758. const urlExpected = 'proxied/api/v1/query_range?query=test' + '&start=0&end=500&step=100';
  759. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  760. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  761. await ctx.ds.query(query);
  762. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  763. expect(res.method).toBe('GET');
  764. expect(res.url).toBe(urlExpected);
  765. });
  766. it('should not not be affected by the 11000 data points limit when large enough', async () => {
  767. const query = {
  768. // 1 week range
  769. range: { from: time({}), to: time({ hours: 7 * 24 }) },
  770. targets: [
  771. {
  772. expr: 'test',
  773. intervalFactor: 10,
  774. },
  775. ],
  776. interval: '10s',
  777. };
  778. const end = 7 * 24 * 60 * 60;
  779. const start = 0;
  780. const urlExpected = 'proxied/api/v1/query_range?query=test' + '&start=' + start + '&end=' + end + '&step=100';
  781. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  782. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  783. await ctx.ds.query(query);
  784. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  785. expect(res.method).toBe('GET');
  786. expect(res.url).toBe(urlExpected);
  787. });
  788. it('should be determined by the 11000 data points limit when too small', async () => {
  789. const query = {
  790. // 1 week range
  791. range: { from: time({}), to: time({ hours: 7 * 24 }) },
  792. targets: [
  793. {
  794. expr: 'test',
  795. intervalFactor: 10,
  796. },
  797. ],
  798. interval: '5s',
  799. };
  800. const end = 7 * 24 * 60 * 60;
  801. const start = 0;
  802. const urlExpected = 'proxied/api/v1/query_range?query=test' + '&start=' + start + '&end=' + end + '&step=60';
  803. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  804. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  805. await ctx.ds.query(query);
  806. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  807. expect(res.method).toBe('GET');
  808. expect(res.url).toBe(urlExpected);
  809. });
  810. });
  811. describe('The __interval and __interval_ms template variables', () => {
  812. const response = {
  813. status: 'success',
  814. data: {
  815. data: {
  816. resultType: 'matrix',
  817. result: [],
  818. },
  819. },
  820. };
  821. it('should be unchanged when auto interval is greater than min interval', async () => {
  822. const query = {
  823. // 6 minute range
  824. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  825. targets: [
  826. {
  827. expr: 'rate(test[$__interval])',
  828. interval: '5s',
  829. },
  830. ],
  831. interval: '10s',
  832. scopedVars: {
  833. __interval: { text: '10s', value: '10s' },
  834. __interval_ms: { text: 10 * 1000, value: 10 * 1000 },
  835. },
  836. };
  837. const urlExpected =
  838. 'proxied/api/v1/query_range?query=' +
  839. encodeURIComponent('rate(test[$__interval])') +
  840. '&start=60&end=420&step=10';
  841. templateSrv.replace = jest.fn(str => str);
  842. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  843. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  844. await ctx.ds.query(query);
  845. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  846. expect(res.method).toBe('GET');
  847. expect(res.url).toBe(urlExpected);
  848. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  849. __interval: {
  850. text: '10s',
  851. value: '10s',
  852. },
  853. __interval_ms: {
  854. text: 10000,
  855. value: 10000,
  856. },
  857. });
  858. });
  859. it('should be min interval when it is greater than auto interval', async () => {
  860. const query = {
  861. // 6 minute range
  862. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  863. targets: [
  864. {
  865. expr: 'rate(test[$__interval])',
  866. interval: '10s',
  867. },
  868. ],
  869. interval: '5s',
  870. scopedVars: {
  871. __interval: { text: '5s', value: '5s' },
  872. __interval_ms: { text: 5 * 1000, value: 5 * 1000 },
  873. },
  874. };
  875. const urlExpected =
  876. 'proxied/api/v1/query_range?query=' +
  877. encodeURIComponent('rate(test[$__interval])') +
  878. '&start=60&end=420&step=10';
  879. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  880. templateSrv.replace = jest.fn(str => str);
  881. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  882. await ctx.ds.query(query);
  883. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  884. expect(res.method).toBe('GET');
  885. expect(res.url).toBe(urlExpected);
  886. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  887. __interval: {
  888. text: '5s',
  889. value: '5s',
  890. },
  891. __interval_ms: {
  892. text: 5000,
  893. value: 5000,
  894. },
  895. });
  896. });
  897. it('should account for intervalFactor', async () => {
  898. const query = {
  899. // 6 minute range
  900. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  901. targets: [
  902. {
  903. expr: 'rate(test[$__interval])',
  904. interval: '5s',
  905. intervalFactor: 10,
  906. },
  907. ],
  908. interval: '10s',
  909. scopedVars: {
  910. __interval: { text: '10s', value: '10s' },
  911. __interval_ms: { text: 10 * 1000, value: 10 * 1000 },
  912. },
  913. };
  914. const urlExpected =
  915. 'proxied/api/v1/query_range?query=' +
  916. encodeURIComponent('rate(test[$__interval])') +
  917. '&start=0&end=500&step=100';
  918. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  919. templateSrv.replace = jest.fn(str => str);
  920. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  921. await ctx.ds.query(query);
  922. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  923. expect(res.method).toBe('GET');
  924. expect(res.url).toBe(urlExpected);
  925. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  926. __interval: {
  927. text: '10s',
  928. value: '10s',
  929. },
  930. __interval_ms: {
  931. text: 10000,
  932. value: 10000,
  933. },
  934. });
  935. expect(query.scopedVars.__interval.text).toBe('10s');
  936. expect(query.scopedVars.__interval.value).toBe('10s');
  937. expect(query.scopedVars.__interval_ms.text).toBe(10 * 1000);
  938. expect(query.scopedVars.__interval_ms.value).toBe(10 * 1000);
  939. });
  940. it('should be interval * intervalFactor when greater than min interval', async () => {
  941. const query = {
  942. // 6 minute range
  943. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  944. targets: [
  945. {
  946. expr: 'rate(test[$__interval])',
  947. interval: '10s',
  948. intervalFactor: 10,
  949. },
  950. ],
  951. interval: '5s',
  952. scopedVars: {
  953. __interval: { text: '5s', value: '5s' },
  954. __interval_ms: { text: 5 * 1000, value: 5 * 1000 },
  955. },
  956. };
  957. const urlExpected =
  958. 'proxied/api/v1/query_range?query=' +
  959. encodeURIComponent('rate(test[$__interval])') +
  960. '&start=50&end=450&step=50';
  961. templateSrv.replace = jest.fn(str => str);
  962. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  963. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  964. await ctx.ds.query(query);
  965. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  966. expect(res.method).toBe('GET');
  967. expect(res.url).toBe(urlExpected);
  968. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  969. __interval: {
  970. text: '5s',
  971. value: '5s',
  972. },
  973. __interval_ms: {
  974. text: 5000,
  975. value: 5000,
  976. },
  977. });
  978. });
  979. it('should be min interval when greater than interval * intervalFactor', async () => {
  980. const query = {
  981. // 6 minute range
  982. range: { from: time({ minutes: 1 }), to: time({ minutes: 7 }) },
  983. targets: [
  984. {
  985. expr: 'rate(test[$__interval])',
  986. interval: '15s',
  987. intervalFactor: 2,
  988. },
  989. ],
  990. interval: '5s',
  991. scopedVars: {
  992. __interval: { text: '5s', value: '5s' },
  993. __interval_ms: { text: 5 * 1000, value: 5 * 1000 },
  994. },
  995. };
  996. const urlExpected =
  997. 'proxied/api/v1/query_range?query=' +
  998. encodeURIComponent('rate(test[$__interval])') +
  999. '&start=60&end=420&step=15';
  1000. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  1001. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  1002. await ctx.ds.query(query);
  1003. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  1004. expect(res.method).toBe('GET');
  1005. expect(res.url).toBe(urlExpected);
  1006. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  1007. __interval: {
  1008. text: '5s',
  1009. value: '5s',
  1010. },
  1011. __interval_ms: {
  1012. text: 5000,
  1013. value: 5000,
  1014. },
  1015. });
  1016. });
  1017. it('should be determined by the 11000 data points limit, accounting for intervalFactor', async () => {
  1018. const query = {
  1019. // 1 week range
  1020. range: { from: time({}), to: time({ hours: 7 * 24 }) },
  1021. targets: [
  1022. {
  1023. expr: 'rate(test[$__interval])',
  1024. intervalFactor: 10,
  1025. },
  1026. ],
  1027. interval: '5s',
  1028. scopedVars: {
  1029. __interval: { text: '5s', value: '5s' },
  1030. __interval_ms: { text: 5 * 1000, value: 5 * 1000 },
  1031. },
  1032. };
  1033. const end = 7 * 24 * 60 * 60;
  1034. const start = 0;
  1035. const urlExpected =
  1036. 'proxied/api/v1/query_range?query=' +
  1037. encodeURIComponent('rate(test[$__interval])') +
  1038. '&start=' +
  1039. start +
  1040. '&end=' +
  1041. end +
  1042. '&step=60';
  1043. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  1044. templateSrv.replace = jest.fn(str => str);
  1045. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  1046. await ctx.ds.query(query);
  1047. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  1048. expect(res.method).toBe('GET');
  1049. expect(res.url).toBe(urlExpected);
  1050. expect(templateSrv.replace.mock.calls[0][1]).toEqual({
  1051. __interval: {
  1052. text: '5s',
  1053. value: '5s',
  1054. },
  1055. __interval_ms: {
  1056. text: 5000,
  1057. value: 5000,
  1058. },
  1059. });
  1060. });
  1061. });
  1062. });
  1063. describe('PrometheusDatasource for POST', () => {
  1064. // const ctx = new helpers.ServiceTestContext();
  1065. const instanceSettings = {
  1066. url: 'proxied',
  1067. directUrl: 'direct',
  1068. user: 'test',
  1069. password: 'mupp',
  1070. jsonData: { httpMethod: 'POST' },
  1071. };
  1072. describe('When querying prometheus with one target using query editor target spec', () => {
  1073. let results;
  1074. const urlExpected = 'proxied/api/v1/query_range';
  1075. const dataExpected = {
  1076. query: 'test{job="testjob"}',
  1077. start: 1 * 60,
  1078. end: 3 * 60,
  1079. step: 60,
  1080. };
  1081. const query = {
  1082. range: { from: time({ minutes: 1, seconds: 3 }), to: time({ minutes: 2, seconds: 3 }) },
  1083. targets: [{ expr: 'test{job="testjob"}', format: 'time_series' }],
  1084. interval: '60s',
  1085. };
  1086. beforeEach(async () => {
  1087. const response = {
  1088. status: 'success',
  1089. data: {
  1090. data: {
  1091. resultType: 'matrix',
  1092. result: [
  1093. {
  1094. metric: { __name__: 'test', job: 'testjob' },
  1095. values: [[2 * 60, '3846']],
  1096. },
  1097. ],
  1098. },
  1099. },
  1100. };
  1101. backendSrv.datasourceRequest = jest.fn(() => Promise.resolve(response));
  1102. ctx.ds = new PrometheusDatasource(instanceSettings, q, backendSrv as any, templateSrv, timeSrv);
  1103. await ctx.ds.query(query).then(data => {
  1104. results = data;
  1105. });
  1106. });
  1107. it('should generate the correct query', () => {
  1108. const res = backendSrv.datasourceRequest.mock.calls[0][0];
  1109. expect(res.method).toBe('POST');
  1110. expect(res.url).toBe(urlExpected);
  1111. expect(res.data).toEqual(dataExpected);
  1112. });
  1113. it('should return series list', () => {
  1114. expect(results.data.length).toBe(1);
  1115. expect(results.data[0].target).toBe('test{job="testjob"}');
  1116. });
  1117. });
  1118. });