From e7627becec12f019e5279940e3fce8988a3c0f2f Mon Sep 17 00:00:00 2001 From: Tymoteusz Czech <2625371+Tymek@users.noreply.github.com> Date: Fri, 12 Jul 2024 14:44:46 +0200 Subject: [PATCH] feat: API Tokens limit - UI (#7561) When approaching limit or limit reached for the number of API tokens, we show a corresponding message. --- .../CreateApiToken/CreateApiToken.test.tsx | 64 +++++++++++++++++++ .../CreateApiToken/CreateApiToken.tsx | 49 +++++++++++++- frontend/src/component/common/Limit/Limit.tsx | 5 +- 3 files changed, 114 insertions(+), 4 deletions(-) create mode 100644 frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.test.tsx diff --git a/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.test.tsx b/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.test.tsx new file mode 100644 index 000000000000..387e1f0f83e7 --- /dev/null +++ b/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.test.tsx @@ -0,0 +1,64 @@ +import { render } from 'utils/testRenderer'; +import { screen, waitFor } from '@testing-library/react'; +import { testServerRoute, testServerSetup } from 'utils/testServer'; +import { CreateApiToken } from './CreateApiToken'; +import { + ADMIN, + CREATE_CLIENT_API_TOKEN, + CREATE_FRONTEND_API_TOKEN, +} from '@server/types/permissions'; + +const permissions = [ + { permission: CREATE_CLIENT_API_TOKEN }, + { permission: CREATE_FRONTEND_API_TOKEN }, + { permission: ADMIN }, +]; + +const server = testServerSetup(); + +const setupApi = (existingTokensCount: number) => { + testServerRoute(server, '/api/admin/ui-config', { + flags: { + resourceLimits: true, + }, + resourceLimits: { + apiTokens: 1, + }, + versionInfo: { + current: { enterprise: 'version' }, + }, + }); + + testServerRoute(server, '/api/admin/api-tokens', { + tokens: [...Array(existingTokensCount).keys()].map((_, i) => ({ + secret: `token${i}`, + })), + }); +}; + +test('Enabled new token button when limits, version and permission allow for it', async () => { + setupApi(0); + render(, { + permissions, + }); + + const button = await screen.findByText('Create token'); + expect(button).toBeDisabled(); + + await waitFor(async () => { + const button = await screen.findByText('Create token'); + expect(button).not.toBeDisabled(); + }); +}); + +test('Token limit reached', async () => { + setupApi(1); + render(, { + permissions, + }); + + await screen.findByText('You have reached the limit for API tokens'); + + const button = await screen.findByText('Create token'); + expect(button).toBeDisabled(); +}); diff --git a/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.tsx b/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.tsx index e5d30fd576ef..a820a61008d3 100644 --- a/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.tsx +++ b/frontend/src/component/admin/apiToken/CreateApiToken/CreateApiToken.tsx @@ -1,5 +1,8 @@ import { useState } from 'react'; import { useNavigate } from 'react-router-dom'; +import { styled } from '@mui/material'; +import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender'; +import { useUiFlag } from 'hooks/useUiFlag'; import FormTemplate from 'component/common/FormTemplate/FormTemplate'; import ApiTokenForm from '../ApiTokenForm/ApiTokenForm'; import { CreateButton } from 'component/common/CreateButton/CreateButton'; @@ -22,17 +25,45 @@ import { CREATE_CLIENT_API_TOKEN, CREATE_FRONTEND_API_TOKEN, } from '@server/types/permissions'; +import { Limit } from 'component/common/Limit/Limit'; const pageTitle = 'Create API token'; interface ICreateApiTokenProps { modal?: boolean; } + +const StyledLimit = styled(Limit)(({ theme }) => ({ + margin: theme.spacing(2, 0, 4), +})); + +const useApiTokenLimit = () => { + const resourceLimitsEnabled = useUiFlag('resourceLimits'); + const { tokens, loading: loadingTokens } = useApiTokens(); + const { uiConfig, loading: loadingConfig } = useUiConfig(); + const apiTokensLimit = uiConfig.resourceLimits.apiTokens; + + return { + resourceLimitsEnabled, + limit: apiTokensLimit, + currentValue: tokens.length, + limitReached: resourceLimitsEnabled && tokens.length >= apiTokensLimit, + loading: loadingConfig || loadingTokens, + }; +}; + export const CreateApiToken = ({ modal = false }: ICreateApiTokenProps) => { const { setToastApiError } = useToast(); const { uiConfig } = useUiConfig(); const navigate = useNavigate(); const [showConfirm, setShowConfirm] = useState(false); const [token, setToken] = useState(''); + const { + resourceLimitsEnabled, + limit, + currentValue, + limitReached, + loading: loadingLimit, + } = useApiTokenLimit(); const { getApiTokenPayload, @@ -50,7 +81,7 @@ export const CreateApiToken = ({ modal = false }: ICreateApiTokenProps) => { apiTokenTypes, } = useApiTokenForm(); - const { createToken, loading } = useApiTokensApi(); + const { createToken, loading: loadingCreateToken } = useApiTokensApi(); const { refetch } = useApiTokens(); usePageTitle(pageTitle); @@ -96,7 +127,7 @@ export const CreateApiToken = ({ modal = false }: ICreateApiTokenProps) => { return ( { CREATE_CLIENT_API_TOKEN, CREATE_FRONTEND_API_TOKEN, ]} + disabled={ + limitReached || loadingLimit || loadingCreateToken + } /> } > @@ -142,6 +176,17 @@ export const CreateApiToken = ({ modal = false }: ICreateApiTokenProps) => { environment={environment} setEnvironment={setEnvironment} /> + + } + /> void; -}> = ({ name, shortName, limit, currentValue, onClose }) => { + className?: string; +}> = ({ name, shortName, limit, currentValue, onClose, className }) => { const percentageLimit = Math.floor((currentValue / limit) * 100); const belowLimit = currentValue < limit; const threshold = 80; @@ -78,7 +79,7 @@ export const Limit: FC<{ } return ( - +