Files
supabase/apps/studio/hooks/analytics/useLogsQuery.tsx
Charis 180ce515f6 style: require @ imports and sort imports for studio/hooks (#44444)
* **Chores**
* Updated internal module import paths across hook files to use
standardized path aliases for improved code consistency and
maintainability.
2026-04-01 11:48:02 -04:00

137 lines
3.8 KiB
TypeScript

import { useQuery } from '@tanstack/react-query'
import { IS_PLATFORM } from 'common'
import { Dispatch, SetStateAction, useEffect, useState } from 'react'
import {
EXPLORER_DATEPICKER_HELPERS,
getDefaultHelper,
} from '@/components/interfaces/Settings/Logs/Logs.constants'
import type {
LogData,
Logs,
LogsEndpointParams,
} from '@/components/interfaces/Settings/Logs/Logs.types'
import {
checkForILIKEClause,
checkForWithClause,
} from '@/components/interfaces/Settings/Logs/Logs.utils'
import { get } from '@/data/fetchers'
import { useIsFeatureEnabled } from '@/hooks/misc/useIsFeatureEnabled'
import { DOCS_URL } from '@/lib/constants'
export interface LogsQueryHook {
params: LogsEndpointParams
isLoading: boolean
logData: LogData[]
data?: never
error: string | Object | null
changeQuery: (newQuery?: string) => void
runQuery: () => void
setParams: Dispatch<SetStateAction<LogsEndpointParams>>
enabled?: boolean
}
export const useLogsQuery = (
projectRef: string,
initialParams: Partial<LogsEndpointParams> = {},
enabled = true
): LogsQueryHook => {
const defaultHelper = getDefaultHelper(EXPLORER_DATEPICKER_HELPERS)
const [params, setParams] = useState<LogsEndpointParams>({
sql: initialParams?.sql || '',
iso_timestamp_start: initialParams.iso_timestamp_start
? initialParams.iso_timestamp_start
: defaultHelper.calcFrom(),
iso_timestamp_end: initialParams.iso_timestamp_end
? initialParams.iso_timestamp_end
: defaultHelper.calcTo(),
})
const { logsMetadata } = useIsFeatureEnabled(['logs:metadata'])
useEffect(() => {
setParams((prev) => ({
...prev,
...initialParams,
sql: initialParams?.sql ?? prev.sql,
iso_timestamp_start: initialParams.iso_timestamp_start ?? prev.iso_timestamp_start,
iso_timestamp_end: initialParams.iso_timestamp_end ?? prev.iso_timestamp_end,
}))
}, [initialParams.sql, initialParams.iso_timestamp_start, initialParams.iso_timestamp_end])
const _enabled = enabled && typeof projectRef !== 'undefined' && Boolean(params.sql)
const usesWith = checkForWithClause(params.sql || '')
const usesILIKE = checkForILIKEClause(params.sql || '')
const {
data,
error: rqError,
isPending: isLoading,
isRefetching,
refetch,
} = useQuery({
queryKey: ['projects', projectRef, 'logs', params],
queryFn: async ({ signal }) => {
const { data, error } = await get(`/platform/projects/{ref}/analytics/endpoints/logs.all`, {
params: {
path: { ref: projectRef },
query: params,
},
signal,
})
if (error) {
throw error
}
return data as unknown as Logs
},
enabled: _enabled,
refetchOnWindowFocus: false,
})
let error: null | string | object = rqError ? (rqError as any).message : null
if (!error && data?.error) {
error = data?.error
}
if (IS_PLATFORM) {
if (usesWith) {
error = {
message: 'The parser does not yet support WITH and subquery statements.',
docs: `${DOCS_URL}/guides/platform/advanced-log-filtering#the-with-keyword-and-subqueries-are-not-supported`,
}
}
if (usesILIKE) {
error = {
message: 'BigQuery does not support ILIKE. Use REGEXP_CONTAINS instead.',
docs: `${DOCS_URL}/guides/platform/advanced-log-filtering#the-ilike-and-similar-to-keywords-are-not-supported`,
}
}
}
const changeQuery = (newQuery = '') => {
setParams((prev) => ({ ...prev, sql: newQuery }))
}
const logData = (data?.result ?? []).map((x) => {
if (logsMetadata) {
return x
} else {
const { metadata, ...log } = x
return log
}
})
return {
params,
isLoading: (_enabled && isLoading) || isRefetching,
logData: logData,
error,
changeQuery,
runQuery: () => refetch(),
setParams,
}
}
export default useLogsQuery