mirror of
https://github.com/langgenius/dify.git
synced 2026-04-27 11:06:46 +08:00
Co-authored-by: CodingOnStar <hanxujiang@dify.com> Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
615 lines
22 KiB
TypeScript
615 lines
22 KiB
TypeScript
import { act, render, screen, waitFor } from '@testing-library/react'
|
|
import userEvent from '@testing-library/user-event'
|
|
import SecretKeyModal from './secret-key-modal'
|
|
|
|
// Mock the app context
|
|
const mockCurrentWorkspace = vi.fn().mockReturnValue({
|
|
id: 'workspace-1',
|
|
name: 'Test Workspace',
|
|
})
|
|
const mockIsCurrentWorkspaceManager = vi.fn().mockReturnValue(true)
|
|
const mockIsCurrentWorkspaceEditor = vi.fn().mockReturnValue(true)
|
|
|
|
vi.mock('@/context/app-context', () => ({
|
|
useAppContext: () => ({
|
|
currentWorkspace: mockCurrentWorkspace(),
|
|
isCurrentWorkspaceManager: mockIsCurrentWorkspaceManager(),
|
|
isCurrentWorkspaceEditor: mockIsCurrentWorkspaceEditor(),
|
|
}),
|
|
}))
|
|
|
|
// Mock the timestamp hook
|
|
vi.mock('@/hooks/use-timestamp', () => ({
|
|
default: () => ({
|
|
formatTime: vi.fn((value: number, _format: string) => `Formatted: ${value}`),
|
|
formatDate: vi.fn((value: string, _format: string) => `Formatted: ${value}`),
|
|
}),
|
|
}))
|
|
|
|
// Mock API services
|
|
const mockCreateAppApikey = vi.fn().mockResolvedValue({ token: 'new-app-token-123' })
|
|
const mockDelAppApikey = vi.fn().mockResolvedValue({})
|
|
vi.mock('@/service/apps', () => ({
|
|
createApikey: (...args: unknown[]) => mockCreateAppApikey(...args),
|
|
delApikey: (...args: unknown[]) => mockDelAppApikey(...args),
|
|
}))
|
|
|
|
const mockCreateDatasetApikey = vi.fn().mockResolvedValue({ token: 'new-dataset-token-123' })
|
|
const mockDelDatasetApikey = vi.fn().mockResolvedValue({})
|
|
vi.mock('@/service/datasets', () => ({
|
|
createApikey: (...args: unknown[]) => mockCreateDatasetApikey(...args),
|
|
delApikey: (...args: unknown[]) => mockDelDatasetApikey(...args),
|
|
}))
|
|
|
|
// Mock React Query hooks for apps
|
|
const mockAppApiKeysData = vi.fn().mockReturnValue({ data: [] })
|
|
const mockIsAppApiKeysLoading = vi.fn().mockReturnValue(false)
|
|
const mockInvalidateAppApiKeys = vi.fn()
|
|
|
|
vi.mock('@/service/use-apps', () => ({
|
|
useAppApiKeys: (_appId: string, _options: unknown) => ({
|
|
data: mockAppApiKeysData(),
|
|
isLoading: mockIsAppApiKeysLoading(),
|
|
}),
|
|
useInvalidateAppApiKeys: () => mockInvalidateAppApiKeys,
|
|
}))
|
|
|
|
// Mock React Query hooks for datasets
|
|
const mockDatasetApiKeysData = vi.fn().mockReturnValue({ data: [] })
|
|
const mockIsDatasetApiKeysLoading = vi.fn().mockReturnValue(false)
|
|
const mockInvalidateDatasetApiKeys = vi.fn()
|
|
|
|
vi.mock('@/service/knowledge/use-dataset', () => ({
|
|
useDatasetApiKeys: (_options: unknown) => ({
|
|
data: mockDatasetApiKeysData(),
|
|
isLoading: mockIsDatasetApiKeysLoading(),
|
|
}),
|
|
useInvalidateDatasetApiKeys: () => mockInvalidateDatasetApiKeys,
|
|
}))
|
|
|
|
describe('SecretKeyModal', () => {
|
|
const defaultProps = {
|
|
isShow: true,
|
|
onClose: vi.fn(),
|
|
}
|
|
|
|
beforeEach(() => {
|
|
vi.clearAllMocks()
|
|
mockCurrentWorkspace.mockReturnValue({ id: 'workspace-1', name: 'Test Workspace' })
|
|
mockIsCurrentWorkspaceManager.mockReturnValue(true)
|
|
mockIsCurrentWorkspaceEditor.mockReturnValue(true)
|
|
mockAppApiKeysData.mockReturnValue({ data: [] })
|
|
mockIsAppApiKeysLoading.mockReturnValue(false)
|
|
mockDatasetApiKeysData.mockReturnValue({ data: [] })
|
|
mockIsDatasetApiKeysLoading.mockReturnValue(false)
|
|
})
|
|
|
|
describe('rendering when shown', () => {
|
|
it('should render the modal when isShow is true', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
expect(screen.getByText('appApi.apiKeyModal.apiSecretKey')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render the tips text', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
expect(screen.getByText('appApi.apiKeyModal.apiSecretKeyTips')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render the create new key button', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
expect(screen.getByText('appApi.apiKeyModal.createNewSecretKey')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render the close icon', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
// Modal renders via portal, so we need to query from document.body
|
|
const closeIcon = document.body.querySelector('svg.cursor-pointer')
|
|
expect(closeIcon).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('rendering when hidden', () => {
|
|
it('should not render content when isShow is false', () => {
|
|
render(<SecretKeyModal {...defaultProps} isShow={false} />)
|
|
expect(screen.queryByText('appApi.apiKeyModal.apiSecretKey')).not.toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('loading state', () => {
|
|
it('should show loading when app API keys are loading', () => {
|
|
mockIsAppApiKeysLoading.mockReturnValue(true)
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.getByRole('status')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should show loading when dataset API keys are loading', () => {
|
|
mockIsDatasetApiKeysLoading.mockReturnValue(true)
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
expect(screen.getByRole('status')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should not show loading when data is loaded', () => {
|
|
mockIsAppApiKeysLoading.mockReturnValue(false)
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.queryByRole('status')).not.toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('API keys list for app', () => {
|
|
const apiKeys = [
|
|
{ id: 'key-1', token: 'sk-abc123def456ghi789', created_at: 1700000000, last_used_at: 1700100000 },
|
|
{ id: 'key-2', token: 'sk-xyz987wvu654tsr321', created_at: 1700050000, last_used_at: null },
|
|
]
|
|
|
|
beforeEach(() => {
|
|
mockAppApiKeysData.mockReturnValue({ data: apiKeys })
|
|
})
|
|
|
|
it('should render API keys when available', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
// Token 'sk-abc123def456ghi789' (21 chars) -> first 3 'sk-' + '...' + last 20 'k-abc123def456ghi789'
|
|
expect(screen.getByText('sk-...k-abc123def456ghi789')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render created time for keys', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.getByText('Formatted: 1700000000')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render last used time for keys', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.getByText('Formatted: 1700100000')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render "never" for keys without last_used_at', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.getByText('appApi.never')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render delete button for managers', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
// Delete button contains RiDeleteBinLine SVG - look for SVGs with h-4 w-4 class within buttons
|
|
const buttons = screen.getAllByRole('button')
|
|
// There should be at least 3 buttons: copy feedback, delete, and create
|
|
expect(buttons.length).toBeGreaterThanOrEqual(2)
|
|
// Check for delete icon SVG - Modal renders via portal
|
|
const deleteIcon = document.body.querySelector('svg[class*="h-4"][class*="w-4"]')
|
|
expect(deleteIcon).toBeInTheDocument()
|
|
})
|
|
|
|
it('should not render delete button for non-managers', () => {
|
|
mockIsCurrentWorkspaceManager.mockReturnValue(false)
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
// The specific delete action button should not be present
|
|
const actionButtons = screen.getAllByRole('button')
|
|
// Should only have copy and create buttons, not delete
|
|
expect(actionButtons.length).toBeGreaterThan(0)
|
|
})
|
|
|
|
it('should render table headers', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
expect(screen.getByText('appApi.apiKeyModal.secretKey')).toBeInTheDocument()
|
|
expect(screen.getByText('appApi.apiKeyModal.created')).toBeInTheDocument()
|
|
expect(screen.getByText('appApi.apiKeyModal.lastUsed')).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('API keys list for dataset', () => {
|
|
const datasetKeys = [
|
|
{ id: 'dk-1', token: 'dk-abc123def456ghi789', created_at: 1700000000, last_used_at: 1700100000 },
|
|
]
|
|
|
|
beforeEach(() => {
|
|
mockDatasetApiKeysData.mockReturnValue({ data: datasetKeys })
|
|
})
|
|
|
|
it('should render dataset API keys when no appId', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
// Token 'dk-abc123def456ghi789' (21 chars) -> first 3 'dk-' + '...' + last 20 'k-abc123def456ghi789'
|
|
expect(screen.getByText('dk-...k-abc123def456ghi789')).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('close functionality', () => {
|
|
it('should call onClose when X icon is clicked', async () => {
|
|
const user = userEvent.setup()
|
|
const onClose = vi.fn()
|
|
render(<SecretKeyModal {...defaultProps} onClose={onClose} />)
|
|
|
|
// Modal renders via portal, so we need to query from document.body
|
|
const closeIcon = document.body.querySelector('svg.cursor-pointer')
|
|
expect(closeIcon).toBeInTheDocument()
|
|
|
|
await act(async () => {
|
|
await user.click(closeIcon!)
|
|
})
|
|
|
|
expect(onClose).toHaveBeenCalledTimes(1)
|
|
})
|
|
})
|
|
|
|
describe('create new key', () => {
|
|
it('should call create API for app when button is clicked', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockCreateAppApikey).toHaveBeenCalledWith({
|
|
url: '/apps/app-123/api-keys',
|
|
body: {},
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should call create API for dataset when no appId', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockCreateDatasetApikey).toHaveBeenCalledWith({
|
|
url: '/datasets/api-keys',
|
|
body: {},
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should show generate modal after creating key', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
// The SecretKeyGenerateModal should be shown with the new token
|
|
expect(screen.getByText('appApi.apiKeyModal.generateTips')).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
it('should invalidate app API keys after creating', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockInvalidateAppApiKeys).toHaveBeenCalledWith('app-123')
|
|
})
|
|
})
|
|
|
|
it('should invalidate dataset API keys after creating (no appId)', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockInvalidateDatasetApiKeys).toHaveBeenCalled()
|
|
})
|
|
})
|
|
|
|
it('should disable create button when no workspace', () => {
|
|
mockCurrentWorkspace.mockReturnValue(null)
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey').closest('button')
|
|
expect(createButton).toBeDisabled()
|
|
})
|
|
|
|
it('should disable create button when not editor', () => {
|
|
mockIsCurrentWorkspaceEditor.mockReturnValue(false)
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey').closest('button')
|
|
expect(createButton).toBeDisabled()
|
|
})
|
|
})
|
|
|
|
describe('delete key', () => {
|
|
const apiKeys = [
|
|
{ id: 'key-1', token: 'sk-abc123def456ghi789', created_at: 1700000000, last_used_at: 1700100000 },
|
|
]
|
|
|
|
beforeEach(() => {
|
|
mockAppApiKeysData.mockReturnValue({ data: apiKeys })
|
|
})
|
|
|
|
it('should render delete button for managers', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find buttons that contain SVG (delete/copy buttons)
|
|
const actionButtons = screen.getAllByRole('button')
|
|
// There should be at least copy, delete, and create buttons
|
|
expect(actionButtons.length).toBeGreaterThanOrEqual(3)
|
|
})
|
|
|
|
it('should render API key row with actions', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Verify the truncated token is rendered
|
|
expect(screen.getByText('sk-...k-abc123def456ghi789')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should have action buttons in the key row', () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Check for action button containers - Modal renders via portal
|
|
const actionContainers = document.body.querySelectorAll('[class*="space-x-2"]')
|
|
expect(actionContainers.length).toBeGreaterThan(0)
|
|
})
|
|
|
|
it('should have delete button visible for managers', async () => {
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find the delete button by looking for the button with the delete icon
|
|
const deleteIcon = document.body.querySelector('svg[class*="h-4"][class*="w-4"]')
|
|
const deleteButton = deleteIcon?.closest('button')
|
|
expect(deleteButton).toBeInTheDocument()
|
|
})
|
|
|
|
it('should show confirm dialog when delete button is clicked', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find delete button by action-btn class (second action button after copy)
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
// The delete button is the second action button (first is copy)
|
|
const deleteButton = actionButtons[1]
|
|
expect(deleteButton).toBeInTheDocument()
|
|
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Confirm dialog should appear
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTips')).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
it('should call delete API for app when confirmed', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find and click delete button
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
const deleteButton = actionButtons[1]
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Wait for confirm dialog and click confirm
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
})
|
|
|
|
// Find and click the confirm button
|
|
const confirmButton = screen.getByText('common.operation.confirm')
|
|
await act(async () => {
|
|
await user.click(confirmButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockDelAppApikey).toHaveBeenCalledWith({
|
|
url: '/apps/app-123/api-keys/key-1',
|
|
params: {},
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should invalidate app API keys after deleting', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find and click delete button
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
const deleteButton = actionButtons[1]
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Wait for confirm dialog and click confirm
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
})
|
|
|
|
const confirmButton = screen.getByText('common.operation.confirm')
|
|
await act(async () => {
|
|
await user.click(confirmButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockInvalidateAppApiKeys).toHaveBeenCalledWith('app-123')
|
|
})
|
|
})
|
|
|
|
it('should close confirm dialog and clear delKeyId when cancel is clicked', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Find and click delete button
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
const deleteButton = actionButtons[1]
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Wait for confirm dialog
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
})
|
|
|
|
// Click cancel button
|
|
const cancelButton = screen.getByText('common.operation.cancel')
|
|
await act(async () => {
|
|
await user.click(cancelButton)
|
|
})
|
|
|
|
// Confirm dialog should close
|
|
await waitFor(() => {
|
|
expect(screen.queryByText('appApi.actionMsg.deleteConfirmTitle')).not.toBeInTheDocument()
|
|
})
|
|
|
|
// Delete API should not be called
|
|
expect(mockDelAppApikey).not.toHaveBeenCalled()
|
|
})
|
|
})
|
|
|
|
describe('delete key for dataset', () => {
|
|
const datasetKeys = [
|
|
{ id: 'dk-1', token: 'dk-abc123def456ghi789', created_at: 1700000000, last_used_at: 1700100000 },
|
|
]
|
|
|
|
beforeEach(() => {
|
|
mockDatasetApiKeysData.mockReturnValue({ data: datasetKeys })
|
|
})
|
|
|
|
it('should call delete API for dataset when no appId', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
// Find and click delete button
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
const deleteButton = actionButtons[1]
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Wait for confirm dialog and click confirm
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
})
|
|
|
|
const confirmButton = screen.getByText('common.operation.confirm')
|
|
await act(async () => {
|
|
await user.click(confirmButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockDelDatasetApikey).toHaveBeenCalledWith({
|
|
url: '/datasets/api-keys/dk-1',
|
|
params: {},
|
|
})
|
|
})
|
|
})
|
|
|
|
it('should invalidate dataset API keys after deleting', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
|
|
// Find and click delete button
|
|
const actionButtons = document.body.querySelectorAll('button.action-btn')
|
|
const deleteButton = actionButtons[1]
|
|
await act(async () => {
|
|
await user.click(deleteButton!)
|
|
})
|
|
|
|
// Wait for confirm dialog and click confirm
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.actionMsg.deleteConfirmTitle')).toBeInTheDocument()
|
|
})
|
|
|
|
const confirmButton = screen.getByText('common.operation.confirm')
|
|
await act(async () => {
|
|
await user.click(confirmButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(mockInvalidateDatasetApiKeys).toHaveBeenCalled()
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('token truncation', () => {
|
|
it('should truncate token correctly', () => {
|
|
const apiKeys = [
|
|
{ id: 'key-1', token: 'sk-abcdefghijklmnopqrstuvwxyz1234567890', created_at: 1700000000, last_used_at: null },
|
|
]
|
|
mockAppApiKeysData.mockReturnValue({ data: apiKeys })
|
|
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Token format: first 3 chars + ... + last 20 chars
|
|
// 'sk-abcdefghijklmnopqrstuvwxyz1234567890' -> 'sk-...qrstuvwxyz1234567890'
|
|
expect(screen.getByText('sk-...qrstuvwxyz1234567890')).toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('styling', () => {
|
|
it('should render modal with expected structure', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
// Modal should render and contain the title
|
|
expect(screen.getByText('appApi.apiKeyModal.apiSecretKey')).toBeInTheDocument()
|
|
})
|
|
|
|
it('should render create button with flex styling', () => {
|
|
render(<SecretKeyModal {...defaultProps} />)
|
|
// Modal renders via portal, so query from document.body
|
|
const flexContainers = document.body.querySelectorAll('[class*="flex"]')
|
|
expect(flexContainers.length).toBeGreaterThan(0)
|
|
})
|
|
})
|
|
|
|
describe('empty state', () => {
|
|
it('should not render table when no keys', () => {
|
|
mockAppApiKeysData.mockReturnValue({ data: [] })
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
expect(screen.queryByText('appApi.apiKeyModal.secretKey')).not.toBeInTheDocument()
|
|
})
|
|
|
|
it('should not render table when data is null', () => {
|
|
mockAppApiKeysData.mockReturnValue(null)
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
expect(screen.queryByText('appApi.apiKeyModal.secretKey')).not.toBeInTheDocument()
|
|
})
|
|
})
|
|
|
|
describe('SecretKeyGenerateModal', () => {
|
|
it('should close generate modal on close', async () => {
|
|
const user = userEvent.setup()
|
|
render(<SecretKeyModal {...defaultProps} appId="app-123" />)
|
|
|
|
// Create a new key to open generate modal
|
|
const createButton = screen.getByText('appApi.apiKeyModal.createNewSecretKey')
|
|
await act(async () => {
|
|
await user.click(createButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(screen.getByText('appApi.apiKeyModal.generateTips')).toBeInTheDocument()
|
|
})
|
|
|
|
// Find and click the close/OK button in generate modal
|
|
const okButton = screen.getByText('appApi.actionMsg.ok')
|
|
await act(async () => {
|
|
await user.click(okButton)
|
|
})
|
|
|
|
await waitFor(() => {
|
|
expect(screen.queryByText('appApi.apiKeyModal.generateTips')).not.toBeInTheDocument()
|
|
})
|
|
})
|
|
})
|
|
})
|