mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
286 lines
10 KiB
TypeScript
286 lines
10 KiB
TypeScript
import React from 'react';
|
||
import { render, screen, waitFor } from '@testing-library/react';
|
||
import { within } from '@testing-library/dom';
|
||
import userEvent from '@testing-library/user-event';
|
||
import { PanelPluginMeta, PluginType } from '@grafana/data';
|
||
|
||
import { LibraryPanelsSearch, LibraryPanelsSearchProps } from './LibraryPanelsSearch';
|
||
import * as api from '../../state/api';
|
||
import { LibraryElementKind, LibraryElementsSearchResult } from '../../types';
|
||
import { backendSrv } from '../../../../core/services/backend_srv';
|
||
import * as viztypepicker from '../../../panel/components/VizTypePicker/VizTypePicker';
|
||
|
||
jest.mock('@grafana/runtime', () => ({
|
||
...((jest.requireActual('@grafana/runtime') as unknown) as object),
|
||
config: {
|
||
panels: {
|
||
timeseries: {
|
||
info: { logos: { small: '' } },
|
||
name: 'Time Series',
|
||
},
|
||
},
|
||
},
|
||
}));
|
||
|
||
jest.mock('debounce-promise', () => {
|
||
const debounce = (fn: any) => {
|
||
const debounced = () =>
|
||
Promise.resolve([
|
||
{ label: 'General', value: { id: 0, title: 'General' } },
|
||
{ label: 'Folder1', value: { id: 1, title: 'Folder1' } },
|
||
{ label: 'Folder2', value: { id: 2, title: 'Folder2' } },
|
||
]);
|
||
return debounced;
|
||
};
|
||
|
||
return debounce;
|
||
});
|
||
|
||
async function getTestContext(
|
||
propOverrides: Partial<LibraryPanelsSearchProps> = {},
|
||
searchResult: LibraryElementsSearchResult = { elements: [], perPage: 40, page: 1, totalCount: 0 }
|
||
) {
|
||
jest.clearAllMocks();
|
||
const pluginInfo: any = { logos: { small: '', large: '' } };
|
||
const graph: PanelPluginMeta = {
|
||
name: 'Graph',
|
||
id: 'graph',
|
||
info: pluginInfo,
|
||
baseUrl: '',
|
||
type: PluginType.panel,
|
||
module: '',
|
||
sort: 0,
|
||
};
|
||
const timeseries: PanelPluginMeta = {
|
||
name: 'Time Series',
|
||
id: 'timeseries',
|
||
info: pluginInfo,
|
||
baseUrl: '',
|
||
type: PluginType.panel,
|
||
module: '',
|
||
sort: 1,
|
||
};
|
||
const getSpy = jest
|
||
.spyOn(backendSrv, 'get')
|
||
.mockResolvedValue({ sortOptions: [{ displaName: 'Desc', name: 'alpha-desc' }] });
|
||
const getLibraryPanelsSpy = jest.spyOn(api, 'getLibraryPanels').mockResolvedValue(searchResult);
|
||
const getAllPanelPluginMetaSpy = jest
|
||
.spyOn(viztypepicker, 'getAllPanelPluginMeta')
|
||
.mockReturnValue([graph, timeseries]);
|
||
|
||
const props: LibraryPanelsSearchProps = {
|
||
onClick: jest.fn(),
|
||
};
|
||
|
||
Object.assign(props, propOverrides);
|
||
const { rerender } = render(<LibraryPanelsSearch {...props} />);
|
||
|
||
await waitFor(() => expect(getLibraryPanelsSpy).toHaveBeenCalled());
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledTimes(1);
|
||
|
||
return { rerender, getLibraryPanelsSpy, getSpy, getAllPanelPluginMetaSpy };
|
||
}
|
||
|
||
describe('LibraryPanelsSearch', () => {
|
||
describe('when mounted with default options', () => {
|
||
it('should show input filter and library panels view', async () => {
|
||
await getTestContext();
|
||
|
||
expect(screen.getByPlaceholderText(/search by name/i)).toBeInTheDocument();
|
||
expect(screen.getByText(/no library panels found./i)).toBeInTheDocument();
|
||
});
|
||
|
||
describe('and user searches for library panel by name or description', () => {
|
||
it('should call api with correct params', async () => {
|
||
const { getLibraryPanelsSpy } = await getTestContext();
|
||
getLibraryPanelsSpy.mockClear();
|
||
|
||
await userEvent.type(screen.getByPlaceholderText(/search by name/i), 'a');
|
||
await waitFor(() => expect(getLibraryPanelsSpy).toHaveBeenCalled());
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledTimes(1);
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledWith({
|
||
searchString: 'a',
|
||
folderFilter: [],
|
||
page: 0,
|
||
typeFilter: [],
|
||
perPage: 40,
|
||
});
|
||
});
|
||
});
|
||
});
|
||
|
||
describe('when mounted with showSort', () => {
|
||
it('should show input filter and library panels view and sort', async () => {
|
||
await getTestContext({ showSort: true });
|
||
|
||
expect(screen.getByPlaceholderText(/search by name/i)).toBeInTheDocument();
|
||
expect(screen.getByText(/no library panels found./i)).toBeInTheDocument();
|
||
expect(screen.getByText(/sort \(default a–z\)/i)).toBeInTheDocument();
|
||
});
|
||
|
||
describe('and user changes sorting', () => {
|
||
it('should call api with correct params', async () => {
|
||
const { getLibraryPanelsSpy } = await getTestContext({ showSort: true });
|
||
getLibraryPanelsSpy.mockClear();
|
||
|
||
await userEvent.type(screen.getByText(/sort \(default a–z\)/i), 'Desc{enter}');
|
||
await waitFor(() => expect(getLibraryPanelsSpy).toHaveBeenCalledTimes(1));
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledWith({
|
||
searchString: '',
|
||
sortDirection: 'alpha-desc',
|
||
folderFilter: [],
|
||
page: 0,
|
||
typeFilter: [],
|
||
perPage: 40,
|
||
});
|
||
});
|
||
});
|
||
});
|
||
|
||
describe('when mounted with showPanelFilter', () => {
|
||
it('should show input filter and library panels view and panel filter', async () => {
|
||
await getTestContext({ showPanelFilter: true });
|
||
|
||
expect(screen.getByPlaceholderText(/search by name/i)).toBeInTheDocument();
|
||
expect(screen.getByText(/no library panels found./i)).toBeInTheDocument();
|
||
expect(screen.getByRole('textbox', { name: /panel type filter/i })).toBeInTheDocument();
|
||
});
|
||
|
||
describe('and user changes panel filter', () => {
|
||
it('should call api with correct params', async () => {
|
||
const { getLibraryPanelsSpy } = await getTestContext({ showPanelFilter: true });
|
||
getLibraryPanelsSpy.mockClear();
|
||
|
||
await userEvent.type(screen.getByRole('textbox', { name: /panel type filter/i }), 'Graph{enter}');
|
||
await userEvent.type(screen.getByRole('textbox', { name: /panel type filter/i }), 'Time Series{enter}');
|
||
await waitFor(() => expect(getLibraryPanelsSpy).toHaveBeenCalledTimes(1));
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledWith({
|
||
searchString: '',
|
||
folderFilter: [],
|
||
page: 0,
|
||
typeFilter: ['graph', 'timeseries'],
|
||
perPage: 40,
|
||
});
|
||
});
|
||
});
|
||
});
|
||
|
||
describe('when mounted with showPanelFilter', () => {
|
||
it('should show input filter and library panels view and folder filter', async () => {
|
||
await getTestContext({ showFolderFilter: true });
|
||
|
||
expect(screen.getByPlaceholderText(/search by name/i)).toBeInTheDocument();
|
||
expect(screen.getByText(/no library panels found./i)).toBeInTheDocument();
|
||
expect(screen.getByRole('textbox', { name: /folder filter/i })).toBeInTheDocument();
|
||
});
|
||
|
||
describe('and user changes folder filter', () => {
|
||
it('should call api with correct params', async () => {
|
||
const { getLibraryPanelsSpy } = await getTestContext({ showFolderFilter: true });
|
||
getLibraryPanelsSpy.mockClear();
|
||
|
||
userEvent.click(screen.getByRole('textbox', { name: /folder filter/i }));
|
||
await userEvent.type(screen.getByRole('textbox', { name: /folder filter/i }), '{enter}', {
|
||
skipClick: true,
|
||
});
|
||
await waitFor(() => expect(getLibraryPanelsSpy).toHaveBeenCalledTimes(1));
|
||
expect(getLibraryPanelsSpy).toHaveBeenCalledWith({
|
||
searchString: '',
|
||
folderFilter: ['0'],
|
||
page: 0,
|
||
typeFilter: [],
|
||
perPage: 40,
|
||
});
|
||
});
|
||
});
|
||
});
|
||
|
||
describe('when mounted without showSecondaryActions and there is one panel', () => {
|
||
it('should show correct row and no delete button', async () => {
|
||
await getTestContext(
|
||
{},
|
||
{
|
||
page: 1,
|
||
totalCount: 1,
|
||
perPage: 40,
|
||
elements: [
|
||
{
|
||
id: 1,
|
||
name: 'Library Panel Name',
|
||
kind: LibraryElementKind.Panel,
|
||
uid: 'uid',
|
||
description: 'Library Panel Description',
|
||
folderId: 0,
|
||
model: { type: 'timeseries', title: 'A title' },
|
||
type: 'timeseries',
|
||
orgId: 1,
|
||
version: 1,
|
||
meta: {
|
||
folderName: 'General',
|
||
folderUid: '',
|
||
connectedDashboards: 0,
|
||
created: '2021-01-01 12:00:00',
|
||
createdBy: { id: 1, name: 'Admin', avatarUrl: '' },
|
||
updated: '2021-01-01 12:00:00',
|
||
updatedBy: { id: 1, name: 'Admin', avatarUrl: '' },
|
||
},
|
||
},
|
||
],
|
||
}
|
||
);
|
||
|
||
const card = () => screen.getByLabelText(/plugin visualization item time series/i);
|
||
|
||
expect(screen.queryByText(/no library panels found./i)).not.toBeInTheDocument();
|
||
expect(card()).toBeInTheDocument();
|
||
expect(within(card()).getByText(/library panel name/i)).toBeInTheDocument();
|
||
expect(within(card()).getByText(/library panel description/i)).toBeInTheDocument();
|
||
expect(within(card()).queryByLabelText(/delete button on panel type card/i)).not.toBeInTheDocument();
|
||
});
|
||
});
|
||
|
||
describe('when mounted with showSecondaryActions and there is one panel', () => {
|
||
it('should show correct row and delete button', async () => {
|
||
await getTestContext(
|
||
{ showSecondaryActions: true },
|
||
{
|
||
page: 1,
|
||
totalCount: 1,
|
||
perPage: 40,
|
||
elements: [
|
||
{
|
||
id: 1,
|
||
name: 'Library Panel Name',
|
||
kind: LibraryElementKind.Panel,
|
||
uid: 'uid',
|
||
description: 'Library Panel Description',
|
||
folderId: 0,
|
||
model: { type: 'timeseries', title: 'A title' },
|
||
type: 'timeseries',
|
||
orgId: 1,
|
||
version: 1,
|
||
meta: {
|
||
folderName: 'General',
|
||
folderUid: '',
|
||
connectedDashboards: 0,
|
||
created: '2021-01-01 12:00:00',
|
||
createdBy: { id: 1, name: 'Admin', avatarUrl: '' },
|
||
updated: '2021-01-01 12:00:00',
|
||
updatedBy: { id: 1, name: 'Admin', avatarUrl: '' },
|
||
},
|
||
},
|
||
],
|
||
}
|
||
);
|
||
|
||
const card = () => screen.getByLabelText(/plugin visualization item time series/i);
|
||
|
||
expect(screen.queryByText(/no library panels found./i)).not.toBeInTheDocument();
|
||
expect(card()).toBeInTheDocument();
|
||
expect(within(card()).getByText(/library panel name/i)).toBeInTheDocument();
|
||
expect(within(card()).getByText(/library panel description/i)).toBeInTheDocument();
|
||
expect(within(card()).getByLabelText(/delete button on panel type card/i)).toBeInTheDocument();
|
||
});
|
||
});
|
||
});
|