Files
Inventory/src/Mcp/Tool/SearchInventoryTool.php
Matthieu add3a9a21f 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>
2026-03-16 17:24:04 +01:00

115 lines
4.1 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Mcp\Tool;
use App\Repository\ComposantRepository;
use App\Repository\ConstructeurRepository;
use App\Repository\MachineRepository;
use App\Repository\PieceRepository;
use App\Repository\ProductRepository;
use App\Repository\SiteRepository;
use Mcp\Capability\Attribute\McpTool;
use Mcp\Schema\Result\CallToolResult;
#[McpTool(
name: 'search_inventory',
description: 'Global search across all inventory entities (machines, pieces, composants, products, sites, constructeurs). Searches by name and reference (when available). Returns a flat list of matching results.',
)]
class SearchInventoryTool
{
use McpToolHelper;
private const ALLOWED_TYPES = ['machine', 'piece', 'composant', 'product', 'site', 'constructeur'];
public function __construct(
private readonly MachineRepository $machines,
private readonly PieceRepository $pieces,
private readonly ComposantRepository $composants,
private readonly ProductRepository $products,
private readonly SiteRepository $sites,
private readonly ConstructeurRepository $constructeurs,
) {}
public function __invoke(string $query, string $types = '', int $limit = 20): CallToolResult
{
$query = trim($query);
if ('' === $query) {
return $this->jsonResponse([]);
}
$limit = min(100, max(1, $limit));
$searchTypes = $this->resolveTypes($types);
$results = [];
foreach ($searchTypes as $type) {
$results = array_merge($results, match ($type) {
'machine' => $this->searchWithReference($this->machines, 'm', 'machine', $query),
'piece' => $this->searchWithReference($this->pieces, 'p', 'piece', $query),
'composant' => $this->searchWithReference($this->composants, 'c', 'composant', $query),
'product' => $this->searchWithReference($this->products, 'p', 'product', $query),
'site' => $this->searchNameOnly($this->sites, 's', 'site', $query),
'constructeur' => $this->searchNameOnly($this->constructeurs, 'c', 'constructeur', $query),
});
}
$results = array_slice($results, 0, $limit);
return $this->jsonResponse($results);
}
/**
* @return list<string>
*/
private function resolveTypes(string $types): array
{
if ('' === trim($types)) {
return self::ALLOWED_TYPES;
}
$requested = array_map('trim', explode(',', strtolower($types)));
return array_values(array_intersect($requested, self::ALLOWED_TYPES));
}
private function searchWithReference(object $repository, string $alias, string $type, string $search): array
{
$qb = $repository->createQueryBuilder($alias)
->select("{$alias}.id", "{$alias}.name", "{$alias}.reference")
->where("LOWER({$alias}.name) LIKE LOWER(:search)")
->orWhere("LOWER({$alias}.reference) LIKE LOWER(:search)")
->setParameter('search', "%{$search}%")
->orderBy("{$alias}.name", 'ASC')
;
$rows = $qb->getQuery()->getArrayResult();
return array_map(fn (array $row) => [
'type' => $type,
'id' => $row['id'],
'name' => $row['name'],
'reference' => $row['reference'] ?? null,
], $rows);
}
private function searchNameOnly(object $repository, string $alias, string $type, string $search): array
{
$qb = $repository->createQueryBuilder($alias)
->select("{$alias}.id", "{$alias}.name")
->where("LOWER({$alias}.name) LIKE LOWER(:search)")
->setParameter('search', "%{$search}%")
->orderBy("{$alias}.name", 'ASC')
;
$rows = $qb->getQuery()->getArrayResult();
return array_map(fn (array $row) => [
'type' => $type,
'id' => $row['id'],
'name' => $row['name'],
'reference' => null,
], $rows);
}
}