mirror of
https://github.com/supabase/supabase.git
synced 2026-05-08 18:00:20 -04:00
12989ba7fe
## I have read the [CONTRIBUTING.md](https://github.com/supabase/supabase/blob/master/CONTRIBUTING.md) file. YES ## What kind of change does this PR introduce? Some small styling brush ups and experimental for internal telemetry tools. <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **New Features** * Developer toolbar redesigned with compact event/flag lists, “Copy JSON” per event, and a fixed draggable trigger that snaps and remembers its position. Toolbar is now available in staging and local environments. * **Bug Fixes** * ConfigCat readiness wait ensures flags load correctly. * Feature flag loading made resilient so one provider’s failure won’t block the other. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Sean Oliver <882952+seanoliver@users.noreply.github.com>
217 lines
5.4 KiB
TypeScript
217 lines
5.4 KiB
TypeScript
'use client'
|
|
|
|
import { ensurePlatformSuffix, posthogClient, type ClientTelemetryEvent } from 'common'
|
|
import {
|
|
createContext,
|
|
useCallback,
|
|
useContext,
|
|
useEffect,
|
|
useRef,
|
|
useState,
|
|
type ReactNode,
|
|
} from 'react'
|
|
|
|
import type {
|
|
DevTelemetryEvent,
|
|
DevTelemetryToolbarContextType,
|
|
ServerTelemetryEvent,
|
|
} from './types'
|
|
import { getCookie } from './utils'
|
|
|
|
// Duplicated for tree-shaking — bundler must see literal process.env reference.
|
|
// Keep in sync: index.ts, DevToolbar.tsx, DevToolbarTrigger.tsx, feature-flags.tsx
|
|
const env = process.env.NEXT_PUBLIC_ENVIRONMENT
|
|
const IS_TOOLBAR_ENABLED = env === 'local' || env === 'staging'
|
|
const IS_LOCAL_DEV = env === 'local'
|
|
const MAX_EVENTS = 200
|
|
const STORAGE_KEY = 'dev-telemetry-toolbar-enabled'
|
|
|
|
const SSE_INITIAL_RETRY_MS = 1000
|
|
const SSE_MAX_RETRY_MS = 30000
|
|
const SSE_BACKOFF_MULTIPLIER = 2
|
|
|
|
declare global {
|
|
interface Window {
|
|
devTelemetry?: () => void
|
|
}
|
|
}
|
|
|
|
const DevToolbarContext = createContext<DevTelemetryToolbarContextType | null>(null)
|
|
|
|
interface DevToolbarProviderProps {
|
|
children: ReactNode
|
|
apiUrl: string
|
|
}
|
|
|
|
export function DevToolbarProvider({ children, apiUrl }: DevToolbarProviderProps) {
|
|
const [isEnabled, setIsEnabled] = useState(false)
|
|
const [isOpen, setIsOpen] = useState(false)
|
|
const [events, setEvents] = useState<DevTelemetryEvent[]>([])
|
|
|
|
const sseRetryDelayRef = useRef(SSE_INITIAL_RETRY_MS)
|
|
const sseRetryTimeoutRef = useRef<NodeJS.Timeout | null>(null)
|
|
|
|
const dismissToolbar = useCallback(() => {
|
|
try {
|
|
localStorage.removeItem(STORAGE_KEY)
|
|
} catch {}
|
|
setIsEnabled(false)
|
|
setIsOpen(false)
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
if (!IS_TOOLBAR_ENABLED) return
|
|
|
|
let stored: string | null = null
|
|
try {
|
|
stored = localStorage.getItem(STORAGE_KEY)
|
|
} catch {}
|
|
if (stored === 'true') {
|
|
setIsEnabled(true)
|
|
}
|
|
|
|
window.devTelemetry = () => {
|
|
try {
|
|
localStorage.setItem(STORAGE_KEY, 'true')
|
|
} catch {}
|
|
setIsEnabled(true)
|
|
}
|
|
|
|
return () => {
|
|
delete window.devTelemetry
|
|
}
|
|
}, [])
|
|
|
|
const appendEvent = useCallback((event: DevTelemetryEvent) => {
|
|
setEvents((prev) => {
|
|
const key = `${event.source}-${event.id}`
|
|
if (prev.some((e) => `${e.source}-${e.id}` === key)) return prev
|
|
return [...prev.slice(-(MAX_EVENTS - 1)), event]
|
|
})
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
if (!isEnabled) return
|
|
|
|
const unsubscribe = posthogClient.subscribeToEvents((clientEvent: ClientTelemetryEvent) => {
|
|
appendEvent({
|
|
id: clientEvent.id,
|
|
timestamp: clientEvent.timestamp,
|
|
source: 'client',
|
|
eventType: clientEvent.eventType,
|
|
eventName: clientEvent.eventName,
|
|
distinctId: clientEvent.distinctId,
|
|
properties: clientEvent.properties,
|
|
})
|
|
})
|
|
|
|
return unsubscribe
|
|
}, [appendEvent, isEnabled])
|
|
|
|
useEffect(() => {
|
|
if (!IS_LOCAL_DEV || !isEnabled || typeof EventSource === 'undefined') return
|
|
|
|
let eventSource: EventSource | null = null
|
|
let isMounted = true
|
|
|
|
const connect = () => {
|
|
if (!isMounted) return
|
|
|
|
const sessionId = getCookie('session_id')
|
|
const streamUrl = `${ensurePlatformSuffix(apiUrl)}/telemetry/stream${
|
|
sessionId ? `?session_id=${encodeURIComponent(sessionId)}` : ''
|
|
}`
|
|
|
|
eventSource = new EventSource(streamUrl, { withCredentials: true })
|
|
|
|
eventSource.onopen = () => {
|
|
sseRetryDelayRef.current = SSE_INITIAL_RETRY_MS
|
|
}
|
|
|
|
eventSource.onmessage = (event) => {
|
|
try {
|
|
const data = JSON.parse(event.data) as ServerTelemetryEvent
|
|
appendEvent({
|
|
id: data.id,
|
|
timestamp: data.timestamp,
|
|
source: 'server',
|
|
eventType: data.eventType,
|
|
eventName: data.eventName,
|
|
distinctId: data.distinctId,
|
|
properties: data.properties,
|
|
})
|
|
} catch (e) {
|
|
console.error('[DevToolbar] Failed to parse SSE event:', e)
|
|
}
|
|
}
|
|
|
|
eventSource.onerror = () => {
|
|
if (!isMounted) return
|
|
|
|
eventSource?.close()
|
|
eventSource = null
|
|
|
|
const delay = sseRetryDelayRef.current
|
|
console.warn(`[DevToolbar] SSE connection error, reconnecting in ${delay}ms...`)
|
|
|
|
if (sseRetryTimeoutRef.current) {
|
|
clearTimeout(sseRetryTimeoutRef.current)
|
|
sseRetryTimeoutRef.current = null
|
|
}
|
|
sseRetryTimeoutRef.current = setTimeout(() => {
|
|
if (isMounted) {
|
|
connect()
|
|
}
|
|
}, delay)
|
|
|
|
sseRetryDelayRef.current = Math.min(delay * SSE_BACKOFF_MULTIPLIER, SSE_MAX_RETRY_MS)
|
|
}
|
|
}
|
|
|
|
connect()
|
|
|
|
return () => {
|
|
isMounted = false
|
|
eventSource?.close()
|
|
if (sseRetryTimeoutRef.current) {
|
|
clearTimeout(sseRetryTimeoutRef.current)
|
|
sseRetryTimeoutRef.current = null
|
|
}
|
|
}
|
|
}, [apiUrl, appendEvent, isEnabled])
|
|
|
|
if (!IS_TOOLBAR_ENABLED) {
|
|
return <>{children}</>
|
|
}
|
|
|
|
return (
|
|
<DevToolbarContext.Provider
|
|
value={{
|
|
isEnabled,
|
|
isOpen,
|
|
setIsOpen,
|
|
events,
|
|
setEvents,
|
|
dismissToolbar,
|
|
}}
|
|
>
|
|
{children}
|
|
</DevToolbarContext.Provider>
|
|
)
|
|
}
|
|
|
|
export function useDevToolbar() {
|
|
const context = useContext(DevToolbarContext)
|
|
if (!context) {
|
|
return {
|
|
isEnabled: false,
|
|
isOpen: false,
|
|
setIsOpen: () => {},
|
|
events: [],
|
|
setEvents: () => {},
|
|
dismissToolbar: () => {},
|
|
}
|
|
}
|
|
return context
|
|
}
|