organizationContextContainer.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. import {Component, Fragment} from 'react';
  2. import {PlainRoute, RouteComponentProps} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import * as Sentry from '@sentry/react';
  5. import {openSudo} from 'sentry/actionCreators/modal';
  6. import {fetchOrganizationDetails} from 'sentry/actionCreators/organization';
  7. import ProjectActions from 'sentry/actions/projectActions';
  8. import {Client} from 'sentry/api';
  9. import Alert from 'sentry/components/alert';
  10. import LoadingError from 'sentry/components/loadingError';
  11. import LoadingTriangle from 'sentry/components/loadingTriangle';
  12. import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
  13. import Sidebar from 'sentry/components/sidebar';
  14. import {ORGANIZATION_FETCH_ERROR_TYPES} from 'sentry/constants';
  15. import {t} from 'sentry/locale';
  16. import SentryTypes from 'sentry/sentryTypes';
  17. import ConfigStore from 'sentry/stores/configStore';
  18. import HookStore from 'sentry/stores/hookStore';
  19. import OrganizationStore from 'sentry/stores/organizationStore';
  20. import space from 'sentry/styles/space';
  21. import {Organization} from 'sentry/types';
  22. import {metric} from 'sentry/utils/analytics';
  23. import {callIfFunction} from 'sentry/utils/callIfFunction';
  24. import getRouteStringFromRoutes from 'sentry/utils/getRouteStringFromRoutes';
  25. import RequestError from 'sentry/utils/requestError/requestError';
  26. import withApi from 'sentry/utils/withApi';
  27. import withOrganizations from 'sentry/utils/withOrganizations';
  28. import {OrganizationContext} from './organizationContext';
  29. type Props = RouteComponentProps<{orgId: string}, {}> & {
  30. api: Client;
  31. includeSidebar: boolean;
  32. organizations: Organization[];
  33. organizationsLoading: boolean;
  34. routes: PlainRoute[];
  35. useLastOrganization: boolean;
  36. };
  37. type State = {
  38. loading: boolean;
  39. organization: Organization | null;
  40. prevProps: {
  41. location: RouteComponentProps<{orgId: string}, {}>['location'];
  42. orgId: string;
  43. organizationsLoading: boolean;
  44. };
  45. dirty?: boolean;
  46. error?: RequestError | null;
  47. errorType?: string | null;
  48. hooks?: React.ReactNode[];
  49. };
  50. class OrganizationContextContainer extends Component<Props, State> {
  51. static getDerivedStateFromProps(props: Readonly<Props>, prevState: State): State {
  52. const {prevProps} = prevState;
  53. if (OrganizationContextContainer.shouldRemount(prevProps, props)) {
  54. return OrganizationContextContainer.getDefaultState(props);
  55. }
  56. const {organizationsLoading, location, params} = props;
  57. const {orgId} = params;
  58. return {
  59. ...prevState,
  60. prevProps: {
  61. orgId,
  62. organizationsLoading,
  63. location,
  64. },
  65. };
  66. }
  67. static shouldRemount(prevProps: State['prevProps'], props: Props): boolean {
  68. const hasOrgIdAndChanged =
  69. prevProps.orgId && props.params.orgId && prevProps.orgId !== props.params.orgId;
  70. const hasOrgId =
  71. props.params.orgId ||
  72. (props.useLastOrganization && ConfigStore.get('lastOrganization'));
  73. // protect against the case where we finish fetching org details
  74. // and then `OrganizationsStore` finishes loading:
  75. // only fetch in the case where we don't have an orgId
  76. //
  77. // Compare `getOrganizationSlug` because we may have a last used org from server
  78. // if there is no orgId in the URL
  79. const organizationLoadingChanged =
  80. prevProps.organizationsLoading !== props.organizationsLoading &&
  81. props.organizationsLoading === false;
  82. return (
  83. hasOrgIdAndChanged ||
  84. (!hasOrgId && organizationLoadingChanged) ||
  85. (props.location.state === 'refresh' && prevProps.location.state !== 'refresh')
  86. );
  87. }
  88. static getDefaultState(props: Props): State {
  89. const prevProps = {
  90. orgId: props.params.orgId,
  91. organizationsLoading: props.organizationsLoading,
  92. location: props.location,
  93. };
  94. if (OrganizationContextContainer.isOrgStorePopulatedCorrectly(props)) {
  95. // retrieve initial state from store
  96. return {
  97. ...OrganizationStore.get(),
  98. prevProps,
  99. };
  100. }
  101. return {
  102. loading: true,
  103. error: null,
  104. errorType: null,
  105. organization: null,
  106. prevProps,
  107. };
  108. }
  109. static getOrganizationSlug(props: Props) {
  110. return (
  111. props.params.orgId ||
  112. ((props.useLastOrganization &&
  113. (ConfigStore.get('lastOrganization') ||
  114. props.organizations?.[0]?.slug)) as string)
  115. );
  116. }
  117. static isOrgChanging(props: Props): boolean {
  118. const {organization} = OrganizationStore.get();
  119. if (!organization) {
  120. return false;
  121. }
  122. return organization.slug !== OrganizationContextContainer.getOrganizationSlug(props);
  123. }
  124. static isOrgStorePopulatedCorrectly(props: Props) {
  125. const {organization, dirty} = OrganizationStore.get();
  126. return !dirty && organization && !OrganizationContextContainer.isOrgChanging(props);
  127. }
  128. static childContextTypes = {
  129. organization: SentryTypes.Organization,
  130. };
  131. constructor(props: Props) {
  132. super(props);
  133. this.state = OrganizationContextContainer.getDefaultState(props);
  134. }
  135. getChildContext() {
  136. return {
  137. organization: this.state.organization,
  138. };
  139. }
  140. componentDidMount() {
  141. this.fetchData(true);
  142. }
  143. componentDidUpdate(prevProps: Props) {
  144. const remountPrevProps: State['prevProps'] = {
  145. orgId: prevProps.params.orgId,
  146. organizationsLoading: prevProps.organizationsLoading,
  147. location: prevProps.location,
  148. };
  149. if (OrganizationContextContainer.shouldRemount(remountPrevProps, this.props)) {
  150. this.remountComponent();
  151. }
  152. }
  153. componentWillUnmount() {
  154. this.unlisteners.forEach(callIfFunction);
  155. }
  156. unlisteners = [
  157. ProjectActions.createSuccess.listen(() => this.onProjectCreation(), undefined),
  158. OrganizationStore.listen(data => this.loadOrganization(data), undefined),
  159. ];
  160. remountComponent = () => {
  161. this.setState(
  162. OrganizationContextContainer.getDefaultState(this.props),
  163. this.fetchData
  164. );
  165. };
  166. onProjectCreation() {
  167. // If a new project was created, we need to re-fetch the
  168. // org details endpoint, which will propagate re-rendering
  169. // for the entire component tree
  170. fetchOrganizationDetails(
  171. this.props.api,
  172. OrganizationContextContainer.getOrganizationSlug(this.props),
  173. true,
  174. false
  175. );
  176. }
  177. isLoading() {
  178. // In the absence of an organization slug, the loading state should be
  179. // derived from this.props.organizationsLoading from OrganizationsStore
  180. if (!OrganizationContextContainer.getOrganizationSlug(this.props)) {
  181. return this.props.organizationsLoading;
  182. }
  183. return this.state.loading;
  184. }
  185. fetchData(isInitialFetch = false) {
  186. if (!OrganizationContextContainer.getOrganizationSlug(this.props)) {
  187. return;
  188. }
  189. // fetch from the store, then fetch from the API if necessary
  190. if (OrganizationContextContainer.isOrgStorePopulatedCorrectly(this.props)) {
  191. return;
  192. }
  193. metric.mark({name: 'organization-details-fetch-start'});
  194. fetchOrganizationDetails(
  195. this.props.api,
  196. OrganizationContextContainer.getOrganizationSlug(this.props),
  197. !OrganizationContextContainer.isOrgChanging(this.props), // if true, will preserve a lightweight org that was fetched,
  198. isInitialFetch
  199. );
  200. }
  201. loadOrganization(orgData: State) {
  202. const {organization, error} = orgData;
  203. const hooks: React.ReactNode[] = [];
  204. if (organization && !error) {
  205. HookStore.get('organization:header').forEach(cb => {
  206. hooks.push(cb(organization));
  207. });
  208. // Configure scope to have organization tag
  209. Sentry.configureScope(scope => {
  210. // XXX(dcramer): this is duplicated in sdk.py on the backend
  211. scope.setTag('organization', organization.id);
  212. scope.setTag('organization.slug', organization.slug);
  213. scope.setContext('organization', {id: organization.id, slug: organization.slug});
  214. });
  215. } else if (error) {
  216. // If user is superuser, open sudo window
  217. const user = ConfigStore.get('user');
  218. if (!user || !user.isSuperuser || error.status !== 403) {
  219. // This `catch` can swallow up errors in development (and tests)
  220. // So let's log them. This may create some noise, especially the test case where
  221. // we specifically test this branch
  222. console.error(error); // eslint-disable-line no-console
  223. } else {
  224. openSudo({
  225. retryRequest: () => Promise.resolve(this.fetchData()),
  226. isSuperuser: true,
  227. needsReload: true,
  228. });
  229. }
  230. }
  231. this.setState({...orgData, hooks}, () => {
  232. // Take a measurement for when organization details are done loading and the new state is applied
  233. if (organization) {
  234. metric.measure({
  235. name: 'app.component.perf',
  236. start: 'organization-details-fetch-start',
  237. data: {
  238. name: 'org-details',
  239. route: getRouteStringFromRoutes(this.props.routes),
  240. organization_id: parseInt(organization.id, 10),
  241. },
  242. });
  243. }
  244. });
  245. }
  246. getOrganizationDetailsEndpoint() {
  247. return `/organizations/${OrganizationContextContainer.getOrganizationSlug(
  248. this.props
  249. )}/`;
  250. }
  251. getTitle() {
  252. return this.state.organization?.name ?? 'Sentry';
  253. }
  254. renderSidebar(): React.ReactNode {
  255. if (!this.props.includeSidebar) {
  256. return null;
  257. }
  258. const {children: _, ...props} = this.props;
  259. return <Sidebar {...props} organization={this.state.organization as Organization} />;
  260. }
  261. renderError() {
  262. let errorComponent: React.ReactElement;
  263. switch (this.state.errorType) {
  264. case ORGANIZATION_FETCH_ERROR_TYPES.ORG_NO_ACCESS:
  265. // We can still render when an org can't be loaded due to 401. The
  266. // backend will handle redirects when this is a problem.
  267. return this.renderBody();
  268. case ORGANIZATION_FETCH_ERROR_TYPES.ORG_NOT_FOUND:
  269. errorComponent = (
  270. <Alert type="error">
  271. {t('The organization you were looking for was not found.')}
  272. </Alert>
  273. );
  274. break;
  275. default:
  276. errorComponent = <LoadingError onRetry={this.remountComponent} />;
  277. }
  278. return <ErrorWrapper>{errorComponent}</ErrorWrapper>;
  279. }
  280. renderBody() {
  281. return (
  282. <SentryDocumentTitle noSuffix title={this.getTitle()}>
  283. <OrganizationContext.Provider value={this.state.organization}>
  284. <div className="app">
  285. {this.state.hooks}
  286. {this.renderSidebar()}
  287. {this.props.children}
  288. </div>
  289. </OrganizationContext.Provider>
  290. </SentryDocumentTitle>
  291. );
  292. }
  293. render() {
  294. if (this.isLoading()) {
  295. return (
  296. <LoadingTriangle>{t('Loading data for your organization.')}</LoadingTriangle>
  297. );
  298. }
  299. if (this.state.error) {
  300. return (
  301. <Fragment>
  302. {this.renderSidebar()}
  303. {this.renderError()}
  304. </Fragment>
  305. );
  306. }
  307. return this.renderBody();
  308. }
  309. }
  310. export default withApi(
  311. withOrganizations(Sentry.withProfiler(OrganizationContextContainer))
  312. );
  313. export {OrganizationContextContainer as OrganizationLegacyContext, OrganizationContext};
  314. const ErrorWrapper = styled('div')`
  315. padding: ${space(3)};
  316. `;