import type {Location} from 'history';
import {OrganizationFixture} from 'sentry-fixture/organization';
import {ProjectFixture} from 'sentry-fixture/project';
import {render, screen, waitFor, within} from 'sentry-test/reactTestingLibrary';
import {browserHistory} from 'sentry/utils/browserHistory';
import localStorage from 'sentry/utils/localStorage';
import {useLocation} from 'sentry/utils/useLocation';
import usePageFilters from 'sentry/utils/usePageFilters';
import useProjects from 'sentry/utils/useProjects';
import {useOnboardingProject} from 'sentry/views/performance/browser/webVitals/utils/useOnboardingProject';
import ScreenLoadSpans from 'sentry/views/performance/mobile/screenload/screenLoadSpans';
jest.mock('sentry/views/performance/browser/webVitals/utils/useOnboardingProject');
jest.mock('sentry/utils/useLocation');
jest.mock('sentry/utils/usePageFilters');
jest.mock('sentry/utils/useProjects');
function mockResponses(organization, project) {
jest.mocked(useOnboardingProject).mockReturnValue(undefined);
jest.mocked(useProjects).mockReturnValue({
fetchError: null,
fetching: false,
hasMore: false,
initiallyLoaded: false,
onSearch: jest.fn(),
placeholders: [],
projects: [project],
});
jest.mocked(useLocation).mockReturnValue({
action: 'PUSH',
hash: '',
key: '',
pathname: '/organizations/org-slug/performance/mobile/screens/spans/',
query: {
project: project.id,
transaction: 'MainActivity',
primaryRelease: 'com.example.vu.android@2.10.5',
secondaryRelease: 'com.example.vu.android@2.10.3+42',
},
search: '',
state: undefined,
} as Location);
jest.mocked(usePageFilters).mockReturnValue({
isReady: true,
desyncedFilters: new Set(),
pinnedFilters: new Set(),
shouldPersist: true,
selection: {
datetime: {
period: '10d',
start: null,
end: null,
utc: false,
},
environments: [],
projects: [parseInt(project.id, 10)],
},
});
MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/releases/`,
body: [
{
id: 970136705,
version: 'com.example.vu.android@2.10.5',
dateCreated: '2023-12-19T21:37:53.895495Z',
},
{
id: 969902997,
version: 'com.example.vu.android@2.10.3+42',
dateCreated: '2023-12-19T18:04:06.953025Z',
},
],
});
MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events/`,
query: {
dataset: 'metrics',
environment: [],
field: ['release', 'count()'],
per_page: 50,
project: ['2'],
query:
'transaction.op:ui.load release:["com.example.vu.android@2.10.5","com.example.vu.android@2.10.3+42"]',
referrer: 'api.starfish.mobile-release-selector',
statsPeriod: '10d',
},
body: {
meta: {},
data: [
{
release: 'com.example.vu.android@2.10.5',
'count()': 9768,
},
{
release: 'com.example.vu.android@2.10.3+42',
'count()': 826,
},
],
},
});
}
describe('Screen Summary', function () {
describe('Cross Platform Project', function () {
let eventsMock;
let eventsStatsMock;
let organization;
beforeEach(function () {
const project = ProjectFixture({platform: 'react-native'});
organization = OrganizationFixture({
features: ['insights-initial-modules'],
projects: [project],
});
mockResponses(organization, project);
localStorage.clear();
browserHistory.push = jest.fn();
eventsMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events/`,
});
eventsStatsMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events-stats/`,
});
});
afterEach(function () {
MockApiClient.clearMockResponses();
jest.clearAllMocks();
});
it('appends os.name filter for react native projects', async function () {
render(, {organization});
// Event samples
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'discover',
query:
'transaction.op:ui.load transaction:MainActivity release:com.example.vu.android@2.10.5 os.name:Android',
}),
})
);
});
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'discover',
query:
'transaction.op:ui.load transaction:MainActivity release:com.example.vu.android@2.10.3+42 os.name:Android',
}),
})
);
});
// Span Table
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'spansMetrics',
query:
'transaction.op:ui.load transaction:MainActivity has:span.description span.op:[file.read,file.write,ui.load,http.client,db,db.sql.room,db.sql.query,db.sql.transaction] os.name:Android release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
// Chart
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'metrics',
query:
'event.type:transaction transaction.op:ui.load transaction:MainActivity os.name:Android release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
await waitFor(() => {
expect(eventsStatsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'metrics',
query:
'event.type:transaction transaction.op:ui.load transaction:MainActivity os.name:Android release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
});
});
describe('Native Project', function () {
let eventsMock;
let eventsStatsMock;
let organization;
beforeEach(function () {
const project = ProjectFixture({platform: 'android'});
organization = OrganizationFixture({
features: ['insights-initial-modules'],
projects: [project],
});
mockResponses(organization, project);
localStorage.clear();
browserHistory.push = jest.fn();
eventsMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events/`,
});
eventsStatsMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events-stats/`,
});
});
afterEach(function () {
MockApiClient.clearMockResponses();
jest.clearAllMocks();
});
it('does not append os.name filter for native projects', async function () {
render(, {organization});
// Event samples
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'discover',
query:
'transaction.op:ui.load transaction:MainActivity release:com.example.vu.android@2.10.5',
}),
})
);
});
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'discover',
query:
'transaction.op:ui.load transaction:MainActivity release:com.example.vu.android@2.10.3+42',
}),
})
);
});
// Span Table
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'spansMetrics',
query:
'transaction.op:ui.load transaction:MainActivity has:span.description span.op:[file.read,file.write,ui.load,http.client,db,db.sql.room,db.sql.query,db.sql.transaction] release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
// Chart
await waitFor(() => {
expect(eventsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'metrics',
query:
'event.type:transaction transaction.op:ui.load transaction:MainActivity release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
await waitFor(() => {
expect(eventsStatsMock).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: expect.objectContaining({
dataset: 'metrics',
query:
'event.type:transaction transaction.op:ui.load transaction:MainActivity release:[com.example.vu.android@2.10.5,com.example.vu.android@2.10.3+42]',
}),
})
);
});
});
it('renders the top level metrics data correctly', async function () {
eventsMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/events/`,
body: {
data: [
{
'avg_if(measurements.time_to_initial_display,release,com.example.vu.android@2.10.5)': 1000,
'avg_if(measurements.time_to_initial_display,release,com.example.vu.android@2.10.3+42)': 2000,
'avg_if(measurements.time_to_full_display,release,com.example.vu.android@2.10.5)': 3000,
'avg_if(measurements.time_to_full_display,release,com.example.vu.android@2.10.3+42)': 4000,
'count()': 20,
},
],
},
match: [
MockApiClient.matchQuery({referrer: 'api.starfish.mobile-screen-totals'}),
],
});
render(, {organization});
await waitFor(() => {
expect(eventsMock).toHaveBeenCalled();
});
const blocks = [
{header: 'TTID (R1)', value: '1.00s'},
{header: 'TTID (R2)', value: '2.00s'},
{header: 'TTFD (R1)', value: '3.00s'},
{header: 'TTFD (R2)', value: '4.00s'},
{header: 'Count', value: '20'},
];
for (const block of blocks) {
const blockEl = screen.getByRole('heading', {name: block.header}).closest('div');
await within(blockEl!).findByText(block.value);
}
});
});
});