2016-05-04 16:26:48 +03:00
|
|
|
<?php
|
2019-12-03 21:57:53 +03:00
|
|
|
|
2018-01-17 15:42:02 +03:00
|
|
|
declare(strict_types=1);
|
2019-12-03 21:57:53 +03:00
|
|
|
|
2016-05-04 16:26:48 +03:00
|
|
|
/**
|
|
|
|
* @copyright Copyright (c) 2016, ownCloud, Inc.
|
2016-07-21 17:49:16 +03:00
|
|
|
*
|
2020-04-29 12:57:22 +03:00
|
|
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
2016-07-21 17:49:16 +03:00
|
|
|
* @author Joas Schilling <coding@schilljs.com>
|
2017-11-06 17:56:42 +03:00
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
2016-07-21 17:49:16 +03:00
|
|
|
*
|
2016-05-04 16:26:48 +03:00
|
|
|
* @license AGPL-3.0
|
|
|
|
*
|
|
|
|
* This code is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
|
|
* as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License, version 3,
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
2016-05-04 16:26:48 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OCA\UpdateNotification\Notification;
|
|
|
|
|
|
|
|
use OC\BackgroundJob\TimedJob;
|
|
|
|
use OC\Installer;
|
|
|
|
use OC\Updater\VersionCheck;
|
|
|
|
use OCP\App\IAppManager;
|
|
|
|
use OCP\Http\Client\IClientService;
|
|
|
|
use OCP\IConfig;
|
2016-05-09 10:43:06 +03:00
|
|
|
use OCP\IGroup;
|
2016-05-04 16:26:48 +03:00
|
|
|
use OCP\IGroupManager;
|
|
|
|
use OCP\Notification\IManager;
|
|
|
|
|
|
|
|
class BackgroundJob extends TimedJob {
|
2017-03-17 14:47:26 +03:00
|
|
|
protected $connectionNotifications = [3, 7, 14, 30];
|
|
|
|
|
2016-05-04 16:26:48 +03:00
|
|
|
/** @var IConfig */
|
|
|
|
protected $config;
|
|
|
|
|
|
|
|
/** @var IManager */
|
|
|
|
protected $notificationManager;
|
|
|
|
|
|
|
|
/** @var IGroupManager */
|
|
|
|
protected $groupManager;
|
|
|
|
|
|
|
|
/** @var IAppManager */
|
|
|
|
protected $appManager;
|
|
|
|
|
|
|
|
/** @var IClientService */
|
|
|
|
protected $client;
|
|
|
|
|
2017-11-24 12:41:51 +03:00
|
|
|
/** @var Installer */
|
|
|
|
protected $installer;
|
|
|
|
|
2017-01-09 13:18:59 +03:00
|
|
|
/** @var string[] */
|
2016-05-04 16:26:48 +03:00
|
|
|
protected $users;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* NotificationBackgroundJob constructor.
|
|
|
|
*
|
|
|
|
* @param IConfig $config
|
|
|
|
* @param IManager $notificationManager
|
|
|
|
* @param IGroupManager $groupManager
|
|
|
|
* @param IAppManager $appManager
|
|
|
|
* @param IClientService $client
|
2017-11-24 12:41:51 +03:00
|
|
|
* @param Installer $installer
|
2016-05-04 16:26:48 +03:00
|
|
|
*/
|
2017-11-24 12:41:51 +03:00
|
|
|
public function __construct(IConfig $config, IManager $notificationManager, IGroupManager $groupManager, IAppManager $appManager, IClientService $client, Installer $installer) {
|
2016-05-04 16:26:48 +03:00
|
|
|
// Run once a day
|
|
|
|
$this->setInterval(60 * 60 * 24);
|
|
|
|
|
|
|
|
$this->config = $config;
|
|
|
|
$this->notificationManager = $notificationManager;
|
|
|
|
$this->groupManager = $groupManager;
|
|
|
|
$this->appManager = $appManager;
|
|
|
|
$this->client = $client;
|
2017-11-24 12:41:51 +03:00
|
|
|
$this->installer = $installer;
|
2016-05-04 16:26:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function run($argument) {
|
|
|
|
$this->checkCoreUpdate();
|
|
|
|
$this->checkAppUpdates();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check for ownCloud update
|
|
|
|
*/
|
|
|
|
protected function checkCoreUpdate() {
|
2018-01-17 15:42:02 +03:00
|
|
|
if (\in_array($this->getChannel(), ['daily', 'git'], true)) {
|
2016-05-04 16:26:48 +03:00
|
|
|
// "These aren't the update channels you're looking for." - Ben Obi-Wan Kenobi
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-05-09 13:04:32 +03:00
|
|
|
$updater = $this->createVersionCheck();
|
2016-05-04 16:26:48 +03:00
|
|
|
|
|
|
|
$status = $updater->check();
|
2017-03-17 14:47:26 +03:00
|
|
|
if ($status === false) {
|
|
|
|
$errors = 1 + (int) $this->config->getAppValue('updatenotification', 'update_check_errors', 0);
|
|
|
|
$this->config->setAppValue('updatenotification', 'update_check_errors', $errors);
|
|
|
|
|
2018-01-17 15:42:02 +03:00
|
|
|
if (\in_array($errors, $this->connectionNotifications, true)) {
|
2017-03-17 14:47:26 +03:00
|
|
|
$this->sendErrorNotifications($errors);
|
|
|
|
}
|
2020-04-10 11:35:09 +03:00
|
|
|
} elseif (\is_array($status)) {
|
2017-03-17 14:47:26 +03:00
|
|
|
$this->config->setAppValue('updatenotification', 'update_check_errors', 0);
|
|
|
|
$this->clearErrorNotifications();
|
|
|
|
|
2017-03-17 23:40:39 +03:00
|
|
|
if (isset($status['version'])) {
|
|
|
|
$this->createNotifications('core', $status['version'], $status['versionstring']);
|
|
|
|
}
|
2016-05-04 16:26:48 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-17 14:47:26 +03:00
|
|
|
/**
|
|
|
|
* Send a message to the admin when the update server could not be reached
|
|
|
|
* @param int $numDays
|
|
|
|
*/
|
|
|
|
protected function sendErrorNotifications($numDays) {
|
|
|
|
$this->clearErrorNotifications();
|
|
|
|
|
|
|
|
$notification = $this->notificationManager->createNotification();
|
|
|
|
try {
|
|
|
|
$notification->setApp('updatenotification')
|
|
|
|
->setDateTime(new \DateTime())
|
|
|
|
->setObject('updatenotification', 'error')
|
|
|
|
->setSubject('connection_error', ['days' => $numDays]);
|
|
|
|
|
|
|
|
foreach ($this->getUsersToNotify() as $uid) {
|
|
|
|
$notification->setUser($uid);
|
|
|
|
$this->notificationManager->notify($notification);
|
|
|
|
}
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove error notifications again
|
|
|
|
*/
|
|
|
|
protected function clearErrorNotifications() {
|
|
|
|
$notification = $this->notificationManager->createNotification();
|
|
|
|
try {
|
|
|
|
$notification->setApp('updatenotification')
|
|
|
|
->setSubject('connection_error')
|
|
|
|
->setObject('updatenotification', 'error');
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$this->notificationManager->markProcessed($notification);
|
|
|
|
}
|
|
|
|
|
2016-05-04 16:26:48 +03:00
|
|
|
/**
|
|
|
|
* Check all installed apps for updates
|
|
|
|
*/
|
|
|
|
protected function checkAppUpdates() {
|
|
|
|
$apps = $this->appManager->getInstalledApps();
|
|
|
|
foreach ($apps as $app) {
|
2016-05-09 13:04:32 +03:00
|
|
|
$update = $this->isUpdateAvailable($app);
|
2016-05-04 16:26:48 +03:00
|
|
|
if ($update !== false) {
|
2017-01-09 13:18:59 +03:00
|
|
|
$this->createNotifications($app, $update);
|
2016-05-04 16:26:48 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create notifications for this app version
|
|
|
|
*
|
|
|
|
* @param string $app
|
|
|
|
* @param string $version
|
2016-10-04 13:48:17 +03:00
|
|
|
* @param string $visibleVersion
|
2016-05-04 16:26:48 +03:00
|
|
|
*/
|
2017-01-09 13:18:59 +03:00
|
|
|
protected function createNotifications($app, $version, $visibleVersion = '') {
|
2016-05-04 16:26:48 +03:00
|
|
|
$lastNotification = $this->config->getAppValue('updatenotification', $app, false);
|
|
|
|
if ($lastNotification === $version) {
|
|
|
|
// We already notified about this update
|
|
|
|
return;
|
2018-01-17 15:42:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($lastNotification !== false) {
|
2016-05-04 16:26:48 +03:00
|
|
|
// Delete old updates
|
|
|
|
$this->deleteOutdatedNotifications($app, $lastNotification);
|
|
|
|
}
|
|
|
|
|
|
|
|
$notification = $this->notificationManager->createNotification();
|
2018-01-17 15:42:02 +03:00
|
|
|
try {
|
|
|
|
$notification->setApp('updatenotification')
|
|
|
|
->setDateTime(new \DateTime())
|
|
|
|
->setObject($app, $version);
|
2016-10-04 13:48:17 +03:00
|
|
|
|
2018-01-17 15:42:02 +03:00
|
|
|
if ($visibleVersion !== '') {
|
|
|
|
$notification->setSubject('update_available', ['version' => $visibleVersion]);
|
|
|
|
} else {
|
|
|
|
$notification->setSubject('update_available');
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this->getUsersToNotify() as $uid) {
|
|
|
|
$notification->setUser($uid);
|
|
|
|
$this->notificationManager->notify($notification);
|
|
|
|
}
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
return;
|
2016-05-04 16:26:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->config->setAppValue('updatenotification', $app, $version);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-05-09 13:04:32 +03:00
|
|
|
* @return string[]
|
2016-05-04 16:26:48 +03:00
|
|
|
*/
|
2018-01-17 15:42:02 +03:00
|
|
|
protected function getUsersToNotify(): array {
|
2016-05-04 16:26:48 +03:00
|
|
|
if ($this->users !== null) {
|
|
|
|
return $this->users;
|
|
|
|
}
|
|
|
|
|
2018-01-17 15:42:02 +03:00
|
|
|
$notifyGroups = (array) json_decode($this->config->getAppValue('updatenotification', 'notify_groups', '["admin"]'), true);
|
2016-05-09 13:04:32 +03:00
|
|
|
$this->users = [];
|
2016-05-09 10:43:06 +03:00
|
|
|
foreach ($notifyGroups as $group) {
|
|
|
|
$groupToNotify = $this->groupManager->get($group);
|
|
|
|
if ($groupToNotify instanceof IGroup) {
|
2016-05-09 13:04:32 +03:00
|
|
|
foreach ($groupToNotify->getUsers() as $user) {
|
|
|
|
$this->users[$user->getUID()] = true;
|
|
|
|
}
|
2016-05-09 10:43:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-09 13:04:32 +03:00
|
|
|
$this->users = array_keys($this->users);
|
|
|
|
|
2016-05-04 16:26:48 +03:00
|
|
|
return $this->users;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete notifications for old updates
|
|
|
|
*
|
|
|
|
* @param string $app
|
|
|
|
* @param string $version
|
|
|
|
*/
|
|
|
|
protected function deleteOutdatedNotifications($app, $version) {
|
|
|
|
$notification = $this->notificationManager->createNotification();
|
2018-01-17 15:42:02 +03:00
|
|
|
try {
|
|
|
|
$notification->setApp('updatenotification')
|
|
|
|
->setObject($app, $version);
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
return;
|
|
|
|
}
|
2016-05-04 16:26:48 +03:00
|
|
|
$this->notificationManager->markProcessed($notification);
|
|
|
|
}
|
2016-05-09 13:04:32 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return VersionCheck
|
|
|
|
*/
|
2018-01-17 15:42:02 +03:00
|
|
|
protected function createVersionCheck(): VersionCheck {
|
2016-05-09 13:04:32 +03:00
|
|
|
return new VersionCheck(
|
|
|
|
$this->client,
|
|
|
|
$this->config
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2018-01-17 15:42:02 +03:00
|
|
|
protected function getChannel(): string {
|
2016-05-09 13:04:32 +03:00
|
|
|
return \OC_Util::getChannel();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $app
|
|
|
|
* @return string|false
|
|
|
|
*/
|
|
|
|
protected function isUpdateAvailable($app) {
|
2017-11-24 12:41:51 +03:00
|
|
|
return $this->installer->isUpdateAvailable($app);
|
2016-05-09 13:04:32 +03:00
|
|
|
}
|
2016-05-04 16:26:48 +03:00
|
|
|
}
|