fix(mcp) : accepter les arguments scalaires stringifiés (coercition string->int/bool)
Auto Tag Develop / tag (push) Successful in 11s
Auto Tag Develop / tag (push) Successful in 11s
Certains clients MCP sérialisent tous les arguments JSON-RPC en string (ex: "22" au lieu de 22). Le SDK valide les arguments contre le schéma JSON AVANT de les caster (CallToolHandler), donc un schéma integer strict rejetait "22" en 422 alors que ReferenceHandler::castArgumentType sait le coercer ensuite. CoercingSchemaGenerator enveloppe le SchemaGenerator du SDK et ajoute "string" aux types scalaires integer/number/boolean (et aux items de tableaux), de sorte que opis accepte la valeur stringifiée ; le type PHP réel du paramètre pilote toujours la coercition. Branché sur le builder MCP via McpSchemaGeneratorPass (enregistrée dans Kernel::build). Corrige le rejet 422 sur groupId/effortId/priorityId/statusId/etc. lors de l'appel des tools depuis Claude.
This commit is contained in:
@@ -0,0 +1,75 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Unit\Mcp;
|
||||
|
||||
use App\Mcp\Schema\CoercingSchemaGenerator;
|
||||
use App\Mcp\Tool\Task\CreateTaskTool;
|
||||
use App\Mcp\Tool\Task\ListTasksTool;
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use ReflectionMethod;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class CoercingSchemaGeneratorTest extends TestCase
|
||||
{
|
||||
private CoercingSchemaGenerator $generator;
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$this->generator = new CoercingSchemaGenerator();
|
||||
}
|
||||
|
||||
public function testNullableIntegerScalarAlsoAcceptsString(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(ListTasksTool::class, '__invoke'));
|
||||
|
||||
// ?int $projectId -> ["null","integer"] relaxed with "string".
|
||||
self::assertSame(['null', 'integer', 'string'], $schema['properties']['projectId']['type']);
|
||||
}
|
||||
|
||||
public function testRequiredIntegerScalarAlsoAcceptsString(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(ListTasksTool::class, '__invoke'));
|
||||
|
||||
// int $limit = 100 -> "integer" relaxed to ["integer","string"].
|
||||
self::assertSame(['integer', 'string'], $schema['properties']['limit']['type']);
|
||||
}
|
||||
|
||||
public function testBooleanScalarAlsoAcceptsString(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(CreateTaskTool::class, '__invoke'));
|
||||
|
||||
// ?bool $syncToCalendar -> ["boolean","null"] relaxed with "string".
|
||||
$type = $schema['properties']['syncToCalendar']['type'];
|
||||
self::assertContains('boolean', $type);
|
||||
self::assertContains('string', $type);
|
||||
self::assertContains('null', $type);
|
||||
}
|
||||
|
||||
public function testArrayItemTypeAlsoAcceptsString(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(CreateTaskTool::class, '__invoke'));
|
||||
|
||||
// int[] $tagIds -> items {type: integer} relaxed to {type: [integer, string]}.
|
||||
self::assertSame(['integer', 'string'], $schema['properties']['tagIds']['items']['type']);
|
||||
}
|
||||
|
||||
public function testStringScalarIsLeftUntouched(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(CreateTaskTool::class, '__invoke'));
|
||||
|
||||
// string $title stays a plain string (no spurious relaxation).
|
||||
self::assertSame('string', $schema['properties']['title']['type']);
|
||||
}
|
||||
|
||||
public function testArrayContainerTypeIsNotRelaxed(): void
|
||||
{
|
||||
$schema = $this->generator->generate(new ReflectionMethod(CreateTaskTool::class, '__invoke'));
|
||||
|
||||
// The array container itself must not gain "string".
|
||||
self::assertSame(['array', 'null'], $schema['properties']['tagIds']['type']);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user