feat(mcp) : add business tools — search, history, comments, custom fields, documents, model types
- search_inventory: global search across all 6 entity types - get_entity_history + get_activity_log: audit trail access - 4 comment tools: list, create, resolve, unresolved count - 3 custom field tools: list values, upsert, delete - 2 document tools: list, delete (upload via REST only) - 6 model type tools: list, get, create, update, delete, sync - 69 MCP tests pass total Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
61
src/Mcp/Tool/ActivityLogTool.php
Normal file
61
src/Mcp/Tool/ActivityLogTool.php
Normal file
@@ -0,0 +1,61 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool;
|
||||
|
||||
use App\Repository\AuditLogRepository;
|
||||
use DateTimeInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'get_activity_log',
|
||||
description: 'Get the global activity log with optional filters. Returns paginated audit entries across all entities.',
|
||||
)]
|
||||
class ActivityLogTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly AuditLogRepository $auditLogs,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(int $page = 1, int $limit = 30, string $entityType = '', string $action = ''): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_VIEWER');
|
||||
|
||||
$p = $this->paginationParams($page, $limit);
|
||||
|
||||
$filters = [];
|
||||
if ('' !== $entityType) {
|
||||
$filters['entityType'] = $entityType;
|
||||
}
|
||||
if ('' !== $action) {
|
||||
$filters['action'] = $action;
|
||||
}
|
||||
|
||||
$result = $this->auditLogs->findAllPaginated($p['page'], $p['limit'], $filters);
|
||||
|
||||
$items = array_map(
|
||||
static function ($log) {
|
||||
$snapshot = $log->getSnapshot();
|
||||
|
||||
return [
|
||||
'id' => $log->getId(),
|
||||
'entityType' => $log->getEntityType(),
|
||||
'entityId' => $log->getEntityId(),
|
||||
'entityName' => $snapshot['name'] ?? null,
|
||||
'action' => $log->getAction(),
|
||||
'diff' => $log->getDiff(),
|
||||
'actorProfileId' => $log->getActorProfileId(),
|
||||
'createdAt' => $log->getCreatedAt()->format(DateTimeInterface::ATOM),
|
||||
];
|
||||
},
|
||||
$result['items'],
|
||||
);
|
||||
|
||||
return $this->paginatedResponse(array_values($items), $result['total'], $p['page'], $p['limit']);
|
||||
}
|
||||
}
|
||||
85
src/Mcp/Tool/Comment/CreateCommentTool.php
Normal file
85
src/Mcp/Tool/Comment/CreateCommentTool.php
Normal file
@@ -0,0 +1,85 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Comment;
|
||||
|
||||
use App\Entity\Comment;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ProfileRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'create_comment',
|
||||
description: 'Create a comment on an entity (machine, piece, composant, product…). Requires ROLE_VIEWER.',
|
||||
)]
|
||||
class CreateCommentTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
private readonly ProfileRepository $profiles,
|
||||
) {}
|
||||
|
||||
public function __invoke(
|
||||
string $content,
|
||||
string $entityType,
|
||||
string $entityId,
|
||||
string $entityName = '',
|
||||
): array {
|
||||
$this->requireRole($this->security, 'ROLE_VIEWER');
|
||||
|
||||
$content = trim($content);
|
||||
if ('' === $content) {
|
||||
$this->mcpError('Validation', 'Le contenu est requis.');
|
||||
}
|
||||
|
||||
$allowedTypes = ['machine', 'piece', 'composant', 'product', 'piece_category', 'component_category', 'product_category', 'machine_skeleton'];
|
||||
if (!in_array($entityType, $allowedTypes, true)) {
|
||||
$this->mcpError('Validation', "Type d'entité invalide : {$entityType}.");
|
||||
}
|
||||
|
||||
$entityId = trim($entityId);
|
||||
if ('' === $entityId) {
|
||||
$this->mcpError('Validation', "L'identifiant de l'entité est requis.");
|
||||
}
|
||||
|
||||
$user = $this->security->getUser();
|
||||
$profile = $user ? $this->profiles->find($user->getUserIdentifier()) : null;
|
||||
|
||||
$authorName = 'Inconnu';
|
||||
$authorId = '';
|
||||
if ($profile) {
|
||||
$authorId = $profile->getId();
|
||||
$authorName = trim(sprintf('%s %s', $profile->getFirstName(), $profile->getLastName()));
|
||||
if ('' === $authorName) {
|
||||
$authorName = $profile->getEmail() ?? 'Inconnu';
|
||||
}
|
||||
}
|
||||
|
||||
$comment = new Comment();
|
||||
$comment->setContent($content);
|
||||
$comment->setEntityType($entityType);
|
||||
$comment->setEntityId($entityId);
|
||||
$comment->setEntityName('' !== $entityName ? $entityName : null);
|
||||
$comment->setAuthorId($authorId);
|
||||
$comment->setAuthorName($authorName);
|
||||
|
||||
$this->em->persist($comment);
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse([
|
||||
'id' => $comment->getId(),
|
||||
'content' => $comment->getContent(),
|
||||
'entityType' => $comment->getEntityType(),
|
||||
'entityId' => $comment->getEntityId(),
|
||||
'entityName' => $comment->getEntityName(),
|
||||
'authorName' => $comment->getAuthorName(),
|
||||
'status' => $comment->getStatus(),
|
||||
]);
|
||||
}
|
||||
}
|
||||
67
src/Mcp/Tool/Comment/ListCommentsTool.php
Normal file
67
src/Mcp/Tool/Comment/ListCommentsTool.php
Normal file
@@ -0,0 +1,67 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Comment;
|
||||
|
||||
use App\Entity\Comment;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'list_comments',
|
||||
description: 'List comments for a given entity (machine, piece, composant, product…) with pagination. Filter by entityType and entityId.',
|
||||
)]
|
||||
class ListCommentsTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $entityType, string $entityId, int $page = 1, int $limit = 30): array
|
||||
{
|
||||
$p = $this->paginationParams($page, $limit);
|
||||
|
||||
$repo = $this->em->getRepository(Comment::class);
|
||||
|
||||
$total = (int) $repo->createQueryBuilder('c')
|
||||
->select('COUNT(c.id)')
|
||||
->andWhere('c.entityType = :entityType')
|
||||
->andWhere('c.entityId = :entityId')
|
||||
->setParameter('entityType', $entityType)
|
||||
->setParameter('entityId', $entityId)
|
||||
->getQuery()
|
||||
->getSingleScalarResult()
|
||||
;
|
||||
|
||||
$items = $repo->createQueryBuilder('c')
|
||||
->select(
|
||||
'c.id',
|
||||
'c.content',
|
||||
'c.entityType',
|
||||
'c.entityId',
|
||||
'c.entityName',
|
||||
'c.authorName',
|
||||
'c.authorId',
|
||||
'c.status',
|
||||
'c.resolvedByName',
|
||||
'c.resolvedAt',
|
||||
'c.createdAt',
|
||||
)
|
||||
->andWhere('c.entityType = :entityType')
|
||||
->andWhere('c.entityId = :entityId')
|
||||
->setParameter('entityType', $entityType)
|
||||
->setParameter('entityId', $entityId)
|
||||
->orderBy('c.createdAt', 'DESC')
|
||||
->setFirstResult($p['offset'])
|
||||
->setMaxResults($p['limit'])
|
||||
->getQuery()
|
||||
->getArrayResult()
|
||||
;
|
||||
|
||||
return $this->paginatedResponse($items, $total, $p['page'], $p['limit']);
|
||||
}
|
||||
}
|
||||
65
src/Mcp/Tool/Comment/ResolveCommentTool.php
Normal file
65
src/Mcp/Tool/Comment/ResolveCommentTool.php
Normal file
@@ -0,0 +1,65 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Comment;
|
||||
|
||||
use App\Entity\Comment;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ProfileRepository;
|
||||
use DateTimeImmutable;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'resolve_comment',
|
||||
description: 'Mark a comment as resolved. Sets status to "resolved" with resolver info. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class ResolveCommentTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
private readonly ProfileRepository $profiles,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $commentId): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$comment = $this->em->getRepository(Comment::class)->find($commentId);
|
||||
if (!$comment) {
|
||||
$this->mcpError('NotFound', 'Commentaire introuvable.');
|
||||
}
|
||||
|
||||
$user = $this->security->getUser();
|
||||
$profile = $user ? $this->profiles->find($user->getUserIdentifier()) : null;
|
||||
|
||||
$resolverName = 'Inconnu';
|
||||
$resolverId = null;
|
||||
if ($profile) {
|
||||
$resolverId = $profile->getId();
|
||||
$resolverName = trim(sprintf('%s %s', $profile->getFirstName(), $profile->getLastName()));
|
||||
if ('' === $resolverName) {
|
||||
$resolverName = $profile->getEmail() ?? 'Inconnu';
|
||||
}
|
||||
}
|
||||
|
||||
$comment->setStatus('resolved');
|
||||
$comment->setResolvedById($resolverId);
|
||||
$comment->setResolvedByName($resolverName);
|
||||
$comment->setResolvedAt(new DateTimeImmutable());
|
||||
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse([
|
||||
'id' => $comment->getId(),
|
||||
'status' => $comment->getStatus(),
|
||||
'resolvedById' => $comment->getResolvedById(),
|
||||
'resolvedByName' => $comment->getResolvedByName(),
|
||||
]);
|
||||
}
|
||||
}
|
||||
37
src/Mcp/Tool/Comment/UnresolvedCountTool.php
Normal file
37
src/Mcp/Tool/Comment/UnresolvedCountTool.php
Normal file
@@ -0,0 +1,37 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Comment;
|
||||
|
||||
use App\Entity\Comment;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'get_unresolved_comments_count',
|
||||
description: 'Get the total count of unresolved (open) comments across all entities.',
|
||||
)]
|
||||
class UnresolvedCountTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
) {}
|
||||
|
||||
public function __invoke(): array
|
||||
{
|
||||
$count = (int) $this->em->getRepository(Comment::class)
|
||||
->createQueryBuilder('c')
|
||||
->select('COUNT(c.id)')
|
||||
->andWhere('c.status = :status')
|
||||
->setParameter('status', 'open')
|
||||
->getQuery()
|
||||
->getSingleScalarResult()
|
||||
;
|
||||
|
||||
return $this->jsonResponse(['count' => $count]);
|
||||
}
|
||||
}
|
||||
41
src/Mcp/Tool/CustomField/DeleteCustomFieldValueTool.php
Normal file
41
src/Mcp/Tool/CustomField/DeleteCustomFieldValueTool.php
Normal file
@@ -0,0 +1,41 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\CustomField;
|
||||
|
||||
use App\Entity\CustomFieldValue;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'delete_custom_field_value',
|
||||
description: 'Delete a custom field value by ID. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class DeleteCustomFieldValueTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $customFieldValueId): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$cfv = $this->em->getRepository(CustomFieldValue::class)->find($customFieldValueId);
|
||||
|
||||
if (null === $cfv) {
|
||||
$this->mcpError('not_found', "CustomFieldValue not found: {$customFieldValueId}");
|
||||
}
|
||||
|
||||
$this->em->remove($cfv);
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse(['deleted' => true, 'id' => $customFieldValueId]);
|
||||
}
|
||||
}
|
||||
61
src/Mcp/Tool/CustomField/ListCustomFieldValuesTool.php
Normal file
61
src/Mcp/Tool/CustomField/ListCustomFieldValuesTool.php
Normal file
@@ -0,0 +1,61 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\CustomField;
|
||||
|
||||
use App\Entity\CustomFieldValue;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'list_custom_field_values',
|
||||
description: 'List all custom field values for a given entity (machine, composant, piece or product). Returns each value with its custom field name and type.',
|
||||
)]
|
||||
class ListCustomFieldValuesTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
private const ALLOWED_TYPES = ['machine', 'composant', 'piece', 'product'];
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $entityType, string $entityId): array
|
||||
{
|
||||
$entityType = strtolower($entityType);
|
||||
|
||||
if (!in_array($entityType, self::ALLOWED_TYPES, true)) {
|
||||
$this->mcpError('validation', "entityType must be one of: machine, composant, piece, product. Got '{$entityType}'.");
|
||||
}
|
||||
|
||||
$rows = $this->em->createQueryBuilder()
|
||||
->select(
|
||||
'cfv.id',
|
||||
'cfv.value',
|
||||
'cf.id AS customFieldId',
|
||||
'cf.name AS customFieldName',
|
||||
'cf.type AS customFieldType',
|
||||
'cf.required AS customFieldRequired',
|
||||
'cfv.createdAt',
|
||||
'cfv.updatedAt',
|
||||
)
|
||||
->from(CustomFieldValue::class, 'cfv')
|
||||
->join('cfv.customField', 'cf')
|
||||
->where("IDENTITY(cfv.{$entityType}) = :entityId")
|
||||
->setParameter('entityId', $entityId)
|
||||
->orderBy('cf.name', 'ASC')
|
||||
->getQuery()
|
||||
->getArrayResult()
|
||||
;
|
||||
|
||||
return $this->jsonResponse([
|
||||
'entityType' => $entityType,
|
||||
'entityId' => $entityId,
|
||||
'values' => $rows,
|
||||
'total' => count($rows),
|
||||
]);
|
||||
}
|
||||
}
|
||||
114
src/Mcp/Tool/CustomField/UpsertCustomFieldValuesTool.php
Normal file
114
src/Mcp/Tool/CustomField/UpsertCustomFieldValuesTool.php
Normal file
@@ -0,0 +1,114 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\CustomField;
|
||||
|
||||
use App\Entity\Composant;
|
||||
use App\Entity\CustomField;
|
||||
use App\Entity\CustomFieldValue;
|
||||
use App\Entity\Machine;
|
||||
use App\Entity\Piece;
|
||||
use App\Entity\Product;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'upsert_custom_field_values',
|
||||
description: 'Create or update custom field values for a given entity. Each entry in the fields array needs customFieldId and value. If a value already exists for that custom field + entity, it is updated; otherwise a new one is created. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class UpsertCustomFieldValuesTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
private const ALLOWED_TYPES = ['machine', 'composant', 'piece', 'product'];
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $entityType, string $entityId, array $fields): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$entityType = strtolower($entityType);
|
||||
|
||||
if (!in_array($entityType, self::ALLOWED_TYPES, true)) {
|
||||
$this->mcpError('validation', "entityType must be one of: machine, composant, piece, product. Got '{$entityType}'.");
|
||||
}
|
||||
|
||||
$entityClass = match ($entityType) {
|
||||
'machine' => Machine::class,
|
||||
'composant' => Composant::class,
|
||||
'piece' => Piece::class,
|
||||
'product' => Product::class,
|
||||
};
|
||||
|
||||
$entity = $this->em->getRepository($entityClass)->find($entityId);
|
||||
|
||||
if (null === $entity) {
|
||||
$this->mcpError('not_found', ucfirst($entityType)." not found: {$entityId}");
|
||||
}
|
||||
|
||||
$results = [];
|
||||
|
||||
foreach ($fields as $fieldEntry) {
|
||||
$customFieldId = $fieldEntry['customFieldId'] ?? null;
|
||||
$value = $fieldEntry['value'] ?? '';
|
||||
|
||||
if (null === $customFieldId) {
|
||||
$this->mcpError('validation', 'Each field entry must have a customFieldId.');
|
||||
}
|
||||
|
||||
$customField = $this->em->getRepository(CustomField::class)->find($customFieldId);
|
||||
|
||||
if (null === $customField) {
|
||||
$this->mcpError('not_found', "CustomField not found: {$customFieldId}");
|
||||
}
|
||||
|
||||
$existing = $this->em->getRepository(CustomFieldValue::class)->findOneBy([
|
||||
'customField' => $customField,
|
||||
$entityType => $entity,
|
||||
]);
|
||||
|
||||
if (null !== $existing) {
|
||||
$existing->setValue((string) $value);
|
||||
$results[] = [
|
||||
'id' => $existing->getId(),
|
||||
'customFieldId' => $customField->getId(),
|
||||
'value' => (string) $value,
|
||||
'action' => 'updated',
|
||||
];
|
||||
} else {
|
||||
$cfv = new CustomFieldValue();
|
||||
$cfv->setCustomField($customField);
|
||||
$cfv->setValue((string) $value);
|
||||
|
||||
$setter = 'set'.ucfirst($entityType);
|
||||
$cfv->{$setter}($entity);
|
||||
|
||||
$this->em->persist($cfv);
|
||||
$this->em->flush();
|
||||
|
||||
$results[] = [
|
||||
'id' => $cfv->getId(),
|
||||
'customFieldId' => $customField->getId(),
|
||||
'value' => (string) $value,
|
||||
'action' => 'created',
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse([
|
||||
'entityType' => $entityType,
|
||||
'entityId' => $entityId,
|
||||
'results' => $results,
|
||||
'total' => count($results),
|
||||
]);
|
||||
}
|
||||
}
|
||||
42
src/Mcp/Tool/Document/DeleteDocumentTool.php
Normal file
42
src/Mcp/Tool/Document/DeleteDocumentTool.php
Normal file
@@ -0,0 +1,42 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Document;
|
||||
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\DocumentRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'delete_document',
|
||||
description: 'Delete a document by ID. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class DeleteDocumentTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly DocumentRepository $documents,
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $documentId): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$document = $this->documents->find($documentId);
|
||||
|
||||
if (!$document) {
|
||||
$this->mcpError('not_found', "Document not found: {$documentId}");
|
||||
}
|
||||
|
||||
$this->em->remove($document);
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse(['deleted' => true, 'id' => $documentId]);
|
||||
}
|
||||
}
|
||||
62
src/Mcp/Tool/Document/ListDocumentsTool.php
Normal file
62
src/Mcp/Tool/Document/ListDocumentsTool.php
Normal file
@@ -0,0 +1,62 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\Document;
|
||||
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\DocumentRepository;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'list_documents',
|
||||
description: 'List documents attached to a given entity. entityType must be one of: site, machine, composant, piece, product.',
|
||||
)]
|
||||
class ListDocumentsTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
private const ENTITY_FIELDS = [
|
||||
'site' => 'site',
|
||||
'machine' => 'machine',
|
||||
'composant' => 'composant',
|
||||
'piece' => 'piece',
|
||||
'product' => 'product',
|
||||
];
|
||||
|
||||
public function __construct(
|
||||
private readonly DocumentRepository $documents,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $entityType, string $entityId): array
|
||||
{
|
||||
if (!isset(self::ENTITY_FIELDS[$entityType])) {
|
||||
$this->mcpError('validation', "Invalid entityType '{$entityType}'. Must be one of: site, machine, composant, piece, product.");
|
||||
}
|
||||
|
||||
$field = self::ENTITY_FIELDS[$entityType];
|
||||
|
||||
$docs = $this->documents->findBy([$field => $entityId], ['createdAt' => 'DESC']);
|
||||
|
||||
$items = [];
|
||||
foreach ($docs as $doc) {
|
||||
$items[] = [
|
||||
'id' => $doc->getId(),
|
||||
'name' => $doc->getName(),
|
||||
'filename' => $doc->getFilename(),
|
||||
'fileUrl' => '/api/documents/'.$doc->getId().'/file',
|
||||
'downloadUrl' => '/api/documents/'.$doc->getId().'/download',
|
||||
'mimeType' => $doc->getMimeType(),
|
||||
'size' => $doc->getSize(),
|
||||
'createdAt' => $doc->getCreatedAt()->format('Y-m-d H:i:s'),
|
||||
];
|
||||
}
|
||||
|
||||
return $this->jsonResponse([
|
||||
'entityType' => $entityType,
|
||||
'entityId' => $entityId,
|
||||
'items' => $items,
|
||||
'total' => count($items),
|
||||
]);
|
||||
}
|
||||
}
|
||||
57
src/Mcp/Tool/EntityHistoryTool.php
Normal file
57
src/Mcp/Tool/EntityHistoryTool.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool;
|
||||
|
||||
use App\Repository\AuditLogRepository;
|
||||
use DateTimeInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'get_entity_history',
|
||||
description: 'Get the audit history for a specific entity (machine, piece, composant, product). Returns list of changes with diffs.',
|
||||
)]
|
||||
class EntityHistoryTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
private const VALID_TYPES = ['machine', 'piece', 'composant', 'product'];
|
||||
|
||||
public function __construct(
|
||||
private readonly AuditLogRepository $auditLogs,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $entityType, string $entityId): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_VIEWER');
|
||||
|
||||
if (!in_array($entityType, self::VALID_TYPES, true)) {
|
||||
$this->mcpError('Validation', sprintf(
|
||||
'Invalid entityType "%s". Must be one of: %s',
|
||||
$entityType,
|
||||
implode(', ', self::VALID_TYPES),
|
||||
));
|
||||
}
|
||||
|
||||
$logs = $this->auditLogs->findEntityHistory($entityType, $entityId, 200);
|
||||
|
||||
$items = array_map(
|
||||
static fn ($log) => [
|
||||
'id' => $log->getId(),
|
||||
'action' => $log->getAction(),
|
||||
'diff' => $log->getDiff(),
|
||||
'actorProfileId' => $log->getActorProfileId(),
|
||||
'createdAt' => $log->getCreatedAt()->format(DateTimeInterface::ATOM),
|
||||
],
|
||||
$logs,
|
||||
);
|
||||
|
||||
return $this->jsonResponse([
|
||||
'items' => array_values($items),
|
||||
'total' => count($items),
|
||||
]);
|
||||
}
|
||||
}
|
||||
57
src/Mcp/Tool/ModelType/CreateModelTypeTool.php
Normal file
57
src/Mcp/Tool/ModelType/CreateModelTypeTool.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Entity\ModelType;
|
||||
use App\Enum\ModelCategory;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'create_model_type',
|
||||
description: 'Create a new model type. Category must be one of: composant, piece, product. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class CreateModelTypeTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $name, string $category, string $code = ''): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$enumCategory = match (strtolower($category)) {
|
||||
'composant', 'component' => ModelCategory::COMPONENT,
|
||||
'piece' => ModelCategory::PIECE,
|
||||
'product' => ModelCategory::PRODUCT,
|
||||
default => null,
|
||||
};
|
||||
|
||||
if (null === $enumCategory) {
|
||||
$this->mcpError('validation', "Invalid category '{$category}'. Must be one of: composant, piece, product.");
|
||||
}
|
||||
|
||||
$mt = new ModelType();
|
||||
$mt->setName($name);
|
||||
$mt->setCategory($enumCategory);
|
||||
$mt->setCode('' !== $code ? $code : strtoupper(substr(str_replace(' ', '-', $name), 0, 20)).'-'.bin2hex(random_bytes(3)));
|
||||
|
||||
$this->em->persist($mt);
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse([
|
||||
'id' => $mt->getId(),
|
||||
'name' => $mt->getName(),
|
||||
'code' => $mt->getCode(),
|
||||
'category' => $mt->getCategory()->value,
|
||||
]);
|
||||
}
|
||||
}
|
||||
42
src/Mcp/Tool/ModelType/DeleteModelTypeTool.php
Normal file
42
src/Mcp/Tool/ModelType/DeleteModelTypeTool.php
Normal file
@@ -0,0 +1,42 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ModelTypeRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'delete_model_type',
|
||||
description: 'Delete a model type by ID. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class DeleteModelTypeTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly ModelTypeRepository $modelTypes,
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $modelTypeId): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$mt = $this->modelTypes->find($modelTypeId);
|
||||
|
||||
if (!$mt) {
|
||||
$this->mcpError('not_found', "ModelType not found: {$modelTypeId}");
|
||||
}
|
||||
|
||||
$this->em->remove($mt);
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse(['deleted' => true, 'id' => $modelTypeId]);
|
||||
}
|
||||
}
|
||||
78
src/Mcp/Tool/ModelType/GetModelTypeTool.php
Normal file
78
src/Mcp/Tool/ModelType/GetModelTypeTool.php
Normal file
@@ -0,0 +1,78 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ModelTypeRepository;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'get_model_type',
|
||||
description: 'Get a single model type by ID with full details including skeleton requirements.',
|
||||
)]
|
||||
class GetModelTypeTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly ModelTypeRepository $modelTypes,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $modelTypeId): array
|
||||
{
|
||||
$mt = $this->modelTypes->find($modelTypeId);
|
||||
|
||||
if (!$mt) {
|
||||
$this->mcpError('not_found', "ModelType not found: {$modelTypeId}");
|
||||
}
|
||||
|
||||
$skeletonPieces = [];
|
||||
foreach ($mt->getSkeletonPieceRequirements() as $req) {
|
||||
$skeletonPieces[] = [
|
||||
'id' => $req->getId(),
|
||||
'typePieceId' => $req->getTypePiece()->getId(),
|
||||
'typePiece' => $req->getTypePiece()->getName(),
|
||||
'position' => $req->getPosition(),
|
||||
];
|
||||
}
|
||||
|
||||
$skeletonProducts = [];
|
||||
foreach ($mt->getSkeletonProductRequirements() as $req) {
|
||||
$skeletonProducts[] = [
|
||||
'id' => $req->getId(),
|
||||
'typeProductId' => $req->getTypeProduct()->getId(),
|
||||
'typeProduct' => $req->getTypeProduct()->getName(),
|
||||
'familyCode' => $req->getFamilyCode(),
|
||||
'position' => $req->getPosition(),
|
||||
];
|
||||
}
|
||||
|
||||
$skeletonSubcomponents = [];
|
||||
foreach ($mt->getSkeletonSubcomponentRequirements() as $req) {
|
||||
$skeletonSubcomponents[] = [
|
||||
'id' => $req->getId(),
|
||||
'alias' => $req->getAlias(),
|
||||
'familyCode' => $req->getFamilyCode(),
|
||||
'typeComposantId' => $req->getTypeComposant()?->getId(),
|
||||
'typeComposant' => $req->getTypeComposant()?->getName(),
|
||||
'position' => $req->getPosition(),
|
||||
];
|
||||
}
|
||||
|
||||
return $this->jsonResponse([
|
||||
'id' => $mt->getId(),
|
||||
'name' => $mt->getName(),
|
||||
'code' => $mt->getCode(),
|
||||
'category' => $mt->getCategory()->value,
|
||||
'notes' => $mt->getNotes(),
|
||||
'description' => $mt->getDescription(),
|
||||
'skeletonPieceRequirements' => $skeletonPieces,
|
||||
'skeletonProductRequirements' => $skeletonProducts,
|
||||
'skeletonSubcomponentRequirements' => $skeletonSubcomponents,
|
||||
'createdAt' => $mt->getCreatedAt()->format('Y-m-d H:i:s'),
|
||||
'updatedAt' => $mt->getUpdatedAt()->format('Y-m-d H:i:s'),
|
||||
]);
|
||||
}
|
||||
}
|
||||
79
src/Mcp/Tool/ModelType/ListModelTypesTool.php
Normal file
79
src/Mcp/Tool/ModelType/ListModelTypesTool.php
Normal file
@@ -0,0 +1,79 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Enum\ModelCategory;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ModelTypeRepository;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
|
||||
#[McpTool(
|
||||
name: 'list_model_types',
|
||||
description: 'List model types with pagination. Filterable by category (composant, piece, product).',
|
||||
)]
|
||||
class ListModelTypesTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly ModelTypeRepository $modelTypes,
|
||||
) {}
|
||||
|
||||
public function __invoke(int $page = 1, int $limit = 30, string $category = ''): array
|
||||
{
|
||||
$p = $this->paginationParams($page, $limit);
|
||||
|
||||
$countQb = $this->modelTypes->createQueryBuilder('mt')
|
||||
->select('COUNT(mt.id)')
|
||||
;
|
||||
|
||||
$qb = $this->modelTypes->createQueryBuilder('mt')
|
||||
->select('mt.id', 'mt.name', 'mt.code', 'mt.category')
|
||||
->orderBy('mt.name', 'ASC')
|
||||
;
|
||||
|
||||
if ('' !== $category) {
|
||||
$enumCategory = $this->resolveCategory($category);
|
||||
|
||||
if (null === $enumCategory) {
|
||||
$this->mcpError('validation', "Invalid category '{$category}'. Must be one of: composant, piece, product.");
|
||||
}
|
||||
|
||||
$countQb->andWhere('mt.category = :category')
|
||||
->setParameter('category', $enumCategory)
|
||||
;
|
||||
$qb->andWhere('mt.category = :category')
|
||||
->setParameter('category', $enumCategory)
|
||||
;
|
||||
}
|
||||
|
||||
$total = (int) $countQb->getQuery()->getSingleScalarResult();
|
||||
|
||||
$items = $qb->setFirstResult($p['offset'])
|
||||
->setMaxResults($p['limit'])
|
||||
->getQuery()
|
||||
->getArrayResult()
|
||||
;
|
||||
|
||||
// Convert ModelCategory enum to string in results
|
||||
foreach ($items as &$item) {
|
||||
if ($item['category'] instanceof ModelCategory) {
|
||||
$item['category'] = $item['category']->value;
|
||||
}
|
||||
}
|
||||
|
||||
return $this->paginatedResponse($items, $total, $p['page'], $p['limit']);
|
||||
}
|
||||
|
||||
private function resolveCategory(string $category): ?ModelCategory
|
||||
{
|
||||
return match (strtolower($category)) {
|
||||
'composant', 'component' => ModelCategory::COMPONENT,
|
||||
'piece' => ModelCategory::PIECE,
|
||||
'product' => ModelCategory::PRODUCT,
|
||||
default => null,
|
||||
};
|
||||
}
|
||||
}
|
||||
67
src/Mcp/Tool/ModelType/SyncModelTypeTool.php
Normal file
67
src/Mcp/Tool/ModelType/SyncModelTypeTool.php
Normal file
@@ -0,0 +1,67 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\DTO\SyncConfirmation;
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ModelTypeRepository;
|
||||
use App\Service\ModelTypeSyncService;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'sync_model_type',
|
||||
description: 'Preview or sync a model type structure. Action "preview" shows what would change. Action "sync" applies the pending structure. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class SyncModelTypeTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly ModelTypeRepository $modelTypes,
|
||||
private readonly ModelTypeSyncService $syncService,
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(
|
||||
string $modelTypeId,
|
||||
string $action,
|
||||
?array $structure = null,
|
||||
bool $confirmDeletions = false,
|
||||
bool $confirmTypeChanges = false,
|
||||
): array {
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
if (!in_array($action, ['preview', 'sync'], true)) {
|
||||
$this->mcpError('validation', "Invalid action '{$action}'. Must be 'preview' or 'sync'.");
|
||||
}
|
||||
|
||||
$mt = $this->modelTypes->find($modelTypeId);
|
||||
|
||||
if (!$mt) {
|
||||
$this->mcpError('not_found', "ModelType not found: {$modelTypeId}");
|
||||
}
|
||||
|
||||
if ('preview' === $action) {
|
||||
$result = $this->syncService->preview($mt, $structure ?? []);
|
||||
|
||||
return $this->jsonResponse($result->jsonSerialize());
|
||||
}
|
||||
|
||||
// sync action
|
||||
$confirmation = new SyncConfirmation(
|
||||
confirmDeletions: $confirmDeletions,
|
||||
confirmTypeChanges: $confirmTypeChanges,
|
||||
);
|
||||
|
||||
$result = $this->em->wrapInTransaction(function () use ($mt, $confirmation) {
|
||||
return $this->syncService->execute($mt, $confirmation);
|
||||
});
|
||||
|
||||
return $this->jsonResponse($result->jsonSerialize());
|
||||
}
|
||||
}
|
||||
53
src/Mcp/Tool/ModelType/UpdateModelTypeTool.php
Normal file
53
src/Mcp/Tool/ModelType/UpdateModelTypeTool.php
Normal file
@@ -0,0 +1,53 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Mcp\Tool\McpToolHelper;
|
||||
use App\Repository\ModelTypeRepository;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Mcp\Capability\Attribute\McpTool;
|
||||
use Symfony\Bundle\SecurityBundle\Security;
|
||||
|
||||
#[McpTool(
|
||||
name: 'update_model_type',
|
||||
description: 'Update an existing model type. Only provided fields are changed. Requires ROLE_GESTIONNAIRE.',
|
||||
)]
|
||||
class UpdateModelTypeTool
|
||||
{
|
||||
use McpToolHelper;
|
||||
|
||||
public function __construct(
|
||||
private readonly ModelTypeRepository $modelTypes,
|
||||
private readonly EntityManagerInterface $em,
|
||||
private readonly Security $security,
|
||||
) {}
|
||||
|
||||
public function __invoke(string $modelTypeId, ?string $name = null, ?string $code = null): array
|
||||
{
|
||||
$this->requireRole($this->security, 'ROLE_GESTIONNAIRE');
|
||||
|
||||
$mt = $this->modelTypes->find($modelTypeId);
|
||||
|
||||
if (!$mt) {
|
||||
$this->mcpError('not_found', "ModelType not found: {$modelTypeId}");
|
||||
}
|
||||
|
||||
if (null !== $name) {
|
||||
$mt->setName($name);
|
||||
}
|
||||
if (null !== $code) {
|
||||
$mt->setCode($code);
|
||||
}
|
||||
|
||||
$this->em->flush();
|
||||
|
||||
return $this->jsonResponse([
|
||||
'id' => $mt->getId(),
|
||||
'name' => $mt->getName(),
|
||||
'code' => $mt->getCode(),
|
||||
'category' => $mt->getCategory()->value,
|
||||
]);
|
||||
}
|
||||
}
|
||||
113
src/Mcp/Tool/SearchInventoryTool.php
Normal file
113
src/Mcp/Tool/SearchInventoryTool.php
Normal file
@@ -0,0 +1,113 @@
|
||||
<?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;
|
||||
|
||||
#[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): array
|
||||
{
|
||||
$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);
|
||||
}
|
||||
}
|
||||
98
tests/Mcp/Tool/Comment/CommentsToolTest.php
Normal file
98
tests/Mcp/Tool/Comment/CommentsToolTest.php
Normal file
@@ -0,0 +1,98 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool\Comment;
|
||||
|
||||
use App\Entity\Comment;
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class CommentsToolTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testListComments(): void
|
||||
{
|
||||
$entityId = 'entity-'.uniqid();
|
||||
$this->createComment('First comment', 'machine', $entityId);
|
||||
$this->createComment('Second comment', 'machine', $entityId);
|
||||
$this->createComment('Other entity', 'machine', 'other-id');
|
||||
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'list_comments', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $entityId,
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertSame(2, $data['_parsed']['total']);
|
||||
$this->assertCount(2, $data['_parsed']['items']);
|
||||
}
|
||||
|
||||
public function testCreateComment(): void
|
||||
{
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'create_comment', [
|
||||
'content' => 'A new comment',
|
||||
'entityType' => 'machine',
|
||||
'entityId' => 'some-machine-id',
|
||||
'entityName' => 'Machine Alpha',
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertNotEmpty($data['_parsed']['id']);
|
||||
$this->assertSame('A new comment', $data['_parsed']['content']);
|
||||
$this->assertSame('machine', $data['_parsed']['entityType']);
|
||||
$this->assertSame('open', $data['_parsed']['status']);
|
||||
$this->assertSame('Machine Alpha', $data['_parsed']['entityName']);
|
||||
}
|
||||
|
||||
public function testResolveComment(): void
|
||||
{
|
||||
$comment = $this->createComment('To resolve', 'piece', 'piece-123');
|
||||
$session = $this->createMcpClient('ROLE_GESTIONNAIRE');
|
||||
|
||||
$data = $this->callMcpTool($session, 'resolve_comment', [
|
||||
'commentId' => $comment->getId(),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertSame('resolved', $data['_parsed']['status']);
|
||||
$this->assertNotEmpty($data['_parsed']['resolvedByName']);
|
||||
}
|
||||
|
||||
public function testUnresolvedCount(): void
|
||||
{
|
||||
$entityId = 'entity-'.uniqid();
|
||||
$this->createComment('Open 1', 'machine', $entityId, 'open');
|
||||
$this->createComment('Open 2', 'machine', $entityId, 'open');
|
||||
$this->createComment('Resolved', 'machine', $entityId, 'resolved');
|
||||
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'get_unresolved_comments_count');
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertGreaterThanOrEqual(2, $data['_parsed']['count']);
|
||||
}
|
||||
|
||||
private function createComment(string $content, string $entityType, string $entityId, string $status = 'open'): Comment
|
||||
{
|
||||
$comment = new Comment();
|
||||
$comment->setContent($content);
|
||||
$comment->setEntityType($entityType);
|
||||
$comment->setEntityId($entityId);
|
||||
$comment->setAuthorId('test-author-id');
|
||||
$comment->setAuthorName('Test Author');
|
||||
$comment->setStatus($status);
|
||||
|
||||
$em = $this->getEntityManager();
|
||||
$em->persist($comment);
|
||||
$em->flush();
|
||||
|
||||
return $comment;
|
||||
}
|
||||
}
|
||||
101
tests/Mcp/Tool/CustomField/CustomFieldToolsTest.php
Normal file
101
tests/Mcp/Tool/CustomField/CustomFieldToolsTest.php
Normal file
@@ -0,0 +1,101 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool\CustomField;
|
||||
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class CustomFieldToolsTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testListCustomFieldValues(): void
|
||||
{
|
||||
$machine = $this->createMachine(name: 'Machine CF');
|
||||
$customField = $this->createCustomField(name: 'Serial Number', type: 'text', machine: $machine);
|
||||
$this->createCustomFieldValue(customField: $customField, value: 'SN-12345', machine: $machine);
|
||||
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'list_custom_field_values', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $machine->getId(),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$parsed = $data['_parsed'];
|
||||
$this->assertSame('machine', $parsed['entityType']);
|
||||
$this->assertSame($machine->getId(), $parsed['entityId']);
|
||||
$this->assertSame(1, $parsed['total']);
|
||||
$this->assertSame('SN-12345', $parsed['values'][0]['value']);
|
||||
$this->assertSame('Serial Number', $parsed['values'][0]['customFieldName']);
|
||||
$this->assertSame('text', $parsed['values'][0]['customFieldType']);
|
||||
}
|
||||
|
||||
public function testUpsertCustomFieldValues(): void
|
||||
{
|
||||
$machine = $this->createMachine(name: 'Machine Upsert');
|
||||
$customField = $this->createCustomField(name: 'Voltage', type: 'text', machine: $machine);
|
||||
|
||||
$session = $this->createMcpClient('ROLE_GESTIONNAIRE');
|
||||
|
||||
// Create
|
||||
$data = $this->callMcpTool($session, 'upsert_custom_field_values', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $machine->getId(),
|
||||
'fields' => [
|
||||
['customFieldId' => $customField->getId(), 'value' => '220V'],
|
||||
],
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$parsed = $data['_parsed'];
|
||||
$this->assertSame(1, $parsed['total']);
|
||||
$this->assertSame('created', $parsed['results'][0]['action']);
|
||||
$this->assertSame('220V', $parsed['results'][0]['value']);
|
||||
|
||||
$createdId = $parsed['results'][0]['id'];
|
||||
|
||||
// Update (upsert same field)
|
||||
$data = $this->callMcpTool($session, 'upsert_custom_field_values', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $machine->getId(),
|
||||
'fields' => [
|
||||
['customFieldId' => $customField->getId(), 'value' => '380V'],
|
||||
],
|
||||
]);
|
||||
|
||||
$parsed = $data['_parsed'];
|
||||
$this->assertSame('updated', $parsed['results'][0]['action']);
|
||||
$this->assertSame('380V', $parsed['results'][0]['value']);
|
||||
$this->assertSame($createdId, $parsed['results'][0]['id']);
|
||||
}
|
||||
|
||||
public function testDeleteCustomFieldValue(): void
|
||||
{
|
||||
$machine = $this->createMachine(name: 'Machine Delete CF');
|
||||
$customField = $this->createCustomField(name: 'Weight', type: 'text', machine: $machine);
|
||||
$cfv = $this->createCustomFieldValue(customField: $customField, value: '150kg', machine: $machine);
|
||||
|
||||
$session = $this->createMcpClient('ROLE_GESTIONNAIRE');
|
||||
|
||||
$data = $this->callMcpTool($session, 'delete_custom_field_value', [
|
||||
'customFieldValueId' => $cfv->getId(),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$parsed = $data['_parsed'];
|
||||
$this->assertTrue($parsed['deleted']);
|
||||
$this->assertSame($cfv->getId(), $parsed['id']);
|
||||
|
||||
// Verify it's gone
|
||||
$listData = $this->callMcpTool($session, 'list_custom_field_values', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $machine->getId(),
|
||||
]);
|
||||
|
||||
$this->assertSame(0, $listData['_parsed']['total']);
|
||||
}
|
||||
}
|
||||
41
tests/Mcp/Tool/Document/DocumentToolsTest.php
Normal file
41
tests/Mcp/Tool/Document/DocumentToolsTest.php
Normal file
@@ -0,0 +1,41 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool\Document;
|
||||
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class DocumentToolsTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testListDocuments(): void
|
||||
{
|
||||
$site = $this->createSite(name: 'Doc Site');
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'list_documents', [
|
||||
'entityType' => 'site',
|
||||
'entityId' => $site->getId(),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertSame('site', $data['_parsed']['entityType']);
|
||||
$this->assertSame($site->getId(), $data['_parsed']['entityId']);
|
||||
$this->assertIsArray($data['_parsed']['items']);
|
||||
$this->assertSame(0, $data['_parsed']['total']);
|
||||
}
|
||||
|
||||
public function testDeleteDocumentRequiresGestionnaire(): void
|
||||
{
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'delete_document', [
|
||||
'documentId' => 'nonexistent-id',
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('error', $data);
|
||||
}
|
||||
}
|
||||
44
tests/Mcp/Tool/HistoryToolsTest.php
Normal file
44
tests/Mcp/Tool/HistoryToolsTest.php
Normal file
@@ -0,0 +1,44 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool;
|
||||
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class HistoryToolsTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testGetActivityLog(): void
|
||||
{
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'get_activity_log');
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertArrayHasKey('items', $data['_parsed']);
|
||||
$this->assertArrayHasKey('total', $data['_parsed']);
|
||||
$this->assertArrayHasKey('page', $data['_parsed']);
|
||||
$this->assertArrayHasKey('limit', $data['_parsed']);
|
||||
$this->assertArrayHasKey('pageCount', $data['_parsed']);
|
||||
$this->assertIsArray($data['_parsed']['items']);
|
||||
}
|
||||
|
||||
public function testGetEntityHistory(): void
|
||||
{
|
||||
$machine = $this->createMachine(name: 'History Machine');
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'get_entity_history', [
|
||||
'entityType' => 'machine',
|
||||
'entityId' => $machine->getId(),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertArrayHasKey('items', $data['_parsed']);
|
||||
$this->assertArrayHasKey('total', $data['_parsed']);
|
||||
$this->assertIsArray($data['_parsed']['items']);
|
||||
}
|
||||
}
|
||||
66
tests/Mcp/Tool/ModelType/ModelTypeToolsTest.php
Normal file
66
tests/Mcp/Tool/ModelType/ModelTypeToolsTest.php
Normal file
@@ -0,0 +1,66 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool\ModelType;
|
||||
|
||||
use App\Enum\ModelCategory;
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class ModelTypeToolsTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testListModelTypes(): void
|
||||
{
|
||||
$this->createModelType(name: 'MT Alpha', code: 'MTA-'.bin2hex(random_bytes(3)), category: ModelCategory::COMPONENT);
|
||||
$this->createModelType(name: 'MT Beta', code: 'MTB-'.bin2hex(random_bytes(3)), category: ModelCategory::PIECE);
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'list_model_types');
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertGreaterThanOrEqual(2, $data['_parsed']['total']);
|
||||
}
|
||||
|
||||
public function testGetModelType(): void
|
||||
{
|
||||
$mt = $this->createModelType(name: 'MT Detail', code: 'MTD-'.bin2hex(random_bytes(3)));
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'get_model_type', ['modelTypeId' => $mt->getId()]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertSame('MT Detail', $data['_parsed']['name']);
|
||||
$this->assertSame('COMPONENT', $data['_parsed']['category']);
|
||||
$this->assertIsArray($data['_parsed']['skeletonPieceRequirements']);
|
||||
}
|
||||
|
||||
public function testCreateModelType(): void
|
||||
{
|
||||
$session = $this->createMcpClient('ROLE_GESTIONNAIRE');
|
||||
|
||||
$data = $this->callMcpTool($session, 'create_model_type', [
|
||||
'name' => 'MT Nouveau',
|
||||
'category' => 'composant',
|
||||
'code' => 'MTN-'.bin2hex(random_bytes(3)),
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertSame('MT Nouveau', $data['_parsed']['name']);
|
||||
$this->assertSame('COMPONENT', $data['_parsed']['category']);
|
||||
$this->assertNotEmpty($data['_parsed']['id']);
|
||||
}
|
||||
|
||||
public function testDeleteModelType(): void
|
||||
{
|
||||
$mt = $this->createModelType(name: 'MT To Delete', code: 'MTDEL-'.bin2hex(random_bytes(3)));
|
||||
$session = $this->createMcpClient('ROLE_GESTIONNAIRE');
|
||||
|
||||
$data = $this->callMcpTool($session, 'delete_model_type', ['modelTypeId' => $mt->getId()]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$this->assertTrue($data['_parsed']['deleted']);
|
||||
}
|
||||
}
|
||||
63
tests/Mcp/Tool/SearchInventoryToolTest.php
Normal file
63
tests/Mcp/Tool/SearchInventoryToolTest.php
Normal file
@@ -0,0 +1,63 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace App\Tests\Mcp\Tool;
|
||||
|
||||
use App\Tests\AbstractApiTestCase;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*/
|
||||
class SearchInventoryToolTest extends AbstractApiTestCase
|
||||
{
|
||||
public function testSearchFindsAcrossEntities(): void
|
||||
{
|
||||
$this->createMachine(name: 'Alpha Machine');
|
||||
$this->createPiece(name: 'Alpha Piece');
|
||||
$this->createSite(name: 'Alpha Site');
|
||||
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'search_inventory', ['query' => 'Alpha']);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$results = $data['_parsed'];
|
||||
$this->assertIsArray($results);
|
||||
|
||||
$types = array_unique(array_column($results, 'type'));
|
||||
$this->assertContains('machine', $types);
|
||||
$this->assertContains('piece', $types);
|
||||
$this->assertContains('site', $types);
|
||||
|
||||
foreach ($results as $result) {
|
||||
$this->assertArrayHasKey('type', $result);
|
||||
$this->assertArrayHasKey('id', $result);
|
||||
$this->assertArrayHasKey('name', $result);
|
||||
$this->assertArrayHasKey('reference', $result);
|
||||
$this->assertStringContainsStringIgnoringCase('Alpha', $result['name']);
|
||||
}
|
||||
}
|
||||
|
||||
public function testSearchFiltersByType(): void
|
||||
{
|
||||
$this->createMachine(name: 'Beta Machine');
|
||||
$this->createPiece(name: 'Beta Piece');
|
||||
$this->createSite(name: 'Beta Site');
|
||||
|
||||
$session = $this->createMcpClient('ROLE_VIEWER');
|
||||
|
||||
$data = $this->callMcpTool($session, 'search_inventory', [
|
||||
'query' => 'Beta',
|
||||
'types' => 'machine',
|
||||
]);
|
||||
|
||||
$this->assertArrayHasKey('_parsed', $data);
|
||||
$results = $data['_parsed'];
|
||||
$this->assertIsArray($results);
|
||||
$this->assertNotEmpty($results);
|
||||
|
||||
$types = array_unique(array_column($results, 'type'));
|
||||
$this->assertSame(['machine'], $types);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user