import {Fragment} from 'react'; import styled from '@emotion/styled'; import DeprecatedAsyncComponent from 'sentry/components/deprecatedAsyncComponent'; import ExternalLink from 'sentry/components/links/externalLink'; import {t, tct} from 'sentry/locale'; import ConfigStore from 'sentry/stores/configStore'; import {space} from 'sentry/styles/space'; import type {Event, Frame} from 'sentry/types/event'; import type {TagWithTopValues} from 'sentry/types/group'; import type {Organization} from 'sentry/types/organization'; import type {Project} from 'sentry/types/project'; import {uniq} from 'sentry/utils/array/uniq'; import {safeURL} from 'sentry/utils/url/safeURL'; import OwnerInput from 'sentry/views/settings/project/projectOwnership/ownerInput'; type IssueOwnershipResponse = { autoAssignment: boolean; dateCreated: string; fallthrough: boolean; isActive: boolean; lastUpdated: string; raw: string; }; type Props = DeprecatedAsyncComponent['props'] & { issueId: string; onCancel: () => void; organization: Organization; project: Project; eventData?: Event; }; type State = { ownership: null | IssueOwnershipResponse; urlTagData: null | TagWithTopValues; } & DeprecatedAsyncComponent['state']; function getFrameSuggestions(eventData?: Event) { // pull frame data out of exception or the stacktrace const entry = eventData?.entries?.find(({type}) => ['exception', 'stacktrace'].includes(type) ); let frames: Frame[] = []; if (entry?.type === 'exception') { frames = entry?.data?.values?.[0]?.stacktrace?.frames ?? []; } else if (entry?.type === 'stacktrace') { frames = entry?.data?.frames ?? []; } // Only display in-app frames frames = frames.filter(frame => frame?.inApp).reverse(); return uniq(frames.map(frame => frame.filename || frame.absPath || '')); } /** * Attempt to remove the origin from a URL */ function getUrlPath(maybeUrl?: string) { if (!maybeUrl) { return ''; } const parsedURL = safeURL(maybeUrl); if (!parsedURL) { return maybeUrl; } return `*${parsedURL.pathname}`; } function OwnershipSuggestions({ paths, urls, eventData, }: { paths: string[]; urls: string[]; eventData?: Event; }) { const email = ConfigStore.get('user')?.email; if (!email) { return null; } const pathSuggestion = paths.length ? `path:${paths[0]} ${email}` : null; const urlSuggestion = urls.length ? `url:${getUrlPath(urls[0])} ${email}` : null; const transactionTag = eventData?.tags?.find(({key}) => key === 'transaction'); const transactionSuggestion = transactionTag ? `tags.transaction:${transactionTag.value} ${email}` : null; return ( # {t('Here’s some suggestions based on this issue')}
{[pathSuggestion, urlSuggestion, transactionSuggestion] .filter(x => x) .map(suggestion => ( {suggestion}
))}
); } class ProjectOwnershipModal extends DeprecatedAsyncComponent { getEndpoints(): ReturnType { const {organization, project, issueId} = this.props; return [ ['ownership', `/projects/${organization.slug}/${project.slug}/ownership/`], [ 'urlTagData', `/issues/${issueId}/tags/url/`, {}, { allowError: error => // Allow for 404s error.status === 404, }, ], ]; } renderBody() { const {ownership, urlTagData} = this.state; const {eventData, organization, project, onCancel} = this.props; if (!ownership) { return null; } const urls = urlTagData ? urlTagData.topValues .sort((a, b) => a.count - b.count) .map(i => i.value) .slice(0, 5) : []; const paths = getFrameSuggestions(eventData); return ( {tct( 'Assign issues based on custom rules. To learn more, [docs:read the docs].', { docs: ( ), } )} ); } } const Description = styled('p')` margin-bottom: ${space(1)}; `; const StyledPre = styled('pre')` word-break: break-word; padding: ${space(2)}; line-height: 1.6; color: ${p => p.theme.subText}; `; export default ProjectOwnershipModal;