1
1
Fork 0
mirror of https://code.mensbeam.com/MensBeam/Arsse.git synced 2025-01-25 04:20:34 +00:00
Arsse/lib/Service.php

135 lines
4.5 KiB
PHP
Raw Normal View History

<?php
/** @license MIT
* Copyright 2017 J. King, Dustin Wilson et al.
* See LICENSE and AUTHORS files for details */
declare(strict_types=1);
namespace JKingWeb\Arsse;
2017-08-29 10:50:31 -04:00
use JKingWeb\Arsse\Misc\Date;
class Service {
2020-03-01 18:32:01 -05:00
public const DRIVER_NAMES = [
'serial' => \JKingWeb\Arsse\Service\Serial\Driver::class,
'subprocess' => \JKingWeb\Arsse\Service\Subprocess\Driver::class,
];
2018-10-26 14:58:04 -04:00
/** @var Service\Driver */
protected $drv;
protected $loop = false;
2021-07-04 22:04:05 -04:00
protected $reload = false;
2017-08-29 10:50:31 -04:00
public function __construct() {
$driver = Arsse::$conf->serviceDriver;
2022-05-31 23:55:04 -04:00
$this->drv = new $driver;
}
2017-08-29 10:50:31 -04:00
public function watch(bool $loop = true): \DateTimeInterface {
$this->loop = $loop;
$this->signalInit();
2022-05-31 23:55:04 -04:00
$t = new \DateTime;
2017-07-16 14:55:37 -04:00
do {
$this->checkIn();
static::cleanupPre();
$list = Arsse::$db->feedListStale();
2017-08-29 10:50:31 -04:00
if ($list) {
$this->drv->queue(...$list);
2019-10-19 18:51:01 -04:00
unset($list);
$this->drv->exec();
$this->drv->clean();
}
2017-08-17 22:36:15 -04:00
static::cleanupPost();
2019-10-19 18:51:01 -04:00
$t->add(Arsse::$conf->serviceFrequency);
2019-10-19 12:14:13 -04:00
// @codeCoverageIgnoreStart
if ($this->loop) {
do {
sleep((int) max(0, $t->getTimestamp() - time()));
2021-07-05 21:47:44 -04:00
if (function_exists("pcntl_signal_dispatch")) {
pcntl_signal_dispatch();
if ($this->reload) {
$this->reload();
2021-07-06 10:07:56 -04:00
fwrite(\STDERR, Arsse::$lang->msg("Service.Reload").\PHP_EOL);
2021-07-05 21:47:44 -04:00
}
2021-07-04 22:04:05 -04:00
}
} while ($this->loop && $t->getTimestamp() > time());
}
2019-10-19 12:14:13 -04:00
// @codeCoverageIgnoreEnd
} while ($this->loop);
return $t;
2017-07-16 14:55:37 -04:00
}
2021-07-04 22:04:05 -04:00
public function reload(): void {
$this->reload = false;
2021-07-05 20:21:04 -04:00
Arsse::$user = Arsse::$db = Arsse::$conf = Arsse::$lang = Arsse::$obj = $this->drv = null;
2021-07-04 22:04:05 -04:00
Arsse::bootstrap();
$this->__construct();
}
2017-08-29 10:50:31 -04:00
public function checkIn(): bool {
return Arsse::$db->metaSet("service_last_checkin", time(), "datetime");
2017-07-16 14:55:37 -04:00
}
2017-08-29 10:50:31 -04:00
public static function hasCheckedIn(): bool {
$checkin = Arsse::$db->metaGet("service_last_checkin");
2017-07-16 14:55:37 -04:00
// if the service has never checked in, return false
2017-08-29 10:50:31 -04:00
if (!$checkin) {
2017-07-20 22:40:09 -04:00
return false;
}
2017-07-16 14:55:37 -04:00
// convert the check-in timestamp to a DateTime instance
$checkin = Date::normalize($checkin, "sql");
2017-07-16 14:55:37 -04:00
// get the checking interval
$int = Arsse::$conf->serviceFrequency;
// subtract twice the checking interval from the current time to yield the earliest acceptable check-in time
2022-05-31 23:55:04 -04:00
$limit = new \DateTime;
2017-07-16 14:55:37 -04:00
$limit->sub($int);
$limit->sub($int);
// return whether the check-in time is within the acceptable limit
2020-03-01 15:16:50 -05:00
return $checkin >= $limit;
2017-07-16 14:55:37 -04:00
}
2017-08-29 10:50:31 -04:00
public static function cleanupPre(): bool {
// mark unsubscribed feeds as orphaned and delete orphaned feeds that are beyond their retention period
Arsse::$db->feedCleanup();
2020-11-06 17:06:01 -05:00
// do the same for icons
Arsse::$db->iconCleanup();
// delete expired log-in sessions
Arsse::$db->sessionCleanup();
return true;
2017-07-16 14:55:37 -04:00
}
2017-08-29 10:50:31 -04:00
public static function cleanupPost(): bool {
2019-07-26 09:37:51 -04:00
// delete old articles, according to configured thresholds
$deleted = Arsse::$db->articleCleanup();
// if any articles were deleted, perform database maintenance
if ($deleted) {
Arsse::$db->driverMaintenance();
}
return true;
}
protected function signalInit(): void {
if (function_exists("pcntl_async_signals") && function_exists("pcntl_signal")) {
// receive asynchronous signals if supported
pcntl_async_signals(true);
foreach ([\SIGABRT, \SIGINT, \SIGTERM] as $sig) {
pcntl_signal($sig, [$this, "sigTerm"]);
}
2021-07-04 22:04:05 -04:00
pcntl_signal(\SIGHUP, [$this, "sigHup"]);
}
}
2021-06-06 19:44:36 -04:00
/** Changes the condition for the service loop upon receiving a termination signal
2021-07-05 21:47:44 -04:00
*
2021-06-06 19:44:36 -04:00
* @codeCoverageIgnore */
protected function sigTerm(int $signo): void {
$this->loop = false;
}
2021-07-04 22:04:05 -04:00
/** Changes the condition for the service loop upon receiving a hangup signal
2021-07-05 21:47:44 -04:00
*
2021-07-04 22:04:05 -04:00
* @codeCoverageIgnore */
protected function sigHup(int $signo): void {
2021-07-05 20:21:04 -04:00
$this->reload = true;
2021-07-04 22:04:05 -04:00
}
2017-08-29 10:50:31 -04:00
}