123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208 |
- import {Fragment} from 'react';
- import {browserHistory} from 'react-router';
- import {Location} from 'history';
- import omit from 'lodash/omit';
- import DropdownControl, {DropdownItem} from 'sentry/components/dropdownControl';
- import EmptyStateWarning from 'sentry/components/emptyStateWarning';
- import SearchBar from 'sentry/components/events/searchBar';
- import * as Layout from 'sentry/components/layouts/thirds';
- import LoadingIndicator from 'sentry/components/loadingIndicator';
- import {getParams} from 'sentry/components/organizations/globalSelectionHeader/getParams';
- import Pagination from 'sentry/components/pagination';
- import {t} from 'sentry/locale';
- import {Organization} from 'sentry/types';
- import {defined} from 'sentry/utils';
- import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent';
- import DiscoverQuery from 'sentry/utils/discover/discoverQuery';
- import EventView from 'sentry/utils/discover/eventView';
- import {isAggregateField} from 'sentry/utils/discover/fields';
- import SuspectSpansQuery from 'sentry/utils/performance/suspectSpans/suspectSpansQuery';
- import {decodeScalar} from 'sentry/utils/queryString';
- import {MutableSearch} from 'sentry/utils/tokenizeSearch';
- import {SetStateAction} from '../types';
- import {generateTransactionLink} from '../utils';
- import OpsFilter from './opsFilter';
- import {Actions} from './styles';
- import SuspectSpanCard from './suspectSpanCard';
- import {SpansTotalValues} from './types';
- import {getSuspectSpanSortFromEventView, SPAN_SORT_OPTIONS} from './utils';
- const ANALYTICS_VALUES = {
- spanOp: (organization: Organization, value: string | undefined) =>
- trackAdvancedAnalyticsEvent('performance_views.spans.change_op', {
- organization,
- operation_name: value,
- }),
- sort: (organization: Organization, value: string | undefined) =>
- trackAdvancedAnalyticsEvent('performance_views.spans.change_sort', {
- organization,
- sort_column: value,
- }),
- };
- type Props = {
- location: Location;
- organization: Organization;
- eventView: EventView;
- setError: SetStateAction<string | undefined>;
- transactionName: string;
- };
- function SpansContent(props: Props) {
- const {location, organization, eventView, setError, transactionName} = props;
- const query = decodeScalar(location.query.query, '');
- function handleChange(key: string) {
- return function (value: string | undefined) {
- ANALYTICS_VALUES[key]?.(organization, value);
- const queryParams = getParams({
- ...(location.query || {}),
- [key]: value,
- });
- // do not propagate pagination when making a new search
- const toOmit = ['cursor'];
- if (!defined(value)) {
- toOmit.push(key);
- }
- const searchQueryParams = omit(queryParams, toOmit);
- browserHistory.push({
- ...location,
- query: searchQueryParams,
- });
- };
- }
- const spanOp = decodeScalar(location.query.spanOp);
- const spanGroup = decodeScalar(location.query.spanGroup);
- const sort = getSuspectSpanSortFromEventView(eventView);
- const totalsView = getTotalsView(eventView);
- return (
- <Layout.Main fullWidth>
- <Actions>
- <OpsFilter
- location={location}
- eventView={eventView}
- organization={organization}
- handleOpChange={handleChange('spanOp')}
- transactionName={transactionName}
- />
- <SearchBar
- organization={organization}
- projectIds={eventView.project}
- query={query}
- fields={eventView.fields}
- onSearch={handleChange('query')}
- />
- <DropdownControl buttonProps={{prefix: sort.prefix}} label={sort.label}>
- {SPAN_SORT_OPTIONS.map(option => (
- <DropdownItem
- key={option.field}
- eventKey={option.field}
- isActive={option.field === sort.field}
- onSelect={handleChange('sort')}
- >
- {option.label}
- </DropdownItem>
- ))}
- </DropdownControl>
- </Actions>
- <DiscoverQuery
- eventView={totalsView}
- orgSlug={organization.slug}
- location={location}
- referrer="api.performance.transaction-spans"
- cursor="0:0:1"
- noPagination
- >
- {({tableData}) => {
- const totals: SpansTotalValues | null = tableData?.data?.[0] ?? null;
- return (
- <SuspectSpansQuery
- location={location}
- orgSlug={organization.slug}
- eventView={eventView}
- perSuspect={10}
- spanOps={defined(spanOp) ? [spanOp] : []}
- spanGroups={defined(spanGroup) ? [spanGroup] : []}
- >
- {({suspectSpans, isLoading, error, pageLinks}) => {
- if (error) {
- setError(error);
- return null;
- }
- // make sure to clear the clear the error message
- setError(undefined);
- if (isLoading) {
- return <LoadingIndicator />;
- }
- if (!suspectSpans?.length) {
- return (
- <EmptyStateWarning>
- <p>{t('No span data found')}</p>
- </EmptyStateWarning>
- );
- }
- return (
- <Fragment>
- {suspectSpans.map(suspectSpan => (
- <SuspectSpanCard
- key={`${suspectSpan.op}-${suspectSpan.group}`}
- location={location}
- organization={organization}
- suspectSpan={suspectSpan}
- generateTransactionLink={generateTransactionLink(transactionName)}
- eventView={eventView}
- totals={totals}
- preview={2}
- />
- ))}
- <Pagination pageLinks={pageLinks} />
- </Fragment>
- );
- }}
- </SuspectSpansQuery>
- );
- }}
- </DiscoverQuery>
- </Layout.Main>
- );
- }
- /**
- * For the totals view, we want to get some transaction level stats like
- * the number of transactions and the sum of the transaction duration.
- * This requires the removal of any aggregate conditions as they can result
- * in unexpected empty responses.
- */
- function getTotalsView(eventView: EventView): EventView {
- const totalsView = eventView.withColumns([
- {kind: 'function', function: ['count', '', undefined, undefined]},
- {kind: 'function', function: ['sum', 'transaction.duration', undefined, undefined]},
- ]);
- const conditions = new MutableSearch(eventView.query);
- // filter out any aggregate conditions
- Object.keys(conditions.filters).forEach(field => {
- if (isAggregateField(field)) {
- conditions.removeFilter(field);
- }
- });
- totalsView.query = conditions.formatString();
- return totalsView;
- }
- export default SpansContent;
|