Files
supabase/apps/studio/data/database/table-columns-query.ts
Alaister Young 6c592dec99 chore: remove useExecuteSqlQuery() part 2 (#30467)
* foreign-key-constraints

* update entity-types stale time

* schemas query

* deprecate useExecuteSqlQuery

* users count query

* database size query

* indexes query

* keywords query

* migrations query

* table columns

* database functions

* database roles query

* fdws query

* replication lag query

* ongoing queries query

* vault secrets query

* remove unneeded staleTime: 0

* max connections query

* fix entity types key in tests

* Some fixes

---------

Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
2024-11-18 05:15:37 +00:00

124 lines
3.8 KiB
TypeScript

import { useQuery, UseQueryOptions } from '@tanstack/react-query'
import { executeSql, ExecuteSqlError } from '../sql/execute-sql-query'
import { databaseKeys } from './keys'
export type TableColumn = {
schemaname: string
tablename: string
quoted_name: string
is_table: boolean
columns: any[]
}
export const getTableColumnsSql = ({ table, schema }: { table?: string; schema?: string }) => {
const conditions = []
if (table) {
conditions.push(`tablename = '${table}'`)
}
if (schema) {
conditions.push(`schemaname = '${schema}'`)
}
const whereClause = conditions.length > 0 ? `WHERE ${conditions.join(' AND ')}` : ''
const sql = /* SQL */ `
SELECT
tbl.schemaname,
tbl.tablename,
tbl.quoted_name,
tbl.is_table,
json_agg(a) as columns
FROM
(
SELECT
n.nspname as schemaname,
c.relname as tablename,
(quote_ident(n.nspname) || '.' || quote_ident(c.relname)) as quoted_name,
true as is_table
FROM
pg_catalog.pg_class c
JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
WHERE
c.relkind = 'r'
AND n.nspname not in ('information_schema', 'pg_catalog', 'pg_toast')
AND n.nspname not like 'pg_temp_%'
AND n.nspname not like 'pg_toast_temp_%'
AND has_schema_privilege(n.oid, 'USAGE') = true
AND has_table_privilege(quote_ident(n.nspname) || '.' || quote_ident(c.relname), 'SELECT, INSERT, UPDATE, DELETE, TRUNCATE, REFERENCES, TRIGGER') = true
union all
SELECT
n.nspname as schemaname,
c.relname as tablename,
(quote_ident(n.nspname) || '.' || quote_ident(c.relname)) as quoted_name,
false as is_table
FROM
pg_catalog.pg_class c
JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace
WHERE
c.relkind in ('v', 'm')
AND n.nspname not in ('information_schema', 'pg_catalog', 'pg_toast')
AND n.nspname not like 'pg_temp_%'
AND n.nspname not like 'pg_toast_temp_%'
AND has_schema_privilege(n.oid, 'USAGE') = true
AND has_table_privilege(quote_ident(n.nspname) || '.' || quote_ident(c.relname), 'SELECT, INSERT, UPDATE, DELETE, TRUNCATE, REFERENCES, TRIGGER') = true
) as tbl
LEFT JOIN (
SELECT
attrelid,
attname,
format_type(atttypid, atttypmod) as data_type,
attnum,
attisdropped
FROM
pg_attribute
) as a ON (
a.attrelid = tbl.quoted_name::regclass
AND a.attnum > 0
AND NOT a.attisdropped
AND has_column_privilege(tbl.quoted_name, a.attname, 'SELECT, INSERT, UPDATE, REFERENCES')
)
${whereClause}
GROUP BY schemaname, tablename, quoted_name, is_table;
`.trim()
return sql
}
export type TableColumnsVariables = {
projectRef?: string
connectionString?: string
table?: string
schema?: string
}
export async function getTableColumns(
{ projectRef, connectionString, table, schema }: TableColumnsVariables,
signal?: AbortSignal
) {
const sql = getTableColumnsSql({ table, schema })
const { result } = await executeSql(
{ projectRef, connectionString, sql, queryKey: ['table-columns', schema, table] },
signal
)
return result as TableColumn[]
}
export type TableColumnsData = Awaited<ReturnType<typeof getTableColumns>>
export type TableColumnsError = ExecuteSqlError
export const useTableColumnsQuery = <TData = TableColumnsData>(
{ projectRef, connectionString, schema, table }: TableColumnsVariables,
{ enabled = true, ...options }: UseQueryOptions<TableColumnsData, TableColumnsError, TData> = {}
) =>
useQuery<TableColumnsData, TableColumnsError, TData>(
databaseKeys.tableColumns(projectRef, schema, table),
({ signal }) => getTableColumns({ projectRef, connectionString, schema, table }, signal),
{
enabled: enabled && typeof projectRef !== 'undefined',
...options,
}
)