mirror of
https://code.mensbeam.com/MensBeam/Arsse.git
synced 2025-01-08 17:02:41 +00:00
Add means of unsetting a password in the backend
This commit is contained in:
parent
5bf0b67ec3
commit
e45ba3f0ea
7 changed files with 115 additions and 22 deletions
|
@ -289,27 +289,27 @@ class Database {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Retrieves the hashed password of a user */
|
/** Retrieves the hashed password of a user */
|
||||||
public function userPasswordGet(string $user): string {
|
public function userPasswordGet(string $user) {
|
||||||
if (!Arsse::$user->authorize($user, __FUNCTION__)) {
|
if (!Arsse::$user->authorize($user, __FUNCTION__)) {
|
||||||
throw new User\ExceptionAuthz("notAuthorized", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\ExceptionAuthz("notAuthorized", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
} elseif (!$this->userExists($user)) {
|
} elseif (!$this->userExists($user)) {
|
||||||
throw new User\Exception("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\Exception("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
}
|
}
|
||||||
return (string) $this->db->prepare("SELECT password from arsse_users where id = ?", "str")->run($user)->getValue();
|
return $this->db->prepare("SELECT password from arsse_users where id = ?", "str")->run($user)->getValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Sets the password of an existing user
|
/** Sets the password of an existing user
|
||||||
*
|
*
|
||||||
* @param string $user The user for whom to set the password
|
* @param string $user The user for whom to set the password
|
||||||
* @param string $password The new password, in cleartext. The password will be stored hashed
|
* @param string $password The new password, in cleartext. The password will be stored hashed. If null is passed, the password is unset and authentication not possible
|
||||||
*/
|
*/
|
||||||
public function userPasswordSet(string $user, string $password): bool {
|
public function userPasswordSet(string $user, string $password = null): bool {
|
||||||
if (!Arsse::$user->authorize($user, __FUNCTION__)) {
|
if (!Arsse::$user->authorize($user, __FUNCTION__)) {
|
||||||
throw new User\ExceptionAuthz("notAuthorized", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\ExceptionAuthz("notAuthorized", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
} elseif (!$this->userExists($user)) {
|
} elseif (!$this->userExists($user)) {
|
||||||
throw new User\Exception("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\Exception("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
}
|
}
|
||||||
$hash = (strlen($password) > 0) ? password_hash($password, \PASSWORD_DEFAULT) : "";
|
$hash = (strlen($password ?? "") > 0) ? password_hash($password, \PASSWORD_DEFAULT) : $password;
|
||||||
$this->db->prepare("UPDATE arsse_users set password = ? where id = ?", "str", "str")->run($hash, $user);
|
$this->db->prepare("UPDATE arsse_users set password = ? where id = ?", "str", "str")->run($hash, $user);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
13
lib/User.php
13
lib/User.php
|
@ -114,6 +114,19 @@ class User {
|
||||||
return $out;
|
return $out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function passwordUnset(string $user, $oldPassword = null): bool {
|
||||||
|
$func = "userPasswordUnset";
|
||||||
|
if (!$this->authorize($user, $func)) {
|
||||||
|
throw new User\ExceptionAuthz("notAuthorized", ["action" => $func, "user" => $user]);
|
||||||
|
}
|
||||||
|
$out = $this->u->userPasswordUnset($user, $oldPassword);
|
||||||
|
if (Arsse::$db->userExists($user)) {
|
||||||
|
// if the password change was successful and the user exists, set the internal password to the same value
|
||||||
|
Arsse::$db->userPasswordSet($user, null);
|
||||||
|
}
|
||||||
|
return $out;
|
||||||
|
}
|
||||||
|
|
||||||
public function generatePassword(): string {
|
public function generatePassword(): string {
|
||||||
return (new PassGen)->length(Arsse::$conf->userTempPasswordLength)->get();
|
return (new PassGen)->length(Arsse::$conf->userTempPasswordLength)->get();
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,4 +29,6 @@ interface Driver {
|
||||||
public function userList(): array;
|
public function userList(): array;
|
||||||
// sets a user's password; if the driver does not require the old password, it may be ignored
|
// sets a user's password; if the driver does not require the old password, it may be ignored
|
||||||
public function userPasswordSet(string $user, string $newPassword = null, string $oldPassword = null);
|
public function userPasswordSet(string $user, string $newPassword = null, string $oldPassword = null);
|
||||||
|
// removes a user's password; this makes authentication fail unconditionally
|
||||||
|
public function userPasswordUnset(string $user, string $oldPassword = null): bool;
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,9 @@ class Driver implements \JKingWeb\Arsse\User\Driver {
|
||||||
public function auth(string $user, string $password): bool {
|
public function auth(string $user, string $password): bool {
|
||||||
try {
|
try {
|
||||||
$hash = $this->userPasswordGet($user);
|
$hash = $this->userPasswordGet($user);
|
||||||
|
if (is_null($hash)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -58,7 +61,17 @@ class Driver implements \JKingWeb\Arsse\User\Driver {
|
||||||
return $newPassword;
|
return $newPassword;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function userPasswordGet(string $user): string {
|
public function userPasswordUnset(string $user, string $oldPassword = null): bool {
|
||||||
|
// do nothing: the internal database is updated regardless of what the driver does (assuming it does not throw an exception)
|
||||||
|
// throw an exception if the user does not exist
|
||||||
|
if (!$this->userExists($user)) {
|
||||||
|
throw new Exception("doesNotExist", ['action' => "userPasswordUnset", 'user' => $user]);
|
||||||
|
} else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function userPasswordGet(string $user) {
|
||||||
return Arsse::$db->userPasswordGet($user);
|
return Arsse::$db->userPasswordGet($user);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -127,6 +127,13 @@ trait SeriesUser {
|
||||||
$this->assertTrue(password_verify($pass, $hash), "Failed verifying password of $user '$pass' against hash '$hash'.");
|
$this->assertTrue(password_verify($pass, $hash), "Failed verifying password of $user '$pass' against hash '$hash'.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testUnsetAPassword() {
|
||||||
|
$user = "john.doe@example.com";
|
||||||
|
$this->assertEquals("", Arsse::$db->userPasswordGet($user));
|
||||||
|
$this->assertTrue(Arsse::$db->userPasswordSet($user, null));
|
||||||
|
$this->assertNull(Arsse::$db->userPasswordGet($user));
|
||||||
|
}
|
||||||
|
|
||||||
public function testSetThePasswordOfAMissingUser() {
|
public function testSetThePasswordOfAMissingUser() {
|
||||||
$this->assertException("doesNotExist", "User");
|
$this->assertException("doesNotExist", "User");
|
||||||
Arsse::$db->userPasswordSet("john.doe@example.org", "secret");
|
Arsse::$db->userPasswordSet("john.doe@example.org", "secret");
|
||||||
|
|
|
@ -37,12 +37,13 @@ class TestInternal extends \JKingWeb\Arsse\Test\AbstractTest {
|
||||||
* @dataProvider provideAuthentication
|
* @dataProvider provideAuthentication
|
||||||
* @group slow
|
* @group slow
|
||||||
*/
|
*/
|
||||||
public function testAuthenticateAUser(bool $authorized, string $user, string $password, bool $exp) {
|
public function testAuthenticateAUser(bool $authorized, string $user, $password, bool $exp) {
|
||||||
if ($authorized) {
|
if ($authorized) {
|
||||||
Phake::when(Arsse::$db)->userPasswordGet("john.doe@example.com")->thenReturn('$2y$10$1zbqRJhxM8uUjeSBPp4IhO90xrqK0XjEh9Z16iIYEFRV4U.zeAFom'); // hash of "secret"
|
Phake::when(Arsse::$db)->userPasswordGet("john.doe@example.com")->thenReturn('$2y$10$1zbqRJhxM8uUjeSBPp4IhO90xrqK0XjEh9Z16iIYEFRV4U.zeAFom'); // hash of "secret"
|
||||||
Phake::when(Arsse::$db)->userPasswordGet("jane.doe@example.com")->thenReturn('$2y$10$bK1ljXfTSyc2D.NYvT.Eq..OpehLRXVbglW.23ihVuyhgwJCd.7Im'); // hash of "superman"
|
Phake::when(Arsse::$db)->userPasswordGet("jane.doe@example.com")->thenReturn('$2y$10$bK1ljXfTSyc2D.NYvT.Eq..OpehLRXVbglW.23ihVuyhgwJCd.7Im'); // hash of "superman"
|
||||||
Phake::when(Arsse::$db)->userPasswordGet("owen.hardy@example.com")->thenReturn("");
|
Phake::when(Arsse::$db)->userPasswordGet("owen.hardy@example.com")->thenReturn("");
|
||||||
Phake::when(Arsse::$db)->userPasswordGet("kira.nerys@example.com")->thenThrow(new \JKingWeb\Arsse\User\Exception("doesNotExist"));
|
Phake::when(Arsse::$db)->userPasswordGet("kira.nerys@example.com")->thenThrow(new \JKingWeb\Arsse\User\Exception("doesNotExist"));
|
||||||
|
Phake::when(Arsse::$db)->userPasswordGet("007@example.com")->thenReturn(null);
|
||||||
} else {
|
} else {
|
||||||
Phake::when(Arsse::$db)->userPasswordGet->thenThrow(new \JKingWeb\Arsse\User\ExceptionAuthz("notAuthorized"));
|
Phake::when(Arsse::$db)->userPasswordGet->thenThrow(new \JKingWeb\Arsse\User\ExceptionAuthz("notAuthorized"));
|
||||||
}
|
}
|
||||||
|
@ -54,22 +55,26 @@ class TestInternal extends \JKingWeb\Arsse\Test\AbstractTest {
|
||||||
$jane = "jane.doe@example.com";
|
$jane = "jane.doe@example.com";
|
||||||
$owen = "owen.hardy@example.com";
|
$owen = "owen.hardy@example.com";
|
||||||
$kira = "kira.nerys@example.com";
|
$kira = "kira.nerys@example.com";
|
||||||
|
$bond = "007@example.com";
|
||||||
return [
|
return [
|
||||||
[false, $john, "secret", false],
|
[false, $john, "secret", false],
|
||||||
[false, $jane, "superman", false],
|
[false, $jane, "superman", false],
|
||||||
[false, $owen, "", false],
|
[false, $owen, "", false],
|
||||||
[false, $kira, "ashalla", false],
|
[false, $kira, "ashalla", false],
|
||||||
[true, $john, "secret", true],
|
[false, $bond, "", false],
|
||||||
[true, $jane, "superman", true],
|
[true, $john, "secret", true],
|
||||||
[true, $owen, "", true],
|
[true, $jane, "superman", true],
|
||||||
[true, $kira, "ashalla", false],
|
[true, $owen, "", true],
|
||||||
[true, $john, "top secret", false],
|
[true, $kira, "ashalla", false],
|
||||||
[true, $jane, "clark kent", false],
|
[true, $john, "top secret", false],
|
||||||
[true, $owen, "watchmaker", false],
|
[true, $jane, "clark kent", false],
|
||||||
[true, $kira, "singha", false],
|
[true, $owen, "watchmaker", false],
|
||||||
[true, $john, "", false],
|
[true, $kira, "singha", false],
|
||||||
[true, $jane, "", false],
|
[true, $john, "", false],
|
||||||
[true, $kira, "", false],
|
[true, $jane, "", false],
|
||||||
|
[true, $kira, "", false],
|
||||||
|
[true, $bond, "for England", false],
|
||||||
|
[true, $bond, "", false],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,4 +138,19 @@ class TestInternal extends \JKingWeb\Arsse\Test\AbstractTest {
|
||||||
$this->assertSame("superman", (new Driver)->userPasswordSet($john, "superman"));
|
$this->assertSame("superman", (new Driver)->userPasswordSet($john, "superman"));
|
||||||
$this->assertSame(null, (new Driver)->userPasswordSet($john, null));
|
$this->assertSame(null, (new Driver)->userPasswordSet($john, null));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testUnsetAPassword() {
|
||||||
|
$drv = \Phake::partialMock(Driver::class);
|
||||||
|
\Phake::when($drv)->userExists->thenReturn(true);
|
||||||
|
Phake::verifyNoFurtherInteraction(Arsse::$db);
|
||||||
|
$this->assertTrue($drv->userPasswordUnset("john.doe@example.com"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testUnsetAPasswordForAMssingUser() {
|
||||||
|
$drv = \Phake::partialMock(Driver::class);
|
||||||
|
\Phake::when($drv)->userExists->thenReturn(false);
|
||||||
|
Phake::verifyNoFurtherInteraction(Arsse::$db);
|
||||||
|
$this->assertException("doesNotExist", "User");
|
||||||
|
$drv->userPasswordUnset("john.doe@example.com");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -297,4 +297,42 @@ class TestUser extends \JKingWeb\Arsse\Test\AbstractTest {
|
||||||
[true, $jane, "secret", true, new \JKingWeb\Arsse\User\Exception("doesNotExist")],
|
[true, $jane, "secret", true, new \JKingWeb\Arsse\User\Exception("doesNotExist")],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @dataProvider providePasswordClearings */
|
||||||
|
public function testClearAPassword(bool $authorized, bool $exists, string $user, $exp) {
|
||||||
|
Phake::when($this->drv)->authorize->thenReturn($authorized);
|
||||||
|
Phake::when($this->drv)->userPasswordUnset->thenReturn(true);
|
||||||
|
Phake::when($this->drv)->userPasswordUnset("jane.doe@example.net", null)->thenThrow(new \JKingWeb\Arsse\User\Exception("doesNotExist"));
|
||||||
|
Phake::when(Arsse::$db)->userExists->thenReturn($exists);
|
||||||
|
$u = new User($this->drv);
|
||||||
|
try {
|
||||||
|
if ($exp instanceof \JKingWeb\Arsse\AbstractException) {
|
||||||
|
$this->assertException($exp);
|
||||||
|
$u->passwordUnset($user);
|
||||||
|
} else {
|
||||||
|
$this->assertSame($exp, $u->passwordUnset($user));
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
Phake::verify(Arsse::$db, Phake::times((int) ($authorized && $exists && is_bool($exp))))->userPasswordSet($user, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function providePasswordClearings() {
|
||||||
|
$forbidden = new \JKingWeb\Arsse\User\ExceptionAuthz("notAuthorized");
|
||||||
|
$missing = new \JKingWeb\Arsse\User\Exception("doesNotExist");
|
||||||
|
return [
|
||||||
|
[false, true, "jane.doe@example.com", $forbidden],
|
||||||
|
[false, true, "john.doe@example.com", $forbidden],
|
||||||
|
[false, true, "jane.doe@example.net", $forbidden],
|
||||||
|
[false, false, "jane.doe@example.com", $forbidden],
|
||||||
|
[false, false, "john.doe@example.com", $forbidden],
|
||||||
|
[false, false, "jane.doe@example.net", $forbidden],
|
||||||
|
[true, true, "jane.doe@example.com", true],
|
||||||
|
[true, true, "john.doe@example.com", true],
|
||||||
|
[true, true, "jane.doe@example.net", $missing],
|
||||||
|
[true, false, "jane.doe@example.com", true],
|
||||||
|
[true, false, "john.doe@example.com", true],
|
||||||
|
[true, false, "jane.doe@example.net", $missing],
|
||||||
|
];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue