repositoryProjectPathConfigForm.tsx 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. import {useRef} from 'react';
  2. import pick from 'lodash/pick';
  3. import {FieldFromConfig} from 'sentry/components/forms';
  4. import Form, {FormProps} from 'sentry/components/forms/form';
  5. import FormModel from 'sentry/components/forms/model';
  6. import {Field} from 'sentry/components/forms/types';
  7. import {t} from 'sentry/locale';
  8. import type {
  9. Integration,
  10. IntegrationRepository,
  11. Organization,
  12. Project,
  13. Repository,
  14. RepositoryProjectPathConfig,
  15. } from 'sentry/types';
  16. import {
  17. sentryNameToOption,
  18. trackIntegrationAnalytics,
  19. } from 'sentry/utils/integrationUtil';
  20. import useApi from 'sentry/utils/useApi';
  21. type Props = {
  22. integration: Integration;
  23. onCancel: FormProps['onCancel'];
  24. onSubmitSuccess: FormProps['onSubmitSuccess'];
  25. organization: Organization;
  26. projects: Project[];
  27. repos: Repository[];
  28. existingConfig?: RepositoryProjectPathConfig;
  29. };
  30. function RepositoryProjectPathConfigForm({
  31. existingConfig,
  32. integration,
  33. onCancel,
  34. onSubmitSuccess,
  35. organization,
  36. projects,
  37. repos,
  38. }: Props) {
  39. const api = useApi();
  40. const formRef = useRef(new FormModel());
  41. const repoChoices = repos.map(({name, id}) => ({value: id, label: name}));
  42. /**
  43. * Automatically switch to the default branch for the repo
  44. */
  45. function handleRepoChange(id: string) {
  46. const repo = repos.find(r => r.id === id);
  47. if (!repo) {
  48. return;
  49. }
  50. // Use the integration repo search to get the default branch
  51. api
  52. .requestPromise(
  53. `/organizations/${organization.slug}/integrations/${integration.id}/repos/`,
  54. {query: {search: repo.name}}
  55. )
  56. .then((data: {repos: IntegrationRepository[]}) => {
  57. const {defaultBranch} = data.repos.find(r => r.identifier === repo.name) ?? {};
  58. const isCurrentRepo = formRef.current.getValue('repositoryId') === repo.id;
  59. if (defaultBranch && isCurrentRepo) {
  60. formRef.current.setValue('defaultBranch', defaultBranch);
  61. }
  62. });
  63. }
  64. const formFields: Field[] = [
  65. {
  66. name: 'projectId',
  67. type: 'sentry_project_selector',
  68. required: true,
  69. label: t('Project'),
  70. projects,
  71. },
  72. {
  73. name: 'repositoryId',
  74. type: 'select_async',
  75. required: true,
  76. label: t('Repo'),
  77. placeholder: t('Choose repo'),
  78. url: `/organizations/${organization.slug}/repos/`,
  79. defaultOptions: repoChoices,
  80. onResults: results => results.map(sentryNameToOption),
  81. onChange: handleRepoChange,
  82. },
  83. {
  84. id: 'defaultBranch',
  85. name: 'defaultBranch',
  86. type: 'string',
  87. required: true,
  88. label: t('Branch'),
  89. placeholder: t('Type your branch'),
  90. showHelpInTooltip: true,
  91. help: t(
  92. 'If an event does not have a release tied to a commit, we will use this branch when linking to your source code.'
  93. ),
  94. },
  95. {
  96. name: 'stackRoot',
  97. type: 'string',
  98. required: false,
  99. label: t('Stack Trace Root'),
  100. placeholder: t('Type root path of your stack traces'),
  101. showHelpInTooltip: true,
  102. help: t(
  103. 'Any stack trace starting with this path will be mapped with this rule. An empty string will match all paths.'
  104. ),
  105. },
  106. {
  107. name: 'sourceRoot',
  108. type: 'string',
  109. required: false,
  110. label: t('Source Code Root'),
  111. placeholder: t('Type root path of your source code, e.g. `src/`.'),
  112. showHelpInTooltip: true,
  113. help: t(
  114. 'When a rule matches, the stack trace root is replaced with this path to get the path in your repository. Leaving this empty means replacing the stack trace root with an empty string.'
  115. ),
  116. },
  117. ];
  118. function handlePreSubmit() {
  119. trackIntegrationAnalytics('integrations.stacktrace_submit_config', {
  120. setup_type: 'manual',
  121. view: 'integration_configuration_detail',
  122. provider: integration.provider.key,
  123. organization,
  124. });
  125. }
  126. const initialData = {
  127. defaultBranch: 'master',
  128. stackRoot: '',
  129. sourceRoot: '',
  130. repositoryId: existingConfig?.repoId,
  131. integrationId: integration.id,
  132. ...pick(existingConfig, ['projectId', 'defaultBranch', 'stackRoot', 'sourceRoot']),
  133. };
  134. // endpoint changes if we are making a new row or updating an existing one
  135. const baseEndpoint = `/organizations/${organization.slug}/code-mappings/`;
  136. const endpoint = existingConfig ? `${baseEndpoint}${existingConfig.id}/` : baseEndpoint;
  137. const apiMethod = existingConfig ? 'PUT' : 'POST';
  138. return (
  139. <Form
  140. onSubmitSuccess={onSubmitSuccess}
  141. onPreSubmit={handlePreSubmit}
  142. initialData={initialData}
  143. apiEndpoint={endpoint}
  144. apiMethod={apiMethod}
  145. model={formRef.current}
  146. onCancel={onCancel}
  147. >
  148. {formFields.map(field => (
  149. <FieldFromConfig
  150. key={field.name}
  151. field={field}
  152. inline={false}
  153. stacked
  154. flexibleControlStateSize
  155. />
  156. ))}
  157. </Form>
  158. );
  159. }
  160. export default RepositoryProjectPathConfigForm;