Files
breakpilot-lehrer/admin-lehrer/components/ai/useTrainingMetricsSSE.ts
Benjamin Admin b681ddb131 [split-required] Split 58 monoliths across Python, Go, TypeScript (Phases 1-3)
Phase 1 — Python (klausur-service): 5 monoliths → 36 files
- dsfa_corpus_ingestion.py (1,828 LOC → 5 files)
- cv_ocr_engines.py (2,102 LOC → 7 files)
- cv_layout.py (3,653 LOC → 10 files)
- vocab_worksheet_api.py (2,783 LOC → 8 files)
- grid_build_core.py (1,958 LOC → 6 files)

Phase 2 — Go (edu-search-service, school-service): 8 monoliths → 19 files
- staff_crawler.go (1,402 → 4), policy/store.go (1,168 → 3)
- policy_handlers.go (700 → 2), repository.go (684 → 2)
- search.go (592 → 2), ai_extraction_handlers.go (554 → 2)
- seed_data.go (591 → 2), grade_service.go (646 → 2)

Phase 3 — TypeScript (admin-lehrer): 45 monoliths → 220+ files
- sdk/types.ts (2,108 → 16 domain files)
- ai/rag/page.tsx (2,686 → 14 files)
- 22 page.tsx files split into _components/ + _hooks/
- 11 component files split into sub-components
- 10 SDK data catalogs added to loc-exceptions
- Deleted dead backup index_original.ts (4,899 LOC)

All original public APIs preserved via re-export facades.
Zero new errors: Python imports verified, Go builds clean,
TypeScript tsc --noEmit shows only pre-existing errors.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-24 17:28:57 +02:00

73 lines
1.9 KiB
TypeScript

/**
* useTrainingMetricsSSE Hook
*
* SSE (Server-Sent Events) hook for real-time training metrics.
* Connects to the training metrics stream and provides live updates.
*/
'use client'
import { useState, useEffect, useCallback, useRef } from 'react'
import type { TrainingStatus } from './training-metrics-types'
/**
* Custom hook for SSE-based training metrics
*/
export function useTrainingMetricsSSE(
apiBase: string,
jobId: string | null,
onUpdate?: (status: TrainingStatus) => void
) {
const [status, setStatus] = useState<TrainingStatus | null>(null)
const [connected, setConnected] = useState(false)
const [error, setError] = useState<string | null>(null)
const eventSourceRef = useRef<EventSource | null>(null)
useEffect(() => {
if (!jobId) return
const url = `${apiBase}/api/klausur/trocr/training/metrics/stream?job_id=${jobId}`
const eventSource = new EventSource(url)
eventSourceRef.current = eventSource
eventSource.onopen = () => {
setConnected(true)
setError(null)
}
eventSource.onmessage = (event) => {
try {
const data = JSON.parse(event.data) as TrainingStatus
setStatus(data)
onUpdate?.(data)
// Close connection when training is done
if (data.status === 'completed' || data.status === 'failed' || data.status === 'cancelled') {
eventSource.close()
setConnected(false)
}
} catch (e) {
console.error('Failed to parse SSE message:', e)
}
}
eventSource.onerror = () => {
setError('Verbindung zum Server verloren')
setConnected(false)
eventSource.close()
}
return () => {
eventSource.close()
setConnected(false)
}
}, [apiBase, jobId, onUpdate])
const disconnect = useCallback(() => {
eventSourceRef.current?.close()
setConnected(false)
}, [])
return { status, connected, error, disconnect }
}