import {Fragment} from 'react'; import uniq from 'lodash/uniq'; import AsyncComponent from 'sentry/components/asyncComponent'; import {t} from 'sentry/locale'; import {Frame, Organization, Project, TagWithTopValues} from 'sentry/types'; import {Entry, EventError} from 'sentry/types/event'; 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 = AsyncComponent['props'] & { issueId: string; onSave: () => void; organization: Organization; project: Project; }; type State = { eventData: null | EventError; ownership: null | IssueOwnershipResponse; urlTagData: null | TagWithTopValues; } & AsyncComponent['state']; class ProjectOwnershipModal extends AsyncComponent { 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, }, ], ['eventData', `/issues/${issueId}/events/latest/`], ]; } renderBody() { const {ownership, urlTagData, eventData} = this.state; if (!ownership && !urlTagData && !eventData) { return null; } const urls = urlTagData ? urlTagData.topValues .sort((a, b) => a.count - b.count) .map(i => i.value) .slice(0, 5) : []; // pull frame data out of exception or the stacktrace const entry = (eventData?.entries as Entry[]).find(({type}) => ['exception', 'stacktrace'].includes(type) ); let frames: Frame[] = []; if (entry?.type === 'exception') { frames = entry?.data?.values?.[0]?.stacktrace?.frames ?? []; } if (entry?.type === 'stacktrace') { frames = entry?.data?.frames ?? []; } // filter frames by inApp unless there would be 0 const inAppFrames = frames.filter(frame => frame.inApp); if (inAppFrames.length > 0) { frames = inAppFrames; } const paths = uniq(frames.map(frame => frame.filename || frame.absPath || '')) .filter(i => i) .slice(0, 30); return (

{t('Match against Issue Data: (globbing syntax *, ? supported)')}

); } } export default ProjectOwnershipModal;