import {Fragment} from 'react'; import type {RouteComponentProps} from 'react-router'; import isEqual from 'lodash/isEqual'; import omit from 'lodash/omit'; import {addErrorMessage, addSuccessMessage} from 'sentry/actionCreators/indicator'; import {openModal} from 'sentry/actionCreators/modal'; import {updateOrganization} from 'sentry/actionCreators/organizations'; import {Button} from 'sentry/components/button'; import ExternalLink from 'sentry/components/links/externalLink'; import {IconAdd} from 'sentry/icons'; import {t, tct} from 'sentry/locale'; import type {Organization} from 'sentry/types/organization'; import type {Relay, RelayActivity} from 'sentry/types/relay'; import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView'; import SettingsPageHeader from 'sentry/views/settings/components/settingsPageHeader'; import TextBlock from 'sentry/views/settings/components/text/textBlock'; import PermissionAlert from 'sentry/views/settings/organization/permissionAlert'; import Add from './modals/add'; import Edit from './modals/edit'; import EmptyState from './emptyState'; import List from './list'; const RELAY_DOCS_LINK = 'https://getsentry.github.io/relay/'; type Props = { organization: Organization; } & RouteComponentProps<{}, {}>; type State = { relayActivities: Array; relays: Array; } & DeprecatedAsyncView['state']; class RelayWrapper extends DeprecatedAsyncView { componentDidUpdate(prevProps: Props, prevState: State) { if (!isEqual(prevState.relays, this.state.relays)) { // Fetch fresh activities this.fetchData(); updateOrganization({...prevProps.organization, trustedRelays: this.state.relays}); } super.componentDidUpdate(prevProps, prevState); } getTitle() { return t('Relay'); } getDefaultState() { return { ...super.getDefaultState(), relays: this.props.organization.trustedRelays, }; } getEndpoints(): ReturnType { const {organization} = this.props; return [['relayActivities', `/organizations/${organization.slug}/relay_usage/`]]; } setRelays(trustedRelays: Array) { this.setState({relays: trustedRelays}); } handleDelete = (publicKey: Relay['publicKey']) => async () => { const {relays} = this.state; const trustedRelays = relays .filter(relay => relay.publicKey !== publicKey) .map(relay => omit(relay, ['created', 'lastModified'])); try { const response = await this.api.requestPromise( `/organizations/${this.props.organization.slug}/`, { method: 'PUT', data: {trustedRelays}, } ); addSuccessMessage(t('Successfully deleted Relay public key')); this.setRelays(response.trustedRelays); } catch { addErrorMessage(t('An unknown error occurred while deleting Relay public key')); } }; successfullySaved(response: Organization, successMessage: string) { addSuccessMessage(successMessage); this.setRelays(response.trustedRelays); } handleOpenEditDialog = (publicKey: Relay['publicKey']) => () => { const editRelay = this.state.relays.find(relay => relay.publicKey === publicKey); if (!editRelay) { return; } openModal(modalProps => ( { this.successfullySaved(response, t('Successfully updated Relay public key')); }} /> )); }; handleOpenAddDialog = () => { openModal(modalProps => ( { this.successfullySaved(response, t('Successfully added Relay public key')); }} /> )); }; handleRefresh = () => { // Fetch fresh activities this.fetchData(); }; renderContent(disabled: boolean) { const {relays, relayActivities, loading} = this.state; if (loading) { return this.renderLoading(); } if (!relays.length) { return ; } return ( ); } renderBody() { const {organization} = this.props; const disabled = !organization.access.includes('org:write'); return ( } onClick={this.handleOpenAddDialog} disabled={disabled} > {t('Register Key')} } /> {tct( 'Sentry Relay offers enterprise-grade data security by providing a standalone service that acts as a middle layer between your application and sentry.io. Go to [link:Relay Documentation] for setup and details.', {link: } )} {this.renderContent(disabled)} ); } } export default RelayWrapper;