c5e0e5cfe3
Agents can now use Model Context Protocol servers. End to end: - SharedKernel seam IMcpGateway (ListToolsAsync / CallToolAsync) + McpToolDescriptor / McpToolResult, so the Assembler discovers and can invoke MCP tools without referencing Integrations' tables. - Integrations: McpServerConfig (org-scoped, owner-only; auth headers AES-GCM encrypted, never returned — only their names) + AddMcpServers migration. McpClient: a dependency-free Streamable-HTTP JSON-RPC 2.0 client (initialize → notifications/initialized → tools/list / tools/call), carrying the Mcp-Session-Id and parsing both application/json and text/event-stream replies. McpGateway resolves an org's servers, decrypts headers server-side, and is best-effort: an unreachable server is logged and skipped, never failing the run. CRUD + connectivity-test endpoints (create/test/delete owner-only via ManageApiKeys; list via ConfigureAgents to bind). - OrgBoard: Agent gains McpServerIds (uuid[]; migration backfills existing agents to empty) flowing through ConfigureAgent + AgentRunContext. - Assembler: AgentRunExecutor lists the agent's MCP tools (best-effort) and PromptAssembler renders a "# Tools (MCP)" catalog — labelled as data, never instructions — and records it in the run trace. - Client: SeatsPage gains an MCP servers card (add/test/delete, encrypted auth header) and a per-agent MCP server multi-select; api client gains del(). Note: discovery + the governed call gateway are in place now; the autonomous model-driven tool-call loop (model emits tool_calls → gated execution → feedback) needs a tool-calling model client and is the next increment — the stub model can't drive it. Verified: ArchitectureTests 8/8, IntegrationTests 53/53 (McpClientTests: JSON-RPC handshake/session, json + SSE; McpServerRegistryTests: owner-only, encrypted-header-never-returned, graceful test, Member 403; PromptAssemblerMcpTests: catalog + trace, omitted when empty), client build green. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
60 lines
1.9 KiB
C#
60 lines
1.9 KiB
C#
using TeamUp.SharedKernel.Access;
|
|
using TeamUp.SharedKernel.Domain;
|
|
|
|
namespace TeamUp.Modules.OrgBoard.Domain;
|
|
|
|
/// <summary>
|
|
/// The AI staffing an open seat: identity (name, monogram) + matched skill atoms + autonomy +
|
|
/// the model config + docs. References Skills by key and the BYOK ApiConfig by id — never reaches
|
|
/// into those modules' tables. One agent per seat.
|
|
/// </summary>
|
|
internal sealed class Agent : Entity
|
|
{
|
|
public Guid SeatId { get; private set; }
|
|
public string Name { get; private set; } = null!;
|
|
public string? Monogram { get; private set; }
|
|
public Autonomy Autonomy { get; private set; }
|
|
public Guid ApiConfigId { get; private set; }
|
|
public Guid? FallbackApiConfigId { get; private set; }
|
|
public List<string> SkillKeys { get; private set; } = [];
|
|
|
|
/// <summary>Ids of the org's MCP servers this agent may use (resolved at run time).</summary>
|
|
public List<Guid> McpServerIds { get; private set; } = [];
|
|
public List<string> Docs { get; private set; } = [];
|
|
public DateTimeOffset CreatedAtUtc { get; private set; }
|
|
public DateTimeOffset UpdatedAtUtc { get; private set; }
|
|
|
|
private Agent()
|
|
{
|
|
}
|
|
|
|
public Agent(Guid seatId, DateTimeOffset createdAtUtc)
|
|
{
|
|
SeatId = seatId;
|
|
CreatedAtUtc = createdAtUtc;
|
|
UpdatedAtUtc = createdAtUtc;
|
|
}
|
|
|
|
public void Configure(
|
|
string name,
|
|
string? monogram,
|
|
Autonomy autonomy,
|
|
Guid apiConfigId,
|
|
Guid? fallbackApiConfigId,
|
|
List<string> skillKeys,
|
|
List<Guid> mcpServerIds,
|
|
List<string> docs,
|
|
DateTimeOffset nowUtc)
|
|
{
|
|
Name = name;
|
|
Monogram = monogram;
|
|
Autonomy = autonomy;
|
|
ApiConfigId = apiConfigId;
|
|
FallbackApiConfigId = fallbackApiConfigId;
|
|
SkillKeys = skillKeys;
|
|
McpServerIds = mcpServerIds;
|
|
Docs = docs;
|
|
UpdatedAtUtc = nowUtc;
|
|
}
|
|
}
|