mirror of
https://github.com/supabase/supabase.git
synced 2026-06-18 21:54:18 +08:00
## Problem Our `<Button>` component breaks the default `button` contract by redefining the `type` prop to set its variant (`primary`, `default`, etc) instead of the button type (`submit`, `button`, etc). This is confusing and forces to write more code when using it with shadcn components that expect/inject the standard button props. ## Solution - rename the `type` prop to `variant` - rename the `htmlType` prop to `type` - propagate the changes where necessary - format code ## How to test As this is just prop renaming, if it builds it's ok --------- Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
354 lines
13 KiB
TypeScript
354 lines
13 KiB
TypeScript
import { useParams } from 'common'
|
|
import { sortBy } from 'lodash'
|
|
import { AlertCircle, Search, Trash } from 'lucide-react'
|
|
import { parseAsBoolean, parseAsString, useQueryState } from 'nuqs'
|
|
import { useEffect, useRef, useState } from 'react'
|
|
import { toast } from 'sonner'
|
|
import {
|
|
Button,
|
|
Card,
|
|
SidePanel,
|
|
Table,
|
|
TableBody,
|
|
TableCell,
|
|
TableHead,
|
|
TableHeader,
|
|
TableRow,
|
|
} from 'ui'
|
|
import { Input } from 'ui-patterns/DataInputs/Input'
|
|
import { ConfirmationModal } from 'ui-patterns/Dialogs/ConfirmationModal'
|
|
import { GenericSkeletonLoader, ShimmeringLoader } from 'ui-patterns/ShimmeringLoader'
|
|
|
|
import { ProtectedSchemaWarning } from '../ProtectedSchemaWarning'
|
|
import { CreateIndexSidePanel } from './CreateIndexSidePanel'
|
|
import AlertError from '@/components/ui/AlertError'
|
|
import CodeEditor from '@/components/ui/CodeEditor/CodeEditor'
|
|
import SchemaSelector from '@/components/ui/SchemaSelector'
|
|
import { Shortcut } from '@/components/ui/Shortcut'
|
|
import { useDatabaseIndexDeleteMutation } from '@/data/database-indexes/index-delete-mutation'
|
|
import { useIndexesQuery, type DatabaseIndex } from '@/data/database-indexes/indexes-query'
|
|
import { useSchemasQuery } from '@/data/database/schemas-query'
|
|
import { useQuerySchemaState } from '@/hooks/misc/useSchemaQueryState'
|
|
import { useSelectedProjectQuery } from '@/hooks/misc/useSelectedProject'
|
|
import { useIsProtectedSchema } from '@/hooks/useProtectedSchemas'
|
|
import { onSearchInputEscape } from '@/lib/keyboard'
|
|
import { SHORTCUT_IDS } from '@/state/shortcuts/registry'
|
|
import { useShortcut } from '@/state/shortcuts/useShortcut'
|
|
|
|
export const Indexes = () => {
|
|
const { data: project } = useSelectedProjectQuery()
|
|
const { schema: urlSchema, table } = useParams()
|
|
|
|
const [search, setSearch] = useQueryState('search', parseAsString.withDefault(''))
|
|
const [schemaSelectorOpen, setSchemaSelectorOpen] = useState(false)
|
|
const searchInputRef = useRef<HTMLInputElement>(null)
|
|
const { selectedSchema, setSelectedSchema } = useQuerySchemaState()
|
|
|
|
const {
|
|
data: allIndexes,
|
|
error: indexesError,
|
|
isPending: isLoadingIndexes,
|
|
isSuccess: isSuccessIndexes,
|
|
isError: isErrorIndexes,
|
|
} = useIndexesQuery({
|
|
schema: selectedSchema,
|
|
projectRef: project?.ref,
|
|
connectionString: project?.connectionString,
|
|
})
|
|
|
|
const [showCreateIndex, setShowCreateIndex] = useQueryState(
|
|
'new',
|
|
parseAsBoolean.withDefault(false)
|
|
)
|
|
|
|
const [editIndexId, setEditIndexId] = useQueryState('edit', parseAsString)
|
|
const selectedIndex = allIndexes?.find((idx) => idx.name === editIndexId)
|
|
|
|
const [deleteIndexId, setDeleteIndexId] = useQueryState('delete', parseAsString)
|
|
const selectedIndexToDelete = allIndexes?.find((idx) => idx.name === deleteIndexId)
|
|
|
|
const {
|
|
data: schemas,
|
|
isPending: isLoadingSchemas,
|
|
isSuccess: isSuccessSchemas,
|
|
isError: isErrorSchemas,
|
|
} = useSchemasQuery({
|
|
projectRef: project?.ref,
|
|
connectionString: project?.connectionString,
|
|
})
|
|
|
|
const {
|
|
mutate: deleteIndex,
|
|
isPending: isExecuting,
|
|
isSuccess: isSuccessDelete,
|
|
} = useDatabaseIndexDeleteMutation({
|
|
onSuccess: async () => {
|
|
setDeleteIndexId(null)
|
|
toast.success('Successfully deleted index')
|
|
},
|
|
})
|
|
|
|
const { isSchemaLocked } = useIsProtectedSchema({ schema: selectedSchema })
|
|
|
|
useShortcut(
|
|
SHORTCUT_IDS.LIST_PAGE_FOCUS_SEARCH,
|
|
() => {
|
|
searchInputRef.current?.focus()
|
|
searchInputRef.current?.select()
|
|
},
|
|
{ label: 'Search indexes' }
|
|
)
|
|
|
|
useShortcut(SHORTCUT_IDS.LIST_PAGE_RESET_FILTERS, () => {
|
|
setSearch('')
|
|
})
|
|
|
|
const sortedIndexes = sortBy(allIndexes ?? [], (index) => index.name.toLocaleLowerCase())
|
|
const indexes =
|
|
search.length > 0
|
|
? sortedIndexes.filter((index) => index.name.includes(search) || index.table.includes(search))
|
|
: sortedIndexes
|
|
|
|
const onConfirmDeleteIndex = (index: DatabaseIndex) => {
|
|
if (!project) return console.error('Project is required')
|
|
deleteIndex({
|
|
projectRef: project.ref,
|
|
connectionString: project.connectionString,
|
|
name: index.name,
|
|
schema: selectedSchema,
|
|
})
|
|
}
|
|
|
|
useEffect(() => {
|
|
if (urlSchema !== undefined) {
|
|
const schema = schemas?.find((s) => s.name === urlSchema)
|
|
if (schema !== undefined) setSelectedSchema(schema.name)
|
|
}
|
|
}, [urlSchema, isSuccessSchemas])
|
|
|
|
useEffect(() => {
|
|
if (table !== undefined) setSearch(table)
|
|
}, [table])
|
|
|
|
useEffect(() => {
|
|
if (isSuccessIndexes && !!editIndexId && !selectedIndex) {
|
|
toast('Index not found')
|
|
setEditIndexId(null)
|
|
}
|
|
}, [isSuccessIndexes, editIndexId, selectedIndex, setEditIndexId])
|
|
|
|
useEffect(() => {
|
|
if (isSuccessIndexes && !!deleteIndexId && !selectedIndexToDelete && !isSuccessDelete) {
|
|
toast('Index not found')
|
|
setDeleteIndexId(null)
|
|
}
|
|
}, [isSuccessIndexes, deleteIndexId, selectedIndexToDelete, isSuccessDelete, setDeleteIndexId])
|
|
|
|
return (
|
|
<>
|
|
<div className="pb-8">
|
|
<div className="flex flex-col gap-y-4">
|
|
<div className="flex items-center gap-2 flex-wrap">
|
|
{isLoadingSchemas && <ShimmeringLoader className="w-[260px]" />}
|
|
{isErrorSchemas && (
|
|
<div className="w-[260px] text-foreground-light text-sm border px-3 py-1.5 rounded-sm flex items-center space-x-2">
|
|
<AlertCircle strokeWidth={2} size={16} />
|
|
<p>Failed to load schemas</p>
|
|
</div>
|
|
)}
|
|
{isSuccessSchemas && (
|
|
<Shortcut
|
|
id={SHORTCUT_IDS.LIST_PAGE_FOCUS_SCHEMA}
|
|
onTrigger={() => setSchemaSelectorOpen(true)}
|
|
side="bottom"
|
|
tooltipOpen={schemaSelectorOpen ? false : undefined}
|
|
>
|
|
<SchemaSelector
|
|
className="w-full lg:w-[180px]"
|
|
size="tiny"
|
|
showError={false}
|
|
selectedSchemaName={selectedSchema}
|
|
onSelectSchema={setSelectedSchema}
|
|
open={schemaSelectorOpen}
|
|
onOpenChange={setSchemaSelectorOpen}
|
|
/>
|
|
</Shortcut>
|
|
)}
|
|
<Input
|
|
ref={searchInputRef}
|
|
size="tiny"
|
|
value={search}
|
|
className="w-full lg:w-52"
|
|
onChange={(e) => setSearch(e.target.value)}
|
|
onKeyDown={onSearchInputEscape(search, setSearch)}
|
|
placeholder="Search for an index"
|
|
icon={<Search />}
|
|
/>
|
|
|
|
{!isSchemaLocked && (
|
|
<Shortcut
|
|
id={SHORTCUT_IDS.LIST_PAGE_NEW_ITEM}
|
|
label="Create new index"
|
|
onTrigger={() => setShowCreateIndex(true)}
|
|
options={{ enabled: isSuccessSchemas }}
|
|
side="bottom"
|
|
>
|
|
<Button
|
|
className="ml-auto grow lg:grow-0"
|
|
variant="primary"
|
|
onClick={() => setShowCreateIndex(true)}
|
|
disabled={!isSuccessSchemas}
|
|
>
|
|
Create index
|
|
</Button>
|
|
</Shortcut>
|
|
)}
|
|
</div>
|
|
|
|
{isSchemaLocked && <ProtectedSchemaWarning schema={selectedSchema} entity="indexes" />}
|
|
|
|
{isLoadingIndexes && <GenericSkeletonLoader />}
|
|
|
|
{isErrorIndexes && (
|
|
<AlertError error={indexesError as any} subject="Failed to retrieve database indexes" />
|
|
)}
|
|
|
|
{isSuccessIndexes && (
|
|
<div className="w-full overflow-hidden">
|
|
<Card>
|
|
<Table>
|
|
<TableHeader>
|
|
<TableRow>
|
|
<TableHead key="table">Table</TableHead>
|
|
<TableHead key="columns">Columns</TableHead>
|
|
<TableHead key="name">Name</TableHead>
|
|
<TableHead key="buttons" />
|
|
</TableRow>
|
|
</TableHeader>
|
|
<TableBody>
|
|
{indexes.length === 0 && search.length === 0 && (
|
|
<TableRow>
|
|
<TableCell colSpan={4}>
|
|
<p className="text-sm text-foreground">No indexes created yet</p>
|
|
<p className="text-sm text-foreground-light">
|
|
There are no indexes found in the schema "{selectedSchema}"
|
|
</p>
|
|
</TableCell>
|
|
</TableRow>
|
|
)}
|
|
{indexes.length === 0 && search.length > 0 && (
|
|
<TableRow>
|
|
<TableCell colSpan={4}>
|
|
<p className="text-sm text-foreground">No results found</p>
|
|
<p className="text-sm text-foreground-light">
|
|
Your search for "{search}" did not return any results
|
|
</p>
|
|
</TableCell>
|
|
</TableRow>
|
|
)}
|
|
{indexes.length > 0 &&
|
|
indexes.map((index) => (
|
|
<TableRow key={index.name}>
|
|
<TableCell>
|
|
<p title={index.table}>{index.table}</p>
|
|
</TableCell>
|
|
<TableCell>
|
|
<p title={index.columns}>{index.columns}</p>
|
|
</TableCell>
|
|
<TableCell>
|
|
<p title={index.name}>{index.name}</p>
|
|
</TableCell>
|
|
<TableCell>
|
|
<div className="flex justify-end items-center space-x-2">
|
|
<Button variant="default" onClick={() => setEditIndexId(index.name)}>
|
|
View definition
|
|
</Button>
|
|
{!isSchemaLocked && (
|
|
<Button
|
|
aria-label="Delete index"
|
|
variant="text"
|
|
className="px-1"
|
|
icon={<Trash />}
|
|
onClick={() => setDeleteIndexId(index.name)}
|
|
/>
|
|
)}
|
|
</div>
|
|
</TableCell>
|
|
</TableRow>
|
|
))}
|
|
</TableBody>
|
|
</Table>
|
|
</Card>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
<SidePanel
|
|
size="xlarge"
|
|
visible={!!selectedIndex}
|
|
header={
|
|
<>
|
|
<span>Index:</span>
|
|
<code className="text-sm ml-2">{selectedIndex?.name}</code>
|
|
</>
|
|
}
|
|
onCancel={() => setEditIndexId(null)}
|
|
>
|
|
<div className="h-full">
|
|
<div className="relative h-full">
|
|
<CodeEditor
|
|
isReadOnly
|
|
id={selectedIndex?.name ?? ''}
|
|
language="pgsql"
|
|
defaultValue={selectedIndex?.definition ?? ''}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</SidePanel>
|
|
|
|
<CreateIndexSidePanel visible={showCreateIndex} onClose={() => setShowCreateIndex(false)} />
|
|
|
|
<ConfirmationModal
|
|
variant="warning"
|
|
size="medium"
|
|
loading={isExecuting}
|
|
visible={!!selectedIndexToDelete}
|
|
title={
|
|
<>
|
|
Confirm to delete index{' '}
|
|
<code className="text-code-inline">{selectedIndexToDelete?.name}</code>
|
|
</>
|
|
}
|
|
confirmLabel="Confirm delete"
|
|
confirmLabelLoading="Deleting..."
|
|
onConfirm={() =>
|
|
selectedIndexToDelete !== undefined ? onConfirmDeleteIndex(selectedIndexToDelete) : {}
|
|
}
|
|
onCancel={() => setDeleteIndexId(null)}
|
|
alert={{
|
|
title: 'This action cannot be undone',
|
|
description:
|
|
'Deleting an index that is still in use will cause queries to slow down, and in some cases causing significant performance issues.',
|
|
}}
|
|
className="pt-0"
|
|
>
|
|
<ul className="mt-4 space-y-5">
|
|
<li className="flex gap-3">
|
|
<div>
|
|
<strong className="text-sm">Before deleting this index, consider:</strong>
|
|
<ul className="space-y-2 mt-2 text-sm text-foreground-light">
|
|
<li className="list-disc ml-6">This index is no longer in use</li>
|
|
<li className="list-disc ml-6">
|
|
The table which the index is on is not currently in use, as dropping an index
|
|
requires a short exclusive access lock on the table.
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</li>
|
|
</ul>
|
|
</ConfirmationModal>
|
|
</>
|
|
)
|
|
}
|