import type {PlatformKey} from 'sentry/data/platformCategories'; import {FieldKind} from 'sentry/utils/fields'; import type {Actor, TimeseriesValue} from './core'; import type {Event, EventMetadata, EventOrGroupType, Level} from './event'; import type {Commit, PullRequest, Repository} from './integrations'; import type {Team} from './organization'; import type {Project} from './project'; import type {Release} from './release'; import type {AvatarUser, User} from './user'; export type EntryData = Record>; /** * Saved issues searches */ export type RecentSearch = { dateCreated: string; id: string; lastSeen: string; organizationId: string; query: string; type: SavedSearchType; }; // XXX: Deprecated Sentry 9 attributes are not included here. export type SavedSearch = { dateCreated: string; id: string; isGlobal: boolean; isPinned: boolean; name: string; query: string; sort: string; type: SavedSearchType; }; export enum SavedSearchType { ISSUE = 0, EVENT = 1, SESSION = 2, REPLAY = 3, } export enum IssueCategory { PERFORMANCE = 'performance', ERROR = 'error', } export enum IssueType { ERROR = 'error', PERFORMANCE_N_PLUS_ONE_DB_QUERIES = 'performance_n_plus_one_db_queries', } type CapabilityInfo = { enabled: boolean; disabledReason?: string; }; /** * Defines what capabilities a category of issue has. Not all categories of * issues work the same. */ export type IssueCategoryCapabilities = { /** * Are codeowner features enabled for this issue */ codeowners: CapabilityInfo; /** * Can the issue be deleted */ delete: CapabilityInfo; /** * Can the issue be deleted and discarded */ deleteAndDiscard: CapabilityInfo; /** * Can the issue be ignored (and the dropdown options) */ ignore: CapabilityInfo; /** * Can the issue be merged */ merge: CapabilityInfo; /** * Can the issue be shared */ share: CapabilityInfo; }; // endpoint: /api/0/issues/:issueId/attachments/?limit=50 export type IssueAttachment = { dateCreated: string; event_id: string; headers: object; id: string; mimetype: string; name: string; sha1: string; size: number; type: string; }; // endpoint: /api/0/projects/:orgSlug/:projSlug/events/:eventId/attachments/ export type EventAttachment = IssueAttachment; /** * Issue Tags */ export type Tag = { key: string; name: string; isInput?: boolean; kind?: FieldKind; /** * How many values should be suggested in autocomplete. * Overrides SmartSearchBar's `maxSearchItems` prop. */ maxSuggestedValues?: number; predefined?: boolean; totalValues?: number; values?: string[]; }; export type TagCollection = Record; export type TagValue = { count: number; firstSeen: string; lastSeen: string; name: string; value: string; email?: string; identifier?: string; ipAddress?: string; key?: string; query?: string; username?: string; } & AvatarUser; type Topvalue = { count: number; firstSeen: string; key: string; lastSeen: string; name: string; value: string; // Might not actually exist. query?: string; }; export type TagWithTopValues = { key: string; name: string; topValues: Array; totalValues: number; uniqueValues: number; canDelete?: boolean; }; /** * Inbox, issue owners and Activity */ export type InboxReasonDetails = { count?: number | null; until?: string | null; user_count?: number | null; user_window?: number | null; window?: number | null; }; export type InboxDetails = { reason_details: InboxReasonDetails; date_added?: string; reason?: number; }; export type SuggestedOwnerReason = | 'suspectCommit' | 'ownershipRule' | 'codeowners' | 'releaseCommit'; // Received from the backend to denote suggested owners of an issue export type SuggestedOwner = { date_added: string; owner: string; type: SuggestedOwnerReason; }; export type IssueOwnership = { autoAssignment: | 'Auto Assign to Suspect Commits' | 'Auto Assign to Issue Owner' | 'Turn off Auto-Assignment'; codeownersAutoSync: boolean; dateCreated: string | null; fallthrough: boolean; isActive: boolean; lastUpdated: string | null; raw: string | null; }; export enum GroupActivityType { NOTE = 'note', SET_RESOLVED = 'set_resolved', SET_RESOLVED_BY_AGE = 'set_resolved_by_age', SET_RESOLVED_IN_RELEASE = 'set_resolved_in_release', SET_RESOLVED_IN_COMMIT = 'set_resolved_in_commit', SET_RESOLVED_IN_PULL_REQUEST = 'set_resolved_in_pull_request', SET_UNRESOLVED = 'set_unresolved', SET_IGNORED = 'set_ignored', SET_PUBLIC = 'set_public', SET_PRIVATE = 'set_private', SET_REGRESSION = 'set_regression', CREATE_ISSUE = 'create_issue', UNMERGE_SOURCE = 'unmerge_source', UNMERGE_DESTINATION = 'unmerge_destination', FIRST_SEEN = 'first_seen', ASSIGNED = 'assigned', UNASSIGNED = 'unassigned', MERGE = 'merge', REPROCESS = 'reprocess', MARK_REVIEWED = 'mark_reviewed', } interface GroupActivityBase { dateCreated: string; id: string; project: Project; assignee?: string; issue?: Group; user?: null | User; } interface GroupActivityNote extends GroupActivityBase { data: { text: string; }; type: GroupActivityType.NOTE; } interface GroupActivitySetResolved extends GroupActivityBase { data: Record; type: GroupActivityType.SET_RESOLVED; } interface GroupActivitySetUnresolved extends GroupActivityBase { data: Record; type: GroupActivityType.SET_UNRESOLVED; } interface GroupActivitySetPublic extends GroupActivityBase { data: Record; type: GroupActivityType.SET_PUBLIC; } interface GroupActivitySetPrivate extends GroupActivityBase { data: Record; type: GroupActivityType.SET_PRIVATE; } interface GroupActivitySetByAge extends GroupActivityBase { data: Record; type: GroupActivityType.SET_RESOLVED_BY_AGE; } interface GroupActivityUnassigned extends GroupActivityBase { data: Record; type: GroupActivityType.UNASSIGNED; } interface GroupActivityFirstSeen extends GroupActivityBase { data: Record; type: GroupActivityType.FIRST_SEEN; } interface GroupActivityMarkReviewed extends GroupActivityBase { data: Record; type: GroupActivityType.MARK_REVIEWED; } interface GroupActivityRegression extends GroupActivityBase { data: { version?: string; }; type: GroupActivityType.SET_REGRESSION; } export interface GroupActivitySetByResolvedInRelease extends GroupActivityBase { data: { current_release_version?: string; version?: string; }; type: GroupActivityType.SET_RESOLVED_IN_RELEASE; } interface GroupActivitySetByResolvedInCommit extends GroupActivityBase { data: { commit: Commit; }; type: GroupActivityType.SET_RESOLVED_IN_COMMIT; } interface GroupActivitySetByResolvedInPullRequest extends GroupActivityBase { data: { pullRequest: PullRequest; }; type: GroupActivityType.SET_RESOLVED_IN_PULL_REQUEST; } export interface GroupActivitySetIgnored extends GroupActivityBase { data: { ignoreCount?: number; ignoreDuration?: number; ignoreUntil?: string; ignoreUserCount?: number; ignoreUserWindow?: number; ignoreWindow?: number; }; type: GroupActivityType.SET_IGNORED; } export interface GroupActivityReprocess extends GroupActivityBase { data: { eventCount: number; newGroupId: number; oldGroupId: number; }; type: GroupActivityType.REPROCESS; } interface GroupActivityUnmergeDestination extends GroupActivityBase { data: { fingerprints: Array; source?: { id: string; shortId: string; }; }; type: GroupActivityType.UNMERGE_DESTINATION; } interface GroupActivityUnmergeSource extends GroupActivityBase { data: { fingerprints: Array; destination?: { id: string; shortId: string; }; }; type: GroupActivityType.UNMERGE_SOURCE; } interface GroupActivityMerge extends GroupActivityBase { data: { issues: Array; }; type: GroupActivityType.MERGE; } export interface GroupActivityAssigned extends GroupActivityBase { data: { assignee: string; assigneeType: string; user: Team | User; assigneeEmail?: string; /** * If the user was assigned via an integration */ integration?: 'projectOwnership' | 'codeowners' | 'slack' | 'msteams'; /** Codeowner or Project owner rule as a string */ rule?: string; }; type: GroupActivityType.ASSIGNED; } export interface GroupActivityCreateIssue extends GroupActivityBase { data: { location: string; provider: string; title: string; }; type: GroupActivityType.CREATE_ISSUE; } export type GroupActivity = | GroupActivityNote | GroupActivitySetResolved | GroupActivitySetUnresolved | GroupActivitySetIgnored | GroupActivitySetByAge | GroupActivitySetByResolvedInRelease | GroupActivitySetByResolvedInCommit | GroupActivitySetByResolvedInPullRequest | GroupActivityFirstSeen | GroupActivityMerge | GroupActivityReprocess | GroupActivityUnassigned | GroupActivityMarkReviewed | GroupActivityUnmergeDestination | GroupActivitySetPublic | GroupActivitySetPrivate | GroupActivityRegression | GroupActivityUnmergeSource | GroupActivityAssigned | GroupActivityCreateIssue; export type Activity = GroupActivity; interface GroupFiltered { count: string; firstSeen: string; lastSeen: string; stats: Record; userCount: number; } export interface GroupStats extends GroupFiltered { filtered: GroupFiltered | null; id: string; lifetime?: GroupFiltered; sessionCount?: string | null; } export interface BaseGroupStatusReprocessing { status: 'reprocessing'; statusDetails: { info: { dateCreated: string; totalEvents: number; } | null; pendingEvents: number; }; } /** * Issue Resolution */ export enum ResolutionStatus { RESOLVED = 'resolved', UNRESOLVED = 'unresolved', IGNORED = 'ignored', } export type ResolutionStatusDetails = { actor?: AvatarUser; autoResolved?: boolean; ignoreCount?: number; // Sent in requests. ignoreUntil is used in responses. ignoreDuration?: number; ignoreUntil?: string; ignoreUserCount?: number; ignoreUserWindow?: number; ignoreWindow?: number; inCommit?: { commit?: string; dateCreated?: string; id?: string; repository?: string | Repository; }; inNextRelease?: boolean; inRelease?: string; repository?: string; }; export type GroupStatusResolution = { status: ResolutionStatus; statusDetails: ResolutionStatusDetails; }; export type GroupRelease = { firstRelease: Release; lastRelease: Release; }; // TODO(ts): incomplete export interface BaseGroup extends GroupRelease { activity: GroupActivity[]; annotations: string[]; assignedTo: Actor; culprit: string; firstSeen: string; hasSeen: boolean; id: string; isBookmarked: boolean; isPublic: boolean; isSubscribed: boolean; isUnhandled: boolean; issueCategory: IssueCategory; issueType: IssueType; lastSeen: string; latestEvent: Event; level: Level; logger: string; metadata: EventMetadata; numComments: number; participants: User[]; permalink: string; platform: PlatformKey; pluginActions: any[]; // TODO(ts) pluginContexts: any[]; // TODO(ts) pluginIssues: any[]; // TODO(ts) project: Project; seenBy: User[]; shareId: string; shortId: string; status: string; subscriptionDetails: {disabled?: boolean; reason?: string} | null; tags: Pick[]; title: string; type: EventOrGroupType; userReportCount: number; inbox?: InboxDetails | null | false; owners?: SuggestedOwner[] | null; } export interface GroupReprocessing // BaseGroupStatusReprocessing status field (enum) incorrectly extends the BaseGroup status field (string) so we omit it. // A proper fix for this would be to make the status field an enum or string and correctly extend it. extends Omit, GroupStats, BaseGroupStatusReprocessing {} export interface GroupResolution // GroupStatusResolution status field (enum) incorrectly extends the BaseGroup status field (string) so we omit it. // A proper fix for this would be to make the status field an enum or string and correctly extend it. extends Omit, GroupStats, GroupStatusResolution {} export type Group = GroupResolution | GroupReprocessing; export interface GroupCollapseRelease extends Omit, Partial {} export type GroupTombstone = { actor: AvatarUser; culprit: string; id: string; level: Level; metadata: EventMetadata; title: string; }; export type ProcessingIssueItem = { checksum: string; data: { // TODO(ts) This type is likely incomplete, but this is what // project processing issues settings uses. _scope: string; image_arch: string; image_path: string; image_uuid: string; }; id: string; lastSeen: string; numEvents: number; type: string; }; export type ProcessingIssue = { hasIssues: boolean; hasMoreResolveableIssues: boolean; issuesProcessing: number; lastSeen: string; numIssues: number; project: string; resolveableIssues: number; signedLink: string; issues?: ProcessingIssueItem[]; }; /** * Datascrubbing */ export type Meta = { chunks: Array; err: Array; len: number; rem: Array; }; export type MetaError = string | [string, any]; export type MetaRemark = Array; export type ChunkType = { rule_id: string | number; text: string; type: string; remark?: string | number; }; /** * User Feedback */ export type UserReport = { comments: string; dateCreated: string; email: string; event: {eventID: string; id: string}; eventID: string; id: string; issue: Group; name: string; user: User; }; export type KeyValueListData = { key: string; subject: string; actionButton?: React.ReactNode; isContextData?: boolean; meta?: Meta; subjectDataTestId?: string; subjectIcon?: React.ReactNode; value?: React.ReactNode; }[]; // Response from ShortIdLookupEndpoint // /organizations/${orgId}/shortids/${query}/ export type ShortIdResponse = { group: Group; groupId: string; organizationSlug: string; projectSlug: string; shortId: string; }; /** * Note used in Group Activity and Alerts for users to comment */ export type Note = { /** * Array of [id, display string] tuples used for @-mentions */ mentions: [string, string][]; /** * Note contents (markdown allowed) */ text: string; };