src/Controller/ReportController.php line 238

Open in your IDE?
  1. <?php
  2. namespace App\Controller;
  3. use App\Entity\Report;
  4. use App\Entity\ReportSnapshot;
  5. use App\Entity\User;
  6. use App\Form\NewReportFormType;
  7. use App\Message\Report as ReportMessage;
  8. use App\Repository\ReportRepository;
  9. use App\Repository\ReportSnapshotRepository;
  10. use App\Service\Badge;
  11. use App\Service\Certification;
  12. use App\Service\Mixpanel;
  13. use App\Service\StatisticsDataAggregation;
  14. use App\Service\WorldMap;
  15. use Doctrine\ORM\EntityManagerInterface;
  16. use GuzzleHttp\Client;
  17. use GuzzleHttp\RequestOptions;
  18. use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
  19. use Symfony\Component\HttpFoundation\JsonResponse;
  20. use Symfony\Component\HttpFoundation\Request;
  21. use Symfony\Component\HttpFoundation\Response;
  22. use Symfony\Component\HttpFoundation\ResponseHeaderBag;
  23. use Symfony\Component\Messenger\MessageBusInterface;
  24. use Symfony\Component\Routing\Annotation\Route;
  25. use voku\helper\HtmlMin;
  26. class ReportController extends AbstractController
  27. {
  28.     private EntityManagerInterface $entityManager;
  29.     private Mixpanel $mixpanel;
  30.     private Certification $certification;
  31.     public function __construct(
  32.         EntityManagerInterface $entityManager,
  33.         Mixpanel $mixpanel,
  34.         Certification $certification,
  35.     ) {
  36.         $this->entityManager $entityManager;
  37.         $this->mixpanel $mixpanel;
  38.         $this->certification $certification;
  39.     }
  40.     /**
  41.      * @Route("/reports/new", name="app_reports_new")
  42.      */
  43.     public function new(Request $requestEntityManagerInterface $entityManagerMessageBusInterface $bus): Response
  44.     {
  45.         /** @var User $user */
  46.         $user $this->getUser();
  47.         $report = new Report();
  48.         $report->setUser($user);
  49.         $form $this->createForm(NewReportFormType::class, $report);
  50.         if ($formData $request->get('new_report_form')) {
  51.             $form->submit($formData);
  52.         } else {
  53.             $form->handleRequest($request);
  54.         }
  55.         if ($form->isSubmitted() && $form->isValid()) {
  56.             $existingReport $entityManager->getRepository(Report::class)->findOneBy(
  57.                 [
  58.                     'user' => $user,
  59.                     'domain' => $report->getDomain(),
  60.                 ]
  61.             );
  62.             if ($existingReport) {
  63.                 $this->addFlash(
  64.                     'error',
  65.                     'Sorry, but we can\'t proceed with your request. You have already created a report with these configurations. Please select a new domain.'
  66.                 );
  67.                 $this->mixpanel->track($user'New Report Duplicate');
  68.                 return $this->render('report/new.html.twig', [
  69.                     'form' => $form->createView(),
  70.                 ]);
  71.             }
  72.             $entityManager->persist($report);
  73.             $entityManager->flush();
  74.             $this->mixpanel->track($user'Report Created');
  75.             $this->mixpanel->increment($user'Report');
  76.             $bus->dispatch(
  77.                 new ReportMessage(
  78.                     $report->getId()
  79.                 )
  80.             );
  81.             return $this->redirectToRoute(
  82.                 'app_report_show', [
  83.                     'id' => $report->getId(),
  84.                 ]
  85.             );
  86.         }
  87.         $this->mixpanel->track($user'New Report');
  88.         return $this->render('report/new.html.twig', [
  89.             'form' => $form->createView(),
  90.         ]);
  91.     }
  92.     /**
  93.      * @Route("/reports", name="app_reports")
  94.      */
  95.     public function index(ReportRepository $repository): Response
  96.     {
  97.         /** @var User $user */
  98.         $user $this->getUser();
  99.         $reports $repository->findByUser($user);
  100.         return $this->render('report/index.html.twig', [
  101.             'reports' => $reports,
  102.         ]);
  103.     }
  104.     protected function getSnapshotDate(array $snapshots): \DateTime
  105.     {
  106.         if (empty($snapshots)) {
  107.             return new \DateTime();
  108.         }
  109.         $createdDates array_map(
  110.             function (ReportSnapshot $snapshot) {
  111.                 return $snapshot->getCreatedAt();
  112.             },
  113.             $snapshots,
  114.         );
  115.         rsort($createdDates);
  116.         return reset($createdDates);
  117.     }
  118.     protected function getReferencePeriod(array $snapshots): string
  119.     {
  120.         if (empty($snapshots)) {
  121.             return (new \DateTime())->format('M Y');
  122.         }
  123.         if (=== count($snapshots)) {
  124.             /** @var ReportSnapshot $snapshot */
  125.             $snapshot reset($snapshots);
  126.             return $snapshot->getPeriodStart()->format('M Y');
  127.         }
  128.         $periods array_map(
  129.             function (ReportSnapshot $snapshot) {
  130.                 return [
  131.                     'periodStart' => $snapshot->getPeriodStart(),
  132.                     'periodEnd' => $snapshot->getPeriodEnd(),
  133.                 ];
  134.             },
  135.             $snapshots,
  136.         );
  137.         usort($periods, fn ($a$b) => $a['periodStart'] <=> $b['periodStart']);
  138.         $periodStart reset($periods)['periodStart'];
  139.         $periodEnd end($periods)['periodEnd'];
  140.         return sprintf('%s - %s'$periodStart->format('M Y'), $periodEnd->format('M Y'));
  141.     }
  142.     protected function getReportData(Report $report, array $snapshots): array
  143.     {
  144.         $snapshotRepository $this->entityManager->getRepository(ReportSnapshot::class);
  145.         $totalPageViews $snapshotRepository->getTotalPageViews($snapshots);
  146.         $totalEmissions $snapshotRepository->getTotals($snapshots);
  147.         $totalEmissionsByNations = [
  148.             ReportSnapshotRepository::SORT_FIELD_TCO2 => $snapshotRepository->getTotalEmissionsByNations($snapshots),
  149.             ReportSnapshotRepository::SORT_FIELD_KWH => $snapshotRepository->getTotalEmissionsByNations($snapshotsReportSnapshotRepository::SORT_FIELD_KWH),
  150.         ];
  151.         $totalEmissionsByPages = [];
  152.         if (=== count($snapshots)) {
  153.             $totalEmissionsByPages reset($snapshots)->getPageStatistics();
  154.         }
  155.         $groupedTotalEmissions WorldMap::mergeTotalEmissions(
  156.             $totalEmissionsByNations['tCO2'],
  157.             $totalEmissionsByNations['kWh'],
  158.         );
  159.         $certification $this->certification->getForDomain($report->getDomain());
  160.         $carbonIndex 0;
  161.         if (count($snapshots) > 0) {
  162.             $carbonIndex reset($snapshots)->getCarbonIndex();
  163.         }
  164.         return [
  165.             // @ToDo Remove
  166.             'isYearly' => 12 === count($snapshots),
  167.             'isEmpty' => 0.00 == $totalEmissions['tCO2'] && 0.00 == $totalEmissions['kWh'],
  168.             'snapshotDate' => $this->getSnapshotDate($snapshots),
  169.             'report' => $report,
  170.             'referencePeriod' => $this->getReferencePeriod($snapshots),
  171.             'totalPageViews' => $totalPageViews,
  172.             'totalEmissions' => $totalEmissions,
  173.             'totalEmissionsByNations' => $totalEmissionsByNations,
  174.             'totalEmissionsByPages' => $totalEmissionsByPages,
  175.             'groupedTotalEmissions' => $groupedTotalEmissions,
  176.             'certification' => $certification,
  177.             'carbonIndex' => $carbonIndex
  178.         ];
  179.     }
  180.     /**
  181.      * @Route("/public/reports/{hash}/badge", name="app_report_public_badge_show")
  182.      */
  183.     public function downloadBadge(Report $reportBadge $badge): Response
  184.     {
  185.         $response = new Response();
  186.         $disposition $response->headers->makeDisposition(ResponseHeaderBag::DISPOSITION_ATTACHMENTsprintf('%s-net-zero-badge.svg'$report->getId()));
  187.         $response->headers->set('Content-Disposition'$disposition);
  188.         $response->headers->set('Content-Type''image/svg+xml');
  189.         $response->setContent(
  190.             $badge->carbonNeutral('dedalo.ai''1b8654')
  191.         );
  192.         return $response;
  193.     }
  194.     /**
  195.      * @Route("/public/reports/{hash}", name="app_report_public_show")
  196.      */
  197.     public function publicShow(Report $report): Response
  198.     {
  199.         $snapshot $report->getLastSnapshot();
  200.         if (null === $snapshot) {
  201.             throw $this->createNotFoundException();
  202.         }
  203.         $data $this->getReportData($report, [$snapshot]);
  204.         return $this->render('report/public_show.html.twig'$data);
  205.     }
  206.     /**
  207.      * @Route("/public/reports/{hash}/yearly", name="app_report_public_show_yearly")
  208.      */
  209.     public function publicShowYearly(Report $report): Response
  210.     {
  211.         $snapshots $this->entityManager->getRepository(ReportSnapshot::class)
  212.             ->getYearly($report);
  213.         if (12 !== count($snapshots)) {
  214.             return $this->render('report/empty.html.twig', [
  215.                 'report' => $report,
  216.             ]);
  217.         }
  218.         $data $this->getReportData($report$snapshots);
  219.         return $this->render('report/public_show.html.twig'$data);
  220.     }
  221.     /**
  222.      * @Route("/reports/{id}/yearly", name="app_report_show_yearly")
  223.      */
  224.     public function showYearly(Report $report): Response
  225.     {
  226.         $user $this->getUser();
  227.         if ($report->getUser() !== $user) {
  228.             throw $this->createNotFoundException();
  229.         }
  230.         $snapshots $this->entityManager->getRepository(ReportSnapshot::class)
  231.             ->getYearly($report);
  232.         if (12 !== count($snapshots)) {
  233.             return $this->render('report/empty.html.twig', [
  234.                 'report' => $report,
  235.             ]);
  236.         }
  237.         $data $this->getReportData($report$snapshots);
  238.         return $this->render('report/show.html.twig'$data);
  239.     }
  240.     /**
  241.      * @Route("/reports/{id}", name="app_report_show")
  242.      */
  243.     public function show(Report $report): Response
  244.     {
  245.         $user $this->getUser();
  246.         if ($report->getUser() !== $user) {
  247.             throw $this->createNotFoundException();
  248.         }
  249.         /** @var ReportSnapshot|null $snapshot */
  250.         $snapshot $report->getLastSnapshot();
  251.         if (null === $snapshot) {
  252.             return $this->render(
  253.                 'report/waiting.html.twig',
  254.                 [
  255.                     'report' => $report,
  256.                 ],
  257.                 new Response(''Response::HTTP_NOT_FOUND)
  258.             );
  259.         }
  260.         if ($snapshot->isEmpty()) {
  261.             return $this->render('report/empty.html.twig', [
  262.                 'report' => $report,
  263.             ]);
  264.         }
  265.         $data $this->getReportData($report, [$snapshot]);
  266.         return $this->render('report/show.html.twig'$data);
  267.     }
  268.     /**
  269.      * @Route("/public/reports/{hash}/cdn-example", name="app_report_cdn_example")
  270.      */
  271.     public function cdnExample(string $hashEntityManagerInterface $entityManager): JsonResponse
  272.     {
  273.         $report $entityManager->getRepository(Report::class)
  274.             ->findOneBy(
  275.                 [
  276.                     'hash' => $hash,
  277.                 ]
  278.             );
  279.         if (null === $report) {
  280.             return new JsonResponse([], Response::HTTP_NOT_FOUND);
  281.         }
  282.         $client = new Client(
  283.             [
  284.                 RequestOptions::HEADERS => [
  285.                     'User-Agent' => 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36',
  286.                     'Accept' => 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.7',
  287.                     'Accept-Encoding' => 'gzip, deflate',
  288.                 ],
  289.             ],
  290.         );
  291.         $content '';
  292.         try {
  293.             $response $client->get(
  294.                 $report->getDomain(),
  295.                 [
  296.                     'verify' => false,
  297.                     'timeout' => 5,
  298.                     'allow_redirects' => [
  299.                         'max' => 3,
  300.                         'strict' => true,
  301.                         'protocols' => ['https''http'],
  302.                     ],
  303.                 ]
  304.             );
  305.             $content $response->getBody()->getContents();
  306.         } catch (\Exception $exception) {
  307.             // nothing to do
  308.         }
  309.         $htmlMin = new HtmlMin();
  310.         $htmlMin->doOptimizeViaHtmlDomParser();               // optimize html via "HtmlDomParser()"
  311.         $htmlMin->doRemoveComments();                         // remove default HTML comments (depends on "doOptimizeViaHtmlDomParser(true)")
  312.         $htmlMin->doSumUpWhitespace();                        // sum-up extra whitespace from the Dom (depends on "doOptimizeViaHtmlDomParser(true)")
  313.         $htmlMin->doRemoveWhitespaceAroundTags();             // remove whitespace around tags (depends on "doOptimizeViaHtmlDomParser(true)")
  314.         $htmlMin->doOptimizeAttributes();                     // optimize html attributes (depends on "doOptimizeViaHtmlDomParser(true)")
  315.         $htmlMin->doRemoveHttpPrefixFromAttributes();         // remove optional "http:"-prefix from attributes (depends on "doOptimizeAttributes(true)")
  316.         $htmlMin->doRemoveHttpsPrefixFromAttributes();        // remove optional "https:"-prefix from attributes (depends on "doOptimizeAttributes(true)")
  317.         $htmlMin->doKeepHttpAndHttpsPrefixOnExternalAttributes(); // keep "http:"- and "https:"-prefix for all external links
  318.         $htmlMin->doRemoveDefaultAttributes();                // remove defaults (depends on "doOptimizeAttributes(true)" | disabled by default)
  319.         $htmlMin->doRemoveDeprecatedAnchorName();             // remove deprecated anchor-jump (depends on "doOptimizeAttributes(true)")
  320.         $htmlMin->doRemoveDeprecatedScriptCharsetAttribute(); // remove deprecated charset-attribute - the browser will use the charset from the HTTP-Header, anyway (depends on "doOptimizeAttributes(true)")
  321.         $htmlMin->doRemoveDeprecatedTypeFromScriptTag();      // remove deprecated script-mime-types (depends on "doOptimizeAttributes(true)")
  322.         $htmlMin->doRemoveDeprecatedTypeFromStylesheetLink(); // remove "type=text/css" for css links (depends on "doOptimizeAttributes(true)")
  323.         $htmlMin->doRemoveDeprecatedTypeFromStyleAndLinkTag(); // remove "type=text/css" from all links and styles
  324.         $htmlMin->doRemoveDefaultMediaTypeFromStyleAndLinkTag(); // remove "media="all" from all links and styles
  325.         $htmlMin->doRemoveDefaultTypeFromButton();            // remove type="submit" from button tags
  326.         $htmlMin->doRemoveEmptyAttributes();                  // remove some empty attributes (depends on "doOptimizeAttributes(true)")
  327.         $htmlMin->doRemoveValueFromEmptyInput();              // remove 'value=""' from empty <input> (depends on "doOptimizeAttributes(true)")
  328.         $htmlMin->doSortCssClassNames();                      // sort css-class-names, for better gzip results (depends on "doOptimizeAttributes(true)")
  329.         $htmlMin->doSortHtmlAttributes();                     // sort html-attributes, for better gzip results (depends on "doOptimizeAttributes(true)")
  330.         $htmlMin->doRemoveSpacesBetweenTags();                // remove more (aggressive) spaces in the dom (disabled by default)
  331.         $htmlMin->doRemoveOmittedQuotes();                    // remove quotes e.g. class="lall" => class=lall
  332.         $htmlMin->doRemoveOmittedHtmlTags();                  // remove ommitted html tags e.g. <p>lall</p> => <p>lall
  333.         $contentMinified $htmlMin->minify($contenttrue);
  334.         $contentBytes mb_strlen($content);
  335.         $contentMinifiedBytes mb_strlen($contentMinified);
  336.         return new JsonResponse([
  337.             'content' => $content,
  338.             'content_bytes' => $contentBytes,
  339.             'content_minified' => $contentMinified,
  340.             'content_minified_bytes' => $contentMinifiedBytes,
  341.             'percentage_reduction' => round(100 - ($contentMinifiedBytes 100 $contentBytes), 2),
  342.         ]);
  343.     }
  344.     /**
  345.      * @Route("/public/reports/{hash}/totals-by-nations", name="app_report_totals_by_nations")
  346.      */
  347.     public function totalsByNations(
  348.         Request $request,
  349.         string $hash,
  350.         EntityManagerInterface $entityManager
  351.     ): JsonResponse {
  352.         $report $entityManager->getRepository(Report::class)
  353.             ->findOneBy(
  354.                 [
  355.                     'hash' => $hash,
  356.                 ]
  357.             );
  358.         if (null === $report) {
  359.             return new JsonResponse([], Response::HTTP_NOT_FOUND);
  360.         }
  361.         $snapshots = [];
  362.         if (null === $request->get('yearly')) {
  363.             $snapshot $report->getLastSnapshot();
  364.             if (null === $snapshot || $snapshot->isEmpty()) {
  365.                 return new JsonResponse([], Response::HTTP_NOT_FOUND);
  366.             }
  367.             $snapshots[] = $snapshot;
  368.         } else {
  369.             $snapshots $this->entityManager->getRepository(ReportSnapshot::class)
  370.                 ->getYearly($report);
  371.             if (12 !== count($snapshots)) {
  372.                 return new JsonResponse([], Response::HTTP_NOT_FOUND);
  373.             }
  374.         }
  375.         $snapshotRepository $entityManager->getRepository(ReportSnapshot::class);
  376.         $validFields = [
  377.             ReportSnapshotRepository::SORT_FIELD_TCO2,
  378.             ReportSnapshotRepository::SORT_FIELD_KWH,
  379.         ];
  380.         $sortField $request->get('field'ReportSnapshotRepository::SORT_FIELD_TCO2);
  381.         if (!in_array($sortField$validFields)) {
  382.             $sortField ReportSnapshotRepository::SORT_FIELD_TCO2;
  383.         }
  384.         $report $snapshotRepository->getTotalEmissionsByNations($snapshots$sortField);
  385.         return new JsonResponse(StatisticsDataAggregation::compact($report));
  386.     }
  387.     /**
  388.      * @Route("/reports/{id}/delete", name="app_report_delete")
  389.      */
  390.     public function delete(Report $reportRequest $requestEntityManagerInterface $entityManager): Response
  391.     {
  392.         $user $this->getUser();
  393.         if ($report->getUser() !== $user) {
  394.             throw $this->createNotFoundException();
  395.         }
  396.         $this->addFlash(
  397.             'success',
  398.             sprintf('Thanks! Your report %s is now deleted.'$report->getDisplayName())
  399.         );
  400.         $entityManager->remove($report);
  401.         $entityManager->flush();
  402.         $this->mixpanel->track($user'Report Deleted');
  403.         $this->mixpanel->decrement($user'Report');
  404.         return $this->redirectToRoute('app_reports');
  405.     }
  406. }