import {OrganizationFixture} from 'sentry-fixture/organization';
import {render, screen, userEvent, within} from 'sentry-test/reactTestingLibrary';
import {DiscoverDatasets} from 'sentry/utils/discover/types';
import {useGroupBys} from 'sentry/views/explore/hooks/useGroupBys';
import {useResultMode} from 'sentry/views/explore/hooks/useResultsMode';
import {useSampleFields} from 'sentry/views/explore/hooks/useSampleFields';
import {useSorts} from 'sentry/views/explore/hooks/useSorts';
import {useVisualizes} from 'sentry/views/explore/hooks/useVisualizes';
import {ExploreToolbar} from 'sentry/views/explore/toolbar';
import {ChartType} from 'sentry/views/insights/common/components/chart';
import {SpanTagsProvider} from '../contexts/spanTagsContext';
describe('ExploreToolbar', function () {
const organization = OrganizationFixture();
beforeEach(function () {
// without this the `CompactSelect` component errors with a bunch of async updates
jest.spyOn(console, 'error').mockImplementation();
MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/spans/fields/`,
method: 'GET',
body: [],
});
});
it('allows changing results mode', async function () {
let resultMode, sampleFields, groupBys;
function Component() {
[resultMode] = useResultMode();
[sampleFields] = useSampleFields();
({groupBys} = useGroupBys());
return ;
}
render(
,
{disableRouterMocks: true}
);
const section = screen.getByTestId('section-result-mode');
const samples = within(section).getByRole('radio', {name: 'Samples'});
const aggregates = within(section).getByRole('radio', {name: 'Aggregates'});
expect(samples).toBeChecked();
expect(aggregates).not.toBeChecked();
expect(resultMode).toEqual('samples');
expect(sampleFields).toEqual([
'span_id',
'project',
'span.op',
'span.description',
'span.duration',
'timestamp',
]); // default
await userEvent.click(aggregates);
expect(samples).not.toBeChecked();
expect(aggregates).toBeChecked();
expect(resultMode).toEqual('aggregate');
// Add a group by, and leave one unselected
const groupBy = screen.getByTestId('section-group-by');
await userEvent.click(within(groupBy).getByRole('button', {name: 'None'}));
await userEvent.click(within(groupBy).getByRole('option', {name: 'release'}));
expect(groupBys).toEqual(['release']);
await userEvent.click(within(groupBy).getByRole('button', {name: 'Add Group'}));
expect(groupBys).toEqual(['release', '']);
await userEvent.click(samples);
expect(samples).toBeChecked();
expect(aggregates).not.toBeChecked();
expect(resultMode).toEqual('samples');
expect(sampleFields).toEqual([
'span_id',
'project',
'span.op',
'span.description',
'span.duration',
'timestamp',
'release',
]);
});
it('allows changing visualizes', async function () {
let visualizes;
function Component() {
[visualizes] = useVisualizes();
return ;
}
render(
,
{disableRouterMocks: true}
);
const section = screen.getByTestId('section-visualizes');
// this is the default
expect(visualizes).toEqual([
{yAxes: ['count(span.duration)'], chartType: ChartType.LINE},
]);
// try changing the field
await userEvent.click(within(section).getByRole('button', {name: 'span.duration'}));
await userEvent.click(within(section).getByRole('option', {name: 'span.self_time'}));
expect(visualizes).toEqual([
{yAxes: ['count(span.self_time)'], chartType: ChartType.LINE},
]);
// try changing the aggregate
await userEvent.click(within(section).getByRole('button', {name: 'count'}));
await userEvent.click(within(section).getByRole('option', {name: 'avg'}));
expect(visualizes).toEqual([
{yAxes: ['avg(span.self_time)'], chartType: ChartType.LINE},
]);
// try adding an overlay
await userEvent.click(within(section).getByRole('button', {name: 'Add Overlay'}));
await userEvent.click(within(section).getByRole('button', {name: 'span.duration'}));
await userEvent.click(within(section).getByRole('option', {name: 'span.self_time'}));
expect(visualizes).toEqual([
{
yAxes: ['avg(span.self_time)', 'count(span.self_time)'],
chartType: ChartType.LINE,
},
]);
// try adding a new chart
await userEvent.click(within(section).getByRole('button', {name: 'Add Chart'}));
expect(visualizes).toEqual([
{
yAxes: ['avg(span.self_time)', 'count(span.self_time)'],
chartType: ChartType.LINE,
},
{yAxes: ['count(span.duration)'], chartType: ChartType.LINE},
]);
// delete first overlay
await userEvent.click(within(section).getAllByLabelText('Remove Overlay')[0]);
expect(visualizes).toEqual([
{yAxes: ['count(span.self_time)'], chartType: ChartType.LINE},
{yAxes: ['count(span.duration)'], chartType: ChartType.LINE},
]);
// delete second chart
await userEvent.click(within(section).getAllByLabelText('Remove Overlay')[1]);
expect(visualizes).toEqual([
{yAxes: ['count(span.self_time)'], chartType: ChartType.LINE},
]);
// only one left so cant be deleted
expect(within(section).getByLabelText('Remove Overlay')).toBeDisabled();
});
it('allows changing sort by', async function () {
let sorts;
function Component() {
const [sampleFields] = useSampleFields();
[sorts] = useSorts({fields: sampleFields});
return ;
}
render(
,
{disableRouterMocks: true}
);
const section = screen.getByTestId('section-sort-by');
// this is the default
expect(within(section).getByRole('button', {name: 'timestamp'})).toBeInTheDocument();
expect(within(section).getByRole('button', {name: 'Desc'})).toBeInTheDocument();
expect(sorts).toEqual([{field: 'timestamp', kind: 'desc'}]);
// check the default field options
const fields = [
'span_id',
'project',
'span.op',
'span.description',
'span.duration',
'timestamp',
];
await userEvent.click(within(section).getByRole('button', {name: 'timestamp'}));
const fieldOptions = await within(section).findAllByRole('option');
expect(fieldOptions).toHaveLength(fields.length);
fieldOptions.forEach((option, i) => {
expect(option).toHaveTextContent(fields[i]);
});
// try changing the field
await userEvent.click(within(section).getByRole('option', {name: 'span.op'}));
expect(within(section).getByRole('button', {name: 'span.op'})).toBeInTheDocument();
expect(within(section).getByRole('button', {name: 'Desc'})).toBeInTheDocument();
expect(sorts).toEqual([{field: 'span.op', kind: 'desc'}]);
// check the kind options
await userEvent.click(within(section).getByRole('button', {name: 'Desc'}));
const kindOptions = await within(section).findAllByRole('option');
expect(kindOptions).toHaveLength(2);
expect(kindOptions[0]).toHaveTextContent('Desc');
expect(kindOptions[1]).toHaveTextContent('Asc');
// try changing the kind
await userEvent.click(within(section).getByRole('option', {name: 'Asc'}));
expect(within(section).getByRole('button', {name: 'span.op'})).toBeInTheDocument();
expect(within(section).getByRole('button', {name: 'Asc'})).toBeInTheDocument();
expect(sorts).toEqual([{field: 'span.op', kind: 'asc'}]);
});
it('allows changing group bys', async function () {
let groupBys;
function Component() {
({groupBys} = useGroupBys());
return ;
}
render(
,
{disableRouterMocks: true}
);
const section = screen.getByTestId('section-group-by');
expect(within(section).getByRole('button', {name: 'None'})).toBeInTheDocument();
expect(groupBys).toEqual(['']);
// disabled in the samples mode
expect(within(section).getByRole('button', {name: 'None'})).toBeDisabled();
// click the aggregates mode to enable
await userEvent.click(
within(screen.getByTestId('section-result-mode')).getByRole('radio', {
name: 'Aggregates',
})
);
expect(within(section).getByRole('button', {name: 'None'})).toBeEnabled();
await userEvent.click(within(section).getByRole('button', {name: 'None'}));
const groupByOptions1 = await within(section).findAllByRole('option');
expect(groupByOptions1.length).toBeGreaterThan(0);
await userEvent.click(within(section).getByRole('option', {name: 'span.op'}));
expect(groupBys).toEqual(['span.op']);
await userEvent.click(within(section).getByRole('button', {name: 'Add Group'}));
expect(groupBys).toEqual(['span.op', '']);
await userEvent.click(within(section).getByRole('button', {name: 'None'}));
const groupByOptions2 = await within(section).findAllByRole('option');
expect(groupByOptions2.length).toBeGreaterThan(0);
await userEvent.click(
within(section).getByRole('option', {name: 'span.description'})
);
expect(groupBys).toEqual(['span.op', 'span.description']);
await userEvent.click(within(section).getAllByLabelText('Remove Column')[0]);
expect(groupBys).toEqual(['span.description']);
// only 1 left but it's not empty
expect(within(section).getByLabelText('Remove Column')).toBeEnabled();
await userEvent.click(within(section).getByLabelText('Remove Column'));
expect(groupBys).toEqual(['']);
// last one and it's empty
expect(within(section).getByLabelText('Remove Column')).toBeDisabled();
});
});