modal.tsx 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. import {Fragment} from 'react';
  2. import uniq from 'lodash/uniq';
  3. import AsyncComponent from 'sentry/components/asyncComponent';
  4. import {t} from 'sentry/locale';
  5. import {Frame, Organization, Project, TagWithTopValues} from 'sentry/types';
  6. import {Entry, EventError} from 'sentry/types/event';
  7. import OwnerInput from 'sentry/views/settings/project/projectOwnership/ownerInput';
  8. type IssueOwnershipResponse = {
  9. autoAssignment: boolean;
  10. dateCreated: string;
  11. fallthrough: boolean;
  12. isActive: boolean;
  13. lastUpdated: string;
  14. raw: string;
  15. };
  16. type Props = AsyncComponent['props'] & {
  17. issueId: string;
  18. onSave: () => void;
  19. organization: Organization;
  20. project: Project;
  21. };
  22. type State = {
  23. eventData: null | EventError;
  24. ownership: null | IssueOwnershipResponse;
  25. urlTagData: null | TagWithTopValues;
  26. } & AsyncComponent['state'];
  27. class ProjectOwnershipModal extends AsyncComponent<Props, State> {
  28. getEndpoints(): ReturnType<AsyncComponent['getEndpoints']> {
  29. const {organization, project, issueId} = this.props;
  30. return [
  31. ['ownership', `/projects/${organization.slug}/${project.slug}/ownership/`],
  32. [
  33. 'urlTagData',
  34. `/issues/${issueId}/tags/url/`,
  35. {},
  36. {
  37. allowError: error =>
  38. // Allow for 404s
  39. error.status === 404,
  40. },
  41. ],
  42. ['eventData', `/issues/${issueId}/events/latest/`],
  43. ];
  44. }
  45. renderBody() {
  46. const {ownership, urlTagData, eventData} = this.state;
  47. if (!ownership && !urlTagData && !eventData) {
  48. return null;
  49. }
  50. const urls = urlTagData
  51. ? urlTagData.topValues
  52. .sort((a, b) => a.count - b.count)
  53. .map(i => i.value)
  54. .slice(0, 5)
  55. : [];
  56. // pull frame data out of exception or the stacktrace
  57. const entry = (eventData?.entries as Entry[]).find(({type}) =>
  58. ['exception', 'stacktrace'].includes(type)
  59. );
  60. let frames: Frame[] = [];
  61. if (entry?.type === 'exception') {
  62. frames = entry?.data?.values?.[0]?.stacktrace?.frames ?? [];
  63. }
  64. if (entry?.type === 'stacktrace') {
  65. frames = entry?.data?.frames ?? [];
  66. }
  67. // filter frames by inApp unless there would be 0
  68. const inAppFrames = frames.filter(frame => frame.inApp);
  69. if (inAppFrames.length > 0) {
  70. frames = inAppFrames;
  71. }
  72. const paths = uniq(frames.map(frame => frame.filename || frame.absPath || ''))
  73. .filter(i => i)
  74. .slice(0, 30);
  75. return (
  76. <Fragment>
  77. <p>{t('Match against Issue Data: (globbing syntax *, ? supported)')}</p>
  78. <OwnerInput
  79. {...this.props}
  80. initialText={ownership?.raw || ''}
  81. urls={urls}
  82. paths={paths}
  83. />
  84. </Fragment>
  85. );
  86. }
  87. }
  88. export default ProjectOwnershipModal;