123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- import {useCallback, useEffect, useMemo} from 'react';
- import {fetchTagValues, loadOrganizationTags} from 'sentry/actionCreators/tags';
- import {SearchQueryBuilder} from 'sentry/components/searchQueryBuilder';
- import SmartSearchBar from 'sentry/components/smartSearchBar';
- import {MAX_QUERY_LENGTH, NEGATION_OPERATOR, SEARCH_WILDCARD} from 'sentry/constants';
- import {t} from 'sentry/locale';
- import type {PageFilters} from 'sentry/types/core';
- import type {Tag, TagCollection, TagValue} from 'sentry/types/group';
- import {SavedSearchType} from 'sentry/types/group';
- import type {Organization} from 'sentry/types/organization';
- import {trackAnalytics} from 'sentry/utils/analytics';
- import {getUtcDateString} from 'sentry/utils/dates';
- 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);
- /**
- * Merges a list of supported tags and replay search fields into one collection.
- */
- function getReplaySearchTags(supportedTags: TagCollection): TagCollection {
- const allTags = {
- ...REPLAY_FIELDS_AS_TAGS,
- ...REPLAY_CLICK_FIELDS_AS_TAGS,
- ...Object.fromEntries(
- Object.keys(supportedTags).map(key => [
- key,
- {
- ...supportedTags[key],
- kind: getReplayFieldDefinition(key)?.kind ?? FieldKind.TAG,
- },
- ])
- ),
- };
- // A hack used to "sort" the dictionary for SearchQueryBuilder.
- // Technically dicts are unordered but this works in dev.
- // To guarantee ordering, we need to implement filterKeySections.
- const keys = Object.keys(allTags);
- keys.sort();
- return Object.fromEntries(keys.map(key => [key, allTags[key]]));
- }
- type Props = React.ComponentProps<typeof SmartSearchBar> & {
- organization: Organization;
- pageFilters: PageFilters;
- };
- function ReplaySearchBar(props: Props) {
- const {organization, pageFilters} = props;
- const api = useApi();
- const projectIds = pageFilters.projects;
- const organizationTags = useTags();
- useEffect(() => {
- loadOrganizationTags(api, organization.slug, pageFilters);
- }, [api, organization.slug, pageFilters]);
- const replayTags = useMemo(
- () => getReplaySearchTags(organizationTags),
- [organizationTags]
- );
- const getTagValues = useCallback(
- (tag: Tag, searchQuery: string): Promise<string[]> => {
- if (isAggregateField(tag.key)) {
- // We can't really auto suggest values for aggregate fields
- // or measurements, so we simply don't
- return Promise.resolve([]);
- }
- const endpointParams = {
- start: pageFilters.datetime.start
- ? getUtcDateString(pageFilters.datetime.start)
- : undefined,
- end: pageFilters.datetime.end
- ? getUtcDateString(pageFilters.datetime.end)
- : undefined,
- statsPeriod: pageFilters.datetime.period,
- };
- return fetchTagValues({
- api,
- orgSlug: organization.slug,
- tagKey: tag.key,
- search: searchQuery,
- projectIds: projectIds?.map(String),
- endpointParams,
- includeReplays: true,
- }).then(
- tagValues => (tagValues as TagValue[]).map(({value}) => value),
- () => {
- throw new Error('Unable to fetch event field values');
- }
- );
- },
- [
- api,
- organization.slug,
- projectIds,
- pageFilters.datetime.end,
- pageFilters.datetime.period,
- pageFilters.datetime.start,
- ]
- );
- const onSearch = props.onSearch;
- const onSearchWithAnalytics = useCallback(
- (query: string) => {
- 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,
- });
- }
- },
- [onSearch, organization]
- );
- if (organization.features.includes('search-query-builder-replays')) {
- return (
- <SearchQueryBuilder
- {...props}
- onChange={undefined} // not implemented and different type from SmartSearchBar
- disallowLogicalOperators={undefined} // ^
- className={props.className}
- fieldDefinitionGetter={getReplayFieldDefinition}
- filterKeys={replayTags}
- filterKeySections={undefined}
- getTagValues={getTagValues}
- initialQuery={props.query ?? props.defaultQuery ?? ''}
- onSearch={onSearchWithAnalytics}
- searchSource={props.searchSource ?? 'replay_index'}
- placeholder={
- props.placeholder ??
- t('Search for users, duration, clicked elements, count_errors, and more')
- }
- recentSearches={SavedSearchType.REPLAY}
- />
- );
- }
- return (
- <SmartSearchBar
- {...props}
- onGetTagValues={getTagValues}
- supportedTags={replayTags}
- placeholder={
- props.placeholder ??
- t('Search for users, duration, clicked elements, count_errors, and more')
- }
- prepareQuery={prepareQuery}
- maxQueryLength={MAX_QUERY_LENGTH}
- searchSource={props.searchSource ?? 'replay_index'}
- savedSearchType={SavedSearchType.REPLAY}
- maxMenuHeight={500}
- hasRecentSearches
- projectIds={projectIds}
- fieldDefinitionGetter={getReplayFieldDefinition}
- mergeSearchGroupWith={{
- click: {
- documentation: t('Search by click selector. (Requires SDK version >= 7.44.0)'),
- },
- }}
- onSearch={onSearchWithAnalytics}
- />
- );
- }
- export default ReplaySearchBar;
|