123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326 |
- import {mountWithTheme} from 'sentry-test/enzyme';
- import {openSudo} from 'sentry/actionCreators/modal';
- import * as OrganizationActionCreator from 'sentry/actionCreators/organization';
- import ProjectActions from 'sentry/actions/projectActions';
- import TeamActions from 'sentry/actions/teamActions';
- import ConfigStore from 'sentry/stores/configStore';
- import OrganizationStore from 'sentry/stores/organizationStore';
- import {OrganizationLegacyContext} from 'sentry/views/organizationContextContainer';
- jest.mock('sentry/stores/configStore', () => ({
- get: jest.fn(),
- }));
- jest.mock('sentry/actionCreators/modal', () => ({
- openSudo: jest.fn(),
- }));
- describe('OrganizationContextContainer', function () {
- let wrapper;
- const org = TestStubs.Organization();
- const teams = [TestStubs.Team()];
- const projects = [TestStubs.Project()];
- const api = new MockApiClient();
- let getOrgMock;
- let getProjectsMock;
- let getTeamsMock;
- const createWrapper = props => {
- wrapper = mountWithTheme(
- <OrganizationLegacyContext
- api={api}
- params={{orgId: 'org-slug'}}
- location={{query: {}}}
- routes={[]}
- {...props}
- >
- <div />
- </OrganizationLegacyContext>
- );
- return wrapper;
- };
- beforeEach(function () {
- MockApiClient.clearMockResponses();
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/org-slug/',
- body: org,
- });
- getProjectsMock = MockApiClient.addMockResponse({
- url: '/organizations/org-slug/projects/',
- body: projects,
- });
- getTeamsMock = MockApiClient.addMockResponse({
- url: '/organizations/org-slug/teams/',
- body: teams,
- });
- jest.spyOn(TeamActions, 'loadTeams');
- jest.spyOn(ProjectActions, 'loadProjects');
- jest.spyOn(OrganizationActionCreator, 'fetchOrganizationDetails');
- });
- afterEach(function () {
- wrapper.unmount();
- OrganizationStore.reset();
- TeamActions.loadTeams.mockRestore();
- ProjectActions.loadProjects.mockRestore();
- ConfigStore.get.mockRestore();
- OrganizationActionCreator.fetchOrganizationDetails.mockRestore();
- });
- it('renders and fetches org, projects, and teams', async function () {
- wrapper = createWrapper();
- // await dispatching the action to org store
- await tick();
- // await resolving the api promise from action creator and updating component
- await tick();
- expect(getOrgMock).toHaveBeenCalled();
- expect(getProjectsMock).toHaveBeenCalled();
- expect(getTeamsMock).toHaveBeenCalled();
- expect(wrapper.state('loading')).toBe(false);
- expect(wrapper.state('error')).toBe(null);
- expect(wrapper.state('organization')).toEqual(org);
- expect(TeamActions.loadTeams).toHaveBeenCalledWith(teams);
- expect(ProjectActions.loadProjects).toHaveBeenCalledWith(projects);
- expect(OrganizationActionCreator.fetchOrganizationDetails).toHaveBeenCalledWith(
- api,
- 'org-slug',
- true,
- true
- );
- });
- it('fetches new org when router params change', async function () {
- const newOrg = TestStubs.Organization({slug: 'new-slug'});
- wrapper = createWrapper();
- const instance = wrapper.instance();
- jest.spyOn(instance, 'render');
- // initial render
- await tick();
- await tick();
- expect(instance.state.organization).toEqual(org);
- expect(instance.render).toHaveBeenCalledTimes(1);
- const mock = MockApiClient.addMockResponse({
- url: '/organizations/new-slug/',
- body: newOrg,
- });
- const projectsMock = MockApiClient.addMockResponse({
- url: '/organizations/new-slug/projects/',
- body: projects,
- });
- const teamsMock = MockApiClient.addMockResponse({
- url: '/organizations/new-slug/teams/',
- body: teams,
- });
- wrapper.setProps({params: {orgId: newOrg.slug}}, () => {
- // state should be reset based on props
- expect(instance.state.organization).toEqual(null);
- expect(instance.render).toHaveBeenCalledTimes(2);
- });
- // await fetching new org
- await tick();
- await tick();
- wrapper.update();
- expect(mock).toHaveBeenLastCalledWith('/organizations/new-slug/', expect.anything());
- expect(projectsMock).toHaveBeenCalled();
- expect(teamsMock).toHaveBeenCalled();
- expect(wrapper.state('loading')).toBe(false);
- expect(wrapper.state('error')).toBe(null);
- expect(wrapper.state('organization')).toEqual(newOrg);
- });
- it('shows loading error for non-superusers on 403s', async function () {
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/org-slug/',
- statusCode: 403,
- });
- console.error = jest.fn(); // eslint-disable-line no-console
- wrapper = createWrapper();
- // await dispatching action
- await tick();
- // await resolving api, and updating component
- await tick();
- await tick();
- wrapper.update();
- expect(wrapper.find('LoadingError')).toHaveLength(1);
- console.error.mockRestore(); // eslint-disable-line no-console
- });
- it('opens sudo modal for superusers on 403s', async function () {
- ConfigStore.get.mockImplementation(() => ({
- isSuperuser: true,
- }));
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/org-slug/',
- statusCode: 403,
- });
- wrapper = createWrapper();
- // await dispatching action
- await tick();
- // await resolving api, and updating component
- await tick();
- await tick();
- wrapper.update();
- expect(openSudo).toHaveBeenCalled();
- });
- it('uses last organization from ConfigStore', async function () {
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/last-org/',
- body: org,
- });
- MockApiClient.addMockResponse({
- url: '/organizations/last-org/projects/',
- body: projects,
- });
- MockApiClient.addMockResponse({
- url: '/organizations/last-org/teams/',
- body: teams,
- });
- // mocking `.get('lastOrganization')`
- ConfigStore.get.mockImplementation(() => 'last-org');
- wrapper = createWrapper({useLastOrganization: true, params: {}});
- // await dispatching action
- await tick();
- // await dispatching the action to org store
- await tick();
- expect(getOrgMock).toHaveBeenLastCalledWith(
- '/organizations/last-org/',
- expect.anything()
- );
- });
- it('uses last organization from `organizations` prop', async function () {
- MockApiClient.addMockResponse({
- url: '/organizations/foo/environments/',
- body: TestStubs.Environments(),
- });
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/foo/',
- body: org,
- });
- getProjectsMock = MockApiClient.addMockResponse({
- url: '/organizations/foo/projects/',
- body: projects,
- });
- getTeamsMock = MockApiClient.addMockResponse({
- url: '/organizations/foo/teams/',
- body: teams,
- });
- ConfigStore.get.mockImplementation(() => '');
- wrapper = createWrapper({
- useLastOrganization: true,
- params: {orgId: ''},
- organizationsLoading: true,
- organizations: [],
- });
- expect(wrapper.find('LoadingTriangle')).toHaveLength(1);
- wrapper.setProps({
- organizationsLoading: false,
- organizations: [
- TestStubs.Organization({slug: 'foo'}),
- TestStubs.Organization({slug: 'bar'}),
- ],
- });
- await tick(); // action to start fetch org
- await tick(); // action after successfully fetching org
- wrapper.update();
- expect(wrapper.find('LoadingTriangle')).toHaveLength(0);
- expect(getOrgMock).toHaveBeenCalled();
- expect(getProjectsMock).toHaveBeenCalled();
- expect(getTeamsMock).toHaveBeenCalled();
- });
- it('uses last organization when no orgId in URL - and fetches org details once', async function () {
- ConfigStore.get.mockImplementation(() => 'my-last-org');
- getOrgMock = MockApiClient.addMockResponse({
- url: '/organizations/my-last-org/',
- body: TestStubs.Organization({slug: 'my-last-org'}),
- });
- getProjectsMock = MockApiClient.addMockResponse({
- url: '/organizations/my-last-org/projects/',
- body: projects,
- });
- getTeamsMock = MockApiClient.addMockResponse({
- url: '/organizations/my-last-org/teams/',
- body: teams,
- });
- wrapper = createWrapper({
- params: {},
- useLastOrganization: true,
- organizations: [],
- });
- // await dispatching action
- await tick();
- // await resolving api, and updating component
- await tick();
- wrapper.update();
- expect(wrapper.find('LoadingTriangle')).toHaveLength(0);
- expect(getOrgMock).toHaveBeenCalledTimes(1);
- // Simulate OrganizationsStore being loaded *after* `OrganizationContext` finishes
- // org details fetch
- wrapper.setProps({
- organizationsLoading: false,
- organizations: [
- TestStubs.Organization({slug: 'foo'}),
- TestStubs.Organization({slug: 'bar'}),
- ],
- });
- expect(getOrgMock).toHaveBeenCalledTimes(1);
- expect(getProjectsMock).toHaveBeenCalledTimes(1);
- expect(getTeamsMock).toHaveBeenCalledTimes(1);
- });
- it('fetches org details only once if organizations loading store changes', async function () {
- wrapper = createWrapper({
- params: {orgId: 'org-slug'},
- organizationsLoading: true,
- organizations: [],
- });
- // await dispatching action
- await tick();
- // await resolving api, and updating component
- await tick();
- wrapper.update();
- expect(wrapper.find('LoadingTriangle')).toHaveLength(0);
- expect(getOrgMock).toHaveBeenCalledTimes(1);
- // Simulate OrganizationsStore being loaded *after* `OrganizationContext` finishes
- // org details fetch
- wrapper.setProps({
- organizationsLoading: false,
- organizations: [
- TestStubs.Organization({slug: 'foo'}),
- TestStubs.Organization({slug: 'bar'}),
- ],
- });
- expect(getOrgMock).toHaveBeenCalledTimes(1);
- expect(getProjectsMock).toHaveBeenCalledTimes(1);
- expect(getTeamsMock).toHaveBeenCalledTimes(1);
- });
- });
|