Files
supabase/apps/studio/components/interfaces/Integrations/Vault/Secrets/SecretsManagement.tsx
Gildas Garcia 5d97339d41 chore: remove <Select> _Shadcn_ suffix (#45988)
## Problem

The `_Shadcn_` suffix isn't needed anymore on `Select` components

## Solution

Remove it. No other changes

<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->

## Summary by CodeRabbit

* **Refactor**
* Updated internal component architecture to standardize and simplify
the codebase. These changes improve code maintainability and consistency
across the application without affecting existing functionality or user
experience.

<!-- review_stack_entry_start -->

[![Review Change
Stack](https://storage.googleapis.com/coderabbit_public_assets/review-stack-in-coderabbit-ui.svg)](https://app.coderabbit.ai/change-stack/supabase/supabase/pull/45988)

<!-- review_stack_entry_end -->

<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-05-15 16:39:57 +02:00

207 lines
6.9 KiB
TypeScript

import { PermissionAction } from '@supabase/shared-types/out/constants'
import { useParams } from 'common'
import { sortBy } from 'lodash'
import { RefreshCw, Search, X } from 'lucide-react'
import { parseAsBoolean, useQueryState } from 'nuqs'
import { useEffect, useMemo, useState } from 'react'
import DataGrid, { Row } from 'react-data-grid'
import {
Button,
cn,
LoadingLine,
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from 'ui'
import { Input } from 'ui-patterns/DataInputs/Input'
import { AddNewSecretModal } from './AddNewSecretModal'
import { DeleteSecretModal } from './DeleteSecretModal'
import { EditSecretModal } from './EditSecretModal'
import { formatSecretColumns } from './Secrets.utils'
import AlertError from '@/components/ui/AlertError'
import { ButtonTooltip } from '@/components/ui/ButtonTooltip'
import { DocsButton } from '@/components/ui/DocsButton'
import { useVaultSecretsQuery } from '@/data/vault/vault-secrets-query'
import { useAsyncCheckPermissions } from '@/hooks/misc/useCheckPermissions'
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
import { DOCS_URL } from '@/lib/constants'
import type { VaultSecret } from '@/types'
export const SecretsManagement = () => {
const { search } = useParams()
const { data: project } = useSelectedProjectQuery()
const [searchValue, setSearchValue] = useState<string>('')
const [, setShowAddSecretModal] = useQueryState('new', parseAsBoolean.withDefault(false))
const [selectedSort, setSelectedSort] = useState<'updated_at' | 'name'>('updated_at')
const { can: canManageSecrets } = useAsyncCheckPermissions(
PermissionAction.TENANT_SQL_ADMIN_WRITE,
'tables'
)
const {
data,
error,
isError,
isPending: isLoading,
isRefetching,
refetch,
} = useVaultSecretsQuery({
projectRef: project?.ref,
connectionString: project?.connectionString,
})
const allSecrets = useMemo(() => data || [], [data])
const secrets = useMemo(() => {
const filtered =
searchValue.length > 0
? allSecrets.filter(
(secret) =>
(secret?.name ?? '').toLowerCase().includes(searchValue.trim().toLowerCase()) ||
(secret?.id ?? '').toLowerCase().includes(searchValue.trim().toLowerCase())
)
: allSecrets
if (selectedSort === 'updated_at') {
return sortBy(filtered, (s) => Number(new Date(s.updated_at))).reverse()
}
return sortBy(filtered, (s) => (s.name || '').toLowerCase())
}, [allSecrets, searchValue, selectedSort])
const columns = useMemo(() => formatSecretColumns(), [])
useEffect(() => {
if (search !== undefined) setSearchValue(search)
}, [search])
return (
<>
<div className="h-full w-full space-y-4">
<div className="h-full w-full flex flex-col relative">
<div className="bg-surface-200 py-3 px-10 flex items-center justify-between flex-wrap">
<div className="flex items-center gap-2">
<Input
size="tiny"
className="w-52"
placeholder="Search by name or key ID"
icon={<Search />}
value={searchValue ?? ''}
onChange={(e) => setSearchValue(e.target.value)}
actions={[
searchValue && (
<Button
key="clear"
size="tiny"
type="text"
icon={<X />}
onClick={() => setSearchValue('')}
className="p-0 h-5 w-5"
/>
),
]}
/>
<Select value={selectedSort} onValueChange={(v) => setSelectedSort(v as any)}>
<SelectTrigger size="tiny" className="w-44">
<SelectValue asChild>
<>Sort by {selectedSort}</>
</SelectValue>
</SelectTrigger>
<SelectContent>
<SelectItem value="updated_at" className="text-xs">
Updated at
</SelectItem>
<SelectItem value="name" className="text-xs">
Name
</SelectItem>
</SelectContent>
</Select>
</div>
<div className="flex items-center gap-x-2">
<Button
type="default"
icon={<RefreshCw />}
loading={isRefetching}
onClick={() => refetch()}
>
Refresh
</Button>
<DocsButton href={`${DOCS_URL}/guides/database/vault`} />
<ButtonTooltip
type="primary"
disabled={!canManageSecrets}
onClick={() => setShowAddSecretModal(true)}
tooltip={{
content: {
side: 'bottom',
text: !canManageSecrets
? 'You need additional permissions to add secrets'
: undefined,
},
}}
>
Add new secret
</ButtonTooltip>
</div>
</div>
<LoadingLine loading={isLoading || isRefetching} />
{isError ? (
<div className="grow p-4">
<AlertError error={error} subject="Failed to load secrets" />
</div>
) : (
<DataGrid
className="grow border-t-0"
rowHeight={52}
headerRowHeight={36}
columns={columns}
rows={secrets}
rowKeyGetter={(row: VaultSecret) => row.id}
rowClass={() => {
return cn(
'cursor-pointer',
'[&>.rdg-cell]:border-box [&>.rdg-cell]:outline-hidden [&>.rdg-cell]:shadow-none',
'[&>.rdg-cell:first-child>div]:pl-8'
)
}}
renderers={{
renderRow(_, props) {
return <Row key={(props.row as VaultSecret).id} {...props} />
},
}}
/>
)}
{secrets.length === 0 && !isLoading && !isError ? (
<div className="absolute top-32 px-6 w-full">
<div className="text-center text-sm flex flex-col gap-y-1">
<p className="text-foreground">
{searchValue ? 'No secrets found' : 'No secrets added yet'}
</p>
<p className="text-foreground-light">
{searchValue
? `There are currently no secrets based on the search "${searchValue}"`
: 'The Vault allows you to store sensitive information like API keys'}
</p>
</div>
</div>
) : null}
</div>
</div>
<AddNewSecretModal />
<EditSecretModal />
<DeleteSecretModal />
</>
)
}