<?php
namespace App\EventSubscriber;
use ApiPlatform\Symfony\EventListener\EventPriorities;
use App\Entity\Job;
use App\Entity\Schedule;
use App\Entity\User;
use App\Service\Message;
use Doctrine\Persistence\ManagerRegistry;
use JetBrains\PhpStorm\ArrayShape;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpKernel\Event\ViewEvent;
use Symfony\Component\HttpKernel\KernelEvents;
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;
use function array_multisort;
use function in_array;
final class JobUnableToCompleteSubscriber implements EventSubscriberInterface
{
/** @var string The relative path inside templates/messages without any extensions */
private const TEMPLATE = 'job/unable-to-complete';
public function __construct(
private readonly TokenStorageInterface $tokenStorage,
private readonly Message $messageService,
private readonly ManagerRegistry $managerRegistry,
)
{
}
#[ArrayShape([KernelEvents::VIEW => "array"])]
public static function getSubscribedEvents(): array
{
return [
KernelEvents::VIEW => ['sendMessage', EventPriorities::POST_WRITE],
];
}
public function sendMessage(ViewEvent $event): void
{
$job = $event->getControllerResult();
$method = $event->getRequest()->getMethod();
# PUT and PATCH methods are used to update instances, and we are only interested in updated jobs
if (!$job instanceof Job || !in_array($method, [Request::METHOD_PUT, Request::METHOD_PATCH])) {
return;
}
$previousData = $event->getRequest()->get('previous_data');
# If the status hasn't changed, we're not interested
if ($previousData->getStatus() === $job->getStatus()) {
return;
}
# If the status hasn't changed to "unable to complete", we're not interested
if ($job->getStatus() !== Job::JOB_STATUS_UNABLE_TO_COMPLETE) {
return;
}
$customer = $job->getPond()->getCustomer();
$schedule = $this->getLatestSchedule($job);
$message = $this->messageService
->withRecipient($customer)
->withSubject('We were unable to complete your job')
->withTemplate(self::TEMPLATE)
->withTemplateData([
'addressLine1' => $job->getPond()->getAddress1(),
'firstName' => $customer->getFirstName() === 'ANNETTE' ? null : $customer->getFirstName(),
'fullAddress' => $job->getPond()->getAddress(),
'reason' => $this->getReasonText($schedule),
'employeeName' => $schedule->getEmployee()->getUser()->getFirstName(),
'notes' => $schedule->getNotes(),
]);
$currentUser = $this->tokenStorage->getToken()->getUser();
if ($currentUser instanceof User) {
$message = $message->withCurrentUser($currentUser);
}
$message->send();
}
private function getLatestSchedule(Job $job): Schedule
{
/** @var iterable<int, Schedule> $schedules */
$schedules = $this->managerRegistry->getRepository(Schedule::class)
->findBy(['job' => $job]);
$sort = [];
foreach ($schedules as $schedule) {
$sort[] = $schedule->getDate()->getTimestamp();
}
array_multisort($schedules, SORT_DESC, $sort);
return current($schedules);
}
private function getReasonText(Schedule $schedule): string|bool
{
return match($schedule->getIncompleteReason()) {
Schedule::INCOMPLETE_REASON_TIME => 'we ran out of time',
Schedule::INCOMPLETE_REASON_MATERIALS => 'we had an issue with materials',
Schedule::INCOMPLETE_REASON_NO_ACCESS => 'we had issues with accessing the pond',
default => false, // we will handle "other" in the template
};
}
}