import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query' import { useAuthStore } from '../../auth/auth-store' // ── Application Config ──────────────────────────────────────────────────── export interface TapDefinition { tapId: string processorId: string target: 'INPUT' | 'OUTPUT' | 'BOTH' expression: string language: string attributeName: string attributeType: 'BUSINESS_OBJECT' | 'CORRELATION' | 'EVENT' | 'CUSTOM' enabled: boolean version: number } export interface ApplicationConfig { application: string version: number updatedAt?: string engineLevel?: string payloadCaptureMode?: string applicationLogLevel?: string agentLogLevel?: string metricsEnabled: boolean samplingRate: number tracedProcessors: Record taps: TapDefinition[] tapVersion: number routeRecording: Record compressSuccess: boolean } /** Authenticated fetch using the JWT from auth store */ function authFetch(url: string, init?: RequestInit): Promise { const token = useAuthStore.getState().accessToken const headers = new Headers(init?.headers) if (token) headers.set('Authorization', `Bearer ${token}`) headers.set('X-Cameleer-Protocol-Version', '1') return fetch(url, { ...init, headers }) } export function useAllApplicationConfigs() { return useQuery({ queryKey: ['applicationConfig', 'all'], queryFn: async () => { const res = await authFetch('/api/v1/config') if (!res.ok) throw new Error('Failed to fetch configs') return res.json() as Promise }, }) } export function useApplicationConfig(application: string | undefined) { return useQuery({ queryKey: ['applicationConfig', application], queryFn: async () => { const res = await authFetch(`/api/v1/config/${application}`) if (!res.ok) throw new Error('Failed to fetch config') return res.json() as Promise }, enabled: !!application, }) } export interface ConfigUpdateResponse { config: ApplicationConfig pushResult: CommandGroupResponse } export function useUpdateApplicationConfig() { const queryClient = useQueryClient() return useMutation({ mutationFn: async (config: ApplicationConfig) => { const res = await authFetch(`/api/v1/config/${config.application}`, { method: 'PUT', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(config), }) if (!res.ok) throw new Error('Failed to update config') return res.json() as Promise }, onSuccess: (result) => { queryClient.setQueryData(['applicationConfig', result.config.application], result.config) queryClient.invalidateQueries({ queryKey: ['applicationConfig', 'all'] }) }, }) } // ── Processor → Route Mapping ───────────────────────────────────────────── export function useProcessorRouteMapping(application?: string) { return useQuery({ queryKey: ['config', application, 'processor-routes'], queryFn: async () => { const res = await authFetch(`/api/v1/config/${application}/processor-routes`) if (!res.ok) throw new Error('Failed to fetch processor-route mapping') return res.json() as Promise> }, enabled: !!application, }) } // ── Group Command Response ─────────────────────────────────────────────── export interface CommandGroupResponse { success: boolean total: number responded: number responses: { agentId: string; status: string; message: string }[] timedOut: string[] } // ── Generic Group Command (kept for non-config commands) ────────────────── interface SendGroupCommandParams { group: string type: string payload: Record } export function useSendGroupCommand() { return useMutation({ mutationFn: async ({ group, type, payload }: SendGroupCommandParams) => { const res = await authFetch(`/api/v1/agents/groups/${encodeURIComponent(group)}/commands`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ type, payload }), }) if (!res.ok) throw new Error('Failed to send command') return res.json() as Promise }, }) } // ── Test Expression ─────────────────────────────────────────────────────── export function useTestExpression() { return useMutation({ mutationFn: async ({ application, expression, language, body, target, }: { application: string expression: string language: string body: string target: string }) => { const res = await authFetch( `/api/v1/config/${encodeURIComponent(application)}/test-expression`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ expression, language, body, target }), }, ) if (!res.ok) { if (res.status === 404) throw new Error('No live agent available') if (res.status === 504) throw new Error('Expression test timed out') throw new Error('Failed to test expression') } return res.json() as Promise<{ result?: string; error?: string }> }, }) } // ── Route Control ──────────────────────────────────────────────────────── export function useSendRouteCommand() { return useMutation({ mutationFn: async ({ application, action, routeId }: { application: string action: 'start' | 'stop' | 'suspend' | 'resume' routeId: string }) => { const res = await authFetch(`/api/v1/agents/groups/${encodeURIComponent(application)}/commands`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ type: 'route-control', payload: { routeId, action, nonce: crypto.randomUUID() } }), }) if (!res.ok) throw new Error('Failed to send route command') return res.json() as Promise }, }) } // ── Replay Exchange ─────────────────────────────────────────────────────── export interface ReplayResult { status: string message: string data?: string } export function useReplayExchange() { return useMutation({ mutationFn: async ({ agentId, routeId, headers, body, originalExchangeId, }: { agentId: string routeId: string headers?: Record body: string originalExchangeId?: string }): Promise => { const res = await authFetch(`/api/v1/agents/${encodeURIComponent(agentId)}/replay`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ routeId, body, headers: headers ?? {}, originalExchangeId }), }) if (!res.ok) { if (res.status === 404) throw new Error('Agent not found') if (res.status === 504) throw new Error('Replay timed out — agent did not respond') throw new Error('Failed to send replay command') } return res.json() as Promise }, }) }