123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292 |
- import selectEvent from 'react-select-event';
- import {act, render, screen, waitFor} from 'sentry-test/reactTestingLibrary';
- import ContextPickerModal from 'sentry/components/contextPickerModal';
- import ConfigStore from 'sentry/stores/configStore';
- import OrganizationsStore from 'sentry/stores/organizationsStore';
- import OrganizationStore from 'sentry/stores/organizationStore';
- import ProjectsStore from 'sentry/stores/projectsStore';
- import {makeCloseButton, ModalBody, ModalFooter} from './globalModal/components';
- describe('ContextPickerModal', function () {
- let project, project2, project4, org, org2;
- const onFinish = jest.fn();
- beforeEach(function () {
- act(() => ProjectsStore.reset());
- MockApiClient.clearMockResponses();
- project = TestStubs.Project();
- org = TestStubs.Organization({projects: [project]});
- project2 = TestStubs.Project({slug: 'project2'});
- org2 = TestStubs.Organization({
- slug: 'org2',
- id: '21',
- });
- project4 = TestStubs.Project({slug: 'project4', isMember: false});
- act(() => OrganizationsStore.load([]));
- act(() => OrganizationStore.reset());
- jest.clearAllMocks();
- });
- const getComponent = (props = {}) => (
- <ContextPickerModal
- Header={() => <div />}
- Body={ModalBody}
- nextPath="/test/:orgId/path/"
- needOrg
- onFinish={onFinish}
- needProject={false}
- CloseButton={makeCloseButton(() => {})}
- Footer={ModalFooter}
- closeModal={jest.fn()}
- {...props}
- />
- );
- it('renders with only org selector when no org is selected', function () {
- render(getComponent());
- expect(screen.getByText('Select an Organization')).toBeInTheDocument();
- expect(screen.queryByText('Select a Project to continue')).not.toBeInTheDocument();
- });
- it('calls onFinish, if project id is not needed, and only 1 org', async function () {
- OrganizationsStore.load([org2]);
- OrganizationStore.onUpdate(org2);
- MockApiClient.addMockResponse({
- url: `/organizations/${org2.slug}/projects/`,
- body: [],
- });
- render(getComponent());
- await waitFor(() => {
- expect(onFinish).toHaveBeenCalledWith('/test/org2/path/');
- });
- });
- it('calls onFinish if there is only 1 org and 1 project', async function () {
- OrganizationsStore.load([org2]);
- OrganizationStore.onUpdate(org2);
- const fetchProjectsForOrg = MockApiClient.addMockResponse({
- url: `/organizations/${org2.slug}/projects/`,
- body: [project2],
- });
- render(
- getComponent({
- needOrg: true,
- needProject: true,
- nextPath: '/test/:orgId/path/:projectId/',
- })
- );
- await waitFor(() => {
- expect(fetchProjectsForOrg).toHaveBeenCalled();
- expect(onFinish).toHaveBeenLastCalledWith('/test/org2/path/project2/');
- });
- });
- it('selects an org and calls `onFinish` with URL with organization slug', async function () {
- OrganizationsStore.load([org]);
- render(getComponent());
- MockApiClient.addMockResponse({
- url: `/organizations/${org.slug}/projects/`,
- body: [],
- });
- await selectEvent.select(screen.getByText('Select an Organization'), 'org-slug');
- await waitFor(() => {
- expect(onFinish).toHaveBeenCalledWith('/test/org-slug/path/');
- });
- });
- it('renders with project selector and org selector selected when org is already selected', async function () {
- OrganizationsStore.load([org]);
- OrganizationStore.onUpdate(org);
- const fetchProjectsForOrg = MockApiClient.addMockResponse({
- url: `/organizations/${org.slug}/projects/`,
- body: [project, project2, project4],
- });
- render(
- getComponent({
- needOrg: true,
- needProject: true,
- })
- );
- await waitFor(() => {
- expect(fetchProjectsForOrg).toHaveBeenCalled();
- });
- // Default to org in latest context
- // Should see 1 selected, and 1 as an option
- expect(screen.getAllByText('org-slug')).toHaveLength(2);
- expect(screen.getByText('My Projects')).toBeInTheDocument();
- expect(screen.getByText(project.slug)).toBeInTheDocument();
- expect(screen.getByText(project2.slug)).toBeInTheDocument();
- expect(screen.getByText('All Projects')).toBeInTheDocument();
- expect(screen.getByText(project4.slug)).toBeInTheDocument();
- });
- it('can select org and project', async function () {
- const organizations = [
- {
- ...org,
- projects: [project],
- },
- {
- ...org2,
- projects: [project2, TestStubs.Project({slug: 'project3'})],
- },
- ];
- const fetchProjectsForOrg = MockApiClient.addMockResponse({
- url: `/organizations/${org2.slug}/projects/`,
- body: organizations[1].projects,
- });
- OrganizationsStore.load(organizations);
- render(
- getComponent({
- needOrg: true,
- needProject: true,
- nextPath: '/test/:orgId/path/:projectId/',
- organizations,
- })
- );
- // Should not have anything selected
- expect(screen.getByText('Select an Organization')).toBeInTheDocument();
- // Select org2
- await selectEvent.select(screen.getByText('Select an Organization'), org2.slug);
- // <Projects> will fetch projects for org2
- expect(fetchProjectsForOrg).toHaveBeenCalled();
- expect(screen.getByText('My Projects')).toBeInTheDocument();
- expect(screen.getByText(project2.slug)).toBeInTheDocument();
- expect(screen.getByText('project3')).toBeInTheDocument();
- expect(screen.queryByText('All Projects')).not.toBeInTheDocument();
- // Select project3
- await selectEvent.select(screen.getByText(/Select a Project/), 'project3');
- expect(onFinish).toHaveBeenCalledWith('/test/org2/path/project3/');
- });
- it('isSuperUser and selects an integrationConfig and calls `onFinish` with URL to that configuration', async function () {
- OrganizationsStore.load([org]);
- OrganizationStore.onUpdate(org);
- ConfigStore.config.user = TestStubs.User({isSuperuser: true});
- const provider = {slug: 'github'};
- const configUrl = `/api/0/organizations/${org.slug}/integrations/?provider_key=${provider.slug}&includeConfig=0`;
- const integration = TestStubs.GitHubIntegration();
- const fetchGithubConfigs = MockApiClient.addMockResponse({
- url: configUrl,
- body: [integration],
- });
- MockApiClient.addMockResponse({
- url: `/organizations/${org.slug}/projects/`,
- body: [],
- });
- render(
- getComponent({
- needOrg: false,
- needProject: false,
- nextPath: `/settings/${org.slug}/integrations/${provider.slug}/`,
- configUrl,
- })
- );
- await waitFor(() => {
- expect(fetchGithubConfigs).toHaveBeenCalled();
- });
- await selectEvent.select(
- screen.getByText(/Select a configuration/i),
- integration.domainName
- );
- expect(onFinish).toHaveBeenCalledWith(
- `/settings/${org.slug}/integrations/github/${integration.id}/`
- );
- });
- it('not superUser and cannot select an integrationConfig and calls `onFinish` with URL to integration overview page', async function () {
- OrganizationsStore.load([org]);
- OrganizationStore.onUpdate(org);
- ConfigStore.config.user = TestStubs.User({isSuperuser: false});
- const provider = {slug: 'github'};
- const configUrl = `/api/0/organizations/${org.slug}/integrations/?provider_key=${provider.slug}&includeConfig=0`;
- const fetchGithubConfigs = MockApiClient.addMockResponse({
- url: configUrl,
- body: [TestStubs.GitHubIntegration()],
- });
- MockApiClient.addMockResponse({
- url: `/organizations/${org.slug}/projects/`,
- body: [],
- });
- render(
- getComponent({
- needOrg: false,
- needProject: false,
- nextPath: `/settings/${org.slug}/integrations/${provider.slug}/`,
- configUrl,
- })
- );
- await waitFor(() => {
- expect(fetchGithubConfigs).toHaveBeenCalled();
- });
- });
- it('is superUser and no integration configurations and calls `onFinish` with URL to integration overview page', async function () {
- OrganizationsStore.load([org]);
- OrganizationStore.onUpdate(org);
- ConfigStore.config = TestStubs.User({isSuperuser: false});
- const provider = {slug: 'github'};
- const configUrl = `/api/0/organizations/${org.slug}/integrations/?provider_key=${provider.slug}&includeConfig=0`;
- const fetchGithubConfigs = MockApiClient.addMockResponse({
- url: configUrl,
- body: [],
- });
- MockApiClient.addMockResponse({
- url: `/organizations/${org.slug}/projects/`,
- body: [],
- });
- render(
- getComponent({
- needOrg: false,
- needProject: false,
- nextPath: `/settings/${org.slug}/integrations/${provider.slug}/`,
- configUrl,
- })
- );
- await waitFor(() => {
- expect(fetchGithubConfigs).toHaveBeenCalled();
- });
- expect(onFinish).toHaveBeenCalledWith(`/settings/${org.slug}/integrations/github/`);
- });
- });
|