Backend: - Add MCP Serializer to centralize entity-to-array conversion (~300 lines deduped) - Fix race condition in task/ticket number generation (SELECT FOR UPDATE + transaction) - Add unique constraint on task (project_id, number) with migration - Fix MIME type validation: use server-detected finfo instead of client-supplied type - Add allowlist of permitted MIME types for uploads - Fix TaskDocumentDownloadController: allow ROLE_CLIENT access, add priority:1 - Fix notification sent even when ticket status unchanged - Remove redundant exception constructors - Simplify services (BookStackApi double fetch, TokenEncryptor, GiteaApi) - Consolidate duplicate checks in processors Frontend: - Fix useApi isHandlingUnauthorized scope (module-level to prevent double 401 redirect) - Fix client-tickets toast key copy-paste bug - Merge duplicated tasks service methods (getByProject + getByProjectArchived) - Extract shared uploadWithRelation helper in task-documents service - Extract formatFileSize utility from duplicated component code - Extract status transition logic into useClientTicketHelpers composable - Remove dead code (unused router, handleLogout, empty script blocks) - Merge duplicate watchers and onMounted calls - Normalize arrow functions to function declarations per convention Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
75 lines
2.3 KiB
PHP
75 lines
2.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Service;
|
|
|
|
use App\Entity\ClientTicket;
|
|
use App\Entity\Notification;
|
|
use App\Repository\UserRepository;
|
|
use DateTimeImmutable;
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
|
|
final readonly class NotificationService
|
|
{
|
|
public function __construct(
|
|
private EntityManagerInterface $entityManager,
|
|
private UserRepository $userRepository,
|
|
) {}
|
|
|
|
/**
|
|
* Notify all ROLE_ADMIN users that a new ticket was created.
|
|
*/
|
|
public function createForTicketCreated(ClientTicket $ticket): void
|
|
{
|
|
$admins = $this->userRepository->findByRole('ROLE_ADMIN');
|
|
$number = sprintf('CT-%03d', $ticket->getNumber());
|
|
$projectName = $ticket->getProject()?->getName() ?? '';
|
|
|
|
foreach ($admins as $admin) {
|
|
$notification = new Notification();
|
|
$notification->setUser($admin);
|
|
$notification->setType('ticket_created');
|
|
$notification->setTitle('Nouveau ticket client '.$number);
|
|
$notification->setMessage($ticket->getTitle().' — '.$projectName);
|
|
$notification->setRelatedTicket($ticket);
|
|
$notification->setCreatedAt(new DateTimeImmutable());
|
|
|
|
$this->entityManager->persist($notification);
|
|
}
|
|
|
|
$this->entityManager->flush();
|
|
}
|
|
|
|
/**
|
|
* Notify the ticket submitter that the status has changed.
|
|
*/
|
|
public function createForStatusChange(ClientTicket $ticket): void
|
|
{
|
|
$submittedBy = $ticket->getSubmittedBy();
|
|
|
|
if (null === $submittedBy) {
|
|
return;
|
|
}
|
|
|
|
$number = sprintf('CT-%03d', $ticket->getNumber());
|
|
$statusComment = $ticket->getStatusComment();
|
|
$message = 'Nouveau statut : '.$ticket->getStatus();
|
|
|
|
if (null !== $statusComment && '' !== $statusComment) {
|
|
$message .= ' — '.$statusComment;
|
|
}
|
|
|
|
$notification = new Notification();
|
|
$notification->setUser($submittedBy);
|
|
$notification->setType('ticket_status_changed');
|
|
$notification->setTitle('Ticket '.$number.' mis à jour');
|
|
$notification->setMessage($message);
|
|
$notification->setRelatedTicket($ticket);
|
|
$notification->setCreatedAt(new DateTimeImmutable());
|
|
|
|
$this->entityManager->persist($notification);
|
|
$this->entityManager->flush();
|
|
}
|
|
}
|