mirror of
https://github.com/supabase/supabase.git
synced 2026-06-30 20:38:14 -04:00
1baaded0bb
## Context Just some clean up as I was going through stuff - `useExecuteSqlQuery` is deprecated and not used at all - As such `execute-sql-query` is technically irrelevant, the more relevant file is `execute-sql-mutation` - Hence opting to consolidate `execute-sql-query` into `execute-sql-mutation` - Also removing `ExecuteSqlError` since its just re-exporting the `ResponseError` type There's a lot of file changes but its essentially just updating the importing statements across the files
160 lines
5.5 KiB
TypeScript
160 lines
5.5 KiB
TypeScript
import { joinSqlFragments, safeSql, type SafeSqlFragment } from '@supabase/pg-meta'
|
|
import { wrapWithTransaction } from '@supabase/pg-meta/src/query'
|
|
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
|
import { toast } from 'sonner'
|
|
|
|
import { tableRowKeys } from './keys'
|
|
import { getTableRowCreateSql } from './table-row-create-mutation'
|
|
import { getTableRowDeleteSql } from './table-row-delete-mutation'
|
|
import { getTableRowUpdateSql } from './table-row-update-mutation'
|
|
import type { PendingAddRow } from '@/components/grid/types'
|
|
import { executeSql } from '@/data/sql/execute-sql-mutation'
|
|
import { RoleImpersonationState, wrapWithRoleImpersonation } from '@/lib/role-impersonation'
|
|
import { isRoleImpersonationEnabled } from '@/state/role-impersonation-state'
|
|
import { QueuedOperation, QueuedOperationType } from '@/state/table-editor-operation-queue.types'
|
|
import type { ResponseError, UseCustomMutationOptions } from '@/types'
|
|
|
|
export type OperationQueueSaveVariables = {
|
|
projectRef: string
|
|
connectionString?: string | null
|
|
operations: readonly QueuedOperation[]
|
|
roleImpersonationState?: RoleImpersonationState
|
|
}
|
|
|
|
/**
|
|
* Generates SQL for a single queued operation.
|
|
* Extend this function as new operation types are added.
|
|
*/
|
|
function getOperationSql(operation: QueuedOperation): SafeSqlFragment {
|
|
switch (operation.type) {
|
|
case QueuedOperationType.EDIT_CELL_CONTENT: {
|
|
const { payload } = operation
|
|
return getTableRowUpdateSql({
|
|
table: {
|
|
id: payload.table.id,
|
|
name: payload.table.name,
|
|
schema: payload.table.schema,
|
|
},
|
|
configuration: { identifiers: payload.rowIdentifiers },
|
|
payload: { [payload.columnName]: payload.newValue },
|
|
enumArrayColumns: payload.enumArrayColumns ?? [],
|
|
returning: false,
|
|
})
|
|
}
|
|
case QueuedOperationType.ADD_ROW: {
|
|
const { payload } = operation
|
|
// Clean internal fields before SQL generation
|
|
const { __tempId, idx, ...cleanRowData } = payload.rowData as PendingAddRow
|
|
return getTableRowCreateSql({
|
|
table: { id: payload.table.id, name: payload.table.name, schema: payload.table.schema },
|
|
payload: cleanRowData,
|
|
enumArrayColumns: payload.enumArrayColumns ?? [],
|
|
returning: false,
|
|
})
|
|
}
|
|
case QueuedOperationType.DELETE_ROW: {
|
|
const { payload } = operation
|
|
// Create a mock row with the row identifiers for the delete SQL
|
|
const mockRow = { idx: 0, ...payload.rowIdentifiers }
|
|
return getTableRowDeleteSql({
|
|
table: payload.table,
|
|
rows: [mockRow],
|
|
})
|
|
}
|
|
default: {
|
|
// Error should never happen, but we'll handle it anyway. cast to never for exhaustive check.
|
|
const _exhaustiveCheck: never = operation
|
|
throw new Error(`Unknown operation: ${(_exhaustiveCheck as { type: string }).type}`)
|
|
}
|
|
}
|
|
}
|
|
|
|
function sortOperations(operations: readonly QueuedOperation[]): QueuedOperation[] {
|
|
const operationOrder: Record<QueuedOperationType, number> = {
|
|
[QueuedOperationType.DELETE_ROW]: 0,
|
|
[QueuedOperationType.ADD_ROW]: 1,
|
|
[QueuedOperationType.EDIT_CELL_CONTENT]: 2,
|
|
}
|
|
|
|
return [...operations].sort((a, b) => {
|
|
return operationOrder[a.type] - operationOrder[b.type]
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Saves all queued operations in a single database transaction.
|
|
* If any operation fails, the entire transaction is rolled back.
|
|
*/
|
|
export async function saveOperationQueue({
|
|
projectRef,
|
|
connectionString,
|
|
operations,
|
|
roleImpersonationState,
|
|
}: OperationQueueSaveVariables) {
|
|
if (operations.length === 0) {
|
|
return { result: [] }
|
|
}
|
|
|
|
const sortedOperations = sortOperations(operations)
|
|
const statements: Array<SafeSqlFragment> = sortedOperations.map((op) => {
|
|
const sql = getOperationSql(op)
|
|
return (sql.endsWith(';') ? sql.slice(0, -1) : sql) as SafeSqlFragment
|
|
})
|
|
|
|
const transactionSql = wrapWithTransaction(safeSql`${joinSqlFragments(statements, ';\n')};`)
|
|
|
|
const sql = wrapWithRoleImpersonation(transactionSql, roleImpersonationState)
|
|
|
|
const { result } = await executeSql({
|
|
projectRef,
|
|
connectionString,
|
|
sql,
|
|
isRoleImpersonationEnabled: isRoleImpersonationEnabled(roleImpersonationState?.role),
|
|
queryKey: ['operation-queue-save'],
|
|
})
|
|
|
|
return { result }
|
|
}
|
|
|
|
type OperationQueueSaveData = Awaited<ReturnType<typeof saveOperationQueue>>
|
|
|
|
export const useOperationQueueSaveMutation = ({
|
|
onSuccess,
|
|
onError,
|
|
...options
|
|
}: Omit<
|
|
UseCustomMutationOptions<OperationQueueSaveData, ResponseError, OperationQueueSaveVariables>,
|
|
'mutationFn'
|
|
> = {}) => {
|
|
const queryClient = useQueryClient()
|
|
|
|
return useMutation<OperationQueueSaveData, ResponseError, OperationQueueSaveVariables>({
|
|
mutationFn: (vars) => saveOperationQueue(vars),
|
|
async onSuccess(data, variables, context) {
|
|
const { projectRef, operations } = variables
|
|
|
|
// Collect all unique table IDs that were affected
|
|
const affectedTableIds = [...new Set(operations.map((op) => op.tableId))]
|
|
|
|
// Invalidate queries for all affected tables (both rows and count)
|
|
await Promise.all(
|
|
affectedTableIds.map((tableId) =>
|
|
queryClient.invalidateQueries({
|
|
queryKey: tableRowKeys.tableRowsAndCount(projectRef, tableId),
|
|
})
|
|
)
|
|
)
|
|
|
|
await onSuccess?.(data, variables, context)
|
|
},
|
|
async onError(data, variables, context) {
|
|
if (onError === undefined) {
|
|
toast.error(`Failed to save changes: ${data.message}`)
|
|
} else {
|
|
onError(data, variables, context)
|
|
}
|
|
},
|
|
...options,
|
|
})
|
|
}
|