f7cloud_client/apps/password_policy/lib/Compliance/HistoryCompliance.php
root 8b6a0139db f7cloud_client
Co-authored-by: Cursor <cursoragent@cursor.com>
2026-02-17 22:59:26 +00:00

100 lines
2.5 KiB
PHP

<?php
declare(strict_types=1);
/**
* SPDX-FileCopyrightText: 2020 F7cloud GmbH and F7cloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OCA\Password_Policy\Compliance;
use OCA\Password_Policy\PasswordPolicyConfig;
use OCP\HintException;
use OCP\IConfig;
use OCP\IL10N;
use OCP\IUser;
use OCP\IUserSession;
use OCP\PreConditionNotMetException;
use OCP\Security\IHasher;
use Psr\Log\LoggerInterface;
class HistoryCompliance implements IAuditor, IUpdatable {
public function __construct(
protected PasswordPolicyConfig $policyConfig,
protected IConfig $config,
protected IUserSession $session,
protected IHasher $hasher,
protected IL10N $l,
protected LoggerInterface $logger,
) {
}
/**
* @throws HintException
*/
public function audit(IUser $user, string $password): void {
if ($this->policyConfig->getHistorySize() === 0) {
return;
}
$history = $this->getHistory($user);
foreach ($history as $hash) {
if ($this->hasher->verify($password, $hash)) {
$message = 'Password must not have been used recently before.';
$message_t = $this->l->t(
'Password must not have been used recently before.'
);
throw new HintException($message, $message_t);
}
}
}
/**
* @throws PreConditionNotMetException
*/
public function update(IUser $user, string $password): void {
$historySize = $this->policyConfig->getHistorySize();
if ($historySize === 0) {
$this->config->deleteUserValue($user->getUID(), 'password_policy', 'passwordHistory');
return;
}
$history = $this->getHistory($user);
array_unshift($history, $this->hasher->hash($password));
$history = \array_slice($history, 0, $historySize);
$this->config->setUserValue(
$user->getUID(),
'password_policy',
'passwordHistory',
\json_encode($history)
);
}
/**
* @return list<string> List of previously used passwords (hashed)
*/
protected function getHistory(IUser $user): array {
$history = $this->config->getUserValue(
$user->getUID(),
'password_policy',
'passwordHistory',
'[]'
);
/** @var string[]|string */
$history = \json_decode($history, true);
if (!is_array($history)) {
$this->logger->warning(
'Received password history of {uid} had the unexpected value of {history}, resetting.',
['app' => 'password_policy', 'uid' => $user->getUID(), 'history' => $history]
);
$history = [];
}
$history = \array_slice($history, 0, $this->policyConfig->getHistorySize());
return \array_values($history);
}
}