Compare commits
3 Commits
feat/ms23-
...
test/ms23-
| Author | SHA1 | Date | |
|---|---|---|---|
| 5a2e404f03 | |||
| 76f06d0291 | |||
| 03dd25f028 |
@@ -25,14 +25,14 @@ export class AgentIngestionService {
|
|||||||
where: { sessionId: agentId },
|
where: { sessionId: agentId },
|
||||||
create: {
|
create: {
|
||||||
sessionId: agentId,
|
sessionId: agentId,
|
||||||
parentSessionId: parentAgentId,
|
parentSessionId: parentAgentId ?? null,
|
||||||
missionId,
|
missionId,
|
||||||
taskId,
|
taskId,
|
||||||
agentType,
|
agentType,
|
||||||
status: "spawning",
|
status: "spawning",
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
parentSessionId: parentAgentId,
|
parentSessionId: parentAgentId ?? null,
|
||||||
missionId,
|
missionId,
|
||||||
taskId,
|
taskId,
|
||||||
agentType,
|
agentType,
|
||||||
|
|||||||
140
apps/orchestrator/src/api/agents/agent-control.service.spec.ts
Normal file
140
apps/orchestrator/src/api/agents/agent-control.service.spec.ts
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
|
||||||
|
import { AgentControlService } from "./agent-control.service";
|
||||||
|
import { PrismaService } from "../../prisma/prisma.service";
|
||||||
|
|
||||||
|
describe("AgentControlService", () => {
|
||||||
|
let service: AgentControlService;
|
||||||
|
let prisma: {
|
||||||
|
agentSessionTree: {
|
||||||
|
findUnique: ReturnType<typeof vi.fn>;
|
||||||
|
updateMany: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
agentConversationMessage: {
|
||||||
|
create: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
operatorAuditLog: {
|
||||||
|
create: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
prisma = {
|
||||||
|
agentSessionTree: {
|
||||||
|
findUnique: vi.fn(),
|
||||||
|
updateMany: vi.fn().mockResolvedValue({ count: 1 }),
|
||||||
|
},
|
||||||
|
agentConversationMessage: {
|
||||||
|
create: vi.fn().mockResolvedValue(undefined),
|
||||||
|
},
|
||||||
|
operatorAuditLog: {
|
||||||
|
create: vi.fn().mockResolvedValue(undefined),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
service = new AgentControlService(prisma as unknown as PrismaService);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("injectMessage", () => {
|
||||||
|
it("creates conversation message and audit log when tree entry exists", async () => {
|
||||||
|
prisma.agentSessionTree.findUnique.mockResolvedValue({ id: "tree-1" });
|
||||||
|
|
||||||
|
await service.injectMessage("agent-123", "operator-abc", "Please continue");
|
||||||
|
|
||||||
|
expect(prisma.agentSessionTree.findUnique).toHaveBeenCalledWith({
|
||||||
|
where: { sessionId: "agent-123" },
|
||||||
|
select: { id: true },
|
||||||
|
});
|
||||||
|
expect(prisma.agentConversationMessage.create).toHaveBeenCalledWith({
|
||||||
|
data: {
|
||||||
|
sessionId: "agent-123",
|
||||||
|
role: "operator",
|
||||||
|
content: "Please continue",
|
||||||
|
provider: "internal",
|
||||||
|
metadata: {},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(prisma.operatorAuditLog.create).toHaveBeenCalledWith({
|
||||||
|
data: {
|
||||||
|
sessionId: "agent-123",
|
||||||
|
userId: "operator-abc",
|
||||||
|
provider: "internal",
|
||||||
|
action: "inject",
|
||||||
|
metadata: {
|
||||||
|
payload: {
|
||||||
|
message: "Please continue",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("creates only audit log when no tree entry exists", async () => {
|
||||||
|
prisma.agentSessionTree.findUnique.mockResolvedValue(null);
|
||||||
|
|
||||||
|
await service.injectMessage("agent-456", "operator-def", "Nudge message");
|
||||||
|
|
||||||
|
expect(prisma.agentConversationMessage.create).not.toHaveBeenCalled();
|
||||||
|
expect(prisma.operatorAuditLog.create).toHaveBeenCalledWith({
|
||||||
|
data: {
|
||||||
|
sessionId: "agent-456",
|
||||||
|
userId: "operator-def",
|
||||||
|
provider: "internal",
|
||||||
|
action: "inject",
|
||||||
|
metadata: {
|
||||||
|
payload: {
|
||||||
|
message: "Nudge message",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("pauseAgent", () => {
|
||||||
|
it("updates tree status to paused and creates audit log", async () => {
|
||||||
|
await service.pauseAgent("agent-789", "operator-pause");
|
||||||
|
|
||||||
|
expect(prisma.agentSessionTree.updateMany).toHaveBeenCalledWith({
|
||||||
|
where: { sessionId: "agent-789" },
|
||||||
|
data: { status: "paused" },
|
||||||
|
});
|
||||||
|
expect(prisma.operatorAuditLog.create).toHaveBeenCalledWith({
|
||||||
|
data: {
|
||||||
|
sessionId: "agent-789",
|
||||||
|
userId: "operator-pause",
|
||||||
|
provider: "internal",
|
||||||
|
action: "pause",
|
||||||
|
metadata: {
|
||||||
|
payload: {},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("resumeAgent", () => {
|
||||||
|
it("updates tree status to running and creates audit log", async () => {
|
||||||
|
await service.resumeAgent("agent-321", "operator-resume");
|
||||||
|
|
||||||
|
expect(prisma.agentSessionTree.updateMany).toHaveBeenCalledWith({
|
||||||
|
where: { sessionId: "agent-321" },
|
||||||
|
data: { status: "running" },
|
||||||
|
});
|
||||||
|
expect(prisma.operatorAuditLog.create).toHaveBeenCalledWith({
|
||||||
|
data: {
|
||||||
|
sessionId: "agent-321",
|
||||||
|
userId: "operator-resume",
|
||||||
|
provider: "internal",
|
||||||
|
action: "resume",
|
||||||
|
metadata: {
|
||||||
|
payload: {},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
103
apps/orchestrator/src/api/agents/agent-messages.service.spec.ts
Normal file
103
apps/orchestrator/src/api/agents/agent-messages.service.spec.ts
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
|
||||||
|
import { AgentMessagesService } from "./agent-messages.service";
|
||||||
|
import { PrismaService } from "../../prisma/prisma.service";
|
||||||
|
|
||||||
|
describe("AgentMessagesService", () => {
|
||||||
|
let service: AgentMessagesService;
|
||||||
|
let prisma: {
|
||||||
|
agentConversationMessage: {
|
||||||
|
findMany: ReturnType<typeof vi.fn>;
|
||||||
|
count: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
prisma = {
|
||||||
|
agentConversationMessage: {
|
||||||
|
findMany: vi.fn(),
|
||||||
|
count: vi.fn(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
service = new AgentMessagesService(prisma as unknown as PrismaService);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getMessages", () => {
|
||||||
|
it("returns paginated messages from Prisma", async () => {
|
||||||
|
const sessionId = "agent-123";
|
||||||
|
const messages = [
|
||||||
|
{
|
||||||
|
id: "msg-1",
|
||||||
|
sessionId,
|
||||||
|
provider: "internal",
|
||||||
|
role: "assistant",
|
||||||
|
content: "First message",
|
||||||
|
timestamp: new Date("2026-03-07T16:00:00.000Z"),
|
||||||
|
metadata: {},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "msg-2",
|
||||||
|
sessionId,
|
||||||
|
provider: "internal",
|
||||||
|
role: "user",
|
||||||
|
content: "Second message",
|
||||||
|
timestamp: new Date("2026-03-07T15:59:00.000Z"),
|
||||||
|
metadata: {},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
prisma.agentConversationMessage.findMany.mockResolvedValue(messages);
|
||||||
|
prisma.agentConversationMessage.count.mockResolvedValue(2);
|
||||||
|
|
||||||
|
const result = await service.getMessages(sessionId, 50, 0);
|
||||||
|
|
||||||
|
expect(prisma.agentConversationMessage.findMany).toHaveBeenCalledWith({
|
||||||
|
where: { sessionId },
|
||||||
|
orderBy: { timestamp: "desc" },
|
||||||
|
take: 50,
|
||||||
|
skip: 0,
|
||||||
|
});
|
||||||
|
expect(prisma.agentConversationMessage.count).toHaveBeenCalledWith({ where: { sessionId } });
|
||||||
|
expect(result).toEqual({
|
||||||
|
messages,
|
||||||
|
total: 2,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("applies limit and cursor (skip) correctly", async () => {
|
||||||
|
const sessionId = "agent-456";
|
||||||
|
const limit = 10;
|
||||||
|
const cursor = 20;
|
||||||
|
|
||||||
|
prisma.agentConversationMessage.findMany.mockResolvedValue([]);
|
||||||
|
prisma.agentConversationMessage.count.mockResolvedValue(42);
|
||||||
|
|
||||||
|
await service.getMessages(sessionId, limit, cursor);
|
||||||
|
|
||||||
|
expect(prisma.agentConversationMessage.findMany).toHaveBeenCalledWith({
|
||||||
|
where: { sessionId },
|
||||||
|
orderBy: { timestamp: "desc" },
|
||||||
|
take: limit,
|
||||||
|
skip: cursor,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns empty messages array when no messages exist", async () => {
|
||||||
|
const sessionId = "agent-empty";
|
||||||
|
|
||||||
|
prisma.agentConversationMessage.findMany.mockResolvedValue([]);
|
||||||
|
prisma.agentConversationMessage.count.mockResolvedValue(0);
|
||||||
|
|
||||||
|
const result = await service.getMessages(sessionId, 25, 0);
|
||||||
|
|
||||||
|
expect(result).toEqual({
|
||||||
|
messages: [],
|
||||||
|
total: 0,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
106
apps/orchestrator/src/api/agents/agent-tree.service.spec.ts
Normal file
106
apps/orchestrator/src/api/agents/agent-tree.service.spec.ts
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
|
||||||
|
import { AgentTreeService } from "./agent-tree.service";
|
||||||
|
import { PrismaService } from "../../prisma/prisma.service";
|
||||||
|
|
||||||
|
describe("AgentTreeService", () => {
|
||||||
|
let service: AgentTreeService;
|
||||||
|
let prisma: {
|
||||||
|
agentSessionTree: {
|
||||||
|
findMany: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
prisma = {
|
||||||
|
agentSessionTree: {
|
||||||
|
findMany: vi.fn(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
service = new AgentTreeService(prisma as unknown as PrismaService);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getTree", () => {
|
||||||
|
it("returns mapped entries from Prisma", async () => {
|
||||||
|
prisma.agentSessionTree.findMany.mockResolvedValue([
|
||||||
|
{
|
||||||
|
id: "tree-1",
|
||||||
|
sessionId: "agent-1",
|
||||||
|
parentSessionId: "agent-root",
|
||||||
|
provider: "internal",
|
||||||
|
missionId: "mission-1",
|
||||||
|
taskId: "task-1",
|
||||||
|
taskSource: "queue",
|
||||||
|
agentType: "worker",
|
||||||
|
status: "running",
|
||||||
|
spawnedAt: new Date("2026-03-07T10:00:00.000Z"),
|
||||||
|
completedAt: new Date("2026-03-07T11:00:00.000Z"),
|
||||||
|
metadata: {},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const result = await service.getTree();
|
||||||
|
|
||||||
|
expect(prisma.agentSessionTree.findMany).toHaveBeenCalledWith({
|
||||||
|
orderBy: { spawnedAt: "desc" },
|
||||||
|
take: 200,
|
||||||
|
});
|
||||||
|
expect(result).toEqual([
|
||||||
|
{
|
||||||
|
sessionId: "agent-1",
|
||||||
|
parentSessionId: "agent-root",
|
||||||
|
status: "running",
|
||||||
|
agentType: "worker",
|
||||||
|
taskSource: "queue",
|
||||||
|
spawnedAt: "2026-03-07T10:00:00.000Z",
|
||||||
|
completedAt: "2026-03-07T11:00:00.000Z",
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns empty array when no entries exist", async () => {
|
||||||
|
prisma.agentSessionTree.findMany.mockResolvedValue([]);
|
||||||
|
|
||||||
|
const result = await service.getTree();
|
||||||
|
|
||||||
|
expect(result).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("maps null parentSessionId and completedAt correctly", async () => {
|
||||||
|
prisma.agentSessionTree.findMany.mockResolvedValue([
|
||||||
|
{
|
||||||
|
id: "tree-2",
|
||||||
|
sessionId: "agent-root",
|
||||||
|
parentSessionId: null,
|
||||||
|
provider: "internal",
|
||||||
|
missionId: null,
|
||||||
|
taskId: null,
|
||||||
|
taskSource: null,
|
||||||
|
agentType: null,
|
||||||
|
status: "spawning",
|
||||||
|
spawnedAt: new Date("2026-03-07T09:00:00.000Z"),
|
||||||
|
completedAt: null,
|
||||||
|
metadata: {},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const result = await service.getTree();
|
||||||
|
|
||||||
|
expect(result).toEqual([
|
||||||
|
{
|
||||||
|
sessionId: "agent-root",
|
||||||
|
parentSessionId: null,
|
||||||
|
status: "spawning",
|
||||||
|
agentType: null,
|
||||||
|
taskSource: null,
|
||||||
|
spawnedAt: "2026-03-07T09:00:00.000Z",
|
||||||
|
completedAt: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
30
apps/orchestrator/src/api/agents/agent-tree.service.ts
Normal file
30
apps/orchestrator/src/api/agents/agent-tree.service.ts
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import { Injectable } from "@nestjs/common";
|
||||||
|
import { PrismaService } from "../../prisma/prisma.service";
|
||||||
|
import { AgentTreeResponseDto } from "./dto/agent-tree-response.dto";
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class AgentTreeService {
|
||||||
|
constructor(private readonly prisma: PrismaService) {}
|
||||||
|
|
||||||
|
async getTree(): Promise<AgentTreeResponseDto[]> {
|
||||||
|
const entries = await this.prisma.agentSessionTree.findMany({
|
||||||
|
orderBy: { spawnedAt: "desc" },
|
||||||
|
take: 200,
|
||||||
|
});
|
||||||
|
|
||||||
|
const response: AgentTreeResponseDto[] = [];
|
||||||
|
for (const entry of entries) {
|
||||||
|
response.push({
|
||||||
|
sessionId: entry.sessionId,
|
||||||
|
parentSessionId: entry.parentSessionId ?? null,
|
||||||
|
status: entry.status,
|
||||||
|
agentType: entry.agentType ?? null,
|
||||||
|
taskSource: entry.taskSource ?? null,
|
||||||
|
spawnedAt: entry.spawnedAt.toISOString(),
|
||||||
|
completedAt: entry.completedAt?.toISOString() ?? null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,6 +7,7 @@ import { KillswitchService } from "../../killswitch/killswitch.service";
|
|||||||
import { AgentEventsService } from "./agent-events.service";
|
import { AgentEventsService } from "./agent-events.service";
|
||||||
import { AgentMessagesService } from "./agent-messages.service";
|
import { AgentMessagesService } from "./agent-messages.service";
|
||||||
import { AgentControlService } from "./agent-control.service";
|
import { AgentControlService } from "./agent-control.service";
|
||||||
|
import { AgentTreeService } from "./agent-tree.service";
|
||||||
import type { KillAllResult } from "../../killswitch/killswitch.service";
|
import type { KillAllResult } from "../../killswitch/killswitch.service";
|
||||||
|
|
||||||
describe("AgentsController - Killswitch Endpoints", () => {
|
describe("AgentsController - Killswitch Endpoints", () => {
|
||||||
@@ -41,6 +42,9 @@ describe("AgentsController - Killswitch Endpoints", () => {
|
|||||||
pauseAgent: ReturnType<typeof vi.fn>;
|
pauseAgent: ReturnType<typeof vi.fn>;
|
||||||
resumeAgent: ReturnType<typeof vi.fn>;
|
resumeAgent: ReturnType<typeof vi.fn>;
|
||||||
};
|
};
|
||||||
|
let mockTreeService: {
|
||||||
|
getTree: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mockKillswitchService = {
|
mockKillswitchService = {
|
||||||
@@ -89,6 +93,10 @@ describe("AgentsController - Killswitch Endpoints", () => {
|
|||||||
resumeAgent: vi.fn().mockResolvedValue(undefined),
|
resumeAgent: vi.fn().mockResolvedValue(undefined),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
mockTreeService = {
|
||||||
|
getTree: vi.fn().mockResolvedValue([]),
|
||||||
|
};
|
||||||
|
|
||||||
controller = new AgentsController(
|
controller = new AgentsController(
|
||||||
mockQueueService as unknown as QueueService,
|
mockQueueService as unknown as QueueService,
|
||||||
mockSpawnerService as unknown as AgentSpawnerService,
|
mockSpawnerService as unknown as AgentSpawnerService,
|
||||||
@@ -96,7 +104,8 @@ describe("AgentsController - Killswitch Endpoints", () => {
|
|||||||
mockKillswitchService as unknown as KillswitchService,
|
mockKillswitchService as unknown as KillswitchService,
|
||||||
mockEventsService as unknown as AgentEventsService,
|
mockEventsService as unknown as AgentEventsService,
|
||||||
mockMessagesService as unknown as AgentMessagesService,
|
mockMessagesService as unknown as AgentMessagesService,
|
||||||
mockControlService as unknown as AgentControlService
|
mockControlService as unknown as AgentControlService,
|
||||||
|
mockTreeService as unknown as AgentTreeService
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { KillswitchService } from "../../killswitch/killswitch.service";
|
|||||||
import { AgentEventsService } from "./agent-events.service";
|
import { AgentEventsService } from "./agent-events.service";
|
||||||
import { AgentMessagesService } from "./agent-messages.service";
|
import { AgentMessagesService } from "./agent-messages.service";
|
||||||
import { AgentControlService } from "./agent-control.service";
|
import { AgentControlService } from "./agent-control.service";
|
||||||
|
import { AgentTreeService } from "./agent-tree.service";
|
||||||
import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
|
import { describe, it, expect, beforeEach, afterEach, vi } from "vitest";
|
||||||
|
|
||||||
describe("AgentsController", () => {
|
describe("AgentsController", () => {
|
||||||
@@ -42,6 +43,9 @@ describe("AgentsController", () => {
|
|||||||
pauseAgent: ReturnType<typeof vi.fn>;
|
pauseAgent: ReturnType<typeof vi.fn>;
|
||||||
resumeAgent: ReturnType<typeof vi.fn>;
|
resumeAgent: ReturnType<typeof vi.fn>;
|
||||||
};
|
};
|
||||||
|
let treeService: {
|
||||||
|
getTree: ReturnType<typeof vi.fn>;
|
||||||
|
};
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
// Create mock services
|
// Create mock services
|
||||||
@@ -93,6 +97,10 @@ describe("AgentsController", () => {
|
|||||||
resumeAgent: vi.fn().mockResolvedValue(undefined),
|
resumeAgent: vi.fn().mockResolvedValue(undefined),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
treeService = {
|
||||||
|
getTree: vi.fn().mockResolvedValue([]),
|
||||||
|
};
|
||||||
|
|
||||||
// Create controller with mocked services
|
// Create controller with mocked services
|
||||||
controller = new AgentsController(
|
controller = new AgentsController(
|
||||||
queueService as unknown as QueueService,
|
queueService as unknown as QueueService,
|
||||||
@@ -101,7 +109,8 @@ describe("AgentsController", () => {
|
|||||||
killswitchService as unknown as KillswitchService,
|
killswitchService as unknown as KillswitchService,
|
||||||
eventsService as unknown as AgentEventsService,
|
eventsService as unknown as AgentEventsService,
|
||||||
messagesService as unknown as AgentMessagesService,
|
messagesService as unknown as AgentMessagesService,
|
||||||
controlService as unknown as AgentControlService
|
controlService as unknown as AgentControlService,
|
||||||
|
treeService as unknown as AgentTreeService
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -113,6 +122,27 @@ describe("AgentsController", () => {
|
|||||||
expect(controller).toBeDefined();
|
expect(controller).toBeDefined();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("getAgentTree", () => {
|
||||||
|
it("should return tree entries", async () => {
|
||||||
|
const entries = [
|
||||||
|
{
|
||||||
|
sessionId: "agent-1",
|
||||||
|
parentSessionId: null,
|
||||||
|
status: "running",
|
||||||
|
agentType: "worker",
|
||||||
|
taskSource: "internal",
|
||||||
|
spawnedAt: "2026-03-07T00:00:00.000Z",
|
||||||
|
completedAt: null,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
treeService.getTree.mockResolvedValue(entries);
|
||||||
|
|
||||||
|
await expect(controller.getAgentTree()).resolves.toEqual(entries);
|
||||||
|
expect(treeService.getTree).toHaveBeenCalledTimes(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe("listAgents", () => {
|
describe("listAgents", () => {
|
||||||
it("should return empty array when no agents exist", () => {
|
it("should return empty array when no agents exist", () => {
|
||||||
// Arrange
|
// Arrange
|
||||||
|
|||||||
@@ -30,6 +30,8 @@ import { AgentEventsService } from "./agent-events.service";
|
|||||||
import { GetMessagesQueryDto } from "./dto/get-messages-query.dto";
|
import { GetMessagesQueryDto } from "./dto/get-messages-query.dto";
|
||||||
import { AgentMessagesService } from "./agent-messages.service";
|
import { AgentMessagesService } from "./agent-messages.service";
|
||||||
import { AgentControlService } from "./agent-control.service";
|
import { AgentControlService } from "./agent-control.service";
|
||||||
|
import { AgentTreeService } from "./agent-tree.service";
|
||||||
|
import { AgentTreeResponseDto } from "./dto/agent-tree-response.dto";
|
||||||
import { InjectAgentDto } from "./dto/inject-agent.dto";
|
import { InjectAgentDto } from "./dto/inject-agent.dto";
|
||||||
import { PauseAgentDto, ResumeAgentDto } from "./dto/control-agent.dto";
|
import { PauseAgentDto, ResumeAgentDto } from "./dto/control-agent.dto";
|
||||||
|
|
||||||
@@ -56,7 +58,8 @@ export class AgentsController {
|
|||||||
private readonly killswitchService: KillswitchService,
|
private readonly killswitchService: KillswitchService,
|
||||||
private readonly eventsService: AgentEventsService,
|
private readonly eventsService: AgentEventsService,
|
||||||
private readonly messagesService: AgentMessagesService,
|
private readonly messagesService: AgentMessagesService,
|
||||||
private readonly agentControlService: AgentControlService
|
private readonly agentControlService: AgentControlService,
|
||||||
|
private readonly agentTreeService: AgentTreeService
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -78,6 +81,7 @@ export class AgentsController {
|
|||||||
// Spawn agent using spawner service
|
// Spawn agent using spawner service
|
||||||
const spawnResponse = this.spawnerService.spawnAgent({
|
const spawnResponse = this.spawnerService.spawnAgent({
|
||||||
taskId: dto.taskId,
|
taskId: dto.taskId,
|
||||||
|
...(dto.parentAgentId !== undefined ? { parentAgentId: dto.parentAgentId } : {}),
|
||||||
agentType: dto.agentType,
|
agentType: dto.agentType,
|
||||||
context: dto.context,
|
context: dto.context,
|
||||||
});
|
});
|
||||||
@@ -152,6 +156,13 @@ export class AgentsController {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Get("tree")
|
||||||
|
@UseGuards(OrchestratorApiKeyGuard)
|
||||||
|
@Throttle({ default: { limit: 200, ttl: 60000 } })
|
||||||
|
async getAgentTree(): Promise<AgentTreeResponseDto[]> {
|
||||||
|
return this.agentTreeService.getTree();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* List all agents
|
* List all agents
|
||||||
* @returns Array of all agent sessions with their status
|
* @returns Array of all agent sessions with their status
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import { AgentEventsService } from "./agent-events.service";
|
|||||||
import { PrismaModule } from "../../prisma/prisma.module";
|
import { PrismaModule } from "../../prisma/prisma.module";
|
||||||
import { AgentMessagesService } from "./agent-messages.service";
|
import { AgentMessagesService } from "./agent-messages.service";
|
||||||
import { AgentControlService } from "./agent-control.service";
|
import { AgentControlService } from "./agent-control.service";
|
||||||
|
import { AgentTreeService } from "./agent-tree.service";
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [QueueModule, SpawnerModule, KillswitchModule, ValkeyModule, PrismaModule],
|
imports: [QueueModule, SpawnerModule, KillswitchModule, ValkeyModule, PrismaModule],
|
||||||
@@ -18,6 +19,7 @@ import { AgentControlService } from "./agent-control.service";
|
|||||||
AgentEventsService,
|
AgentEventsService,
|
||||||
AgentMessagesService,
|
AgentMessagesService,
|
||||||
AgentControlService,
|
AgentControlService,
|
||||||
|
AgentTreeService,
|
||||||
],
|
],
|
||||||
})
|
})
|
||||||
export class AgentsModule {}
|
export class AgentsModule {}
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
export class AgentTreeResponseDto {
|
||||||
|
sessionId!: string;
|
||||||
|
parentSessionId!: string | null;
|
||||||
|
status!: string;
|
||||||
|
agentType!: string | null;
|
||||||
|
taskSource!: string | null;
|
||||||
|
spawnedAt!: string;
|
||||||
|
completedAt!: string | null;
|
||||||
|
}
|
||||||
@@ -116,6 +116,10 @@ export class SpawnAgentDto {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsIn(["strict", "standard", "minimal", "custom"])
|
@IsIn(["strict", "standard", "minimal", "custom"])
|
||||||
gateProfile?: GateProfileType;
|
gateProfile?: GateProfileType;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
parentAgentId?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -115,7 +115,13 @@ export class AgentSpawnerService implements OnModuleDestroy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void this.agentIngestionService
|
void this.agentIngestionService
|
||||||
.recordAgentSpawned(agentId, undefined, undefined, request.taskId, request.agentType)
|
.recordAgentSpawned(
|
||||||
|
agentId,
|
||||||
|
request.parentAgentId,
|
||||||
|
undefined,
|
||||||
|
request.taskId,
|
||||||
|
request.agentType
|
||||||
|
)
|
||||||
.catch((error: unknown) => {
|
.catch((error: unknown) => {
|
||||||
const errorMessage = error instanceof Error ? error.message : String(error);
|
const errorMessage = error instanceof Error ? error.message : String(error);
|
||||||
this.logger.error(`Failed to record spawned ingestion for ${agentId}: ${errorMessage}`);
|
this.logger.error(`Failed to record spawned ingestion for ${agentId}: ${errorMessage}`);
|
||||||
|
|||||||
@@ -40,6 +40,8 @@ export interface SpawnAgentOptions {
|
|||||||
export interface SpawnAgentRequest {
|
export interface SpawnAgentRequest {
|
||||||
/** Unique task identifier */
|
/** Unique task identifier */
|
||||||
taskId: string;
|
taskId: string;
|
||||||
|
/** Optional parent session identifier for subagent lineage */
|
||||||
|
parentAgentId?: string;
|
||||||
/** Type of agent to spawn */
|
/** Type of agent to spawn */
|
||||||
agentType: AgentType;
|
agentType: AgentType;
|
||||||
/** Context for task execution */
|
/** Context for task execution */
|
||||||
|
|||||||
@@ -125,8 +125,8 @@ Target version: `v0.0.23`
|
|||||||
| MS23-P0-002 | done | p0-foundation | Agent message ingestion: wire spawner/lifecycle to write messages to DB | #693 | orchestrator | feat/ms23-p0-ingestion | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-06 | 2026-03-07 | 20K | — | |
|
| MS23-P0-002 | done | p0-foundation | Agent message ingestion: wire spawner/lifecycle to write messages to DB | #693 | orchestrator | feat/ms23-p0-ingestion | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-06 | 2026-03-07 | 20K | — | |
|
||||||
| MS23-P0-003 | done | p0-foundation | Orchestrator API: GET /agents/:id/messages + SSE stream endpoint | #693 | orchestrator | feat/ms23-p0-stream | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-06 | 2026-03-07 | 20K | — | |
|
| MS23-P0-003 | done | p0-foundation | Orchestrator API: GET /agents/:id/messages + SSE stream endpoint | #693 | orchestrator | feat/ms23-p0-stream | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-06 | 2026-03-07 | 20K | — | |
|
||||||
| MS23-P0-004 | done | p0-foundation | Orchestrator API: POST /agents/:id/inject + pause/resume endpoints | #693 | orchestrator | feat/ms23-p0-controls | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-07 | 2026-03-07 | 15K | — | |
|
| MS23-P0-004 | done | p0-foundation | Orchestrator API: POST /agents/:id/inject + pause/resume endpoints | #693 | orchestrator | feat/ms23-p0-controls | MS23-P0-001 | MS23-P0-006 | codex | 2026-03-07 | 2026-03-07 | 15K | — | |
|
||||||
| MS23-P0-005 | in-progress | p0-foundation | Subagent tree: parentAgentId on spawn registration + GET /agents/tree | #693 | orchestrator | feat/ms23-p0-tree | MS23-P0-001 | MS23-P0-006 | — | — | — | 15K | — | |
|
| MS23-P0-005 | done | p0-foundation | Subagent tree: parentAgentId on spawn registration + GET /agents/tree | #693 | orchestrator | feat/ms23-p0-tree | MS23-P0-001 | MS23-P0-006 | — | — | — | 15K | — | |
|
||||||
| MS23-P0-006 | not-started | p0-foundation | Unit + integration tests for all P0 orchestrator endpoints | #693 | orchestrator | test/ms23-p0 | MS23-P0-002,MS23-P0-003,MS23-P0-004,MS23-P0-005 | MS23-P1-001 | — | — | — | 20K | — | Phase 0 gate: SSE stream verified via curl |
|
| MS23-P0-006 | in-progress | p0-foundation | Unit + integration tests for all P0 orchestrator endpoints | #693 | orchestrator | test/ms23-p0 | MS23-P0-002,MS23-P0-003,MS23-P0-004,MS23-P0-005 | MS23-P1-001 | codex | 2026-03-07 | — | 20K | — | Phase 0 gate: SSE stream verified via curl |
|
||||||
|
|
||||||
### Phase 1 — Provider Interface (Plugin Architecture)
|
### Phase 1 — Provider Interface (Plugin Architecture)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user