beadboard/src/hooks/use-beads-subscription.ts
copilot-swe-agent[bot] 850335661d fix: remove noisy console logs from useBeadsSubscription hook
Co-authored-by: zenchantlive <103866469+zenchantlive@users.noreply.github.com>
2026-02-14 09:43:31 +00:00

88 lines
2.5 KiB
TypeScript

'use client';
import { useEffect, useRef, useState, useCallback } from 'react';
import type { BeadIssue } from '../lib/types';
interface UseBeadsSubscriptionResult {
issues: BeadIssue[];
refresh: () => Promise<void>;
updateLocal: (issues: BeadIssue[] | ((prev: BeadIssue[]) => BeadIssue[])) => void;
}
interface FetchResponse {
ok: boolean;
issues?: BeadIssue[];
error?: { message?: string };
}
async function fetchIssues(projectRoot: string): Promise<BeadIssue[]> {
const response = await fetch(`/api/beads/read?projectRoot=${encodeURIComponent(projectRoot)}`, {
cache: 'no-store',
});
const payload = (await response.json()) as FetchResponse;
if (!response.ok || !payload.ok || !payload.issues) {
throw new Error(payload.error?.message ?? 'Failed to refresh issues');
}
return payload.issues;
}
export function useBeadsSubscription(
initialIssues: BeadIssue[],
projectRoot: string,
options: { onUpdate?: () => void } = {}
): UseBeadsSubscriptionResult {
const [issues, setIssues] = useState<BeadIssue[]>(initialIssues);
const refreshInFlightRef = useRef(false);
const { onUpdate } = options;
// Allow parent to update local state (e.g. optimistic updates)
const updateLocal = useCallback((newIssues: BeadIssue[] | ((prev: BeadIssue[]) => BeadIssue[])) => {
setIssues(newIssues);
}, []);
// Update local state when initial props change (e.g. server re-render)
useEffect(() => {
setIssues(initialIssues);
}, [initialIssues]);
const refresh = useCallback(async (options: { silent?: boolean } = {}) => {
if (refreshInFlightRef.current) {
return;
}
refreshInFlightRef.current = true;
try {
const reconciled = await fetchIssues(projectRoot);
setIssues(reconciled);
onUpdate?.();
} catch (error) {
if (!options.silent) {
console.error('[BeadsSubscription] Refresh failed:', error);
}
} finally {
refreshInFlightRef.current = false;
}
}, [projectRoot, onUpdate]);
useEffect(() => {
const source = new EventSource(`/api/events?projectRoot=${encodeURIComponent(projectRoot)}`);
source.onerror = (err) => {
console.error('[SSE] Connection error:', err);
};
const onIssues = () => {
onUpdate?.();
void refresh({ silent: true });
};
source.addEventListener('issues', onIssues as EventListener);
return () => {
source.removeEventListener('issues', onIssues as EventListener);
source.close();
};
}, [projectRoot, refresh, onUpdate]);
return { issues, refresh, updateLocal };
}