import {CSSProperties, Fragment, useCallback, useMemo, useState} from 'react'; import {browserHistory} from 'react-router'; import styled from '@emotion/styled'; import {Button} from 'sentry/components/button'; import Count from 'sentry/components/count'; import EmptyStateWarning from 'sentry/components/emptyStateWarning'; import IdBadge from 'sentry/components/idBadge'; import Link from 'sentry/components/links/link'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import Pagination from 'sentry/components/pagination'; import PerformanceDuration from 'sentry/components/performanceDuration'; import ScoreBar from 'sentry/components/scoreBar'; import TextOverflow from 'sentry/components/textOverflow'; import {Tooltip} from 'sentry/components/tooltip'; import {CHART_PALETTE} from 'sentry/constants/chartPalette'; import {IconChevron, IconWarning} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import {EventsResultsDataRow} from 'sentry/utils/profiling/hooks/types'; import {useProfileFunctions} from 'sentry/utils/profiling/hooks/useProfileFunctions'; import {generateProfileFlamechartRouteWithQuery} from 'sentry/utils/profiling/routes'; import {decodeScalar} from 'sentry/utils/queryString'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import {useLocation} from 'sentry/utils/useLocation'; import useOrganization from 'sentry/utils/useOrganization'; import useProjects from 'sentry/utils/useProjects'; import { Accordion, AccordionItem, ContentContainer, HeaderContainer, HeaderTitleLegend, StatusContainer, Subtitle, WidgetContainer, } from './styles'; const MAX_FUNCTIONS = 3; const CURSOR_NAME = 'slowFnCursor'; interface SlowestFunctionsWidgetProps { userQuery?: string; } export function SlowestFunctionsWidget({userQuery}: SlowestFunctionsWidgetProps) { const location = useLocation(); const [expandedIndex, setExpandedIndex] = useState(0); const slowFnCursor = useMemo( () => decodeScalar(location.query[CURSOR_NAME]), [location.query] ); const handleCursor = useCallback((cursor, pathname, query) => { browserHistory.push({ pathname, query: {...query, [CURSOR_NAME]: cursor}, }); }, []); const functionsQuery = useProfileFunctions({ fields: functionsFields, referrer: 'api.profiling.suspect-functions.list', sort: { key: 'sum()', order: 'desc', }, query: userQuery, limit: MAX_FUNCTIONS, cursor: slowFnCursor, }); const hasFunctions = (functionsQuery.data?.data?.length || 0) > 0; const totalsQuery = useProfileFunctions({ fields: totalsFields, referrer: 'api.profiling.suspect-functions.totals', sort: { key: 'sum()', order: 'desc', }, query: userQuery, limit: MAX_FUNCTIONS, // make sure to query for the projects from the top functions projects: functionsQuery.isFetched ? [ ...new Set( (functionsQuery.data?.data ?? []).map(func => func['project.id'] as number) ), ] : [], enabled: functionsQuery.isFetched && hasFunctions, }); const isLoading = functionsQuery.isLoading || (hasFunctions && totalsQuery.isLoading); const isError = functionsQuery.isError || totalsQuery.isError; return ( {t('Suspect Functions')} {t('Slowest functions by total time spent.')} {isLoading && ( )} {isError && ( )} {!isError && !isLoading && !hasFunctions && (

{t('No functions found')}

)} {hasFunctions && totalsQuery.isFetched && ( {(functionsQuery.data?.data ?? []).map((f, i) => { const projectEntry = totalsQuery.data?.data?.find( row => row['project.id'] === f['project.id'] ); const projectTotalDuration = projectEntry?.['sum()'] ?? f['sum()']; return ( setExpandedIndex(i)} func={f} totalDuration={projectTotalDuration as number} query={userQuery ?? ''} /> ); })} )}
); } interface SlowestFunctionEntryProps { func: EventsResultsDataRow; isExpanded: boolean; query: string; setExpanded: () => void; totalDuration: number; } const BARS = 10; function SlowestFunctionEntry({ func, isExpanded, query, setExpanded, totalDuration, }: SlowestFunctionEntryProps) { const organization = useOrganization(); const {projects} = useProjects(); const project = projects.find(p => p.id === String(func['project.id'])); const score = Math.ceil((((func['sum()'] as number) ?? 0) / totalDuration) * BARS); const palette = new Array(BARS).fill([CHART_PALETTE[0][0]]); const userQuery = useMemo(() => { const conditions = new MutableSearch(query); conditions.setFilterValues('project.id', [String(func['project.id'])]); conditions.setFilterValues('package', [String(func.package)]); conditions.setFilterValues('function', [String(func.function)]); return conditions.formatString(); }, [func, query]); const functionTransactionsQuery = useProfileFunctions({ fields: functionTransactionsFields, referrer: 'api.profiling.suspect-functions.transactions', sort: { key: 'sum()', order: 'desc', }, query: userQuery, limit: 5, enabled: isExpanded, }); return ( {project && ( )} {func.function} , totalSelfTime: ( ), })} >