import {render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary';
import {SlowestFunctionsTable} from 'sentry/views/profiling/landing/slowestFunctionsTable';
describe('SlowestFunctionsTable', () => {
it('shows loading state', () => {
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
body: [],
});
render();
expect(screen.getByTestId('loading-indicator')).toBeInTheDocument();
});
it('shows error state', async () => {
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
body: [],
statusCode: 500,
});
render();
expect(await screen.findByTestId('error-indicator')).toBeInTheDocument();
});
it('shows no functions state', async () => {
// @ts-expect-error partial schema mock
const schema: Profiling.Schema = {
metrics: [],
};
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
match: [
MockApiClient.matchQuery({
expand: 'metrics',
}),
],
body: schema,
});
render();
expect(await screen.findByText('No functions found')).toBeInTheDocument();
});
it('renders function fields', async () => {
// @ts-expect-error partial schema mock
const schema: Profiling.Schema = {
metrics: [
{
name: 'slow-function',
package: 'slow-package',
p75: 1500 * 1e6,
p95: 2000 * 1e6,
p99: 3000 * 1e6,
sum: 60_000 * 1e6,
count: 5000,
avg: 0.5 * 1e6,
in_app: true,
fingerprint: 12345,
examples: [
{
project_id: 1,
profile_id: 'profile-id',
},
],
},
],
};
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
match: [
MockApiClient.matchQuery({
expand: 'metrics',
}),
],
body: schema,
});
render();
for (const value of [
'slow-function',
'slow-package',
'5k',
'1.50s',
'2.00s',
'3.00s',
'1.00min',
]) {
expect(await screen.findByText(value)).toBeInTheDocument();
}
});
it('paginates response', async () => {
// @ts-expect-error partial schema mock
const schema: Profiling.Schema = {
metrics: [],
};
for (let i = 0; i < 10; i++) {
schema.metrics?.push({
name: 'slow-function',
package: 'slow-package',
p75: 1500 * 1e6,
p95: 2000 * 1e6,
p99: 3000 * 1e6,
sum: 60_000 * 1e6,
count: 5000,
avg: 0.5 * 1e6,
in_app: true,
fingerprint: 12345,
examples: [
{
project_id: 1,
profile_id: 'profile-id',
},
],
});
}
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
match: [
MockApiClient.matchQuery({
expand: 'metrics',
}),
],
body: schema,
});
render();
expect(await screen.findAllByText('slow-function')).toHaveLength(5);
});
it('paginates results', async () => {
// @ts-expect-error partial schema mock
const schema: Profiling.Schema = {
metrics: [],
};
for (let i = 0; i < 10; i++) {
schema.metrics?.push({
name: 'slow-function-' + i,
package: 'slow-package',
p75: 1500 * 1e6,
p95: 2000 * 1e6,
p99: 3000 * 1e6,
sum: 60_000 * 1e6,
count: 5000,
avg: 0.5 * 1e6,
in_app: true,
fingerprint: 12345,
examples: [
{
project_id: 1,
profile_id: 'profile-id',
},
],
});
}
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
match: [
MockApiClient.matchQuery({
expand: 'metrics',
}),
],
body: schema,
});
render();
expect(await screen.findAllByText('slow-package')).toHaveLength(5);
userEvent.click(screen.getByLabelText('Next'));
for (let i = 6; i < 10; i++) {
expect(await screen.findByText('slow-function-' + i)).toBeInTheDocument();
}
expect(screen.getByLabelText('Next')).toBeDisabled();
userEvent.click(screen.getByLabelText('Previous'));
for (let i = 0; i < 5; i++) {
expect(await screen.findByText('slow-function-' + i)).toBeInTheDocument();
}
expect(screen.getByLabelText('Previous')).toBeDisabled();
});
it('fetches function metrics', async () => {
// @ts-expect-error partial schema mock
const schema: Profiling.Schema = {
metrics: [],
};
for (let i = 0; i < 10; i++) {
schema.metrics?.push({
name: 'slow-function-' + i,
package: 'slow-package',
p75: 1500 * 1e6,
p95: 2000 * 1e6,
p99: 3000 * 1e6,
sum: 60_000 * 1e6,
count: 5000,
avg: 0.5 * 1e6,
in_app: true,
fingerprint: 12345,
examples: [
{
project_id: 1,
profile_id: 'profile-id',
},
],
});
}
MockApiClient.addMockResponse({
url: '/organizations/org-slug/profiling/flamegraph/',
match: [
MockApiClient.matchQuery({
expand: 'metrics',
}),
],
body: schema,
});
const functionMetricsRequest = MockApiClient.addMockResponse({
url: '/organizations/org-slug/events-stats/',
match: [
MockApiClient.matchQuery({
query: 'fingerprint:12345',
dataset: 'profileFunctionsMetrics',
}),
],
body: [],
});
render();
const expandButtons = await screen.findAllByLabelText('View Function Metrics');
expect(expandButtons).toHaveLength(5);
await userEvent.click(expandButtons[0]);
await waitFor(() => {
expect(functionMetricsRequest).toHaveBeenCalled();
});
});
});