2019-05-10 15:21:41 +03:00
|
|
|
<?php
|
2019-12-03 21:57:53 +03:00
|
|
|
|
2019-05-10 15:21:41 +03:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
/**
|
2019-12-03 21:57:53 +03:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* @author Julius Härtl <jus@bitgrid.net>
|
2019-05-10 15:21:41 +03:00
|
|
|
* @author Morris Jobke <hey@morrisjobke.de>
|
2020-08-24 15:54:25 +03:00
|
|
|
* @author Roeland Jago Douma <roeland@famdouma.nl>
|
2019-05-10 15:21:41 +03:00
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* 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
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2019-05-10 15:21:41 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OC\Support\Subscription;
|
|
|
|
|
2020-10-08 12:41:16 +03:00
|
|
|
use OC\User\Backend;
|
2020-07-08 20:38:26 +03:00
|
|
|
use OCP\AppFramework\QueryException;
|
2020-03-06 15:08:51 +03:00
|
|
|
use OCP\IConfig;
|
2020-12-02 12:07:34 +03:00
|
|
|
use OCP\IGroupManager;
|
2020-07-08 20:38:26 +03:00
|
|
|
use OCP\IServerContainer;
|
2020-10-08 12:41:16 +03:00
|
|
|
use OCP\IUserManager;
|
2020-12-02 12:07:34 +03:00
|
|
|
use OCP\Notification\IManager;
|
2019-05-10 15:21:41 +03:00
|
|
|
use OCP\Support\Subscription\Exception\AlreadyRegisteredException;
|
|
|
|
use OCP\Support\Subscription\IRegistry;
|
|
|
|
use OCP\Support\Subscription\ISubscription;
|
|
|
|
use OCP\Support\Subscription\ISupportedApps;
|
2020-10-08 12:41:16 +03:00
|
|
|
use Psr\Log\LoggerInterface;
|
2019-05-10 15:21:41 +03:00
|
|
|
|
|
|
|
class Registry implements IRegistry {
|
|
|
|
|
|
|
|
/** @var ISubscription */
|
|
|
|
private $subscription = null;
|
|
|
|
|
2020-07-08 20:38:26 +03:00
|
|
|
/** @var string */
|
|
|
|
private $subscriptionService = null;
|
|
|
|
|
2020-03-06 15:08:51 +03:00
|
|
|
/** @var IConfig */
|
|
|
|
private $config;
|
|
|
|
|
2020-07-08 20:38:26 +03:00
|
|
|
/** @var IServerContainer */
|
|
|
|
private $container;
|
2020-10-08 12:41:16 +03:00
|
|
|
/** @var IUserManager */
|
|
|
|
private $userManager;
|
2020-12-02 12:07:34 +03:00
|
|
|
/** @var IGroupManager */
|
|
|
|
private $groupManager;
|
2020-10-08 12:41:16 +03:00
|
|
|
/** @var LoggerInterface */
|
|
|
|
private $logger;
|
2020-12-02 12:07:34 +03:00
|
|
|
/** @var IManager */
|
|
|
|
private $notificationManager;
|
2020-10-08 12:41:16 +03:00
|
|
|
|
|
|
|
public function __construct(IConfig $config,
|
|
|
|
IServerContainer $container,
|
|
|
|
IUserManager $userManager,
|
2020-12-02 12:07:34 +03:00
|
|
|
IGroupManager $groupManager,
|
|
|
|
LoggerInterface $logger,
|
|
|
|
IManager $notificationManager) {
|
2020-03-06 15:08:51 +03:00
|
|
|
$this->config = $config;
|
2020-07-08 20:38:26 +03:00
|
|
|
$this->container = $container;
|
2020-10-08 12:41:16 +03:00
|
|
|
$this->userManager = $userManager;
|
2020-12-02 12:07:34 +03:00
|
|
|
$this->groupManager = $groupManager;
|
2020-10-08 12:41:16 +03:00
|
|
|
$this->logger = $logger;
|
2020-12-02 12:07:34 +03:00
|
|
|
$this->notificationManager = $notificationManager;
|
2020-07-08 20:38:26 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private function getSubscription(): ?ISubscription {
|
|
|
|
if ($this->subscription === null && $this->subscriptionService !== null) {
|
|
|
|
try {
|
|
|
|
$this->subscription = $this->container->query($this->subscriptionService);
|
|
|
|
} catch (QueryException $e) {
|
|
|
|
// Ignore this
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->subscription;
|
2020-03-06 15:08:51 +03:00
|
|
|
}
|
|
|
|
|
2019-05-10 15:21:41 +03:00
|
|
|
/**
|
|
|
|
* Register a subscription instance. In case it is called multiple times the
|
|
|
|
* first one is used.
|
|
|
|
*
|
|
|
|
* @param ISubscription $subscription
|
|
|
|
* @throws AlreadyRegisteredException
|
|
|
|
*
|
|
|
|
* @since 17.0.0
|
|
|
|
*/
|
|
|
|
public function register(ISubscription $subscription): void {
|
2020-07-08 20:38:26 +03:00
|
|
|
if ($this->subscription !== null || $this->subscriptionService !== null) {
|
2019-05-10 15:21:41 +03:00
|
|
|
throw new AlreadyRegisteredException();
|
|
|
|
}
|
|
|
|
$this->subscription = $subscription;
|
|
|
|
}
|
|
|
|
|
2020-07-08 20:38:26 +03:00
|
|
|
public function registerService(string $subscriptionService): void {
|
|
|
|
if ($this->subscription !== null || $this->subscriptionService !== null) {
|
|
|
|
throw new AlreadyRegisteredException();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->subscriptionService = $subscriptionService;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 15:21:41 +03:00
|
|
|
/**
|
|
|
|
* Fetches the list of app IDs that are supported by the subscription
|
|
|
|
*
|
|
|
|
* @since 17.0.0
|
|
|
|
*/
|
|
|
|
public function delegateGetSupportedApps(): array {
|
2020-07-08 20:38:26 +03:00
|
|
|
if ($this->getSubscription() instanceof ISupportedApps) {
|
|
|
|
return $this->getSubscription()->getSupportedApps();
|
2019-05-10 15:21:41 +03:00
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if a valid subscription is available
|
|
|
|
*
|
|
|
|
* @since 17.0.0
|
|
|
|
*/
|
|
|
|
public function delegateHasValidSubscription(): bool {
|
2020-03-06 15:08:51 +03:00
|
|
|
// Allow overwriting this manually for environments where the subscription information cannot be fetched
|
|
|
|
if ($this->config->getSystemValueBool('has_valid_subscription')) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-08 20:38:26 +03:00
|
|
|
if ($this->getSubscription() instanceof ISubscription) {
|
|
|
|
return $this->getSubscription()->hasValidSubscription();
|
2019-05-10 15:21:41 +03:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2019-06-11 17:08:26 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if the subscription has extended support
|
|
|
|
*
|
|
|
|
* @since 17.0.0
|
|
|
|
*/
|
|
|
|
public function delegateHasExtendedSupport(): bool {
|
2020-10-08 12:41:16 +03:00
|
|
|
if ($this->getSubscription() instanceof ISubscription) {
|
2020-07-08 20:38:26 +03:00
|
|
|
return $this->getSubscription()->hasExtendedSupport();
|
2019-06-11 17:08:26 +03:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2020-10-08 12:41:16 +03:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if a hard user limit is reached and no new users should be created
|
|
|
|
*
|
|
|
|
* @since 21.0.0
|
|
|
|
*/
|
|
|
|
public function delegateIsHardUserLimitReached(): bool {
|
|
|
|
$subscription = $this->getSubscription();
|
|
|
|
if ($subscription instanceof ISubscription &&
|
|
|
|
$subscription->hasValidSubscription()) {
|
|
|
|
$userLimitReached = $subscription->isHardUserLimitReached();
|
|
|
|
if ($userLimitReached) {
|
|
|
|
$this->notifyAboutReachedUserLimit();
|
|
|
|
}
|
|
|
|
return $userLimitReached;
|
|
|
|
}
|
|
|
|
|
|
|
|
$isOneClickInstance = $this->config->getSystemValueBool('one-click-instance', false);
|
|
|
|
|
|
|
|
if (!$isOneClickInstance) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$userCount = $this->getUserCount();
|
|
|
|
$hardUserLimit = $this->config->getSystemValue('one-click-instance.user-limit', 50);
|
|
|
|
|
|
|
|
$userLimitReached = $userCount >= $hardUserLimit;
|
|
|
|
if ($userLimitReached) {
|
|
|
|
$this->notifyAboutReachedUserLimit();
|
|
|
|
}
|
|
|
|
return $userLimitReached;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getUserCount(): int {
|
|
|
|
$userCount = 0;
|
|
|
|
$backends = $this->userManager->getBackends();
|
|
|
|
foreach ($backends as $backend) {
|
|
|
|
if ($backend->implementsActions(Backend::COUNT_USERS)) {
|
|
|
|
$backendUsers = $backend->countUsers();
|
|
|
|
if ($backendUsers !== false) {
|
|
|
|
$userCount += $backendUsers;
|
|
|
|
} else {
|
|
|
|
// TODO what if the user count can't be determined?
|
|
|
|
$this->logger->warning('Can not determine user count for ' . get_class($backend), ['app' => 'lib']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$disabledUsers = $this->config->getUsersForUserValue('core', 'enabled', 'false');
|
|
|
|
$disabledUsersCount = count($disabledUsers);
|
|
|
|
$userCount = $userCount - $disabledUsersCount;
|
|
|
|
|
|
|
|
if ($userCount < 0) {
|
|
|
|
$userCount = 0;
|
|
|
|
|
|
|
|
// this should never happen
|
|
|
|
$this->logger->warning("Total user count was negative (users: $userCount, disabled: $disabledUsersCount)", ['app' => 'lib']);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $userCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function notifyAboutReachedUserLimit() {
|
2020-12-02 12:07:34 +03:00
|
|
|
$admins = $this->groupManager->get('admin')->getUsers();
|
|
|
|
foreach ($admins as $admin) {
|
|
|
|
$notification = $this->notificationManager->createNotification();
|
|
|
|
|
|
|
|
$notification->setApp('core')
|
|
|
|
->setUser($admin->getUID())
|
|
|
|
->setDateTime(new \DateTime())
|
|
|
|
->setObject('user_limit_reached', '1')
|
|
|
|
->setSubject('user_limit_reached');
|
|
|
|
$this->notificationManager->notify($notification);
|
|
|
|
}
|
|
|
|
|
2020-10-08 12:41:16 +03:00
|
|
|
$this->logger->warning('The user limit was reached and the new user was not created', ['app' => 'lib']);
|
|
|
|
}
|
2019-05-10 15:21:41 +03:00
|
|
|
}
|