index.tsx 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580
  1. import {Fragment, MouseEvent} from 'react';
  2. import {browserHistory} from 'react-router';
  3. import styled from '@emotion/styled';
  4. import {Query} from 'history';
  5. import {bulkDelete, bulkUpdate} from 'sentry/actionCreators/group';
  6. import {addLoadingMessage, clearIndicators} from 'sentry/actionCreators/indicator';
  7. import {
  8. ModalRenderProps,
  9. openModal,
  10. openReprocessEventModal,
  11. } from 'sentry/actionCreators/modal';
  12. import {Client} from 'sentry/api';
  13. import Feature from 'sentry/components/acl/feature';
  14. import FeatureDisabled from 'sentry/components/acl/featureDisabled';
  15. import ArchiveActions, {getArchiveActions} from 'sentry/components/actions/archive';
  16. import ActionButton from 'sentry/components/actions/button';
  17. import IgnoreActions, {getIgnoreActions} from 'sentry/components/actions/ignore';
  18. import ResolveActions from 'sentry/components/actions/resolve';
  19. import GuideAnchor from 'sentry/components/assistant/guideAnchor';
  20. import {Button} from 'sentry/components/button';
  21. import {DropdownMenu} from 'sentry/components/dropdownMenu';
  22. import EnvironmentPageFilter from 'sentry/components/environmentPageFilter';
  23. import {
  24. IconArchive,
  25. IconCheckmark,
  26. IconEllipsis,
  27. IconMute,
  28. IconSubscribed,
  29. IconUnsubscribed,
  30. } from 'sentry/icons';
  31. import {t} from 'sentry/locale';
  32. import GroupStore from 'sentry/stores/groupStore';
  33. import {space} from 'sentry/styles/space';
  34. import {
  35. Group,
  36. GroupStatusResolution,
  37. IssueCategory,
  38. Organization,
  39. Project,
  40. ResolutionStatus,
  41. SavedQueryVersions,
  42. } from 'sentry/types';
  43. import {Event} from 'sentry/types/event';
  44. import {trackAnalytics} from 'sentry/utils/analytics';
  45. import {getUtcDateString} from 'sentry/utils/dates';
  46. import EventView from 'sentry/utils/discover/eventView';
  47. import {DiscoverDatasets} from 'sentry/utils/discover/types';
  48. import {displayReprocessEventAction} from 'sentry/utils/displayReprocessEventAction';
  49. import {getAnalyticsDataForGroup} from 'sentry/utils/events';
  50. import {uniqueId} from 'sentry/utils/guid';
  51. import {getConfigForIssueType} from 'sentry/utils/issueTypeConfig';
  52. import withApi from 'sentry/utils/withApi';
  53. import {normalizeUrl} from 'sentry/utils/withDomainRequired';
  54. import withOrganization from 'sentry/utils/withOrganization';
  55. import ShareIssueModal from './shareModal';
  56. import SubscribeAction from './subscribeAction';
  57. type UpdateData =
  58. | {isBookmarked: boolean}
  59. | {isSubscribed: boolean}
  60. | {inbox: boolean}
  61. | GroupStatusResolution;
  62. const isResolutionStatus = (data: UpdateData): data is GroupStatusResolution => {
  63. return (data as GroupStatusResolution).status !== undefined;
  64. };
  65. type Props = {
  66. api: Client;
  67. disabled: boolean;
  68. group: Group;
  69. organization: Organization;
  70. project: Project;
  71. event?: Event;
  72. query?: Query;
  73. };
  74. export function Actions(props: Props) {
  75. const {api, group, project, organization, disabled, event, query = {}} = props;
  76. const {status, isBookmarked} = group;
  77. const bookmarkKey = isBookmarked ? 'unbookmark' : 'bookmark';
  78. const bookmarkTitle = isBookmarked ? t('Remove bookmark') : t('Bookmark');
  79. const hasRelease = !!project.features?.includes('releases');
  80. const isResolved = status === 'resolved';
  81. const isAutoResolved =
  82. group.status === 'resolved' ? group.statusDetails.autoResolved : undefined;
  83. const isIgnored = status === 'ignored';
  84. const hasEscalatingIssues = organization.features.includes('escalating-issues-ui');
  85. const hasDeleteAccess = organization.access.includes('event:admin');
  86. const disabledMarkReviewed = organization.features.includes('remove-mark-reviewed');
  87. const {
  88. delete: deleteCap,
  89. deleteAndDiscard: deleteDiscardCap,
  90. share: shareCap,
  91. } = getConfigForIssueType(group).actions;
  92. const getDiscoverUrl = () => {
  93. const {title, type, shortId} = group;
  94. const groupIsOccurrenceBacked =
  95. group.issueCategory === IssueCategory.PERFORMANCE && !!event?.occurrence;
  96. const config = getConfigForIssueType(group);
  97. const discoverQuery = {
  98. id: undefined,
  99. name: title || type,
  100. fields: ['title', 'release', 'environment', 'user.display', 'timestamp'],
  101. orderby: '-timestamp',
  102. query: `issue:${shortId}`,
  103. projects: [Number(project.id)],
  104. version: 2 as SavedQueryVersions,
  105. range: '90d',
  106. dataset:
  107. config.usesIssuePlatform || groupIsOccurrenceBacked
  108. ? DiscoverDatasets.ISSUE_PLATFORM
  109. : undefined,
  110. };
  111. const discoverView = EventView.fromSavedQuery(discoverQuery);
  112. return discoverView.getResultsViewUrlTarget(organization.slug);
  113. };
  114. const trackIssueAction = (
  115. action:
  116. | 'shared'
  117. | 'deleted'
  118. | 'bookmarked'
  119. | 'subscribed'
  120. | 'mark_reviewed'
  121. | 'discarded'
  122. | 'open_in_discover'
  123. | ResolutionStatus,
  124. substatus?: string,
  125. statusDetailsKey?: string
  126. ) => {
  127. const {alert_date, alert_rule_id, alert_type} = query;
  128. trackAnalytics('issue_details.action_clicked', {
  129. organization,
  130. project_id: parseInt(project.id, 10),
  131. action_type: action,
  132. action_substatus: substatus,
  133. action_status_details: statusDetailsKey,
  134. // Alert properties track if the user came from email/slack alerts
  135. alert_date:
  136. typeof alert_date === 'string' ? getUtcDateString(Number(alert_date)) : undefined,
  137. alert_rule_id: typeof alert_rule_id === 'string' ? alert_rule_id : undefined,
  138. alert_type: typeof alert_type === 'string' ? alert_type : undefined,
  139. ...getAnalyticsDataForGroup(group),
  140. });
  141. };
  142. const onDelete = () => {
  143. addLoadingMessage(t('Delete event\u2026'));
  144. bulkDelete(
  145. api,
  146. {
  147. orgId: organization.slug,
  148. projectId: project.slug,
  149. itemIds: [group.id],
  150. },
  151. {
  152. complete: () => {
  153. clearIndicators();
  154. browserHistory.push(
  155. normalizeUrl({
  156. pathname: `/organizations/${organization.slug}/issues/`,
  157. query: {project: project.id},
  158. })
  159. );
  160. },
  161. }
  162. );
  163. trackIssueAction('deleted');
  164. };
  165. const onUpdate = (data: UpdateData) => {
  166. addLoadingMessage(t('Saving changes\u2026'));
  167. bulkUpdate(
  168. api,
  169. {
  170. orgId: organization.slug,
  171. projectId: project.slug,
  172. itemIds: [group.id],
  173. data,
  174. },
  175. {
  176. complete: clearIndicators,
  177. }
  178. );
  179. if (isResolutionStatus(data)) {
  180. trackIssueAction(
  181. data.status,
  182. data.substatus,
  183. Object.keys(data.statusDetails || {})[0]
  184. );
  185. }
  186. if ((data as {inbox: boolean}).inbox !== undefined) {
  187. trackIssueAction('mark_reviewed');
  188. }
  189. };
  190. const onReprocessEvent = () => {
  191. openReprocessEventModal({organization, groupId: group.id});
  192. };
  193. const onToggleShare = () => {
  194. const newIsPublic = !group.isPublic;
  195. if (newIsPublic) {
  196. trackAnalytics('issue.shared_publicly', {
  197. organization,
  198. });
  199. }
  200. trackIssueAction('shared');
  201. };
  202. const onToggleBookmark = () => {
  203. onUpdate({isBookmarked: !group.isBookmarked});
  204. trackIssueAction('bookmarked');
  205. };
  206. const onToggleSubscribe = () => {
  207. onUpdate({isSubscribed: !group.isSubscribed});
  208. trackIssueAction('subscribed');
  209. };
  210. const onDiscard = () => {
  211. const id = uniqueId();
  212. addLoadingMessage(t('Discarding event\u2026'));
  213. GroupStore.onDiscard(id, group.id);
  214. api.request(`/issues/${group.id}/`, {
  215. method: 'PUT',
  216. data: {discard: true},
  217. success: response => {
  218. GroupStore.onDiscardSuccess(id, group.id, response);
  219. browserHistory.push(
  220. normalizeUrl({
  221. pathname: `/organizations/${organization.slug}/issues/`,
  222. query: {project: project.id},
  223. })
  224. );
  225. },
  226. error: error => {
  227. GroupStore.onDiscardError(id, group.id, error);
  228. },
  229. complete: clearIndicators,
  230. });
  231. trackIssueAction('discarded');
  232. };
  233. const renderDiscardModal = ({Body, Footer, closeModal}: ModalRenderProps) => {
  234. function renderDiscardDisabled({children, ...innerProps}) {
  235. return children({
  236. ...innerProps,
  237. renderDisabled: ({features}: {features: string[]}) => (
  238. <FeatureDisabled
  239. alert
  240. featureName={t('Discard and Delete')}
  241. features={features}
  242. />
  243. ),
  244. });
  245. }
  246. return (
  247. <Feature
  248. features={['projects:discard-groups']}
  249. hookName="feature-disabled:discard-groups"
  250. organization={organization}
  251. project={project}
  252. renderDisabled={renderDiscardDisabled}
  253. >
  254. {({hasFeature, renderDisabled, ...innerProps}) => (
  255. <Fragment>
  256. <Body>
  257. {!hasFeature &&
  258. typeof renderDisabled === 'function' &&
  259. renderDisabled({...innerProps, hasFeature, children: null})}
  260. {t(
  261. `Discarding this event will result in the deletion of most data associated with this issue and future events being discarded before reaching your stream. Are you sure you wish to continue?`
  262. )}
  263. </Body>
  264. <Footer>
  265. <Button onClick={closeModal}>{t('Cancel')}</Button>
  266. <Button
  267. style={{marginLeft: space(1)}}
  268. priority="primary"
  269. onClick={onDiscard}
  270. disabled={!hasFeature}
  271. >
  272. {t('Discard Future Events')}
  273. </Button>
  274. </Footer>
  275. </Fragment>
  276. )}
  277. </Feature>
  278. );
  279. };
  280. const openDeleteModal = () =>
  281. openModal(({Body, Footer, closeModal}: ModalRenderProps) => (
  282. <Fragment>
  283. <Body>
  284. {t('Deleting this issue is permanent. Are you sure you wish to continue?')}
  285. </Body>
  286. <Footer>
  287. <Button onClick={closeModal}>{t('Cancel')}</Button>
  288. <Button style={{marginLeft: space(1)}} priority="primary" onClick={onDelete}>
  289. {t('Delete')}
  290. </Button>
  291. </Footer>
  292. </Fragment>
  293. ));
  294. const openDiscardModal = () => {
  295. openModal(renderDiscardModal);
  296. };
  297. const openShareModal = () => {
  298. openModal(modalProps => (
  299. <ShareIssueModal
  300. {...modalProps}
  301. organization={organization}
  302. projectSlug={group.project.slug}
  303. groupId={group.id}
  304. onToggle={onToggleShare}
  305. />
  306. ));
  307. };
  308. const handleClick = (onClick: (event?: MouseEvent) => void) => {
  309. return function (innerEvent: MouseEvent) {
  310. if (disabled) {
  311. innerEvent.preventDefault();
  312. innerEvent.stopPropagation();
  313. return;
  314. }
  315. onClick(innerEvent);
  316. };
  317. };
  318. const {dropdownItems, onIgnore} = getIgnoreActions({onUpdate});
  319. const {dropdownItems: archiveDropdownItems} = getArchiveActions({
  320. onUpdate,
  321. });
  322. return (
  323. <ActionWrapper>
  324. <DropdownMenu
  325. triggerProps={{
  326. 'aria-label': t('More Actions'),
  327. icon: <IconEllipsis size="xs" />,
  328. showChevron: false,
  329. size: 'sm',
  330. }}
  331. items={[
  332. ...(isIgnored || hasEscalatingIssues
  333. ? []
  334. : [
  335. {
  336. key: 'ignore',
  337. className: 'hidden-sm hidden-md hidden-lg',
  338. label: t('Ignore'),
  339. isSubmenu: true,
  340. disabled,
  341. children: [
  342. {
  343. key: 'ignore-now',
  344. label: t('Ignore Issue'),
  345. onAction: () => onIgnore(),
  346. },
  347. ...dropdownItems,
  348. ],
  349. },
  350. ]),
  351. ...(hasEscalatingIssues
  352. ? isIgnored
  353. ? []
  354. : [
  355. {
  356. key: 'Archive',
  357. className: 'hidden-sm hidden-md hidden-lg',
  358. label: t('Archive'),
  359. isSubmenu: true,
  360. disabled,
  361. children: archiveDropdownItems,
  362. },
  363. ]
  364. : []),
  365. {
  366. key: 'open-in-discover',
  367. className: 'hidden-sm hidden-md hidden-lg',
  368. label: t('Open in Discover'),
  369. to: disabled ? '' : getDiscoverUrl(),
  370. onAction: () => trackIssueAction('open_in_discover'),
  371. },
  372. {
  373. key: group.isSubscribed ? 'unsubscribe' : 'subscribe',
  374. className: 'hidden-sm hidden-md hidden-lg',
  375. label: group.isSubscribed ? t('Unsubscribe') : t('Subscribe'),
  376. disabled: disabled || group.subscriptionDetails?.disabled,
  377. onAction: onToggleSubscribe,
  378. },
  379. ...(disabledMarkReviewed
  380. ? []
  381. : [
  382. {
  383. key: 'mark-review',
  384. label: t('Mark reviewed'),
  385. disabled: !group.inbox || disabled,
  386. details:
  387. !group.inbox || disabled ? t('Issue has been reviewed') : undefined,
  388. onAction: () => onUpdate({inbox: false}),
  389. },
  390. ]),
  391. {
  392. key: 'share',
  393. label: t('Share'),
  394. disabled: disabled || !shareCap.enabled,
  395. hidden: !organization.features.includes('shared-issues'),
  396. onAction: openShareModal,
  397. },
  398. {
  399. key: bookmarkKey,
  400. label: bookmarkTitle,
  401. onAction: onToggleBookmark,
  402. },
  403. {
  404. key: 'reprocess',
  405. label: t('Reprocess events'),
  406. hidden: !displayReprocessEventAction(organization.features, event),
  407. onAction: onReprocessEvent,
  408. },
  409. {
  410. key: 'delete-issue',
  411. priority: 'danger',
  412. label: t('Delete'),
  413. hidden: !hasDeleteAccess,
  414. disabled: !deleteCap.enabled,
  415. details: deleteCap.disabledReason,
  416. onAction: openDeleteModal,
  417. },
  418. {
  419. key: 'delete-and-discard',
  420. priority: 'danger',
  421. label: t('Delete and discard future events'),
  422. hidden: !hasDeleteAccess,
  423. disabled: !deleteDiscardCap.enabled,
  424. details: deleteDiscardCap.disabledReason,
  425. onAction: openDiscardModal,
  426. },
  427. ]}
  428. />
  429. <SubscribeAction
  430. className="hidden-xs"
  431. disabled={disabled}
  432. disablePriority
  433. group={group}
  434. onClick={handleClick(onToggleSubscribe)}
  435. icon={group.isSubscribed ? <IconSubscribed /> : <IconUnsubscribed />}
  436. size="sm"
  437. />
  438. <div className="hidden-xs">
  439. <EnvironmentPageFilter alignDropdown="right" size="sm" />
  440. </div>
  441. <Feature
  442. hookName="feature-disabled:open-in-discover"
  443. features={['discover-basic']}
  444. organization={organization}
  445. >
  446. <ActionButton
  447. className="hidden-xs"
  448. disabled={disabled}
  449. to={disabled ? '' : getDiscoverUrl()}
  450. onClick={() => trackIssueAction('open_in_discover')}
  451. size="sm"
  452. >
  453. <GuideAnchor target="open_in_discover">{t('Open in Discover')}</GuideAnchor>
  454. </ActionButton>
  455. </Feature>
  456. {isResolved || isIgnored ? (
  457. <ActionButton
  458. priority="primary"
  459. title={
  460. isAutoResolved
  461. ? t(
  462. 'This event is resolved due to the Auto Resolve configuration for this project'
  463. )
  464. : t('Change status to unresolved')
  465. }
  466. size="sm"
  467. icon={
  468. isResolved ? (
  469. <IconCheckmark />
  470. ) : hasEscalatingIssues ? (
  471. <IconArchive />
  472. ) : (
  473. <IconMute />
  474. )
  475. }
  476. disabled={disabled || isAutoResolved}
  477. onClick={() =>
  478. onUpdate({
  479. status: ResolutionStatus.UNRESOLVED,
  480. statusDetails: {},
  481. })
  482. }
  483. >
  484. {isIgnored
  485. ? hasEscalatingIssues
  486. ? t('Archived')
  487. : t('Ignored')
  488. : t('Resolved')}
  489. </ActionButton>
  490. ) : (
  491. <Fragment>
  492. {hasEscalatingIssues ? (
  493. <GuideAnchor target="issue_details_archive_button" position="bottom">
  494. <ArchiveActions
  495. className="hidden-xs"
  496. size="sm"
  497. isArchived={isIgnored}
  498. onUpdate={onUpdate}
  499. disabled={disabled}
  500. hideIcon
  501. disableTooltip
  502. />
  503. </GuideAnchor>
  504. ) : (
  505. <IgnoreActions
  506. className="hidden-xs"
  507. isIgnored={isIgnored}
  508. onUpdate={onUpdate}
  509. disabled={disabled}
  510. size="sm"
  511. hideIcon
  512. disableTooltip
  513. />
  514. )}
  515. <GuideAnchor target="resolve" position="bottom" offset={20}>
  516. <ResolveActions
  517. disableTooltip
  518. disabled={disabled}
  519. disableDropdown={disabled}
  520. hasRelease={hasRelease}
  521. latestRelease={project.latestRelease}
  522. onUpdate={onUpdate}
  523. projectSlug={project.slug}
  524. isResolved={isResolved}
  525. isAutoResolved={isAutoResolved}
  526. size="sm"
  527. hideIcon
  528. priority="primary"
  529. />
  530. </GuideAnchor>
  531. </Fragment>
  532. )}
  533. </ActionWrapper>
  534. );
  535. }
  536. const ActionWrapper = styled('div')`
  537. display: flex;
  538. align-items: center;
  539. gap: ${space(0.5)};
  540. `;
  541. export default withApi(withOrganization(Actions));