Fix state propragation of the backup codes provider
Starting with Nextcloud 14, the server knows the enabled/disabled state of 2fa providers. While it will query that information if it's unknown (on first use), it won't notice any changes. Thus, providers have to propagate that information themselves. Ref https://github.com/nextcloud/twofactor_totp/pull/263 Ref https://github.com/nextcloud/twofactor_u2f/pull/210 Signed-off-by: Christoph Wurst <christoph@winzerhof-wurst.at>
This commit is contained in:
parent
b5c5faebfc
commit
f8fe7488a5
|
@ -11,6 +11,10 @@ return array(
|
||||||
'OCA\\TwoFactorBackupCodes\\Controller\\SettingsController' => $baseDir . '/../lib/Controller/SettingsController.php',
|
'OCA\\TwoFactorBackupCodes\\Controller\\SettingsController' => $baseDir . '/../lib/Controller/SettingsController.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Db\\BackupCode' => $baseDir . '/../lib/Db/BackupCode.php',
|
'OCA\\TwoFactorBackupCodes\\Db\\BackupCode' => $baseDir . '/../lib/Db/BackupCode.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Db\\BackupCodeMapper' => $baseDir . '/../lib/Db/BackupCodeMapper.php',
|
'OCA\\TwoFactorBackupCodes\\Db\\BackupCodeMapper' => $baseDir . '/../lib/Db/BackupCodeMapper.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Event\\CodesGenerated' => $baseDir . '/../lib/Event/CodesGenerated.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\ActivityPublisher' => $baseDir . '/../lib/Listener/ActivityPublisher.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\IListener' => $baseDir . '/../lib/Listener/IListener.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\RegistryUpdater' => $baseDir . '/../lib/Listener/RegistryUpdater.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607104347' => $baseDir . '/../lib/Migration/Version1002Date20170607104347.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607104347' => $baseDir . '/../lib/Migration/Version1002Date20170607104347.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607113030' => $baseDir . '/../lib/Migration/Version1002Date20170607113030.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607113030' => $baseDir . '/../lib/Migration/Version1002Date20170607113030.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170919123342' => $baseDir . '/../lib/Migration/Version1002Date20170919123342.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170919123342' => $baseDir . '/../lib/Migration/Version1002Date20170919123342.php',
|
||||||
|
|
|
@ -26,6 +26,10 @@ class ComposerStaticInitTwoFactorBackupCodes
|
||||||
'OCA\\TwoFactorBackupCodes\\Controller\\SettingsController' => __DIR__ . '/..' . '/../lib/Controller/SettingsController.php',
|
'OCA\\TwoFactorBackupCodes\\Controller\\SettingsController' => __DIR__ . '/..' . '/../lib/Controller/SettingsController.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Db\\BackupCode' => __DIR__ . '/..' . '/../lib/Db/BackupCode.php',
|
'OCA\\TwoFactorBackupCodes\\Db\\BackupCode' => __DIR__ . '/..' . '/../lib/Db/BackupCode.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Db\\BackupCodeMapper' => __DIR__ . '/..' . '/../lib/Db/BackupCodeMapper.php',
|
'OCA\\TwoFactorBackupCodes\\Db\\BackupCodeMapper' => __DIR__ . '/..' . '/../lib/Db/BackupCodeMapper.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Event\\CodesGenerated' => __DIR__ . '/..' . '/../lib/Event/CodesGenerated.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\ActivityPublisher' => __DIR__ . '/..' . '/../lib/Listener/ActivityPublisher.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\IListener' => __DIR__ . '/..' . '/../lib/Listener/IListener.php',
|
||||||
|
'OCA\\TwoFactorBackupCodes\\Listener\\RegistryUpdater' => __DIR__ . '/..' . '/../lib/Listener/RegistryUpdater.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607104347' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170607104347.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607104347' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170607104347.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607113030' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170607113030.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170607113030' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170607113030.php',
|
||||||
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170919123342' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170919123342.php',
|
'OCA\\TwoFactorBackupCodes\\Migration\\Version1002Date20170919123342' => __DIR__ . '/..' . '/../lib/Migration/Version1002Date20170919123342.php',
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @copyright Copyright (c) 2017 Joas Schilling <coding@schilljs.com>
|
* @copyright Copyright (c) 2017 Joas Schilling <coding@schilljs.com>
|
||||||
*
|
*
|
||||||
* @author Joas Schilling <coding@schilljs.com>
|
* @author Joas Schilling <coding@schilljs.com>
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
*
|
*
|
||||||
* @license GNU AGPL version 3 or any later version
|
* @license GNU AGPL version 3 or any later version
|
||||||
*
|
*
|
||||||
|
@ -24,11 +26,16 @@
|
||||||
namespace OCA\TwoFactorBackupCodes\AppInfo;
|
namespace OCA\TwoFactorBackupCodes\AppInfo;
|
||||||
|
|
||||||
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCA\TwoFactorBackupCodes\Listener\ActivityPublisher;
|
||||||
|
use OCA\TwoFactorBackupCodes\Listener\IListener;
|
||||||
|
use OCA\TwoFactorBackupCodes\Listener\RegistryUpdater;
|
||||||
use OCP\AppFramework\App;
|
use OCP\AppFramework\App;
|
||||||
use OCP\Util;
|
use OCP\Util;
|
||||||
|
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
|
||||||
|
|
||||||
class Application extends App {
|
class Application extends App {
|
||||||
public function __construct () {
|
public function __construct() {
|
||||||
parent::__construct('twofactor_backupcodes');
|
parent::__construct('twofactor_backupcodes');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,6 +51,21 @@ class Application extends App {
|
||||||
*/
|
*/
|
||||||
public function registerHooksAndEvents() {
|
public function registerHooksAndEvents() {
|
||||||
Util::connectHook('OC_User', 'post_deleteUser', $this, 'deleteUser');
|
Util::connectHook('OC_User', 'post_deleteUser', $this, 'deleteUser');
|
||||||
|
|
||||||
|
$container = $this->getContainer();
|
||||||
|
/** @var EventDispatcherInterface $eventDispatcher */
|
||||||
|
$eventDispatcher = $container->query(EventDispatcherInterface::class);
|
||||||
|
$eventDispatcher->addListener(CodesGenerated::class, function (CodesGenerated $event) use ($container) {
|
||||||
|
/** @var IListener[] $listeners */
|
||||||
|
$listeners = [
|
||||||
|
$container->query(ActivityPublisher::class),
|
||||||
|
$container->query(RegistryUpdater::class),
|
||||||
|
];
|
||||||
|
|
||||||
|
foreach ($listeners as $listener) {
|
||||||
|
$listener->handle($event);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public function deleteUser($params) {
|
public function deleteUser($params) {
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Event;
|
||||||
|
|
||||||
|
use OCP\IUser;
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
|
||||||
|
class CodesGenerated extends Event {
|
||||||
|
|
||||||
|
/** @var IUser */
|
||||||
|
private $user;
|
||||||
|
|
||||||
|
public function __construct(IUser $user) {
|
||||||
|
$this->user = $user;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return IUser
|
||||||
|
*/
|
||||||
|
public function getUser(): IUser {
|
||||||
|
return $this->user;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,66 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Listener;
|
||||||
|
|
||||||
|
use BadMethodCallException;
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCP\Activity\IManager;
|
||||||
|
use OCP\ILogger;
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
|
||||||
|
class ActivityPublisher implements IListener {
|
||||||
|
|
||||||
|
/** @var IManager */
|
||||||
|
private $activityManager;
|
||||||
|
|
||||||
|
/** @var ILogger */
|
||||||
|
private $logger;
|
||||||
|
|
||||||
|
public function __construct(IManager $activityManager, ILogger $logger) {
|
||||||
|
$this->activityManager = $activityManager;
|
||||||
|
$this->logger = $logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Push an event to the user's activity stream
|
||||||
|
*/
|
||||||
|
public function handle(Event $event) {
|
||||||
|
if ($event instanceof CodesGenerated) {
|
||||||
|
$activity = $this->activityManager->generateEvent();
|
||||||
|
$activity->setApp('twofactor_backupcodes')
|
||||||
|
->setType('security')
|
||||||
|
->setAuthor($event->getUser()->getUID())
|
||||||
|
->setAffectedUser($event->getUser()->getUID())
|
||||||
|
->setSubject('codes_generated');
|
||||||
|
try {
|
||||||
|
$this->activityManager->publish($activity);
|
||||||
|
} catch (BadMethodCallException $e) {
|
||||||
|
$this->logger->warning('could not publish backup code creation activity', ['app' => 'twofactor_backupcodes']);
|
||||||
|
$this->logger->logException($e, ['app' => 'twofactor_backupcodes']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,33 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Listener;
|
||||||
|
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
|
||||||
|
interface IListener {
|
||||||
|
|
||||||
|
public function handle(Event $event);
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Listener;
|
||||||
|
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCA\TwoFactorBackupCodes\Provider\BackupCodesProvider;
|
||||||
|
use OCP\Authentication\TwoFactorAuth\IRegistry;
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
|
||||||
|
class RegistryUpdater implements IListener {
|
||||||
|
|
||||||
|
/** @var IRegistry */
|
||||||
|
private $registry;
|
||||||
|
|
||||||
|
/** @var BackupCodesProvider */
|
||||||
|
private $provider;
|
||||||
|
|
||||||
|
public function __construct(IRegistry $registry, BackupCodesProvider $provider) {
|
||||||
|
$this->registry = $registry;
|
||||||
|
$this->provider = $provider;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function handle(Event $event) {
|
||||||
|
if ($event instanceof CodesGenerated) {
|
||||||
|
$this->registry->enableProviderFor($this->provider, $event->getUser());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,11 +25,13 @@ namespace OCA\TwoFactorBackupCodes\Service;
|
||||||
use BadMethodCallException;
|
use BadMethodCallException;
|
||||||
use OCA\TwoFactorBackupCodes\Db\BackupCode;
|
use OCA\TwoFactorBackupCodes\Db\BackupCode;
|
||||||
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
use OCP\Activity\IManager;
|
use OCP\Activity\IManager;
|
||||||
use OCP\ILogger;
|
use OCP\ILogger;
|
||||||
use OCP\IUser;
|
use OCP\IUser;
|
||||||
use OCP\Security\IHasher;
|
use OCP\Security\IHasher;
|
||||||
use OCP\Security\ISecureRandom;
|
use OCP\Security\ISecureRandom;
|
||||||
|
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
|
||||||
|
|
||||||
class BackupCodeStorage {
|
class BackupCodeStorage {
|
||||||
|
|
||||||
|
@ -44,26 +46,17 @@ class BackupCodeStorage {
|
||||||
/** @var ISecureRandom */
|
/** @var ISecureRandom */
|
||||||
private $random;
|
private $random;
|
||||||
|
|
||||||
/** @var IManager */
|
/** @var EventDispatcherInterface */
|
||||||
private $activityManager;
|
private $eventDispatcher;
|
||||||
|
|
||||||
/** @var ILogger */
|
public function __construct(BackupCodeMapper $mapper,
|
||||||
private $logger;
|
ISecureRandom $random,
|
||||||
|
IHasher $hasher,
|
||||||
/**
|
EventDispatcherInterface $eventDispatcher) {
|
||||||
* @param BackupCodeMapper $mapper
|
|
||||||
* @param ISecureRandom $random
|
|
||||||
* @param IHasher $hasher
|
|
||||||
* @param IManager $activityManager
|
|
||||||
* @param ILogger $logger
|
|
||||||
*/
|
|
||||||
public function __construct(BackupCodeMapper $mapper, ISecureRandom $random, IHasher $hasher,
|
|
||||||
IManager $activityManager, ILogger $logger) {
|
|
||||||
$this->mapper = $mapper;
|
$this->mapper = $mapper;
|
||||||
$this->hasher = $hasher;
|
$this->hasher = $hasher;
|
||||||
$this->random = $random;
|
$this->random = $random;
|
||||||
$this->activityManager = $activityManager;
|
$this->eventDispatcher = $eventDispatcher;
|
||||||
$this->logger = $logger;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -89,32 +82,11 @@ class BackupCodeStorage {
|
||||||
$result[] = $code;
|
$result[] = $code;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->publishEvent($user, 'codes_generated');
|
$this->eventDispatcher->dispatch(CodesGenerated::class, new CodesGenerated($user));
|
||||||
|
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Push an event the user's activity stream
|
|
||||||
*
|
|
||||||
* @param IUser $user
|
|
||||||
* @param string $event
|
|
||||||
*/
|
|
||||||
private function publishEvent(IUser $user, $event) {
|
|
||||||
$activity = $this->activityManager->generateEvent();
|
|
||||||
$activity->setApp('twofactor_backupcodes')
|
|
||||||
->setType('security')
|
|
||||||
->setAuthor($user->getUID())
|
|
||||||
->setAffectedUser($user->getUID())
|
|
||||||
->setSubject($event);
|
|
||||||
try {
|
|
||||||
$this->activityManager->publish($activity);
|
|
||||||
} catch (BadMethodCallException $e) {
|
|
||||||
$this->logger->warning('could not publish backup code creation activity', ['app' => 'twofactor_backupcodes']);
|
|
||||||
$this->logger->logException($e, ['app' => 'twofactor_backupcodes']);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param IUser $user
|
* @param IUser $user
|
||||||
* @return bool
|
* @return bool
|
||||||
|
@ -133,7 +105,7 @@ class BackupCodeStorage {
|
||||||
$total = count($codes);
|
$total = count($codes);
|
||||||
$used = 0;
|
$used = 0;
|
||||||
array_walk($codes, function (BackupCode $code) use (&$used) {
|
array_walk($codes, function (BackupCode $code) use (&$used) {
|
||||||
if (1 === (int) $code->getUsed()) {
|
if (1 === (int)$code->getUsed()) {
|
||||||
$used++;
|
$used++;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -153,7 +125,7 @@ class BackupCodeStorage {
|
||||||
$dbCodes = $this->mapper->getBackupCodes($user);
|
$dbCodes = $this->mapper->getBackupCodes($user);
|
||||||
|
|
||||||
foreach ($dbCodes as $dbCode) {
|
foreach ($dbCodes as $dbCode) {
|
||||||
if (0 === (int) $dbCode->getUsed() && $this->hasher->verify($code, $dbCode->getCode())) {
|
if (0 === (int)$dbCode->getUsed() && $this->hasher->verify($code, $dbCode->getCode())) {
|
||||||
$dbCode->setUsed(1);
|
$dbCode->setUsed(1);
|
||||||
$this->mapper->update($dbCode);
|
$this->mapper->update($dbCode);
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Tests\Unit\Event;
|
||||||
|
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCP\IUser;
|
||||||
|
use Test\TestCase;
|
||||||
|
|
||||||
|
class CodesGeneratedTest extends TestCase {
|
||||||
|
|
||||||
|
public function testCodeGeneratedEvent() {
|
||||||
|
$user = $this->createMock(IUser::class);
|
||||||
|
|
||||||
|
$event = new CodesGenerated($user);
|
||||||
|
|
||||||
|
$this->assertSame($user, $event->getUser());
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,97 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Tests\Unit\Listener;
|
||||||
|
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCA\TwoFactorBackupCodes\Listener\ActivityPublisher;
|
||||||
|
use OCP\Activity\IEvent;
|
||||||
|
use OCP\Activity\IManager;
|
||||||
|
use OCP\ILogger;
|
||||||
|
use OCP\IUser;
|
||||||
|
use PHPUnit_Framework_MockObject_MockObject;
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
use Test\TestCase;
|
||||||
|
|
||||||
|
class ActivityPublisherTest extends TestCase {
|
||||||
|
|
||||||
|
/** @var IManager|PHPUnit_Framework_MockObject_MockObject */
|
||||||
|
private $activityManager;
|
||||||
|
|
||||||
|
/** @var ILogger */
|
||||||
|
|
||||||
|
private $logger;
|
||||||
|
|
||||||
|
/** @var ActivityPublisher */
|
||||||
|
private $listener;
|
||||||
|
|
||||||
|
protected function setUp() {
|
||||||
|
parent::setUp();
|
||||||
|
|
||||||
|
$this->activityManager = $this->createMock(IManager::class);
|
||||||
|
$this->logger = $this->createMock(ILogger::class);
|
||||||
|
|
||||||
|
$this->listener = new ActivityPublisher($this->activityManager, $this->logger);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testHandleGenericEvent() {
|
||||||
|
$event = $this->createMock(Event::class);
|
||||||
|
$this->activityManager->expects($this->never())
|
||||||
|
->method('publish');
|
||||||
|
|
||||||
|
$this->listener->handle($event);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testHandleCodesGeneratedEvent() {
|
||||||
|
$user = $this->createMock(IUser::class);
|
||||||
|
$user->method('getUID')->willReturn('fritz');
|
||||||
|
$event = new CodesGenerated($user);
|
||||||
|
$activityEvent = $this->createMock(IEvent::class);
|
||||||
|
$this->activityManager->expects($this->once())
|
||||||
|
->method('generateEvent')
|
||||||
|
->will($this->returnValue($activityEvent));
|
||||||
|
$activityEvent->expects($this->once())
|
||||||
|
->method('setApp')
|
||||||
|
->with('twofactor_backupcodes')
|
||||||
|
->will($this->returnSelf());
|
||||||
|
$activityEvent->expects($this->once())
|
||||||
|
->method('setType')
|
||||||
|
->with('security')
|
||||||
|
->will($this->returnSelf());
|
||||||
|
$activityEvent->expects($this->once())
|
||||||
|
->method('setAuthor')
|
||||||
|
->with('fritz')
|
||||||
|
->will($this->returnSelf());
|
||||||
|
$activityEvent->expects($this->once())
|
||||||
|
->method('setAffectedUser')
|
||||||
|
->with('fritz')
|
||||||
|
->will($this->returnSelf());
|
||||||
|
$this->activityManager->expects($this->once())
|
||||||
|
->method('publish')
|
||||||
|
->will($this->returnValue($activityEvent));
|
||||||
|
|
||||||
|
$this->listener->handle($event);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,75 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
||||||
|
*
|
||||||
|
* @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
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace OCA\TwoFactorBackupCodes\Tests\Unit\Listener;
|
||||||
|
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
|
use OCA\TwoFactorBackupCodes\Listener\RegistryUpdater;
|
||||||
|
use OCA\TwoFactorBackupCodes\Provider\BackupCodesProvider;
|
||||||
|
use OCP\Authentication\TwoFactorAuth\IRegistry;
|
||||||
|
use OCP\IUser;
|
||||||
|
use Symfony\Component\EventDispatcher\Event;
|
||||||
|
use Test\TestCase;
|
||||||
|
|
||||||
|
class RegistryUpdaterTest extends TestCase {
|
||||||
|
|
||||||
|
/** @var IRegistry */
|
||||||
|
private $registry;
|
||||||
|
|
||||||
|
/** @var BackupCodesProvider */
|
||||||
|
private $provider;
|
||||||
|
|
||||||
|
/** @var RegistryUpdater */
|
||||||
|
private $listener;
|
||||||
|
|
||||||
|
protected function setUp() {
|
||||||
|
parent::setUp();
|
||||||
|
|
||||||
|
$this->registry = $this->createMock(IRegistry::class);
|
||||||
|
$this->provider = $this->createMock(BackupCodesProvider::class);
|
||||||
|
|
||||||
|
$this->listener = new RegistryUpdater($this->registry, $this->provider);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testHandleGenericEvent() {
|
||||||
|
$event = $this->createMock(Event::class);
|
||||||
|
$this->registry->expects($this->never())
|
||||||
|
->method('enableProviderFor');
|
||||||
|
|
||||||
|
$this->listener->handle($event);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testHandleCodesGeneratedEvent() {
|
||||||
|
$user = $this->createMock(IUser::class);
|
||||||
|
$event = new CodesGenerated($user);
|
||||||
|
$this->registry->expects($this->once())
|
||||||
|
->method('enableProviderFor')
|
||||||
|
->with(
|
||||||
|
$this->provider,
|
||||||
|
$user
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->listener->handle($event);
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,14 +24,13 @@ namespace OCA\TwoFactorBackupCodes\Tests\Unit\Service;
|
||||||
|
|
||||||
use OCA\TwoFactorBackupCodes\Db\BackupCode;
|
use OCA\TwoFactorBackupCodes\Db\BackupCode;
|
||||||
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
use OCA\TwoFactorBackupCodes\Db\BackupCodeMapper;
|
||||||
|
use OCA\TwoFactorBackupCodes\Event\CodesGenerated;
|
||||||
use OCA\TwoFactorBackupCodes\Service\BackupCodeStorage;
|
use OCA\TwoFactorBackupCodes\Service\BackupCodeStorage;
|
||||||
use OCP\Activity\IEvent;
|
|
||||||
use OCP\Activity\IManager;
|
|
||||||
use OCP\ILogger;
|
|
||||||
use OCP\IUser;
|
use OCP\IUser;
|
||||||
use OCP\Security\IHasher;
|
use OCP\Security\IHasher;
|
||||||
use OCP\Security\ISecureRandom;
|
use OCP\Security\ISecureRandom;
|
||||||
use PHPUnit_Framework_MockObject_MockObject;
|
use PHPUnit_Framework_MockObject_MockObject;
|
||||||
|
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
|
||||||
use Test\TestCase;
|
use Test\TestCase;
|
||||||
|
|
||||||
class BackupCodeStorageTest extends TestCase {
|
class BackupCodeStorageTest extends TestCase {
|
||||||
|
@ -45,11 +44,8 @@ class BackupCodeStorageTest extends TestCase {
|
||||||
/** @var IHasher|PHPUnit_Framework_MockObject_MockObject */
|
/** @var IHasher|PHPUnit_Framework_MockObject_MockObject */
|
||||||
private $hasher;
|
private $hasher;
|
||||||
|
|
||||||
/** @var IManager|PHPUnit_Framework_MockObject_MockObject */
|
/** @var EventDispatcherInterface|PHPUnit_Framework_MockObject_MockObject */
|
||||||
private $activityManager;
|
private $eventDispatcher;
|
||||||
|
|
||||||
/** @var ILogger|PHPUnit_Framework_MockObject_MockObject */
|
|
||||||
private $logger;
|
|
||||||
|
|
||||||
/** @var BackupCodeStorage */
|
/** @var BackupCodeStorage */
|
||||||
private $storage;
|
private $storage;
|
||||||
|
@ -60,20 +56,15 @@ class BackupCodeStorageTest extends TestCase {
|
||||||
$this->mapper = $this->createMock(BackupCodeMapper::class);
|
$this->mapper = $this->createMock(BackupCodeMapper::class);
|
||||||
$this->random = $this->createMock(ISecureRandom::class);
|
$this->random = $this->createMock(ISecureRandom::class);
|
||||||
$this->hasher = $this->createMock(IHasher::class);
|
$this->hasher = $this->createMock(IHasher::class);
|
||||||
$this->activityManager = $this->createMock(IManager::class);
|
$this->eventDispatcher = $this->createMock(EventDispatcherInterface::class);
|
||||||
$this->logger = $this->createMock(ILogger::class);
|
|
||||||
|
|
||||||
$this->storage = new BackupCodeStorage($this->mapper, $this->random, $this->hasher, $this->activityManager, $this->logger);
|
$this->storage = new BackupCodeStorage($this->mapper, $this->random, $this->hasher, $this->eventDispatcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCreateCodes() {
|
public function testCreateCodes() {
|
||||||
$user = $this->createMock(IUser::class);
|
$user = $this->createMock(IUser::class);
|
||||||
$number = 5;
|
$number = 5;
|
||||||
$event = $this->createMock(IEvent::class);
|
$user->method('getUID')->willReturn('fritz');
|
||||||
|
|
||||||
$user->expects($this->any())
|
|
||||||
->method('getUID')
|
|
||||||
->will($this->returnValue('fritz'));
|
|
||||||
$this->random->expects($this->exactly($number))
|
$this->random->expects($this->exactly($number))
|
||||||
->method('generate')
|
->method('generate')
|
||||||
->with(16, 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789')
|
->with(16, 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789')
|
||||||
|
@ -89,28 +80,12 @@ class BackupCodeStorageTest extends TestCase {
|
||||||
$this->mapper->expects($this->exactly($number))
|
$this->mapper->expects($this->exactly($number))
|
||||||
->method('insert')
|
->method('insert')
|
||||||
->with($this->equalTo($row));
|
->with($this->equalTo($row));
|
||||||
$this->activityManager->expects($this->once())
|
$this->eventDispatcher->expects($this->once())
|
||||||
->method('generateEvent')
|
->method('dispatch')
|
||||||
->will($this->returnValue($event));
|
->with(
|
||||||
$event->expects($this->once())
|
$this->equalTo(CodesGenerated::class),
|
||||||
->method('setApp')
|
$this->equalTo(new CodesGenerated($user))
|
||||||
->with('twofactor_backupcodes')
|
);
|
||||||
->will($this->returnSelf());
|
|
||||||
$event->expects($this->once())
|
|
||||||
->method('setType')
|
|
||||||
->with('security')
|
|
||||||
->will($this->returnSelf());
|
|
||||||
$event->expects($this->once())
|
|
||||||
->method('setAuthor')
|
|
||||||
->with('fritz')
|
|
||||||
->will($this->returnSelf());
|
|
||||||
$event->expects($this->once())
|
|
||||||
->method('setAffectedUser')
|
|
||||||
->with('fritz')
|
|
||||||
->will($this->returnSelf());
|
|
||||||
$this->activityManager->expects($this->once())
|
|
||||||
->method('publish')
|
|
||||||
->will($this->returnValue($event));
|
|
||||||
|
|
||||||
$codes = $this->storage->createCodes($user, $number);
|
$codes = $this->storage->createCodes($user, $number);
|
||||||
$this->assertCount($number, $codes);
|
$this->assertCount($number, $codes);
|
||||||
|
|
Loading…
Reference in New Issue