domain-watchdog/src/Controller/DomainRefreshController.php

92 lines
3.5 KiB
PHP
Raw Normal View History

2024-07-17 18:18:11 +02:00
<?php
namespace App\Controller;
use App\Entity\Domain;
use App\Entity\WatchList;
use App\Message\ProcessDomainTrigger;
use App\Repository\DomainRepository;
2024-07-17 18:18:11 +02:00
use App\Service\RDAPService;
2024-08-04 14:45:27 +02:00
use Psr\Log\LoggerInterface;
use Random\Randomizer;
2024-07-17 18:18:11 +02:00
use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
use Symfony\Component\HttpKernel\Exception\TooManyRequestsHttpException;
2024-07-21 19:47:25 +02:00
use Symfony\Component\HttpKernel\KernelInterface;
use Symfony\Component\Messenger\Exception\ExceptionInterface;
use Symfony\Component\Messenger\MessageBusInterface;
use Symfony\Component\RateLimiter\RateLimiterFactory;
use Symfony\Contracts\HttpClient\Exception\DecodingExceptionInterface;
use Symfony\Contracts\HttpClient\Exception\HttpExceptionInterface;
use Symfony\Contracts\HttpClient\Exception\TransportExceptionInterface;
2024-07-17 18:18:11 +02:00
class DomainRefreshController extends AbstractController
{
2024-08-02 23:24:52 +02:00
public function __construct(private readonly DomainRepository $domainRepository,
private readonly RDAPService $RDAPService,
2024-08-05 01:30:27 +02:00
private readonly RateLimiterFactory $rdapRequestsLimiter,
2024-08-04 14:45:27 +02:00
private readonly MessageBusInterface $bus,
private readonly LoggerInterface $logger
) {
}
2024-07-17 18:18:11 +02:00
/**
* @throws TransportExceptionInterface
* @throws DecodingExceptionInterface
* @throws ExceptionInterface
2024-08-02 23:24:52 +02:00
* @throws \Exception
2024-07-17 18:18:11 +02:00
*/
2024-07-21 19:47:25 +02:00
public function __invoke(string $ldhName, KernelInterface $kernel): ?Domain
2024-07-17 18:18:11 +02:00
{
$idnDomain = strtolower(idn_to_ascii($ldhName));
2024-08-04 14:45:27 +02:00
$userId = $this->getUser()->getUserIdentifier();
$this->logger->info('User {username} wants to update the domain name {idnDomain}.', [
'username' => $userId,
'idnDomain' => $idnDomain,
]);
2024-08-03 00:30:03 +02:00
/** @var ?Domain $domain */
$domain = $this->domainRepository->findOneBy(['ldhName' => $idnDomain]);
2024-07-25 20:21:57 +02:00
// If the domain name exists in the database, recently updated and not important, we return the stored Domain
2024-08-02 23:24:52 +02:00
if (null !== $domain
&& !$domain->getDeleted()
&& ($domain->getUpdatedAt()->diff(new \DateTimeImmutable('now'))->days < 7)
&& !$this->RDAPService::isToBeWatchClosely($domain, $domain->getUpdatedAt())
) {
2024-08-04 14:45:27 +02:00
$this->logger->info('It is not necessary to update the information of the domain name {idnDomain} with the RDAP protocol.', [
'idnDomain' => $idnDomain,
]);
2024-08-02 23:24:52 +02:00
return $domain;
}
if (false === $kernel->isDebug() && true === $this->getParameter('limited_features')) {
2024-08-05 01:30:27 +02:00
$limiter = $this->rdapRequestsLimiter->create($userId);
2024-08-05 22:41:08 +02:00
$limit = $limiter->consume();
if (!$limit->isAccepted()) {
throw new TooManyRequestsHttpException($limit->getRetryAfter()->getTimestamp() - time());
2024-08-02 23:24:52 +02:00
}
}
2024-08-03 00:30:03 +02:00
$updatedAt = null === $domain ? new \DateTimeImmutable('now') : $domain->getUpdatedAt();
try {
$domain = $this->RDAPService->registerDomain($idnDomain);
} catch (HttpExceptionInterface) {
throw new NotFoundHttpException('This domain name cannot be found in the WHOIS database');
}
2024-07-25 20:21:57 +02:00
$randomizer = new Randomizer();
$watchLists = $randomizer->shuffleArray($domain->getWatchLists()->toArray());
/** @var WatchList $watchList */
foreach ($watchLists as $watchList) {
$this->bus->dispatch(new ProcessDomainTrigger($watchList->getToken(), $domain->getLdhName(), $updatedAt));
}
return $domain;
2024-07-17 18:18:11 +02:00
}
2024-08-02 23:24:52 +02:00
}