ruvector/studio/data/table-rows/table-row-delete-mutation.tsx
rUv 814f595995 feat(studio): Add complete RuVector Studio application
Major additions:
- Complete Next.js studio application with 1600+ components
- Docker support (Dockerfile.combined, docker-compose.yml)
- GCP deployment documentation and benchmarks
- SQL benchmark scripts for performance testing
- Sentry integration for monitoring
- Comprehensive test suite and mocks

Studio features:
- Dashboard and admin interfaces
- Data visualization components
- Authentication and user management
- API integration with RuVector backend
- Static data and public assets

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-06 23:04:48 +00:00

140 lines
5.2 KiB
TypeScript

import { useMutation, useQueryClient } from '@tanstack/react-query'
import { toast } from 'sonner'
import { Query } from '@supabase/pg-meta/src/query'
import type { SupaRow } from 'components/grid/types'
import { Markdown } from 'components/interfaces/Markdown'
import { DocsButton } from 'components/ui/DocsButton'
import { executeSql } from 'data/sql/execute-sql-query'
import { Entity } from 'data/table-editor/table-editor-types'
import { DOCS_URL } from 'lib/constants'
import { RoleImpersonationState, wrapWithRoleImpersonation } from 'lib/role-impersonation'
import { isRoleImpersonationEnabled } from 'state/role-impersonation-state'
import type { ResponseError, UseCustomMutationOptions } from 'types'
import { tableRowKeys } from './keys'
import { getPrimaryKeys } from './utils'
export type TableRowDeleteVariables = {
projectRef: string
connectionString?: string | null
table: Entity
rows: SupaRow[]
roleImpersonationState?: RoleImpersonationState
}
export function getTableRowDeleteSql({
table,
rows,
}: Pick<TableRowDeleteVariables, 'table' | 'rows'>) {
const { primaryKeys, error } = getPrimaryKeys({ table })
if (error) throw error
let queryChains = new Query().from(table.name, table.schema ?? undefined).delete()
primaryKeys?.forEach((key) => {
const primaryKeyValues = rows.map((x) => x[key])
queryChains = queryChains.filter(key, 'in', primaryKeyValues)
})
return queryChains.toSql()
}
export async function deleteTableRow({
projectRef,
connectionString,
table,
rows,
roleImpersonationState,
}: TableRowDeleteVariables) {
const sql = wrapWithRoleImpersonation(
getTableRowDeleteSql({ table, rows }),
roleImpersonationState
)
const { result } = await executeSql({
projectRef,
connectionString,
sql,
isRoleImpersonationEnabled: isRoleImpersonationEnabled(roleImpersonationState?.role),
})
return result
}
type TableRowDeleteData = Awaited<ReturnType<typeof deleteTableRow>>
export const useTableRowDeleteMutation = ({
onSuccess,
onError,
...options
}: Omit<
UseCustomMutationOptions<TableRowDeleteData, ResponseError, TableRowDeleteVariables>,
'mutationFn'
> = {}) => {
const queryClient = useQueryClient()
return useMutation<TableRowDeleteData, ResponseError, TableRowDeleteVariables>({
mutationFn: (vars) => deleteTableRow(vars),
async onSuccess(data, variables, context) {
const { projectRef, table } = variables
await queryClient.invalidateQueries({
queryKey: tableRowKeys.tableRowsAndCount(projectRef, table.id),
})
await onSuccess?.(data, variables, context)
},
async onError(data, variables, context) {
if (onError === undefined) {
const { table, rows } = variables
const isPkError = data.message.includes('Please add a primary key column')
const isFkError = data.message.includes('violates foreign key constraint')
const isMultipleRows = rows.length > 1
if (isFkError) {
const sourceTable = table.name
const referencingTable = data.message.split('on table ')[2].replaceAll('"', '')
const fkName = data.message
.split('foreign key constraint')[1]
.split('on table')[0]
.replaceAll('"', '')
const initialMessage = isMultipleRows
? `Unable to delete rows as one of them is currently referenced by a foreign key constraint from the table \`${referencingTable}\`.`
: `Unable to delete row as it is currently referenced by a foreign key constraint from the table \`${referencingTable}\`.`
const resolutionCTA = `Set an on delete behavior on the foreign key relation \`${fkName}\` in the \`${referencingTable}\` table to automatically respond when row(s) are being deleted in the \`${sourceTable}\` table.`
toast(initialMessage, {
description: <Markdown content={resolutionCTA} className="[&>p]:m-0" />,
action: (
<div className="w-full flex gap-x-2 !mx-0 mt-3">
{/* [Joshen] Ideally we also are able to add this CTA but we can't guarantee this info without an on-demand fetch */}
{/* <Button asChild key="cta-1" type="default">
<Link href={`/project/${projectRef}/editor`}>
View "{referencingTable}" table
</Link>
</Button> */}
<DocsButton href={`${DOCS_URL}/guides/database/postgres/cascade-deletes`} />
</div>
),
})
} else if (isPkError) {
toast('Unable to delete row(s) as table has no primary keys', {
description: (
<div>
<p className="text-sm text-foreground-light">
Add a primary key column to your table first to serve as a unique identifier for
each row before updating or deleting the row.
</p>
<div className="mt-3">
<DocsButton href={`${DOCS_URL}/guides/database/tables#primary-keys`} />
</div>
</div>
),
})
} else {
toast.error(`Failed to delete table row: ${data.message}`)
}
} else {
onError(data, variables, context)
}
},
...options,
})
}