Additive-only — no existing files modified. Services (php/Mcp/Services/): - CircuitBreaker (3-state, Cache::add trial lock) - DataRedactor (28 sensitive + 16 PII keys, partial-redact algorithm) - McpHealthService (YAML registry + JSON-RPC stdio ping protocolVersion 2024-11-05) - McpMetricsService (p50/p95/p99 linear interpolation) - McpWebhookDispatcher (mcp.tool.executed → WebhookEndpoints) - OpenApiGenerator (OpenAPI 3.0.3) - ToolRateLimiter (Cache::put first, Cache::increment after — no reset) - AgentSessionService (php/Mod/Mcp/Services/ namespace per spec) Transport (php/Mcp/Transport/): - McpContext (transport-agnostic callbacks) - Contracts/McpToolHandler interface Resources (php/Mcp/Resources/): - AppConfig, ContentResource, DatabaseSchema Config: php/resources/mcp/registry.yaml. Pest Feature tests _Good/_Bad/_Ugly per AX-10 for each new class. Co-authored-by: Codex <noreply@openai.com> Closes tasks.lthn.sh/view.php?id=842
52 lines
2 KiB
PHP
52 lines
2 KiB
PHP
<?php
|
|
|
|
// SPDX-License-Identifier: EUPL-1.2
|
|
|
|
declare(strict_types=1);
|
|
|
|
require_once dirname(__DIR__).'/Support/bootstrap.php';
|
|
|
|
mcpRequire('Mod/Mcp/Services/AgentSessionService.php');
|
|
|
|
use Core\Mod\Agentic\Models\AgentPlan;
|
|
use Core\Mod\Agentic\Models\AgentSession;
|
|
use Illuminate\Support\Facades\Cache;
|
|
use Mod\Mcp\Services\AgentSessionService;
|
|
|
|
beforeEach(function (): void {
|
|
Cache::flush();
|
|
});
|
|
|
|
test('AgentSessionService_start_Good_creates_and_caches_a_new_active_session', function (): void {
|
|
$workspace = createWorkspace();
|
|
$plan = AgentPlan::factory()->create(['workspace_id' => $workspace->id]);
|
|
$service = new AgentSessionService;
|
|
|
|
$session = $service->start('opus', $plan, $workspace->id, ['task' => 'draft']);
|
|
|
|
expect($session)->toBeInstanceOf(AgentSession::class)
|
|
->and($service->exists($session->session_id))->toBeTrue()
|
|
->and(Cache::get('mcp_session:active:'.$session->session_id)['workspace_id'])->toBe($workspace->id);
|
|
});
|
|
|
|
test('AgentSessionService_end_Bad_returns_null_for_unknown_sessions', function (): void {
|
|
$service = new AgentSessionService;
|
|
|
|
expect($service->end('missing-session'))->toBeNull()
|
|
->and($service->getHandoffContext('missing-session'))->toBeNull();
|
|
});
|
|
|
|
test('AgentSessionService_continueFrom_Ugly_hands_off_the_previous_session_and_starts_a_follow_up', function (): void {
|
|
$workspace = createWorkspace();
|
|
$plan = AgentPlan::factory()->create(['workspace_id' => $workspace->id]);
|
|
$service = new AgentSessionService;
|
|
$first = $service->start('opus', $plan, $workspace->id, ['task' => 'phase-one']);
|
|
|
|
$service->prepareHandoff($first->session_id, 'Done', ['next'], ['none'], ['checkpoint' => 'alpha']);
|
|
$second = $service->continueFrom($first->session_id, 'sonnet');
|
|
|
|
expect($second)->toBeInstanceOf(AgentSession::class)
|
|
->and($first->fresh()->status)->toBe(AgentSession::STATUS_HANDED_OFF)
|
|
->and($second->context_summary['continued_from'])->toBe($first->session_id)
|
|
->and($second->context_summary['previous_agent'])->toBe('opus');
|
|
});
|