/** * AccountSecurityDetails is only displayed when user is enrolled in the 2fa method. * It displays created + last used time of the 2fa method. * * Also displays 2fa method specific details. */ import {Fragment} from 'react'; import {RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import {addErrorMessage, addSuccessMessage} from 'sentry/actionCreators/indicator'; import Button from 'sentry/components/button'; import CircleIndicator from 'sentry/components/circleIndicator'; import DateTime from 'sentry/components/dateTime'; import Tooltip from 'sentry/components/tooltip'; import {t} from 'sentry/locale'; import space from 'sentry/styles/space'; import {Authenticator, AuthenticatorDevice} from 'sentry/types'; import AsyncView from 'sentry/views/asyncView'; import RecoveryCodes from 'sentry/views/settings/account/accountSecurity/components/recoveryCodes'; import RemoveConfirm from 'sentry/views/settings/account/accountSecurity/components/removeConfirm'; import U2fEnrolledDetails from 'sentry/views/settings/account/accountSecurity/components/u2fEnrolledDetails'; import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader'; import TextBlock from 'sentry/views/settings/components/text/textBlock'; const ENDPOINT = '/users/me/authenticators/'; type AuthenticatorDateProps = { /** * Can be null or a Date object. * Component will have value "never" if it is null */ date: string | null; label: string; }; function AuthenticatorDate({label, date}: AuthenticatorDateProps) { return ( {label}
{date ? : t('never')}
); } type Props = { deleteDisabled: boolean; onRegenerateBackupCodes: () => void; } & RouteComponentProps<{authId: string}, {}>; type State = { authenticator: Authenticator | null; } & AsyncView['state']; class AccountSecurityDetails extends AsyncView { getTitle() { return t('Security'); } getEndpoints(): ReturnType { const {params} = this.props; const {authId} = params; return [['authenticator', `${ENDPOINT}${authId}/`]]; } handleRemove = async (device?: AuthenticatorDevice) => { const {authenticator} = this.state; if (!authenticator || !authenticator.authId) { return; } // if the device is defined, it means that U2f is being removed // reason for adding a trailing slash is a result of the endpoint on line 109 needing it but it can't be set there as if deviceId is None, the route will end with '//' const deviceId = device ? `${device.key_handle}/` : ''; const deviceName = device ? device.name : t('Authenticator'); this.setState({loading: true}); try { await this.api.requestPromise(`${ENDPOINT}${authenticator.authId}/${deviceId}`, { method: 'DELETE', }); this.props.router.push('/settings/account/security'); addSuccessMessage(t('%s has been removed', deviceName)); } catch { // Error deleting authenticator this.setState({loading: false}); addErrorMessage(t('Error removing %s', deviceName)); } }; handleRename = async (device: AuthenticatorDevice, deviceName: string) => { const {authenticator} = this.state; if (!authenticator?.authId) { return; } // if the device is defined, it means that U2f is being renamed // reason for adding a trailing slash is a result of the endpoint on line 109 needing it but it can't be set there as if deviceId is None, the route will end with '//' const deviceId = device ? `${device.key_handle}/` : ''; this.setState({loading: true}); const data = { name: deviceName, }; try { await this.api.requestPromise(`${ENDPOINT}${authenticator.authId}/${deviceId}`, { method: 'PUT', data, }); this.props.router.push(`/settings/account/security/mfa/${authenticator.authId}`); addSuccessMessage(t('Device was renamed')); } catch { this.setState({loading: false}); addErrorMessage(t('Error renaming the device')); } }; renderBody() { const {authenticator} = this.state; if (!authenticator) { return null; } const {deleteDisabled, onRegenerateBackupCodes} = this.props; return ( {authenticator.name} } action={ {authenticator.isEnrolled && authenticator.allowRotationInPlace && ( )} {authenticator.isEnrolled && authenticator.removeButton && ( )} } /> {authenticator.description} {authenticator.isEnrolled && authenticator.phone && ( {t('Confirmation codes are sent to the following phone number')}: {authenticator.phone} )} ); } } export default AccountSecurityDetails; const AuthenticatorStatus = styled(CircleIndicator)` margin-left: ${space(1)}; `; const AuthenticatorActions = styled('div')` display: flex; justify-content: center; align-items: center; > * { margin-left: ${space(1)}; } `; const AuthenticatorDates = styled('div')` display: grid; gap: ${space(2)}; grid-template-columns: max-content auto; `; const DateLabel = styled('span')` font-weight: bold; `; const PhoneWrapper = styled('div')` margin-top: ${space(4)}; `; const Phone = styled('span')` font-weight: bold; margin-left: ${space(1)}; `;