serviceFrequency); } catch (\Exception $e) { return new \DateInterval("PT2M"); } } public function __construct() { $driver = Arsse::$conf->serviceDriver; $this->drv = new $driver(); $this->interval = static::interval(); } public function watch(bool $loop = true): \DateTimeInterface { $t = new \DateTime(); do { $this->checkIn(); static::cleanupPre(); $list = Arsse::$db->feedListStale(); if ($list) { $this->drv->queue(...$list); $this->drv->exec(); $this->drv->clean(); unset($list); } static::cleanupPost(); $t->add($this->interval); if ($loop) { do { @time_sleep_until($t->getTimestamp()); } while ($t->getTimestamp() > time()); } } while ($loop); return $t; } public function checkIn(): bool { return Arsse::$db->metaSet("service_last_checkin", time(), "datetime"); } public static function hasCheckedIn(): bool { $checkin = Arsse::$db->metaGet("service_last_checkin"); // if the service has never checked in, return false if (!$checkin) { return false; } // convert the check-in timestamp to a DateTime instance $checkin = Date::normalize($checkin, "sql"); // get the checking interval $int = static::interval(); // subtract twice the checking interval from the current time to the earliest acceptable check-in time $limit = new \DateTime(); $limit->sub($int); $limit->sub($int); // return whether the check-in time is within the acceptable limit return ($checkin >= $limit); } public static function cleanupPre(): bool { // mark unsubscribed feeds as orphaned and delete orphaned feeds that are beyond their retention period return Arsse::$db->feedCleanup(); } public static function cleanupPost(): bool { // delete old articles, according to configured threasholds return Arsse::$db->articleCleanup(); } }