Files
supabase/apps/studio/data/table-rows/table-rows-query.ts
Jonathan Summers-Muir a4b4167a5b fix: temporary remove default sorting (#28072)
* Update table-rows-query.ts

* Add comment

---------

Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
2024-07-19 11:20:58 +08:00

244 lines
7.4 KiB
TypeScript

import type { QueryKey, UseQueryOptions } from '@tanstack/react-query'
import { IS_PLATFORM } from 'common'
import { Query } from 'components/grid/query/Query'
import { Filter, Sort, SupaRow, SupaTable } from 'components/grid/types'
import {
JSON_TYPES,
TEXT_TYPES,
} from 'components/interfaces/TableGridEditor/SidePanelEditor/SidePanelEditor.constants'
import { KB } from 'lib/constants'
import {
ImpersonationRole,
ROLE_IMPERSONATION_NO_RESULTS,
wrapWithRoleImpersonation,
} from 'lib/role-impersonation'
import { useIsRoleImpersonationEnabled } from 'state/role-impersonation-state'
import {
ExecuteSqlData,
ExecuteSqlError,
executeSql,
useExecuteSqlQuery,
} from '../sql/execute-sql-query'
import { getPagination } from '../utils/pagination'
import { formatFilterValue } from './utils'
type GetTableRowsArgs = {
table?: SupaTable
filters?: Filter[]
sorts?: Sort[]
limit?: number
page?: number
impersonatedRole?: ImpersonationRole
}
// [Joshen] We can probably make this reasonably high, but for now max aim to load 10kb
export const MAX_CHARACTERS = 10 * KB
// return the primary key column if exists, otherwise return the first column
// to use as a default sort
const getDefaultOrderByColumn = (table: SupaTable) => {
return table.columns.find((column) => column.isPrimaryKey)?.name || table.columns[0]?.name
}
// Updated fetchAllTableRows function
export const fetchAllTableRows = async ({
projectRef,
connectionString,
table,
filters = [],
sorts = [],
impersonatedRole,
}: {
projectRef: string
connectionString?: string
table: SupaTable
filters?: Filter[]
sorts?: Sort[]
impersonatedRole?: ImpersonationRole
}) => {
if (IS_PLATFORM && !connectionString) {
console.error('Connection string is required')
return []
}
const rows: any[] = []
const query = new Query()
let queryChains = query.from(table.name, table.schema ?? undefined).select()
filters
.filter((filter) => filter.value && filter.value !== '')
.forEach((filter) => {
const value = formatFilterValue(table, filter)
queryChains = queryChains.filter(filter.column, filter.operator, value)
})
/**
* [Joshen] JFYI 190724 We've commented the default sort logic out below and in getTableRowsSqlQuery
* as the default sort causes resource issues in particular for very large tables (e.g 200M rows)
* The long term fix here should be to only apply default sort if table has < 50k rows
* - this is the same threshold that we do for retrieving the exact count on the table (ref table-rows-count-query)
* - we can check the estimate row count when we get the table details via pg-meta under the live_rows_estimate property
* - i.e the larger the table, the less hand holding the table editor does (which i think is fair from a scalability pov)
*/
// If sorts is empty, use the primary key as the default sort
// if (sorts.length === 0) {
// const primaryKey = getDefaultOrderByColumn(table)
// if (primaryKey) {
// queryChains = queryChains.order(table.name, primaryKey, true, true)
// }
// } else {
sorts.forEach((sort) => {
queryChains = queryChains.order(sort.table, sort.column, sort.ascending, sort.nullsFirst)
})
// }
// Starting from page 0, fetch 500 records per call
let page = -1
let from = 0
let to = 0
let pageData = []
const rowsPerPage = 500
await (async () => {
do {
page += 1
from = page * rowsPerPage
to = (page + 1) * rowsPerPage - 1
const query = wrapWithRoleImpersonation(queryChains.range(from, to).toSql(), {
projectRef,
role: impersonatedRole,
})
try {
const { result } = await executeSql({ projectRef, connectionString, sql: query })
rows.push(...result)
pageData = result
} catch (error) {
return { data: { rows: [] } }
}
} while (pageData.length === rowsPerPage)
})()
return rows.filter((row) => row[ROLE_IMPERSONATION_NO_RESULTS] !== 1)
}
export const getTableRowsSqlQuery = ({
table,
filters = [],
sorts = [],
page,
limit,
}: GetTableRowsArgs) => {
const query = new Query()
if (!table) return ``
const arrayBasedColumns = table.columns
.filter(
(column) => (column?.enum ?? []).length > 0 && column.dataType.toLowerCase() === 'array'
)
.map((column) => `"${column.name}"::text[]`)
let queryChains = query
.from(table.name, table.schema ?? undefined)
.select(arrayBasedColumns.length > 0 ? `*,${arrayBasedColumns.join(',')}` : '*')
filters
.filter((x) => x.value && x.value != '')
.forEach((x) => {
const value = formatFilterValue(table, x)
queryChains = queryChains.filter(x.column, x.operator, value)
})
// If sorts is empty, use the primary key as the default sort
// if (sorts.length === 0) {
// const defaultOrderByColumn = getDefaultOrderByColumn(table)
// if (defaultOrderByColumn) {
// queryChains = queryChains.order(table.name, defaultOrderByColumn, true, true)
// }
// } else {
sorts.forEach((x) => {
queryChains = queryChains.order(x.table, x.column, x.ascending, x.nullsFirst)
})
// }
// getPagination is expecting to start from 0
const { from, to } = getPagination((page ?? 1) - 1, limit)
const baseSql = queryChains.range(from, to).toSql()
// [Joshen] Only truncate text/json based columns as their length could go really big
// Note: Risk of payload being too large if the user has many many text/json based columns
// although possibly negligible risk.
const truncatedColumns = table.columns
.filter((column) => TEXT_TYPES.includes(column.format) || JSON_TYPES.includes(column.format))
.map((column) => {
return `case when length("${column.name}"::text) > ${MAX_CHARACTERS} then concat(left("${column.name}"::text, ${MAX_CHARACTERS}), '...') else "${column.name}"::text end "${column.name}"`
})
const outputSql =
truncatedColumns.length > 0
? `with _temp as (${baseSql.slice(0, -1)}) select *, ${truncatedColumns.join(',')} from _temp`
: baseSql
return outputSql
}
export type TableRows = {
rows: SupaRow[]
}
export type TableRowsVariables = GetTableRowsArgs & {
projectRef?: string
connectionString?: string
queryKey?: QueryKey
}
export type TableRowsData = TableRows
export type TableRowsError = ExecuteSqlError
export const useTableRowsQuery = <TData extends TableRowsData = TableRowsData>(
{ projectRef, connectionString, queryKey, table, impersonatedRole, ...args }: TableRowsVariables,
options: UseQueryOptions<ExecuteSqlData, TableRowsError, TData> = {}
) => {
const isRoleImpersonationEnabled = useIsRoleImpersonationEnabled()
return useExecuteSqlQuery(
{
projectRef,
connectionString,
sql: wrapWithRoleImpersonation(getTableRowsSqlQuery({ table, ...args }), {
projectRef: projectRef ?? 'ref',
role: impersonatedRole,
}),
queryKey: [
...(queryKey ?? []),
{
table: {
name: table?.name,
schema: table?.schema,
columns: table?.columns.map((c) => c.name),
},
impersonatedRole,
...args,
},
],
isRoleImpersonationEnabled,
},
{
select(data) {
const rows = data.result.map((x: any, index: number) => {
return { idx: index, ...x } as SupaRow
})
return {
rows,
} as TData
},
enabled: typeof projectRef !== 'undefined' && typeof table !== 'undefined',
...options,
}
)
}