mission-control/src/app/api/adapters/route.ts

119 lines
4.2 KiB
TypeScript

import { NextRequest, NextResponse } from 'next/server'
import { requireRole } from '@/lib/auth'
import { getAdapter, listAdapters } from '@/lib/adapters'
import { agentHeartbeatLimiter } from '@/lib/rate-limit'
import { logger } from '@/lib/logger'
/**
* GET /api/adapters — List available framework adapters.
*/
export async function GET(request: NextRequest) {
const auth = requireRole(request, 'viewer')
if ('error' in auth) return NextResponse.json({ error: auth.error }, { status: auth.status })
return NextResponse.json({ adapters: listAdapters() })
}
/**
* POST /api/adapters — Framework-agnostic agent action dispatcher.
*
* Body: { framework, action, payload }
*
* Actions:
* register — Register an agent via its framework adapter
* heartbeat — Send a heartbeat/status update
* report — Report task progress
* assignments — Get pending task assignments
* disconnect — Disconnect an agent
*/
export async function POST(request: NextRequest) {
const auth = requireRole(request, 'operator')
if ('error' in auth) return NextResponse.json({ error: auth.error }, { status: auth.status })
const rateLimited = agentHeartbeatLimiter(request)
if (rateLimited) return rateLimited
let body: any
try {
body = await request.json()
} catch {
return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })
}
const framework = typeof body?.framework === 'string' ? body.framework.trim() : ''
const action = typeof body?.action === 'string' ? body.action.trim() : ''
const payload = body?.payload ?? {}
if (!framework || !action) {
return NextResponse.json({ error: 'framework and action are required' }, { status: 400 })
}
let adapter
try {
adapter = getAdapter(framework)
} catch {
return NextResponse.json({
error: `Unknown framework: ${framework}. Available: ${listAdapters().join(', ')}`,
}, { status: 400 })
}
try {
switch (action) {
case 'register': {
const { agentId, name, metadata } = payload
if (!agentId || !name) {
return NextResponse.json({ error: 'payload.agentId and payload.name required' }, { status: 400 })
}
await adapter.register({ agentId, name, framework, metadata })
return NextResponse.json({ ok: true, action: 'register', framework })
}
case 'heartbeat': {
const { agentId, status, metrics } = payload
if (!agentId) {
return NextResponse.json({ error: 'payload.agentId required' }, { status: 400 })
}
await adapter.heartbeat({ agentId, status: status || 'online', metrics })
return NextResponse.json({ ok: true, action: 'heartbeat', framework })
}
case 'report': {
const { taskId, agentId, progress, status: taskStatus, output } = payload
if (!taskId || !agentId) {
return NextResponse.json({ error: 'payload.taskId and payload.agentId required' }, { status: 400 })
}
await adapter.reportTask({ taskId, agentId, progress: progress ?? 0, status: taskStatus || 'in_progress', output })
return NextResponse.json({ ok: true, action: 'report', framework })
}
case 'assignments': {
const { agentId } = payload
if (!agentId) {
return NextResponse.json({ error: 'payload.agentId required' }, { status: 400 })
}
const assignments = await adapter.getAssignments(agentId)
return NextResponse.json({ assignments, framework })
}
case 'disconnect': {
const { agentId } = payload
if (!agentId) {
return NextResponse.json({ error: 'payload.agentId required' }, { status: 400 })
}
await adapter.disconnect(agentId)
return NextResponse.json({ ok: true, action: 'disconnect', framework })
}
default:
return NextResponse.json({
error: `Unknown action: ${action}. Use: register, heartbeat, report, assignments, disconnect`,
}, { status: 400 })
}
} catch (error) {
logger.error({ err: error, framework, action }, 'POST /api/adapters error')
return NextResponse.json({ error: 'Adapter action failed' }, { status: 500 })
}
}
export const dynamic = 'force-dynamic'