123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 |
- import {Fragment, useCallback, useEffect, useState} from 'react';
- import {
- addErrorMessage,
- addLoadingMessage,
- addSuccessMessage,
- } from 'sentry/actionCreators/indicator';
- import Access from 'sentry/components/acl/access';
- import {Button} from 'sentry/components/button';
- import Confirm from 'sentry/components/confirm';
- import DateTime from 'sentry/components/dateTime';
- import FieldGroup from 'sentry/components/forms/fieldGroup';
- import BooleanField from 'sentry/components/forms/fields/booleanField';
- import SelectField from 'sentry/components/forms/fields/selectField';
- import TextField from 'sentry/components/forms/fields/textField';
- import Form from 'sentry/components/forms/form';
- import ExternalLink from 'sentry/components/links/externalLink';
- import {
- Panel,
- PanelAlert,
- PanelBody,
- PanelFooter,
- PanelHeader,
- } from 'sentry/components/panels';
- import TextCopyInput from 'sentry/components/textCopyInput';
- import {t, tct} from 'sentry/locale';
- import {Organization} from 'sentry/types';
- import getDynamicText from 'sentry/utils/getDynamicText';
- import handleXhrErrorResponse from 'sentry/utils/handleXhrErrorResponse';
- import useApi from 'sentry/utils/useApi';
- import KeyRateLimitsForm from 'sentry/views/settings/project/projectKeys/details/keyRateLimitsForm';
- import ProjectKeyCredentials from 'sentry/views/settings/project/projectKeys/projectKeyCredentials';
- import {ProjectKey} from 'sentry/views/settings/project/projectKeys/types';
- type Props = {
- data: ProjectKey;
- onRemove: () => void;
- organization: Organization;
- params: {
- keyId: string;
- projectId: string;
- };
- };
- export enum DynamicSDKLoaderOption {
- HAS_DEBUG = 'hasDebug',
- HAS_PERFORMANCE = 'hasPerformance',
- HAS_REPLAY = 'hasReplay',
- }
- export const sdkLoaderOptions = {
- [DynamicSDKLoaderOption.HAS_PERFORMANCE]: {
- label: t('Enable Performance Monitoring'),
- requiresV7: true,
- },
- [DynamicSDKLoaderOption.HAS_REPLAY]: {
- label: t('Enable Session Replay'),
- requiresV7: true,
- },
- [DynamicSDKLoaderOption.HAS_DEBUG]: {
- label: t('Enable Debug Bundles'),
- requiresV7: false,
- },
- };
- export function KeySettings({onRemove, organization, params, data}: Props) {
- const api = useApi();
- const [browserSdkVersion, setBrowserSdkVersion] = useState(data.browserSdkVersion);
- const [dynamicSDKLoaderOptions, setDynamicSDKLoaderOptions] = useState(
- data.dynamicSdkLoaderOptions
- );
- useEffect(() => {
- setBrowserSdkVersion(data.browserSdkVersion);
- }, [data.browserSdkVersion]);
- useEffect(() => {
- setDynamicSDKLoaderOptions(data.dynamicSdkLoaderOptions);
- }, [data.dynamicSdkLoaderOptions]);
- const {keyId, projectId} = params;
- const apiEndpoint = `/projects/${organization.slug}/${projectId}/keys/${keyId}/`;
- const loaderLink = getDynamicText({
- value: data.dsn.cdn,
- fixed: '__JS_SDK_LOADER_URL__',
- });
- const handleRemove = useCallback(async () => {
- addLoadingMessage(t('Revoking key\u2026'));
- try {
- await api.requestPromise(
- `/projects/${organization.slug}/${projectId}/keys/${keyId}/`,
- {
- method: 'DELETE',
- }
- );
- onRemove();
- addSuccessMessage(t('Revoked key'));
- } catch (_err) {
- addErrorMessage(t('Unable to revoke key'));
- }
- }, [organization, api, onRemove, keyId, projectId]);
- const handleToggleDynamicSDKLoaderOption = useCallback(
- async <T extends typeof dynamicSDKLoaderOptions, K extends keyof T>(
- dynamicSdkLoaderOption: K,
- value: T[K]
- ) => {
- const newDynamicSdkLoaderOptions = Object.keys(dynamicSDKLoaderOptions).reduce(
- (acc, key) => {
- if (key === dynamicSdkLoaderOption) {
- return {...acc, [key]: value};
- }
- return {...acc, [key]: dynamicSDKLoaderOptions[key]};
- },
- {}
- );
- addLoadingMessage();
- try {
- const response = await api.requestPromise(apiEndpoint, {
- method: 'PUT',
- data: {
- dynamicSdkLoaderOptions: newDynamicSdkLoaderOptions,
- },
- });
- setDynamicSDKLoaderOptions(response.dynamicSdkLoaderOptions);
- addSuccessMessage(t('Successfully updated dynamic SDK loader configuration'));
- } catch (error) {
- const message = t('Unable to updated dynamic SDK loader configuration');
- handleXhrErrorResponse(message)(error);
- addErrorMessage(message);
- }
- },
- [api, apiEndpoint, dynamicSDKLoaderOptions, setDynamicSDKLoaderOptions]
- );
- const handleUpdateBrowserSDKVersion = useCallback(
- async (newBrowserSDKVersion: typeof browserSdkVersion) => {
- addLoadingMessage();
- const apiData: {
- browserSdkVersion: typeof browserSdkVersion;
- dynamicSdkLoaderOptions?: Partial<Record<DynamicSDKLoaderOption, boolean>>;
- } = {
- browserSdkVersion: newBrowserSDKVersion,
- };
- const shouldRestrictDynamicSdkLoaderOptions =
- !sdkVersionSupportsPerformanceAndReplay(newBrowserSDKVersion);
- if (shouldRestrictDynamicSdkLoaderOptions) {
- // Performance & Replay are not supported before 7.x
- const newDynamicSdkLoaderOptions = {
- ...dynamicSDKLoaderOptions,
- hasPerformance: false,
- hasReplay: false,
- };
- apiData.dynamicSdkLoaderOptions = newDynamicSdkLoaderOptions;
- }
- try {
- const response = await api.requestPromise(apiEndpoint, {
- method: 'PUT',
- data: apiData,
- });
- setBrowserSdkVersion(response.browserSdkVersion);
- if (shouldRestrictDynamicSdkLoaderOptions) {
- setDynamicSDKLoaderOptions(response.dynamicSdkLoaderOptions);
- }
- addSuccessMessage(t('Successfully updated SDK version'));
- } catch (error) {
- const message = t('Unable to updated SDK version');
- handleXhrErrorResponse(message)(error);
- addErrorMessage(message);
- }
- },
- [
- api,
- apiEndpoint,
- setBrowserSdkVersion,
- setDynamicSDKLoaderOptions,
- dynamicSDKLoaderOptions,
- ]
- );
- return (
- <Access access={['project:write']}>
- {({hasAccess}) => (
- <Fragment>
- <Form
- saveOnBlur
- allowUndo
- apiEndpoint={apiEndpoint}
- apiMethod="PUT"
- initialData={data}
- >
- <Panel>
- <PanelHeader>{t('Details')}</PanelHeader>
- <PanelBody>
- <TextField
- name="name"
- label={t('Name')}
- disabled={!hasAccess}
- required={false}
- maxLength={64}
- />
- <BooleanField
- name="isActive"
- label={t('Enabled')}
- required={false}
- disabled={!hasAccess}
- help="Accept events from this key? This may be used to temporarily suspend a key."
- />
- <FieldGroup label={t('Created')}>
- <div className="controls">
- <DateTime date={data.dateCreated} />
- </div>
- </FieldGroup>
- </PanelBody>
- </Panel>
- </Form>
- <KeyRateLimitsForm
- organization={organization}
- params={params}
- data={data}
- disabled={!hasAccess}
- />
- <Panel>
- <PanelHeader>{t('JavaScript Loader')}</PanelHeader>
- <PanelBody>
- <FieldGroup
- help={tct(
- 'Copy this script into your website to setup your JavaScript SDK without any additional configuration. [link]',
- {
- link: (
- <ExternalLink href="https://docs.sentry.io/platforms/javascript/install/lazy-load-sentry/">
- What does the script provide?
- </ExternalLink>
- ),
- }
- )}
- inline={false}
- flexibleControlStateSize
- >
- <TextCopyInput>
- {`<script src='${loaderLink}' crossorigin="anonymous"></script>`}
- </TextCopyInput>
- </FieldGroup>
- <SelectField
- name="browserSdkVersion"
- options={
- data.browserSdk
- ? data.browserSdk.choices.map(([value, label]) => ({
- value,
- label,
- }))
- : []
- }
- value={browserSdkVersion}
- onChange={value => handleUpdateBrowserSDKVersion(value)}
- placeholder={t('4.x')}
- allowClear={false}
- disabled={!hasAccess}
- help={t(
- 'Select the version of the SDK that should be loaded. Note that it can take a few minutes until this change is live.'
- )}
- />
- </PanelBody>
- <PanelFooter>
- {Object.entries(sdkLoaderOptions).map(([key, value]) => {
- const sdkLoaderOption = Object.keys(dynamicSDKLoaderOptions).find(
- dynamicSdkLoaderOption => dynamicSdkLoaderOption === key
- );
- if (!sdkLoaderOption) {
- return null;
- }
- return (
- <BooleanField
- label={value.label}
- key={key}
- name={key}
- value={
- value.requiresV7 &&
- !sdkVersionSupportsPerformanceAndReplay(browserSdkVersion)
- ? false
- : dynamicSDKLoaderOptions[sdkLoaderOption]
- }
- onChange={() =>
- handleToggleDynamicSDKLoaderOption(
- sdkLoaderOption as DynamicSDKLoaderOption,
- !dynamicSDKLoaderOptions[sdkLoaderOption]
- )
- }
- disabled={
- !hasAccess ||
- (value.requiresV7 &&
- !sdkVersionSupportsPerformanceAndReplay(browserSdkVersion))
- }
- help={
- value.requiresV7 &&
- !sdkVersionSupportsPerformanceAndReplay(browserSdkVersion)
- ? t('Only available in SDK version 7.x and above')
- : key === DynamicSDKLoaderOption.HAS_REPLAY &&
- dynamicSDKLoaderOptions[sdkLoaderOption]
- ? t(
- 'When using Replay, the loader will load the ES6 bundle instead of the ES5 bundle.'
- )
- : undefined
- }
- disabledReason={
- !hasAccess
- ? t('You do not have permission to edit this setting')
- : undefined
- }
- />
- );
- })}
- </PanelFooter>
- </Panel>
- <Panel>
- <PanelHeader>{t('Credentials')}</PanelHeader>
- <PanelBody>
- <PanelAlert type="info" showIcon>
- {t(
- 'Your credentials are coupled to a public and secret key. Different clients will require different credentials, so make sure you check the documentation before plugging things in.'
- )}
- </PanelAlert>
- <ProjectKeyCredentials
- projectId={`${data.projectId}`}
- data={data}
- showPublicKey
- showSecretKey
- showProjectId
- />
- </PanelBody>
- </Panel>
- <Access access={['project:admin']}>
- <Panel>
- <PanelHeader>{t('Revoke Key')}</PanelHeader>
- <PanelBody>
- <FieldGroup
- label={t('Revoke Key')}
- help={t(
- 'Revoking this key will immediately remove and suspend the credentials. This action is irreversible.'
- )}
- >
- <div>
- <Confirm
- priority="danger"
- message={t(
- 'Are you sure you want to revoke this key? This will immediately remove and suspend the credentials.'
- )}
- onConfirm={handleRemove}
- confirmText={t('Revoke Key')}
- >
- <Button priority="danger">{t('Revoke Key')}</Button>
- </Confirm>
- </div>
- </FieldGroup>
- </PanelBody>
- </Panel>
- </Access>
- </Fragment>
- )}
- </Access>
- );
- }
- function sdkVersionSupportsPerformanceAndReplay(sdkVersion: string): boolean {
- return sdkVersion === 'latest' || sdkVersion === '7.x';
- }
|