fix(mcp) : return CallToolResult to prevent structuredContent serialization issue

Tools now return CallToolResult directly instead of Content arrays,
preventing the MCP SDK from auto-generating structuredContent as a
JSON array (which Claude Code rejects — expects a JSON object/record).
Also adds Accept header to test helpers and SSE response parsing.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Matthieu
2026-03-16 17:24:04 +01:00
parent f965affc94
commit add3a9a21f
60 changed files with 156 additions and 84 deletions

View File

@@ -10,6 +10,7 @@ use App\Repository\ConstructeurRepository;
use App\Repository\ModelTypeRepository;
use Doctrine\ORM\EntityManagerInterface;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
use Symfony\Bundle\SecurityBundle\Security;
#[McpTool(
@@ -37,7 +38,7 @@ class CreatePieceTool
string $prix = '',
string $modelTypeId = '',
array $constructeurIds = [],
): array {
): CallToolResult {
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
$piece = new Piece();

View File

@@ -8,6 +8,7 @@ use App\Mcp\Tool\McpToolHelper;
use App\Repository\PieceRepository;
use Doctrine\ORM\EntityManagerInterface;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
use Symfony\Bundle\SecurityBundle\Security;
#[McpTool(
@@ -24,7 +25,7 @@ class DeletePieceTool
private readonly Security $security,
) {}
public function __invoke(string $pieceId): array
public function __invoke(string $pieceId): CallToolResult
{
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');

View File

@@ -7,6 +7,7 @@ namespace App\Mcp\Tool\Piece;
use App\Mcp\Tool\McpToolHelper;
use App\Repository\PieceRepository;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
#[McpTool(
name: 'get_piece',
@@ -20,7 +21,7 @@ class GetPieceTool
private readonly PieceRepository $pieces,
) {}
public function __invoke(string $pieceId): array
public function __invoke(string $pieceId): CallToolResult
{
$piece = $this->pieces->find($pieceId);

View File

@@ -7,6 +7,7 @@ namespace App\Mcp\Tool\Piece;
use App\Mcp\Tool\McpToolHelper;
use App\Repository\PieceRepository;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
#[McpTool(
name: 'list_pieces',
@@ -20,7 +21,7 @@ class ListPiecesTool
private readonly PieceRepository $pieces,
) {}
public function __invoke(int $page = 1, int $limit = 30, string $search = ''): array
public function __invoke(int $page = 1, int $limit = 30, string $search = ''): CallToolResult
{
$p = $this->paginationParams($page, $limit);

View File

@@ -10,6 +10,7 @@ use App\Repository\ModelTypeRepository;
use App\Repository\PieceRepository;
use Doctrine\ORM\EntityManagerInterface;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
use Symfony\Bundle\SecurityBundle\Security;
#[McpTool(
@@ -39,7 +40,7 @@ class UpdatePieceTool
?string $prix = null,
?string $modelTypeId = null,
?array $constructeurIds = null,
): array {
): CallToolResult {
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
$piece = $this->pieces->find($pieceId);