import { EntryException, EntryType, Event, ExceptionValue, PlatformType, StacktraceType, Thread, } from 'sentry/types'; const NATIVE_PLATFORMS = ['cocoa', 'native'] as Array; // Finds all frames in a given data blob and returns it's platforms function getPlatforms(exceptionValue: ExceptionValue | StacktraceType | null) { const frames = exceptionValue?.frames ?? []; const stacktraceFrames = (exceptionValue as ExceptionValue)?.stacktrace?.frames ?? []; if (!frames.length && !stacktraceFrames.length) { return []; } return [...frames, ...stacktraceFrames] .map(frame => frame.platform) .filter(platform => !!platform); } function getStackTracePlatforms(event: Event, exceptionEntry: EntryException) { // Fetch platforms in stack traces of an exception entry const exceptionEntryPlatforms = (exceptionEntry.data.values ?? []).flatMap( getPlatforms ); // Fetch platforms in an exception entry const stackTraceEntry = (event.entries.find( entry => entry.type === EntryType.STACKTRACE )?.data ?? {}) as StacktraceType; // Fetch platforms in an exception entry const stackTraceEntryPlatforms = Object.keys(stackTraceEntry).flatMap(key => getPlatforms(stackTraceEntry[key]) ); // Fetch platforms in an thread entry const threadEntry = (event.entries.find(entry => entry.type === EntryType.THREADS)?.data .values ?? []) as Array; // Fetch platforms in a thread entry const threadEntryPlatforms = threadEntry.flatMap(({stacktrace}) => getPlatforms(stacktrace) ); return new Set([ ...exceptionEntryPlatforms, ...stackTraceEntryPlatforms, ...threadEntryPlatforms, ]); } // Checks whether an event indicates that it is a native event. function isNativeEvent(event: Event, exceptionEntry: EntryException) { const {platform} = event; if (platform && NATIVE_PLATFORMS.includes(platform)) { return true; } const stackTracePlatforms = getStackTracePlatforms(event, exceptionEntry); return NATIVE_PLATFORMS.some(nativePlatform => stackTracePlatforms.has(nativePlatform)); } // Checks whether an event indicates that it has an associated minidump. function isMinidumpEvent(exceptionEntry: EntryException) { const {data} = exceptionEntry; return (data.values ?? []).some(value => value.mechanism?.type === 'minidump'); } // Checks whether an event indicates that it has an apple crash report. function isAppleCrashReportEvent(exceptionEntry: EntryException) { const {data} = exceptionEntry; return (data.values ?? []).some(value => value.mechanism?.type === 'applecrashreport'); } export function displayReprocessEventAction(orgFeatures: Array, event?: Event) { if (!event || !orgFeatures.includes('reprocessing-v2')) { return false; } const {entries} = event; const exceptionEntry = entries.find(entry => entry.type === EntryType.EXCEPTION) as | EntryException | undefined; if (!exceptionEntry) { return false; } // We want to show the reprocessing button if the issue in question is native or contains native frames. // The logic is taken from the symbolication pipeline in Python, where it is used to determine whether reprocessing // payloads should be stored: // https://github.com/getsentry/sentry/blob/cb7baef414890336881d67b7a8433ee47198c701/src/sentry/lang/native/processing.py#L425-L426 // It is still not ideal as one can always merge native and non-native events together into one issue, // but it's the best approximation we have. if ( !isMinidumpEvent(exceptionEntry) && !isAppleCrashReportEvent(exceptionEntry) && !isNativeEvent(event, exceptionEntry) ) { return false; } return true; }