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