AzuraCast/src/Controller/Api/Stations/Reports/Overview/ChartsAction.php

198 lines
5.3 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Controller\Api\Stations\Reports\Overview;
use App\Entity\Api\Status;
use App\Entity\Enums\AnalyticsIntervals;
use App\Entity\Repository\AnalyticsRepository;
use App\Http\Response;
use App\Http\ServerRequest;
use Carbon\CarbonImmutable;
use Psr\Http\Message\ResponseInterface;
use stdClass;
final class ChartsAction extends AbstractReportAction
{
public function __construct(
private readonly AnalyticsRepository $analyticsRepo
) {
}
public function __invoke(
ServerRequest $request,
Response $response,
string $station_id
): ResponseInterface {
// Get current analytics level.
if (!$this->isAnalyticsEnabled()) {
return $response->withStatus(400)
->withJson(new Status(false, 'Reporting is restricted due to system analytics level.'));
}
$station = $request->getStation();
$stationTz = $station->getTimezoneObject();
$dateRange = $this->getDateRange($request, $stationTz);
$stats = [];
// Statistics by day.
$dailyStats = $this->analyticsRepo->findForStationInRange(
$station,
$dateRange
);
$daily_chart = new stdClass();
$daily_chart->label = __('Listeners by Day');
$daily_chart->type = 'line';
$daily_chart->fill = false;
$dailyAlt = [
'label' => $daily_chart->label,
'values' => [],
];
$daily_averages = [];
$days_of_week = [];
foreach ($dailyStats as $stat) {
/** @var CarbonImmutable $statTime */
$statTime = $stat['moment'];
$statTime = $statTime->shiftTimezone($stationTz);
$avg_row = new stdClass();
$avg_row->x = $statTime->getTimestampMs();
$avg_row->y = round((float)$stat['number_avg'], 2);
$daily_averages[] = $avg_row;
$row_date = $statTime->format('Y-m-d');
$dailyAlt['values'][] = [
'label' => $row_date,
'type' => 'time',
'original' => $avg_row->x,
'value' => $avg_row->y . ' ' . __('Listeners'),
];
$day_of_week = (int)$statTime->format('N') - 1;
$days_of_week[$day_of_week][] = $stat['number_avg'];
}
$daily_chart->data = $daily_averages;
$stats['daily'] = [
'metrics' => [
$daily_chart,
],
'alt' => [
$dailyAlt,
],
];
$day_of_week_chart = new stdClass();
$day_of_week_chart->label = __('Listeners by Day of Week');
$dayOfWeekAlt = [
'label' => $day_of_week_chart->label,
'values' => [],
];
$days_of_week_names = [
__('Monday'),
__('Tuesday'),
__('Wednesday'),
__('Thursday'),
__('Friday'),
__('Saturday'),
__('Sunday'),
];
$day_of_week_stats = [];
foreach ($days_of_week_names as $day_index => $day_name) {
$day_totals = $days_of_week[$day_index] ?? [0];
$stat_value = round(array_sum($day_totals) / count($day_totals), 2);
$day_of_week_stats[] = $stat_value;
$dayOfWeekAlt['values'][] = [
'label' => $day_name,
'type' => 'string',
'value' => $stat_value . ' ' . __('Listeners'),
];
}
$day_of_week_chart->data = $day_of_week_stats;
$stats['day_of_week'] = [
'labels' => $days_of_week_names,
'metrics' => [
$day_of_week_chart,
],
'alt' => [
$dayOfWeekAlt,
],
];
// Statistics by hour.
$hourlyStats = $this->analyticsRepo->findForStationInRange(
$station,
$dateRange,
AnalyticsIntervals::Hourly
);
$totals_by_hour = [];
foreach ($hourlyStats as $stat) {
/** @var CarbonImmutable $statTime */
$statTime = $stat['moment'];
$statTime = $statTime->shiftTimezone($stationTz);
$hour = $statTime->hour;
$totals_by_hour[$hour][] = $stat['number_avg'];
}
$hourly_labels = [];
$hourly_chart = new stdClass();
$hourly_chart->label = __('Listeners by Hour');
$hourly_rows = [];
$hourlyAlt = [
'label' => $hourly_chart->label,
'values' => [],
];
for ($i = 0; $i < 24; $i++) {
$hourly_labels[] = $i . ':00';
$totals = $totals_by_hour[$i] ?? [0];
$stat_value = round(array_sum($totals) / count($totals), 2);
$hourly_rows[] = $stat_value;
$hourlyAlt['values'][] = [
'label' => $i . ':00',
'type' => 'string',
'value' => $stat_value . ' ' . __('Listeners'),
];
}
$hourly_chart->data = $hourly_rows;
$stats['hourly'] = [
'labels' => $hourly_labels,
'metrics' => [
$hourly_chart,
],
'alt' => [
$hourlyAlt,
],
];
return $response->withJson($stats);
}
}