import { lastValueFrom, of } from 'rxjs'; import { createFetchResponse } from 'test/helpers/createFetchResponse'; import { DataQueryRequest, DataSourceInstanceSettings, DataSourcePluginMeta, FieldType } from '@grafana/data'; import { BackendSrv, TemplateSrv } from '@grafana/runtime'; import { ZipkinDatasource } from './datasource'; import mockJson from './mockJsonResponse.json'; import { ZipkinQuery, ZipkinSpan } from './types'; import { traceFrameFields, zipkinResponse } from './utils/testData'; export const backendSrv = { fetch: jest.fn() } as unknown as BackendSrv; jest.mock('@grafana/runtime', () => ({ ...jest.requireActual('@grafana/runtime'), getBackendSrv: () => backendSrv, })); describe('ZipkinDatasource', () => { describe('query', () => { const templateSrv: TemplateSrv = { replace: jest.fn(), getVariables: jest.fn(), containsTemplate: jest.fn(), updateTimeRange: jest.fn(), }; it('runs query', async () => { setupBackendSrv(zipkinResponse); const ds = new ZipkinDatasource(defaultSettings, templateSrv); await expect(ds.query({ targets: [{ query: '12345' }] } as DataQueryRequest)).toEmitValuesWith( (val) => { expect(val[0].data[0].fields).toMatchObject(traceFrameFields); } ); }); it('runs query with traceId that includes special characters', async () => { setupBackendSrv(zipkinResponse); const ds = new ZipkinDatasource(defaultSettings, templateSrv); await expect(ds.query({ targets: [{ query: 'a/b' }] } as DataQueryRequest)).toEmitValuesWith( (val) => { expect(val[0].data[0].fields).toMatchObject(traceFrameFields); } ); }); it('should handle json file upload', async () => { const ds = new ZipkinDatasource(defaultSettings); ds.uploadedJson = JSON.stringify(mockJson); const response = await lastValueFrom( ds.query({ targets: [{ queryType: 'upload', refId: 'A' }], } as DataQueryRequest) ); const field = response.data[0].fields[0]; expect(field.name).toBe('traceID'); expect(field.type).toBe(FieldType.string); expect(field.values.length).toBe(3); }); it('should fail on invalid json file upload', async () => { const ds = new ZipkinDatasource(defaultSettings); ds.uploadedJson = JSON.stringify({ key: 'value', arr: [] }); const response = await lastValueFrom( ds.query({ targets: [{ queryType: 'upload', refId: 'A' }], } as DataQueryRequest) ); expect(response.error?.message).toBeDefined(); expect(response.data.length).toBe(0); }); }); describe('metadataRequest', () => { it('runs query', async () => { setupBackendSrv(['service 1', 'service 2'] as unknown as ZipkinSpan[]); const ds = new ZipkinDatasource(defaultSettings); const response = await ds.metadataRequest('/api/v2/services'); expect(response).toEqual(['service 1', 'service 2']); }); }); }); function setupBackendSrv(response: ZipkinSpan[]) { const defaultMock = () => of(createFetchResponse(response)); const fetchMock = jest.spyOn(backendSrv, 'fetch'); fetchMock.mockImplementation(defaultMock); } const defaultSettings: DataSourceInstanceSettings = { id: 1, uid: '1', type: 'tracing', name: 'zipkin', meta: {} as DataSourcePluginMeta, jsonData: {}, access: 'proxy', readOnly: false, };