src/Controller/Admin/Tech/HealthController.php line 174

Open in your IDE?
  1. <?php
  2. namespace App\Controller\Admin\Tech;
  3. use App\Command\CsvDbUtil;
  4. use App\Command\RouteExtCsv;
  5. use App\Command\StopExtCsv;
  6. use App\Controller\Admin\BaseController;
  7. use App\Entity\RouteExt;
  8. use App\Entity\StopExt;
  9. use App\Repository\RouteExtRepository;
  10. use App\Repository\StopExtRepository;
  11. use App\Service\Client\ClientRepository;
  12. use App\Service\Client\Dumb\DumbClient;
  13. use App\Service\Client\FakeBus\ClientFakeBus;
  14. use App\Service\Client\IClient;
  15. use App\Service\Client\OneWayClient;
  16. use App\Service\Client\Trip;
  17. use App\Service\E2E\SearchResults;
  18. use App\Utils\StringUtils;
  19. use App\Utils\Utils;
  20. use ReflectionClass;
  21. use Sensio\Bundle\FrameworkExtraBundle\Configuration\ParamConverter;
  22. use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
  23. use Symfony\Component\HttpFoundation\JsonResponse;
  24. use Symfony\Component\HttpFoundation\Request;
  25. use Symfony\Component\HttpFoundation\StreamedResponse;
  26. use Symfony\Component\Routing\Annotation\Route;
  27. use Throwable;
  28. class HealthController extends BaseController {
  29.     private ClientRepository $clientRepo;
  30.     private StopExtRepository $stopExtRepo;
  31.     private RouteExtRepository $routeExtRepo;
  32.     /** @required */
  33.     public function setRepos(ClientRepository $clientRepoStopExtRepository $stopExtRepoRouteExtRepository $routeExtRepo): void {
  34.         $this->clientRepo $clientRepo;
  35.         $this->stopExtRepo $stopExtRepo;
  36.         $this->routeExtRepo $routeExtRepo;
  37.     }
  38.     /**
  39.      * @Route("health", name="tech_health_index")
  40.      * @Template("Admin/tech/health_index.html.twig")
  41.      */
  42.     public static function healthIndexAction(Request $request) {
  43.     }
  44.     /**
  45.      * @Route("health/normal", name="tech_health")
  46.      * @Template("Admin/tech/health.html.twig")
  47.      * @ParamConverter("companyId", converter="querystring")
  48.      * @ParamConverter("dryrun", converter="querystring")
  49.      */
  50.     public function healthAction(?int $companyId nullbool $dryrun false) {
  51.         $tripsPerCompany = [];
  52.         /* @var IClient $companies */
  53.         $companies $companyId ? [$companyId => $this->clientRepo->find($companyId)] : $this->clientRepo->findAll();
  54.         foreach ($companies as $cId => $c) {
  55.             if (!$c instanceof DumbClient && !$c instanceof ClientFakeBus) {
  56.                 $tripsPerCompany[$cId] = $this->getPriceAndInfo($c$dryrun);
  57.             }
  58.         }
  59.         return [
  60.             'tripsPerCompany' => $tripsPerCompany,
  61.         ];
  62.     }
  63.     /**
  64.      * @Route("fetch", name="tech_fetch")
  65.      */
  66.     public function healthContentAction(Request $r) {
  67.         $companyId $r->query->get('companyId');
  68.         $depStopExt $r->query->get('depStopExt');
  69.         $arrStopExt $r->query->get('arrStopExt');
  70.         $date $r->query->get('date');
  71.         $client $this->clientRepo->find($companyId);
  72.         //no dep/arr use default one
  73.         if (!$depStopExt || !$arrStopExt)
  74.             list($depStopExt$arrStopExt) = $client->getRoute();
  75.         //no date, use default one
  76.         $date ?: $client->getHealthCheckingDay();
  77.         $content null;
  78.         if ($client instanceof OneWayClient) {
  79.             $content $client->fetch($depStopExt$arrStopExt$date);
  80.         }
  81.         die($content);
  82.     }
  83.     /**
  84.      * Sent everyday by email : /etc/cron.d/cron-comparabus
  85.      * curl -s 'https://www.comparabus.com/health/streamed?email=1' | mail -aContent-Type:text/html -s "Scrapers Health Report" dev@comparabus.com - > /dev/null
  86.      * @Route("health/streamed", name="tech_health_streamed")
  87.      * @ParamConverter("dryrun", converter="querystring")
  88.      */
  89.     public function heathStreamedAction(bool $dryrun false) {
  90.         if (!defined('STDOUT')) define('STDOUT'fopen('php://stdout''w'));
  91.         try {
  92.             $response = new StreamedResponse();
  93.             $response->setCallback(function () use ($dryrun) {
  94.                 echo $this->renderView("Admin/tech/health_streamed_1_header.html.twig");
  95.                 foreach ($this->clientRepo->findAll() as $c) {
  96.                     if (!$c instanceof DumbClient && !$c instanceof ClientFakeBus) {
  97.                         $prices $this->getPriceAndInfo($c$dryrun);
  98.                         echo $this->renderView("Admin/tech/health_streamed_2_body.html.twig", [
  99.                             'companyId' => $c->getId(),
  100.                             'info' => $prices,
  101.                         ]);
  102.                         ob_flush();
  103.                         flush();
  104.                     }
  105.                 }
  106.                 echo $this->renderView("Admin/tech/health_streamed_3_footer.html.twig");
  107.             });
  108.             return $response;
  109.         } catch (\Exception Throwable | \Error $e) { //Exception and Error could be removed as implement Throwable
  110.             //dirty way to be able to display error in prod as that url is private, because the error in prod is hidden
  111.             // we should be here as the error is already catched before
  112.             die($e->getTraceAsString());
  113.         }
  114.     }
  115.     /**
  116.      * Try to extract price for a company and returns extra info
  117.      */
  118.     private function getPriceAndInfo(IClient $clientbool $dryrun falsebool $showDiff false): array {
  119.         $timestart microtime(true);
  120.         $trips = [];
  121.         $error null;
  122.         $healthCheckingDay $client->getHealthCheckingDay();
  123.         try {
  124.             list($dep$arr) = TechClientListController::getDepArr($client);
  125.             //TODO check $dep / $arr that in DB and route between them
  126.             $trips $dryrun ? [] : $client->getPrice($dep->code$arr->code$healthCheckingDaynull);
  127.         } catch (\Throwable $e) {
  128.             $error $e->getMessage();
  129.         }
  130.         return [
  131.             'company' => $client,
  132.             'class' => new ReflectionClass($client),
  133.             'dep' => $dep,
  134.             'arr' => $arr,
  135.             'link' => count($trips) ? $trips[0]->getLink() : null ,
  136.             'data' => Utils::objectToJsonJMS($trips),
  137.             'error' => $error,
  138.             'count' => $trips == null ? -count($trips),
  139.             'time' => microtime(true) - $timestart,
  140.             'date' => $healthCheckingDay,
  141.             'note' => $this->getHealthNote($client),
  142.             'diffStopDbCsv' => $this->getDifferenceStopDbCsv($client$showDiff)
  143.         ];
  144.     }
  145.     /**
  146.      * Monitoring checking that the keyword 'error' in not in that page
  147.      * @Route("health/e2e", name="tech_health_headless")
  148.      */
  149.     public function healthWebsite() {
  150.         $searchTest = new SearchResults();
  151.         return new JsonResponse([
  152.             'checkWebsiteHasTop6Cities' => $searchTest->checkWebsiteHasTop6Cities()
  153.         ]);
  154.     }
  155.     private function getHealthNote(IClient $c): string {
  156.         $clientClass = new \ReflectionClass($c);
  157.         $comments $clientClass->getDocComment();
  158.         $healthNote '';
  159.         if (StringUtils::contains($comments"HEALTH:")) { //TODO use an annotation instead
  160.             //The note need to be write on a single line
  161.             $healthNote StringUtils::substringBetween($comments'HEALTH:'"\n");
  162.         }
  163.         return $healthNote;
  164.     }
  165.     /**
  166.      * @param bool $showDiff debug param to see which element is wrong in csv file
  167.      * @return array NbStopDB/NbStopCsv
  168.      */
  169.     private function getDifferenceStopDbCsv(IClient $cbool $showDiff false): array {
  170.         $stopsCsvSum CsvDbUtil::stopExtSumStopIdByCompanyId($c->getId());
  171.         $stopsCsvCount CsvDbUtil::stopExtCountByCompanyId($c->getId());
  172.         $stopsDbSum $this->stopExtRepo->sumStopIdOfCompanyId($c->getId());
  173.         $stopsDbCount $this->stopExtRepo->count(['companyId' => $c->getId()]);
  174.         $routesCsvCount CsvDbUtil::countRouteExt($c->getId());
  175.         $routesDbCount $this->routeExtRepo->count(['company_id' => $c->getId()]);
  176.         $diffAsJson '';
  177.         if ($showDiff) {
  178.             $dbItems $this->stopExtRepo->findBy(['companyId' => $c->getId()]);
  179.             $csvItems CsvDbUtil::getStopExts($c->getId());
  180.             $diffAsJson $this->showStopDiffCsvDb($dbItems$csvItems);
  181.         }
  182.         //Cannot check routes because RouteExt csv contains not-existing StopExt.code AND StopExt.code == 0 / null
  183.         // Use Csv query to be able to check that
  184.         return [
  185.             'info' => "$stopsDbCount/$stopsCsvCount $diffAsJson",
  186.             'title' => "StopExt: sum=$stopsDbSum/$stopsCsvSum ; count=$stopsDbCount/$stopsCsvCount - RouteExt: count=$routesDbCount/$routesCsvCount",
  187.             'warning' => $stopsDbSum !== $stopsCsvSum || $stopsDbCount !== $stopsCsvCount,
  188.         ];
  189.     }
  190.     /**
  191.      * @param StopExt[] $dbItems
  192.      * @param StopExtCsv[] $csvItems
  193.      * @return string
  194.      */
  195.     private function showStopDiffCsvDb(array $dbItems, array $csvItems): string {
  196.         $tabA = [];
  197.         foreach ($csvItems as $item) {
  198.             $tabA[] = $item->getCode();
  199.         }
  200.         //Check if there are duplicate
  201.         $tabDuplicate = [];
  202.         $tabAWithoutDuplicate array_unique($tabA);
  203.         if (count($tabA) != count($tabAWithoutDuplicate)) {
  204.             //There are duplicate
  205.             $keysDuplicates array_diff(array_keys($tabA), array_keys($tabAWithoutDuplicate));
  206.             foreach ($keysDuplicates as $key) {
  207.                 $tabDuplicate[] = $tabA[$key];
  208.             }
  209.         }
  210.         //Check if csv diff from db
  211.         $tabB = [];
  212.         foreach ($dbItems as $item) {
  213.             $tabB[] = $item->getCode();
  214.         }
  215.         $tabDiff array_diff($tabA$tabB);
  216.         $tabDiff array_merge($tabDiff$tabDuplicate);
  217.         return json_encode(array_values($tabDiff));
  218.     }
  219.     /**
  220.      * @param RouteExt[] $dbItems
  221.      * @param RouteExtCsv[] $csvItems
  222.      * @return string
  223.      */
  224.     private function showRouteDiffCsvDb(array $dbItems, array $csvItems): string {
  225.         $tabA = [];
  226.         foreach ($csvItems as $item) {
  227.             /**
  228.              * @var RouteExtCsv $item
  229.              */
  230.             $tabA[] = $item->getDepStopCode() . " - " $item->getArrStopCode();
  231.         }
  232.         $tabDuplicate = [];
  233.         $tabAWithoutDuplicate Utils::array_unique($tabA);
  234.         if (count($tabA) != count($tabAWithoutDuplicate)) {
  235.             //There are duplicate
  236.             $keysDuplicates array_diff(array_keys($tabA), array_keys($tabAWithoutDuplicate));
  237.             foreach ($keysDuplicates as $key) {
  238.                 $tabDuplicate[] = $tabA[$key];
  239.             }
  240.         }
  241.         $tabB = [];
  242.         foreach ($dbItems as $item) {
  243.             /**
  244.              * @var RouteExt $item
  245.              */
  246.             $tabB[] = $item->getStopExtDep() . " - " $item->getStopExtArr();
  247.         }
  248.         $tabDiff array_diff($tabA$tabB);
  249.         $tabDiff array_merge($tabDiff$tabDuplicate);
  250.         return json_encode(array_values($tabDiff));
  251.     }
  252. }