180 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			180 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
declare(strict_types=1);
 | 
						|
require_once 'vendor/autoload.php';
 | 
						|
 | 
						|
use ICal\ICal;
 | 
						|
use Symfony\Component\HttpClient\CachingHttpClient;
 | 
						|
use Symfony\Component\HttpClient\HttpClient;
 | 
						|
use Symfony\Component\HttpKernel\HttpCache\Store;
 | 
						|
use Twig\TwigFilter;
 | 
						|
use Twig\NodeVisitor\OptimizerNodeVisitor;
 | 
						|
 | 
						|
const VISITORS_FILE = '/opt/dorf.jetzt_visitors';
 | 
						|
const ICAL_URL = 'https://chaosdorf.de/~derf/cccd_all.ics';
 | 
						|
const HTTP_CACHE = '/tmp/dorf.jetzt/http_cache';
 | 
						|
const TMPL_CACHE = '/tmp/dorf.jetzt/twig_cache';
 | 
						|
const ROOM_STATE_URL = 'https://chaosdorf.de/raumstatus/status.png';
 | 
						|
 | 
						|
const INVALID_UAS = [
 | 
						|
	"AhrefsBot",
 | 
						|
	"Googlebot",
 | 
						|
	"Yahoo",
 | 
						|
	"Go-http-client/",
 | 
						|
	"bingbot",
 | 
						|
	"CheckMarkNetwork",
 | 
						|
	"SemrushBot",
 | 
						|
	"BingPreview",
 | 
						|
	"facebookexternalhit",
 | 
						|
	"hetrix.tools",
 | 
						|
];
 | 
						|
 | 
						|
const HASH_TO_STATE = [
 | 
						|
	'bff0167ed8aba031c49122ef4046cf1b' => 'closed',
 | 
						|
	'd8ec899c69283bc775952a767db9d5f5' => 'maybe_open',
 | 
						|
	'2c2672c641425e5b2acd6ee74f39ae60' => 'open',
 | 
						|
	'66aece8ae27ffd3a656d42005fa3efbd' => 'private',
 | 
						|
	'86c75c0ad413b06ff8291673162d0b64' => 'unknown',
 | 
						|
	'0' => 'error',
 | 
						|
];
 | 
						|
 | 
						|
$STATE_MAP = [
 | 
						|
	'closed' => (object) [
 | 
						|
		'state_string' => 'Das Dorf ist gerade <em>geschlossen</em>.',
 | 
						|
		'svg' => 'lock',
 | 
						|
		'color' => 'red',
 | 
						|
	],
 | 
						|
	'maybe_open' => (object) [
 | 
						|
		'state_string' => 'Das Dorf ist gerade <em>vielleicht geöffnet</em>: </p><p>Der Clubraum ist offen, aber es findet keine Veranstaltung statt.</p><p>
 | 
						|
		Der Status kann sich also kurzfristig ändern.',
 | 
						|
		'svg' => 'done',
 | 
						|
		'color' => 'brown',
 | 
						|
	],
 | 
						|
	'open' => (object) [
 | 
						|
		'state_string' => 'Das Dorf ist gerade <em>geöffnet</em>.</p><p>
 | 
						|
		Komm gerne vorbei.',
 | 
						|
		'svg' => 'done',
 | 
						|
		'color' => 'green',
 | 
						|
	],
 | 
						|
	'private' => (object) [
 | 
						|
		'state_string' => 'Das Dorf ist gerade <em>privat</em>: </p><p>Es sind Leute da, aber der Clubraum ist nicht geöffnet.</p><p>
 | 
						|
		Komm gerne vorbei (aber frag lieber vorher, wie lange noch Leute da sind).',
 | 
						|
		'svg' => 'lock',
 | 
						|
		'color' => 'fdd835',
 | 
						|
	],
 | 
						|
	'unknown' => (object) [
 | 
						|
		'state_string' => 'Der Status vom Dorf ist gerade <em>unbekannt</em>',
 | 
						|
		'svg' => 'warning',
 | 
						|
		'color' => 'orange',
 | 
						|
	],
 | 
						|
	'error' => (object) [
 | 
						|
		'state_string' => 'Der Server konnte den Status vom Dorf nicht abrufen.',
 | 
						|
		'svg' => 'error',
 | 
						|
		'color' => 'blue',
 | 
						|
	],
 | 
						|
];
 | 
						|
 | 
						|
function hasValidUa(): bool
 | 
						|
{
 | 
						|
	if (isset($_SERVER['HTTP_USER_AGENT'])) {
 | 
						|
		if (in_array(true, array_map(fn ($ua) => str_contains($_SERVER['HTTP_USER_AGENT'], $ua), INVALID_UAS))) {
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
function dateTimeFromEvent(ICal $ical, object $event): DateTimeImmutable {
 | 
						|
	return DateTimeImmutable::createFromMutable($ical->iCalDateToDateTime($event->dtstart_array[3]));
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * @return array<int, array<string, DateTime|string|bool>>
 | 
						|
 */
 | 
						|
function prepare_events(ICal $ical, array $events): array
 | 
						|
{
 | 
						|
	$returns = [];
 | 
						|
	foreach ($events as $event) {
 | 
						|
		$start = dateTimeFromEvent($ical, $event);
 | 
						|
		$end = $start->add(new DateInterval($event->duration));
 | 
						|
		$returns[] = [
 | 
						|
			'summary' => $event->summary,
 | 
						|
			'url' => $event->url,
 | 
						|
			'start' => $start,
 | 
						|
			'end' => $end,
 | 
						|
		];
 | 
						|
	}
 | 
						|
	return $returns;
 | 
						|
}
 | 
						|
 | 
						|
$store = new Store(HTTP_CACHE);
 | 
						|
$client = HttpClient::create();
 | 
						|
$client = new CachingHttpClient($client, $store, ['default_ttl' => 60, 'allow_revalidate' => true]);
 | 
						|
 | 
						|
$state = 'error';
 | 
						|
try {
 | 
						|
	$response = $client->request('GET', ROOM_STATE_URL);
 | 
						|
	$hash = md5($response->getContent());
 | 
						|
} catch (\Exception $e) {
 | 
						|
	$hash = 0;
 | 
						|
}
 | 
						|
$state = HASH_TO_STATE[$hash];
 | 
						|
$ical = new ICal(false, [
 | 
						|
	'defaultSpan' => 2,
 | 
						|
	'defaultTimeZone' => 'Europe/Berlin',
 | 
						|
	'defaultWeekStart' => 'MO',
 | 
						|
	'filterDaysBefore' => '1',
 | 
						|
]);
 | 
						|
$ical->initUrl(ICAL_URL, $acceptLanguage = 'de');
 | 
						|
$events = $ical->eventsFromInterval('2 week');
 | 
						|
if (isset($_SERVER['HTTP_ACCEPT_LANGUAGE'])) {
 | 
						|
	$locale = locale_accept_from_http($_SERVER['HTTP_ACCEPT_LANGUAGE']);
 | 
						|
	setlocale(LC_TIME, $locale);
 | 
						|
}
 | 
						|
$visitors = file_get_contents(VISITORS_FILE);
 | 
						|
if (is_string($visitors)) {
 | 
						|
	$visitors = intval($visitors);
 | 
						|
} else {
 | 
						|
	$visitors = 0;
 | 
						|
}
 | 
						|
$state_obj = $STATE_MAP[$state];
 | 
						|
 | 
						|
$loader = new \Twig\Loader\FilesystemLoader('templates');
 | 
						|
$twig = new \Twig\Environment($loader, [
 | 
						|
	'cache' => TMPL_CACHE,
 | 
						|
	'auto_reload' => true,
 | 
						|
	// The 'raw' optimizer sometimes eats the only 'raw' that is used
 | 
						|
	'optimizations' => OptimizerNodeVisitor::OPTIMIZE_ALL ^ OptimizerNodeVisitor::OPTIMIZE_RAW_FILTER,
 | 
						|
]);
 | 
						|
 | 
						|
function formatEndDt(?DateTimeImmutable $end, ?DateTimeImmutable $start, string $timezone = 'Europe/Berlin'): string
 | 
						|
{
 | 
						|
	if ($end == null || $start == null) return "ERROR";
 | 
						|
	$tz = new DateTimeZone($timezone);
 | 
						|
	$daySame = $end->setTimeZone($tz)->format('d.m.Y') == $start->setTimeZone($tz)->format('d.m.Y');
 | 
						|
	$endIsMidnight = $end->setTimeZone($tz)->format('H:i') == '00:00' && $start->setTimeZone($tz)->format('H:i') != '00:00';
 | 
						|
	if ($daySame || $endIsMidnight) {
 | 
						|
		return $end->format('H:i');
 | 
						|
	} else {
 | 
						|
		return $end->format('d.m.Y H:i');
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
$twig->addFilter(new TwigFilter('end_datetime', 'formatEndDt'));
 | 
						|
//$twig->addExtension(new \Twig\Extra\Intl\IntlExtension());
 | 
						|
$twig->getExtension(\Twig\Extension\CoreExtension::class)->setTimezone('Europe/Berlin');
 | 
						|
$render_evts = prepare_events($ical, $events);
 | 
						|
echo ($twig->render("Main.twig", [
 | 
						|
	'visitors' => $visitors,
 | 
						|
	'state_svg' => $state_obj->svg,
 | 
						|
	'state_color' => $state_obj->color,
 | 
						|
	'state_string' => $state_obj->state_string,
 | 
						|
	'events' => $render_evts,
 | 
						|
]));
 | 
						|
/* Initialising values */
 | 
						|
if (hasValidUa()) {
 | 
						|
	$visitors++;
 | 
						|
	file_put_contents(VISITORS_FILE, strval($visitors));
 | 
						|
}
 |