import ExternalLink from 'sentry/components/links/externalLink'; import {DEFAULT_QUERY} from 'sentry/constants'; import {t, tct} from 'sentry/locale'; import {Organization} from 'sentry/types'; export enum Query { FOR_REVIEW = 'is:unresolved is:for_review assigned_or_suggested:[me, none]', UNRESOLVED = 'is:unresolved', IGNORED = 'is:ignored', NEW = 'is:new', ARCHIVED = 'is:archived', ESCALATING = 'is:escalating', ONGOING = 'is:ongoing', REPROCESSING = 'is:reprocessing', } type OverviewTab = { /** * Emitted analytics event tab name */ analyticsName: string; /** * Will fetch a count to display on this tab */ count: boolean; /** * Tabs can be disabled via flag */ enabled: boolean; name: string; /** * Tooltip text to be hoverable when text has links */ tooltipHoverable?: boolean; /** * Tooltip text for each tab */ tooltipTitle?: React.ReactNode; }; /** * Get a list of currently active tabs */ export function getTabs(organization: Organization) { const hasEscalatingIssuesUi = organization.features.includes('escalating-issues-ui'); const tabs: Array<[string, OverviewTab]> = [ [ Query.UNRESOLVED, { name: hasEscalatingIssuesUi ? t('Unresolved') : t('All Unresolved'), analyticsName: 'unresolved', count: true, enabled: true, }, ], [ Query.FOR_REVIEW, { name: t('For Review'), analyticsName: 'needs_review', count: true, enabled: true, tooltipTitle: hasEscalatingIssuesUi ? t( 'Issues are marked for review if they are new or escalating, and have not been resolved or archived. Issues are automatically marked reviewed in 7 days.' ) : t(`Issues are marked for review when they are created, unresolved, or unignored. Mark an issue reviewed to move it out of this list. Issues are automatically marked reviewed in 7 days.`), }, ], [ Query.NEW, { name: t('New'), analyticsName: 'new', count: true, enabled: hasEscalatingIssuesUi, }, ], [ Query.ESCALATING, { name: t('Escalating'), analyticsName: 'escalating', count: true, enabled: hasEscalatingIssuesUi, }, ], [ Query.ONGOING, { name: t('Ongoing'), analyticsName: 'ongoing', count: true, enabled: hasEscalatingIssuesUi, }, ], [ Query.ARCHIVED, { name: t('Archived'), analyticsName: 'archived', count: true, enabled: hasEscalatingIssuesUi, }, ], [ Query.IGNORED, { name: t('Ignored'), analyticsName: 'ignored', count: true, enabled: !hasEscalatingIssuesUi, tooltipTitle: t(`Ignored issues don’t trigger alerts. When their ignore conditions are met they become Unresolved and are flagged for review.`), }, ], [ Query.REPROCESSING, { name: t('Reprocessing'), analyticsName: 'reprocessing', count: true, enabled: organization.features.includes('reprocessing-v2'), tooltipTitle: tct( `These [link:reprocessing issues] will take some time to complete. Any new issues that are created during reprocessing will be flagged for review.`, { link: ( ), } ), tooltipHoverable: true, }, ], ]; return tabs.filter(([_query, tab]) => tab.enabled); } /** * @returns queries that should have counts fetched */ export function getTabsWithCounts(organization: Organization) { const tabs = getTabs(organization); return tabs.filter(([_query, tab]) => tab.count).map(([query]) => query); } export function isForReviewQuery(query: string | undefined) { return !!query && /\bis:for_review\b/.test(query); } // the tab counts will look like 99+ export const TAB_MAX_COUNT = 99; type QueryCount = { count: number; hasMore: boolean; }; export type QueryCounts = Partial>; export enum IssueSortOptions { DATE = 'date', NEW = 'new', PRIORITY = 'priority', FREQ = 'freq', USER = 'user', INBOX = 'inbox', } export const DEFAULT_ISSUE_STREAM_SORT = IssueSortOptions.DATE; export function isDefaultIssueStreamSearch({query, sort}: {query: string; sort: string}) { return query === DEFAULT_QUERY && sort === DEFAULT_ISSUE_STREAM_SORT; } export function getSortLabel(key: string) { switch (key) { case IssueSortOptions.NEW: return t('First Seen'); case IssueSortOptions.PRIORITY: return t('Priority'); case IssueSortOptions.FREQ: return t('Events'); case IssueSortOptions.USER: return t('Users'); case IssueSortOptions.INBOX: return t('Date Added'); case IssueSortOptions.DATE: default: return t('Last Seen'); } } export const DISCOVER_EXCLUSION_FIELDS: string[] = [ 'query', 'status', 'bookmarked_by', 'assigned', 'assigned_to', 'unassigned', 'subscribed_by', 'active_at', 'first_release', 'first_seen', 'is', '__text', ]; export const SAVED_SEARCHES_SIDEBAR_OPEN_LOCALSTORAGE_KEY = 'issue-stream-saved-searches-sidebar-open';