import {useCallback, useEffect} from 'react'; import {fetchTagValues, loadOrganizationTags} from 'sentry/actionCreators/tags'; import SmartSearchBar from 'sentry/components/smartSearchBar'; import {MAX_QUERY_LENGTH, NEGATION_OPERATOR, SEARCH_WILDCARD} from 'sentry/constants'; import {t} from 'sentry/locale'; import type {Organization, PageFilters, Tag, TagCollection, TagValue} from 'sentry/types'; import {SavedSearchType} from 'sentry/types'; import {trackAnalytics} from 'sentry/utils/analytics'; import {isAggregateField} from 'sentry/utils/discover/fields'; import { FieldKind, getFieldDefinition, REPLAY_CLICK_FIELDS, REPLAY_FIELDS, } from 'sentry/utils/fields'; import {MutableSearch} from 'sentry/utils/tokenizeSearch'; import useApi from 'sentry/utils/useApi'; import useTags from 'sentry/utils/useTags'; const SEARCH_SPECIAL_CHARS_REGEXP = new RegExp( `^${NEGATION_OPERATOR}|\\${SEARCH_WILDCARD}`, 'g' ); /** * Prepare query string (e.g. strip special characters like negation operator) */ function prepareQuery(searchQuery: string) { return searchQuery.replace(SEARCH_SPECIAL_CHARS_REGEXP, ''); } const getReplayFieldDefinition = (key: string) => getFieldDefinition(key, 'replay'); function fieldDefinitionsToTagCollection(fieldKeys: string[]): TagCollection { return Object.fromEntries( fieldKeys.map(key => [ key, { key, name: key, ...getReplayFieldDefinition(key), }, ]) ); } const REPLAY_FIELDS_AS_TAGS = fieldDefinitionsToTagCollection(REPLAY_FIELDS); const REPLAY_CLICK_FIELDS_AS_TAGS = fieldDefinitionsToTagCollection(REPLAY_CLICK_FIELDS); function getSupportedTags(supportedTags: TagCollection) { return { ...Object.fromEntries( Object.keys(supportedTags).map(key => [ key, { ...supportedTags[key], kind: getReplayFieldDefinition(key)?.kind ?? FieldKind.TAG, }, ]) ), ...REPLAY_CLICK_FIELDS_AS_TAGS, ...REPLAY_FIELDS_AS_TAGS, }; } type Props = React.ComponentProps & { organization: Organization; pageFilters: PageFilters; placeholder?: string; }; function ReplaySearchBar(props: Props) { const {organization, pageFilters, placeholder} = props; const api = useApi(); const projectIds = pageFilters.projects; const tags = useTags(); useEffect(() => { loadOrganizationTags(api, organization.slug, pageFilters); }, [api, organization.slug, pageFilters]); const getTagValues = useCallback( (tag: Tag, searchQuery: string, _params: object): Promise => { if (isAggregateField(tag.key)) { // We can't really auto suggest values for aggregate fields // or measurements, so we simply don't return Promise.resolve([]); } return fetchTagValues({ api, orgSlug: organization.slug, tagKey: tag.key, search: searchQuery, projectIds: projectIds?.map(String), includeReplays: true, }).then( tagValues => (tagValues as TagValue[]).map(({value}) => value), () => { throw new Error('Unable to fetch event field values'); } ); }, [api, organization.slug, projectIds] ); return ( = 7.44.0)'), }, }} onSearch={(query: string) => { props.onSearch?.(query); const conditions = new MutableSearch(query); const searchKeys = conditions.tokens.map(({key}) => key).filter(Boolean); if (searchKeys.length > 0) { trackAnalytics('replay.search', { search_keys: searchKeys.join(','), organization, }); } }} /> ); } export default ReplaySearchBar;