index.tsx 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. import {useCallback} from 'react';
  2. import styled from '@emotion/styled';
  3. import {addErrorMessage, addLoadingMessage} from 'sentry/actionCreators/indicator';
  4. import CheckboxField from 'sentry/components/forms/fields/checkboxField';
  5. import SelectField from 'sentry/components/forms/fields/selectField';
  6. import TextField from 'sentry/components/forms/fields/textField';
  7. import Form from 'sentry/components/forms/form';
  8. import type {OnSubmitCallback} from 'sentry/components/forms/types';
  9. import HookOrDefault from 'sentry/components/hookOrDefault';
  10. import ExternalLink from 'sentry/components/links/externalLink';
  11. import NarrowLayout from 'sentry/components/narrowLayout';
  12. import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle';
  13. import {t, tct} from 'sentry/locale';
  14. import ConfigStore from 'sentry/stores/configStore';
  15. import HookStore from 'sentry/stores/hookStore';
  16. import type {OrganizationSummary} from 'sentry/types/organization';
  17. import {getRegionChoices, shouldDisplayRegions} from 'sentry/utils/regions';
  18. import normalizeUrl from 'sentry/utils/url/normalizeUrl';
  19. import useApi from 'sentry/utils/useApi';
  20. export const DATA_STORAGE_DOCS_LINK =
  21. 'https://docs.sentry.io/product/accounts/choose-your-data-center';
  22. function removeDataStorageLocationFromFormData(
  23. formData: Record<string, any>
  24. ): Record<string, any> {
  25. const shallowFormDataClone = {...formData};
  26. delete shallowFormDataClone.dataStorageLocation;
  27. return shallowFormDataClone;
  28. }
  29. const DataConsentCheck = HookOrDefault({
  30. hookName: 'component:data-consent-org-creation-checkbox',
  31. defaultComponent: null,
  32. });
  33. function OrganizationCreate() {
  34. const termsUrl = ConfigStore.get('termsUrl');
  35. const privacyUrl = ConfigStore.get('privacyUrl');
  36. const isSelfHosted = ConfigStore.get('isSelfHosted');
  37. const relocationUrl = normalizeUrl(`/relocation/`);
  38. const regionChoices = getRegionChoices();
  39. const client = useApi();
  40. const hasDataConsent =
  41. HookStore.get('component:data-consent-org-creation-checkbox').length !== 0;
  42. // This is a trimmed down version of the logic in ApiForm. It validates the
  43. // form data prior to submitting the request, and overrides the request host
  44. // with the selected region's URL if one is provided.
  45. const submitOrganizationCreate: OnSubmitCallback = useCallback(
  46. (data, onSubmitSuccess, onSubmitError, _event, formModel) => {
  47. if (!formModel.validateForm()) {
  48. return;
  49. }
  50. const regionUrl = data.dataStorageLocation;
  51. addLoadingMessage(t('Creating Organization\u2026'));
  52. formModel.setFormSaving();
  53. client.request('/organizations/', {
  54. method: 'POST',
  55. data: removeDataStorageLocationFromFormData(data),
  56. host: regionUrl,
  57. success: onSubmitSuccess,
  58. error: onSubmitError,
  59. });
  60. },
  61. [client]
  62. );
  63. return (
  64. <SentryDocumentTitle title={t('Create Organization')}>
  65. <NarrowLayout showLogout>
  66. <h3>{t('Create a New Organization')}</h3>
  67. <p>
  68. {t(
  69. "Organizations represent the top level in your hierarchy. You'll be able to bundle a collection of teams within an organization as well as give organization-wide permissions to users."
  70. )}
  71. </p>
  72. <Form
  73. initialData={{defaultTeam: true}}
  74. submitLabel={t('Create Organization')}
  75. apiEndpoint="/organizations/"
  76. apiMethod="POST"
  77. onSubmit={submitOrganizationCreate}
  78. onSubmitSuccess={(createdOrg: OrganizationSummary) => {
  79. const hasCustomerDomain =
  80. ConfigStore.get('features').has('system:multi-region');
  81. let nextUrl = normalizeUrl(
  82. `/organizations/${createdOrg.slug}/projects/new/`,
  83. {forceCustomerDomain: hasCustomerDomain}
  84. );
  85. if (hasCustomerDomain) {
  86. nextUrl = `${createdOrg.links.organizationUrl}${nextUrl}`;
  87. }
  88. // redirect to project creation *(BYPASS REACT ROUTER AND FORCE PAGE REFRESH TO GRAB CSRF TOKEN)*
  89. // browserHistory.pushState(null, `/organizations/${data.slug}/projects/new/`);
  90. window.location.assign(nextUrl);
  91. }}
  92. onSubmitError={error => {
  93. addErrorMessage(
  94. error.responseJSON?.detail ?? t('Unable to create organization.')
  95. );
  96. }}
  97. requireChanges
  98. >
  99. <TextField
  100. id="organization-name"
  101. name="name"
  102. label={t('Organization Name')}
  103. placeholder={t('e.g. My Company')}
  104. inline={false}
  105. flexibleControlStateSize
  106. stacked
  107. required
  108. />
  109. {shouldDisplayRegions() && (
  110. <SelectField
  111. name="dataStorageLocation"
  112. label={t('Data Storage Location')}
  113. help={tct(
  114. "Choose where to store your organization's data. Please note, you won't be able to change locations once your organization has been created. [learnMore:Learn More]",
  115. {learnMore: <a href={DATA_STORAGE_DOCS_LINK} />}
  116. )}
  117. choices={regionChoices}
  118. inline={false}
  119. stacked
  120. required
  121. />
  122. )}
  123. {termsUrl && privacyUrl && (
  124. <TermsWrapper hasDataConsent={hasDataConsent}>
  125. <CheckboxField
  126. name="agreeTerms"
  127. label={tct(
  128. 'I agree to the [termsLink:Terms of Service] and the [privacyLink:Privacy Policy]',
  129. {
  130. termsLink: <ExternalLink href={termsUrl} />,
  131. privacyLink: <ExternalLink href={privacyUrl} />,
  132. }
  133. )}
  134. inline={false}
  135. stacked
  136. required
  137. />
  138. </TermsWrapper>
  139. )}
  140. <DataConsentCheck />
  141. {!isSelfHosted && ConfigStore.get('features').has('relocation:enabled') && (
  142. <div>
  143. {tct('[relocationLink:Relocating from self-hosted?]', {
  144. relocationLink: <a href={relocationUrl} />,
  145. })}
  146. </div>
  147. )}
  148. </Form>
  149. </NarrowLayout>
  150. </SentryDocumentTitle>
  151. );
  152. }
  153. export default OrganizationCreate;
  154. const TermsWrapper = styled('div')<{hasDataConsent?: boolean}>`
  155. margin-bottom: ${p => (p.hasDataConsent ? '0' : '16px')};
  156. `;