mirror of
https://github.com/supabase/supabase.git
synced 2026-05-23 10:21:37 +08:00
* wip: updated UI and design for row renderer * updated components * wip: updated row count to remove useless estimates * wip: updated colors for seq and index scans * updated UI * updated utility functions * updated padding * made circle smaller * fixed breaking test
188 lines
5.4 KiB
TypeScript
188 lines
5.4 KiB
TypeScript
import {
|
|
Activity,
|
|
Database,
|
|
GitMerge,
|
|
Hash,
|
|
ListFilter,
|
|
SortAsc,
|
|
Zap,
|
|
Layers,
|
|
type LucideIcon,
|
|
} from 'lucide-react'
|
|
|
|
// Get human-readable description for an operation
|
|
export function getOperationDescription(operation: string): string {
|
|
const op = operation.toLowerCase()
|
|
|
|
if (op.includes('seq scan')) {
|
|
return 'Reads entire table row by row'
|
|
}
|
|
if (op.includes('index only scan')) {
|
|
return 'Reads data directly from index (fastest)'
|
|
}
|
|
if (op.includes('bitmap index scan')) {
|
|
return 'Builds bitmap of matching rows from index'
|
|
}
|
|
if (op.includes('bitmap heap scan')) {
|
|
return 'Fetches rows using bitmap'
|
|
}
|
|
if (op.includes('index scan')) {
|
|
return 'Uses index to find matching rows'
|
|
}
|
|
if (op.includes('hash left join')) {
|
|
return 'Returns all left rows with matching right rows via hash'
|
|
}
|
|
if (op.includes('hash right join')) {
|
|
return 'Returns all right rows with matching left rows via hash'
|
|
}
|
|
if (op.includes('hash full join')) {
|
|
return 'Returns all rows from both tables via hash'
|
|
}
|
|
if (op.includes('hash anti join')) {
|
|
return 'Returns rows without matches via hash'
|
|
}
|
|
if (op.includes('hash semi join')) {
|
|
return 'Returns rows with at least one match via hash'
|
|
}
|
|
if (op.includes('hash join')) {
|
|
return 'Joins tables using hash lookup'
|
|
}
|
|
if (op.includes('merge left join')) {
|
|
return 'Returns all left rows with matching right rows via merge'
|
|
}
|
|
if (op.includes('merge right join')) {
|
|
return 'Returns all right rows with matching left rows via merge'
|
|
}
|
|
if (op.includes('merge full join')) {
|
|
return 'Returns all rows from both tables via merge'
|
|
}
|
|
if (op.includes('merge anti join')) {
|
|
return 'Returns rows without matches via merge'
|
|
}
|
|
if (op.includes('merge semi join')) {
|
|
return 'Returns rows with at least one match via merge'
|
|
}
|
|
if (op.includes('merge join')) {
|
|
return 'Joins pre-sorted tables'
|
|
}
|
|
if (op.includes('nested loop left join')) {
|
|
return 'Returns all left rows with matching right rows via loop'
|
|
}
|
|
if (op.includes('nested loop anti join')) {
|
|
return 'Returns rows without matches via loop'
|
|
}
|
|
if (op.includes('nested loop semi join')) {
|
|
return 'Returns rows with at least one match via loop'
|
|
}
|
|
if (op.includes('nested loop')) {
|
|
return 'Joins by looping through each row'
|
|
}
|
|
if (op === 'hash') {
|
|
return 'Builds hash table for fast lookups'
|
|
}
|
|
if (op.includes('sort')) {
|
|
return 'Sorts rows for output or join'
|
|
}
|
|
if (op.includes('aggregate') || op.includes('group')) {
|
|
return 'Groups rows and calculates aggregates'
|
|
}
|
|
if (op.includes('limit')) {
|
|
return 'Returns only first N rows'
|
|
}
|
|
if (op.includes('materialize')) {
|
|
return 'Stores results in memory for reuse'
|
|
}
|
|
if (op.includes('gather')) {
|
|
return 'Collects results from parallel workers'
|
|
}
|
|
|
|
return ''
|
|
}
|
|
|
|
// Get an icon for the operation type
|
|
export function getOperationIcon(operation: string): LucideIcon {
|
|
const op = operation.toLowerCase()
|
|
if (op === 'hash') return Hash
|
|
if (op.includes('hash join')) return GitMerge
|
|
if (op.includes('merge join')) return GitMerge
|
|
if (op.includes('nested loop')) return GitMerge
|
|
if (op.includes('join')) return Layers
|
|
if (op.includes('index')) return Zap
|
|
if (op.includes('seq scan')) return Database
|
|
if (op.includes('scan')) return Database
|
|
if (op.includes('filter')) return ListFilter
|
|
if (op.includes('sort')) return SortAsc
|
|
if (op.includes('aggregate') || op.includes('group')) return Activity
|
|
return Database
|
|
}
|
|
|
|
// Get a color class for the operation type
|
|
export function getOperationColor(operation: string): string {
|
|
const op = operation.toLowerCase()
|
|
if (op.includes('seq scan')) return 'text-warning'
|
|
if (op.includes('index')) return 'text-brand'
|
|
if (op.includes('join')) return 'text-foreground-light'
|
|
if (op.includes('sort') || op.includes('aggregate')) return 'text-foreground-light'
|
|
return 'text-foreground-light'
|
|
}
|
|
|
|
export function isExplainQuery(rows: readonly any[]): boolean {
|
|
return (
|
|
rows.length > 0 && rows[0].hasOwnProperty('QUERY PLAN') && Object.keys(rows[0]).length === 1
|
|
)
|
|
}
|
|
|
|
export function formatNodeDuration(ms: number | undefined): string {
|
|
if (ms === undefined) return '-'
|
|
if (ms >= 1000) return `${(ms / 1000).toFixed(2)}s`
|
|
if (ms >= 1) return `${ms.toFixed(2)}ms`
|
|
if (ms >= 0.01) return `${ms.toFixed(2)}ms`
|
|
if (ms >= 0.001) return `${ms.toFixed(3)}ms`
|
|
|
|
const us = ms * 1000
|
|
if (us >= 0.1) return `${us.toFixed(1)}µs`
|
|
return `${us.toFixed(2)}µs`
|
|
}
|
|
|
|
export function getScanBarColor(operation: string): string {
|
|
const op = operation.toLowerCase()
|
|
|
|
// Index scans are green
|
|
if (
|
|
op.includes('index scan') ||
|
|
op.includes('index only scan') ||
|
|
op.includes('bitmap index scan')
|
|
) {
|
|
return 'bg-brand/20'
|
|
}
|
|
|
|
// Sequential scans are yellow
|
|
if (op.includes('seq scan') || op.includes('sequential scan')) {
|
|
return 'bg-warning/20'
|
|
}
|
|
|
|
// Default neutral color for other operations
|
|
return 'bg-foreground/[0.06]'
|
|
}
|
|
|
|
export function getScanBorderColor(operation: string): string {
|
|
const op = operation.toLowerCase()
|
|
|
|
// Index scans are green
|
|
if (
|
|
op.includes('index scan') ||
|
|
op.includes('index only scan') ||
|
|
op.includes('bitmap index scan')
|
|
) {
|
|
return 'border-l-brand'
|
|
}
|
|
|
|
// Sequential scans are yellow
|
|
if (op.includes('seq scan') || op.includes('sequential scan')) {
|
|
return 'border-l-warning'
|
|
}
|
|
|
|
// Default neutral color for other operations
|
|
return 'border-l-border-muted'
|
|
}
|