import {Component} from 'react';
import type {RouteContextInterface} from 'react-router';
import {LocationFixture} from 'sentry-fixture/locationFixture';
import {OrganizationFixture} from 'sentry-fixture/organization';
import {ProjectFixture} from 'sentry-fixture/project';
import {RouterFixture} from 'sentry-fixture/routerFixture';
import {TeamFixture} from 'sentry-fixture/team';
import {UserFixture} from 'sentry-fixture/user';
import {render, screen, waitFor} from 'sentry-test/reactTestingLibrary';
import * as orgsActionCreators from 'sentry/actionCreators/organizations';
import {openSudo} from 'sentry/actionCreators/sudoModal';
import {SentryPropTypeValidators} from 'sentry/sentryPropTypeValidators';
import ConfigStore from 'sentry/stores/configStore';
import OrganizationStore from 'sentry/stores/organizationStore';
import ProjectsStore from 'sentry/stores/projectsStore';
import TeamStore from 'sentry/stores/teamStore';
import type {Organization} from 'sentry/types';
import useOrganization from 'sentry/utils/useOrganization';
import {OrganizationContextProvider, useEnsureOrganization} from './organizationContext';
import {RouteContext} from './routeContext';
jest.mock('sentry/actionCreators/sudoModal');
describe('OrganizationContext', function () {
let getOrgMock: jest.Mock;
let getProjectsMock: jest.Mock;
let getTeamsMock: jest.Mock;
const organization = OrganizationFixture();
const project = ProjectFixture();
const team = TeamFixture();
const router: RouteContextInterface = {
router: RouterFixture(),
location: LocationFixture(),
routes: [],
params: {orgId: organization.slug},
};
function setupOrgMocks(org: Organization) {
const orgMock = MockApiClient.addMockResponse({
url: `/organizations/${org.slug}/`,
body: org,
});
const projectMock = MockApiClient.addMockResponse({
url: `/organizations/${org.slug}/projects/`,
body: [project],
});
const teamMock = MockApiClient.addMockResponse({
url: `/organizations/${org.slug}/teams/`,
body: [team],
});
return {orgMock, projectMock, teamMock};
}
beforeEach(function () {
MockApiClient.clearMockResponses();
const {orgMock, projectMock, teamMock} = setupOrgMocks(organization);
getOrgMock = orgMock;
getProjectsMock = projectMock;
getTeamsMock = teamMock;
jest.spyOn(TeamStore, 'loadInitialData');
jest.spyOn(ProjectsStore, 'loadInitialData');
ConfigStore.init();
OrganizationStore.reset();
jest.spyOn(console, 'error').mockImplementation(jest.fn());
});
afterEach(function () {
// eslint-disable-next-line no-console
jest.mocked(console.error).mockRestore();
});
function OrganizationLoaderStub() {
useEnsureOrganization();
return null;
}
/**
* Used to test that the organization context is propegated
*/
function OrganizationName() {
const org = useOrganization({allowNull: true});
return
{org?.slug ?? 'no-org'}
;
}
/**
* Used to test the legacy organization context behavior
*/
class OrganizationNameLegacyConsumer extends Component {
static contextTypes = {
organization: SentryPropTypeValidators.isOrganization,
};
declare context: {organization: Organization | undefined};
render() {
return {this.context.organization?.slug ?? 'no-org'}
;
}
}
it('fetches org, projects, teams, and provides organization context', async function () {
render(
);
expect(await screen.findByText(organization.slug)).toBeInTheDocument();
expect(getOrgMock).toHaveBeenCalled();
expect(getProjectsMock).toHaveBeenCalled();
expect(getTeamsMock).toHaveBeenCalled();
});
it('provides legacy organization context', async function () {
render(
);
expect(await screen.findByText(organization.slug)).toBeInTheDocument();
});
it('does not fetch if organization is already set', async function () {
OrganizationStore.onUpdate(organization);
render(
);
expect(await screen.findByText(organization.slug)).toBeInTheDocument();
expect(getOrgMock).not.toHaveBeenCalled();
});
it('fetches new org when router params change', async function () {
// First render with org-slug
const {rerender} = render(
);
expect(await screen.findByText(organization.slug)).toBeInTheDocument();
const anotherOrg = OrganizationFixture({slug: 'another-org'});
const {orgMock, projectMock, teamMock} = setupOrgMocks(anotherOrg);
const switchOrganization = jest.spyOn(orgsActionCreators, 'switchOrganization');
// re-render with another-org
rerender(
);
expect(await screen.findByText(anotherOrg.slug)).toBeInTheDocument();
expect(orgMock).toHaveBeenCalled();
expect(projectMock).toHaveBeenCalled();
expect(teamMock).toHaveBeenCalled();
expect(switchOrganization).toHaveBeenCalled();
});
it('opens sudo modal for superusers for nonmember org with active staff', async function () {
ConfigStore.set('user', UserFixture({isSuperuser: true, isStaff: true}));
organization.access = [];
getOrgMock = MockApiClient.addMockResponse({
url: `/organizations/${organization.slug}/`,
body: organization,
});
render(
);
await waitFor(() => !OrganizationStore.getState().loading);
expect(openSudo).toHaveBeenCalled();
});
it('opens sudo modal for superusers on 403s', async function () {
ConfigStore.set('user', UserFixture({isSuperuser: true}));
getOrgMock = MockApiClient.addMockResponse({
url: '/organizations/org-slug/',
statusCode: 403,
});
render(
);
await waitFor(() => !OrganizationStore.getState().loading);
// eslint-disable-next-line no-console
await waitFor(() => expect(console.error).toHaveBeenCalled());
expect(openSudo).toHaveBeenCalled();
});
/**
* This test will rarely happen since most configurations are now using customer domains
*/
it('uses last organization slug from ConfigStore', async function () {
const configStoreOrg = OrganizationFixture({slug: 'config-store-org'});
ConfigStore.set('lastOrganization', configStoreOrg.slug);
const {orgMock, projectMock, teamMock} = setupOrgMocks(configStoreOrg);
// orgId is not present in the router.
render(
);
expect(await screen.findByText(configStoreOrg.slug)).toBeInTheDocument();
expect(orgMock).toHaveBeenCalled();
expect(projectMock).toHaveBeenCalled();
expect(teamMock).toHaveBeenCalled();
});
});