import {act, render, screen, userEvent, waitFor} from 'sentry-test/reactTestingLibrary';
import ProjectsStore from 'sentry/stores/projectsStore';
import Projects from 'sentry/utils/projects';
describe('utils.projects', function () {
const renderer = jest.fn(() => null);
const createWrapper = props =>
render(); // eslint-disable-line
beforeEach(function () {
renderer.mockClear();
MockApiClient.clearMockResponses();
act(() =>
ProjectsStore.loadInitialData([
TestStubs.Project({id: '1', slug: 'foo'}),
TestStubs.Project({id: '2', slug: 'bar'}),
])
);
});
afterEach(async function () {
act(() => ProjectsStore.loadInitialData([]));
await tick();
});
describe('with predefined list of slugs', function () {
it('gets projects that are in the ProjectsStore', function () {
createWrapper({slugs: ['foo', 'bar']});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: null,
projects: [
expect.objectContaining({
id: '1',
slug: 'foo',
}),
expect.objectContaining({
id: '2',
slug: 'bar',
}),
],
})
);
});
it('fetches projects from API if not found in store', async function () {
const request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
query: {
query: 'slug:a slug:b',
},
body: [
TestStubs.Project({
id: '100',
slug: 'a',
}),
TestStubs.Project({
id: '101',
slug: 'b',
}),
],
});
createWrapper({slugs: ['foo', 'a', 'b']});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [
{slug: 'a'},
{slug: 'b'},
expect.objectContaining({
id: '1',
slug: 'foo',
}),
],
})
);
await waitFor(() =>
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
query: 'slug:a slug:b',
collapse: ['latestDeploys'],
},
})
)
);
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: false,
hasMore: null,
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
expect.objectContaining({
id: '1',
slug: 'foo',
}),
],
})
);
});
it('only has partial results from API', async function () {
const request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [
TestStubs.Project({
id: '100',
slug: 'a',
}),
],
});
createWrapper({slugs: ['foo', 'a', 'b']});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [
{slug: 'a'},
{slug: 'b'},
expect.objectContaining({
id: '1',
slug: 'foo',
}),
],
})
);
await waitFor(() =>
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
query: 'slug:a slug:b',
collapse: ['latestDeploys'],
},
})
)
);
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: true,
hasMore: null,
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
}),
{
slug: 'b',
},
expect.objectContaining({
id: '1',
slug: 'foo',
}),
],
})
);
});
it('responds to updated projects from the project store', async function () {
createWrapper({slugs: ['foo', 'bar']});
await waitFor(() =>
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: null,
projects: [
expect.objectContaining({
id: '1',
slug: 'foo',
}),
expect.objectContaining({
id: '2',
slug: 'bar',
}),
],
})
)
);
const newTeam = TestStubs.Team();
act(() => ProjectsStore.onAddTeam(newTeam, 'foo'));
await waitFor(() =>
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: null,
projects: [
expect.objectContaining({
id: '1',
slug: 'foo',
teams: [newTeam],
}),
expect.objectContaining({
id: '2',
slug: 'bar',
}),
],
})
)
);
});
});
describe('with predefined list of project ids', function () {
it('gets project ids that are in the ProjectsStore', function () {
createWrapper({projectIds: [1, 2]});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: null,
projects: [
expect.objectContaining({
id: '1',
slug: 'foo',
}),
expect.objectContaining({
id: '2',
slug: 'bar',
}),
],
})
);
});
it('fetches projects from API if ids not found in store', async function () {
const request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
query: {
all_projects: '1',
collapse: ['latestDeploys'],
},
body: [
TestStubs.Project({
id: '1',
slug: 'foo',
}),
TestStubs.Project({
id: '100',
slug: 'a',
}),
TestStubs.Project({
id: '101',
slug: 'b',
}),
],
});
createWrapper({projectIds: [1, 100, 101]});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [],
})
);
await waitFor(() =>
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
collapse: ['latestDeploys'],
},
})
)
);
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: [
expect.objectContaining({
id: '1',
slug: 'foo',
}),
expect.objectContaining({
id: '100',
slug: 'a',
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
],
})
);
});
});
describe('with no pre-defined projects', function () {
let request;
beforeEach(async function () {
request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [
TestStubs.Project({
id: '100',
slug: 'a',
}),
TestStubs.Project({
id: '101',
slug: 'b',
}),
],
headers: {
Link:
'; rel="previous"; results="true"; cursor="1443575731:0:1", ' +
'; rel="next"; results="true"; cursor="1443575731:0:0',
},
});
act(() => ProjectsStore.loadInitialData([]));
await tick();
});
it('fetches projects from API', async function () {
createWrapper();
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [],
})
);
await waitFor(() =>
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
collapse: ['latestDeploys'],
},
})
)
);
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: true,
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
],
})
);
});
it('queries API for more projects and replaces results', async function () {
const myRenderer = jest.fn(({onSearch}) => (
onSearch(target.value)} />
));
createWrapper({children: myRenderer});
// This is initial state
expect(myRenderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [],
})
);
request.mockClear();
request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [
TestStubs.Project({
id: '102',
slug: 'test1',
}),
TestStubs.Project({
id: '103',
slug: 'test2',
}),
],
});
userEvent.type(screen.getByRole('textbox'), 'test');
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
query: 'test',
collapse: ['latestDeploys'],
},
})
);
await waitFor(() =>
expect(myRenderer).toHaveBeenLastCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: [
expect.objectContaining({
id: '102',
slug: 'test1',
}),
expect.objectContaining({
id: '103',
slug: 'test2',
}),
],
})
)
);
});
it('queries API for more projects and appends results', async function () {
const myRenderer = jest.fn(({onSearch}) => (
onSearch(target.value, {append: true})} />
));
createWrapper({children: myRenderer});
request.mockClear();
request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
body: [
TestStubs.Project({
id: '102',
slug: 'test1',
}),
TestStubs.Project({
id: '103',
slug: 'test2',
}),
],
});
userEvent.type(screen.getByRole('textbox'), 'test');
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {
query: 'test',
collapse: ['latestDeploys'],
},
})
);
await waitFor(() =>
expect(myRenderer).toHaveBeenLastCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
expect.objectContaining({
id: '102',
slug: 'test1',
}),
expect.objectContaining({
id: '103',
slug: 'test2',
}),
],
})
)
);
// Should not have duplicates
userEvent.type(screen.getByRole('textbox'), 'test');
await waitFor(() =>
expect(myRenderer).toHaveBeenLastCalledWith(
expect.objectContaining({
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
expect.objectContaining({
id: '102',
slug: 'test1',
}),
expect.objectContaining({
id: '103',
slug: 'test2',
}),
],
})
)
);
});
});
describe('with all projects prop', function () {
let mockProjects;
let request;
beforeEach(function () {
mockProjects = [
TestStubs.Project({
id: '100',
slug: 'a',
}),
TestStubs.Project({
id: '101',
slug: 'b',
}),
TestStubs.Project({
id: '102',
slug: 'c',
}),
];
request = MockApiClient.addMockResponse({
url: '/organizations/org-slug/projects/',
query: {
all_projects: '1',
collapse: ['latestDeploys'],
},
body: mockProjects,
});
ProjectsStore.reset();
});
it('can query for a list of all projects and save it to the store', async function () {
const loadInitialData = jest.spyOn(ProjectsStore, 'loadInitialData');
createWrapper({allProjects: true});
// This is initial state
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: true,
isIncomplete: null,
hasMore: null,
projects: [],
})
);
// wait for request to resolve
await waitFor(() =>
expect(request).toHaveBeenCalledWith(
expect.anything(),
expect.objectContaining({
query: {all_projects: 1, collapse: ['latestDeploys']},
})
)
);
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: mockProjects,
})
);
// expect the store action to be called
expect(loadInitialData).toHaveBeenCalledWith(mockProjects);
loadInitialData.mockRestore();
});
it('does not refetch projects that are already loaded in the store', async function () {
act(() => ProjectsStore.loadInitialData(mockProjects));
const loadInitialData = jest.spyOn(ProjectsStore, 'loadInitialData');
createWrapper({allProjects: true});
await waitFor(() =>
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: mockProjects,
})
)
);
expect(request).not.toHaveBeenCalled();
expect(loadInitialData).not.toHaveBeenCalled();
loadInitialData.mockRestore();
});
it('responds to updated projects from the project store', async function () {
act(() => ProjectsStore.loadInitialData(mockProjects));
createWrapper({allProjects: true});
await waitFor(() =>
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: mockProjects,
})
)
);
const newTeam = TestStubs.Team();
act(() => ProjectsStore.onAddTeam(newTeam, 'a'));
// Expect new team information to be available
await waitFor(() =>
expect(renderer).toHaveBeenCalledWith(
expect.objectContaining({
fetching: false,
isIncomplete: null,
hasMore: false,
projects: [
expect.objectContaining({
id: '100',
slug: 'a',
teams: [newTeam],
}),
expect.objectContaining({
id: '101',
slug: 'b',
}),
expect.objectContaining({
id: '102',
slug: 'c',
}),
],
})
)
);
});
});
});