123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345 |
- import {Fragment} from 'react';
- import styled from '@emotion/styled';
- import {ActivityAvatar} from 'sentry/components/activity/item/avatar';
- import UserAvatar from 'sentry/components/avatar/userAvatar';
- import DateTime from 'sentry/components/dateTime';
- import SelectControl from 'sentry/components/forms/controls/selectControl';
- import Link from 'sentry/components/links/link';
- import Pagination, {CursorHandler} from 'sentry/components/pagination';
- import {PanelTable} from 'sentry/components/panels';
- import Tag from 'sentry/components/tag';
- import {Tooltip} from 'sentry/components/tooltip';
- import {t, tct} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import {AuditLog, Organization, User} from 'sentry/types';
- import {shouldUse24Hours} from 'sentry/utils/dates';
- import useOrganization from 'sentry/utils/useOrganization';
- import useProjects from 'sentry/utils/useProjects';
- import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader';
- import {retentionPrioritiesLabels} from 'sentry/views/settings/projectPerformance/projectPerformance';
- const avatarStyle = {
- width: 36,
- height: 36,
- marginRight: 8,
- };
- const getAvatarDisplay = (logEntryUser: User | undefined) => {
- // Display Sentry's avatar for system or superuser-initiated events
- if (
- logEntryUser?.isSuperuser ||
- (logEntryUser?.name === 'Sentry' && logEntryUser?.email === undefined)
- ) {
- return <SentryAvatar type="system" size={36} />;
- }
- // Display user's avatar for non-superusers-initiated events
- if (logEntryUser !== undefined) {
- return <UserAvatar style={avatarStyle} user={logEntryUser} />;
- }
- return null;
- };
- const addUsernameDisplay = (logEntryUser: User | undefined) => {
- if (logEntryUser?.isSuperuser) {
- return (
- <Name data-test-id="actor-name">
- {logEntryUser.name}
- <StaffTag>{t('Sentry Staff')}</StaffTag>
- </Name>
- );
- }
- if (logEntryUser !== undefined) {
- return <Name data-test-id="actor-name">{logEntryUser.name}</Name>;
- }
- return null;
- };
- const getTypeDisplay = (event: string) => {
- if (event.startsWith('rule.')) {
- return event.replace('rule.', 'issue-alert.');
- }
- if (event.startsWith('alertrule.')) {
- return event.replace('alertrule.', 'metric-alert.');
- }
- return event;
- };
- const getEventOptions = (eventTypes: string[] | null) =>
- eventTypes
- ?.map(type => {
- // Having both rule.x and alertrule.x may be confusing, so we'll replace their labels to be more descriptive.
- // We need to maintain the values here so we still fetch the correct audit log events from the backend should we want
- // to filter.
- // See https://github.com/getsentry/sentry/issues/46997
- if (type.startsWith('rule.')) {
- return {
- label: type.replace('rule.', 'issue-alert.'),
- value: type,
- };
- }
- if (type.startsWith('alertrule.')) {
- return {
- label: type.replace('alertrule.', 'metric-alert.'),
- value: type,
- };
- }
- return {
- label: type,
- value: type,
- };
- })
- .sort((a, b) => a.label.localeCompare(b.label));
- function AuditNote({
- entry,
- orgSlug,
- }: {
- entry: NonNullable<AuditLog>;
- orgSlug: Organization['slug'];
- }) {
- const {projects} = useProjects();
- const project = projects.find(p => p.id === String(entry.data.id));
- if (entry.event.startsWith('rule.')) {
- return <Note>{entry.note.replace('rule', 'issue alert rule')}</Note>;
- }
- if (!project) {
- return <Note>{entry.note}</Note>;
- }
- if (entry.event === 'project.create') {
- return (
- <Note>
- {tct('Created project [projectSettingsLink]', {
- projectSettingsLink: (
- <Link to={`/settings/${orgSlug}/projects/${project.slug}/`}>
- {entry.data.slug}
- </Link>
- ),
- })}
- </Note>
- );
- }
- if (entry.event === 'project.edit') {
- if (entry.data.old_slug && entry.data.new_slug) {
- return (
- <Note>
- {tct('Renamed project slug from [old-slug] to [new-slug]', {
- 'old-slug': entry.data.old_slug,
- 'new-slug': (
- <Link to={`/settings/${orgSlug}/projects/${entry.data.new_slug}/`}>
- {entry.data.new_slug}
- </Link>
- ),
- })}
- </Note>
- );
- }
- return (
- <Note>
- {tct('Edited project [projectSettingsLink] [note]', {
- projectSettingsLink: (
- <Link to={`/settings/${orgSlug}/projects/${project.slug}/`}>
- {entry.data.slug}
- </Link>
- ),
- note: entry.note.replace('edited project settings ', ''),
- })}
- </Note>
- );
- }
- if (entry.event === 'sampling_priority.enabled') {
- return (
- <Note>
- {tct(
- 'Enabled retention priority "[biasLabel]" in project [samplingInProjectSettingsLink]',
- {
- samplingInProjectSettingsLink: (
- <Link to={`/settings/${orgSlug}/projects/${project.slug}/performance/`}>
- {entry.data.slug}
- </Link>
- ),
- biasLabel: retentionPrioritiesLabels[entry.data.name],
- }
- )}
- </Note>
- );
- }
- if (entry.event === 'sampling_priority.disabled') {
- return (
- <Note>
- {tct(
- 'Disabled retention priority "[biasLabel]" in project [samplingInProjectSettingsLink]',
- {
- samplingInProjectSettingsLink: (
- <Link to={`/settings/${orgSlug}/projects/${project.slug}/performance/`}>
- {entry.data.slug}
- </Link>
- ),
- biasLabel: retentionPrioritiesLabels[entry.data.name],
- }
- )}
- </Note>
- );
- }
- return <Note>{entry.note}</Note>;
- }
- type Props = {
- entries: AuditLog[] | null;
- eventType: string | undefined;
- eventTypes: string[] | null;
- isLoading: boolean;
- onCursor: CursorHandler | undefined;
- onEventSelect: (value: string) => void;
- pageLinks: string | null;
- };
- function AuditLogList({
- isLoading,
- pageLinks,
- entries,
- eventType,
- eventTypes,
- onCursor,
- onEventSelect,
- }: Props) {
- const is24Hours = shouldUse24Hours();
- const organization = useOrganization();
- const hasEntries = entries && entries.length > 0;
- const ipv4Length = 15;
- const action = (
- <EventSelector
- clearable
- isDisabled={isLoading}
- name="eventFilter"
- value={eventType}
- placeholder={t('Select Action: ')}
- options={getEventOptions(eventTypes)}
- onChange={options => {
- onEventSelect(options?.value);
- }}
- />
- );
- return (
- <div>
- <SettingsPageHeader title={t('Audit Log')} action={action} />
- <PanelTable
- headers={[t('Member'), t('Action'), t('IP'), t('Time')]}
- isEmpty={!hasEntries && entries?.length === 0}
- emptyMessage={t('No audit entries available')}
- isLoading={isLoading}
- >
- {(entries ?? []).map(entry => {
- if (!entry) {
- return null;
- }
- return (
- <Fragment key={entry.id}>
- <UserInfo>
- <div>{getAvatarDisplay(entry.actor)}</div>
- <NameContainer>
- {addUsernameDisplay(entry.actor)}
- <AuditNote entry={entry} orgSlug={organization.slug} />
- </NameContainer>
- </UserInfo>
- <FlexCenter>
- <MonoDetail>{getTypeDisplay(entry.event)}</MonoDetail>
- </FlexCenter>
- <FlexCenter>
- {entry.ipAddress && (
- <IpAddressOverflow>
- <Tooltip
- title={entry.ipAddress}
- disabled={entry.ipAddress.length <= ipv4Length}
- >
- <MonoDetail>{entry.ipAddress}</MonoDetail>
- </Tooltip>
- </IpAddressOverflow>
- )}
- </FlexCenter>
- <TimestampInfo>
- <DateTime dateOnly date={entry.dateCreated} />
- <DateTime
- timeOnly
- format={is24Hours ? 'HH:mm zz' : 'LT zz'}
- date={entry.dateCreated}
- />
- </TimestampInfo>
- </Fragment>
- );
- })}
- </PanelTable>
- {pageLinks && <Pagination pageLinks={pageLinks} onCursor={onCursor} />}
- </div>
- );
- }
- const SentryAvatar = styled(ActivityAvatar)`
- margin-right: ${space(1)};
- `;
- const Name = styled('strong')`
- font-size: ${p => p.theme.fontSizeMedium};
- `;
- const StaffTag = styled(Tag)`
- padding: ${space(1)};
- `;
- const EventSelector = styled(SelectControl)`
- width: 250px;
- `;
- const UserInfo = styled('div')`
- display: flex;
- align-items: center;
- line-height: 1.2;
- font-size: ${p => p.theme.fontSizeSmall};
- min-width: 250px;
- `;
- const NameContainer = styled('div')`
- display: flex;
- flex-direction: column;
- justify-content: center;
- `;
- const Note = styled('div')`
- font-size: ${p => p.theme.fontSizeSmall};
- word-break: break-word;
- margin-top: ${space(0.5)};
- `;
- const FlexCenter = styled('div')`
- display: flex;
- align-items: center;
- `;
- const IpAddressOverflow = styled('div')`
- ${p => p.theme.overflowEllipsis};
- min-width: 90px;
- `;
- const MonoDetail = styled('code')`
- font-size: ${p => p.theme.fontSizeMedium};
- white-space: no-wrap;
- `;
- const TimestampInfo = styled('div')`
- display: grid;
- grid-template-rows: auto auto;
- gap: ${space(1)};
- font-size: ${p => p.theme.fontSizeMedium};
- `;
- export default AuditLogList;
|