mirror of
https://code.mensbeam.com/MensBeam/Arsse.git
synced 2024-12-22 21:22:40 +00:00
2946d950f2
- Control characters are now forbidden - Controls and colons are now also forbidden when renaming
532 lines
24 KiB
PHP
532 lines
24 KiB
PHP
<?php
|
|
/** @license MIT
|
|
* Copyright 2017 J. King, Dustin Wilson et al.
|
|
* See LICENSE and AUTHORS files for details */
|
|
|
|
declare(strict_types=1);
|
|
namespace JKingWeb\Arsse\TestCase\User;
|
|
|
|
use JKingWeb\Arsse\Arsse;
|
|
use JKingWeb\Arsse\Database;
|
|
use JKingWeb\Arsse\User;
|
|
use JKingWeb\Arsse\Db\Transaction;
|
|
use JKingWeb\Arsse\User\ExceptionConflict;
|
|
use JKingWeb\Arsse\User\ExceptionInput;
|
|
use JKingWeb\Arsse\User\Driver;
|
|
|
|
/** @covers \JKingWeb\Arsse\User */
|
|
class TestUser extends \JKingWeb\Arsse\Test\AbstractTest {
|
|
public function setUp(): void {
|
|
self::clearData();
|
|
self::setConf();
|
|
// create a mock database interface
|
|
Arsse::$db = \Phake::mock(Database::class);
|
|
\Phake::when(Arsse::$db)->begin->thenReturn(\Phake::mock(\JKingWeb\Arsse\Db\Transaction::class));
|
|
// create a mock user driver
|
|
$this->drv = \Phake::mock(Driver::class);
|
|
}
|
|
|
|
public function tearDown(): void {
|
|
\Phake::verifyNoOtherInteractions($this->drv);
|
|
\Phake::verifyNoOtherInteractions(Arsse::$db);
|
|
}
|
|
|
|
public function testConstruct(): void {
|
|
$this->assertInstanceOf(User::class, new User($this->drv));
|
|
$this->assertInstanceOf(User::class, new User);
|
|
}
|
|
|
|
public function testConversionToString(): void {
|
|
$u = new User;
|
|
$u->id = "john.doe@example.com";
|
|
$this->assertSame("john.doe@example.com", (string) $u);
|
|
$u->id = null;
|
|
$this->assertSame("", (string) $u);
|
|
}
|
|
|
|
public function testStartATransaction(): void {
|
|
\Phake::when(Arsse::$db)->begin->thenReturn(\Phake::mock(Transaction::class));
|
|
$u = new User($this->drv);
|
|
$this->assertInstanceOf(Transaction::class, $u->begin());
|
|
\Phake::verify(Arsse::$db)->begin();
|
|
}
|
|
|
|
public function testGeneratePasswords(): void {
|
|
$u = new User($this->drv);
|
|
$pass1 = $u->generatePassword();
|
|
$pass2 = $u->generatePassword();
|
|
$this->assertNotEquals($pass1, $pass2);
|
|
}
|
|
|
|
/** @dataProvider provideAuthentication */
|
|
public function testAuthenticateAUser(bool $preAuth, string $user, string $password, bool $exp): void {
|
|
Arsse::$conf->userPreAuth = $preAuth;
|
|
\Phake::when($this->drv)->auth->thenReturn(false);
|
|
\Phake::when($this->drv)->auth("john.doe@example.com", "secret")->thenReturn(true);
|
|
\Phake::when($this->drv)->auth("jane.doe@example.com", "superman")->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists("john.doe@example.com")->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists("jane.doe@example.com")->thenReturn(false);
|
|
\Phake::when(Arsse::$db)->userAdd->thenReturn("");
|
|
$u = new User($this->drv);
|
|
$this->assertSame($exp, $u->auth($user, $password));
|
|
$this->assertNull($u->id);
|
|
\Phake::verify($this->drv, \Phake::times((int) !$preAuth))->auth($user, $password);
|
|
\Phake::verify(Arsse::$db, \Phake::times($exp ? 1 : 0))->userExists($user);
|
|
\Phake::verify(Arsse::$db, \Phake::times($exp && $user === "jane.doe@example.com" ? 1 : 0))->userAdd($user, $password);
|
|
}
|
|
|
|
public function provideAuthentication(): iterable {
|
|
$john = "john.doe@example.com";
|
|
$jane = "jane.doe@example.com";
|
|
return [
|
|
[false, $john, "secret", true],
|
|
[false, $john, "superman", false],
|
|
[false, $jane, "secret", false],
|
|
[false, $jane, "superman", true],
|
|
[true, $john, "secret", true],
|
|
[true, $john, "superman", true],
|
|
[true, $jane, "secret", true],
|
|
[true, $jane, "superman", true],
|
|
];
|
|
}
|
|
|
|
public function testListUsers(): void {
|
|
$exp = ["john.doe@example.com", "jane.doe@example.com"];
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userList->thenReturn(["john.doe@example.com", "jane.doe@example.com"]);
|
|
$this->assertSame($exp, $u->list());
|
|
\Phake::verify($this->drv)->userList();
|
|
}
|
|
|
|
public function testLookUpAUserByNumber(): void {
|
|
$exp = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when(Arsse::$db)->userLookup->thenReturn($exp);
|
|
$this->assertSame($exp, $u->lookup(2112));
|
|
\Phake::verify(Arsse::$db)->userLookup(2112);
|
|
}
|
|
|
|
public function testAddAUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userAdd->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertSame($pass, $u->add($user, $pass));
|
|
\Phake::verify($this->drv)->userAdd($user, $pass);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testAddAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userAdd->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertSame($pass, $u->add($user, $pass));
|
|
\Phake::verify($this->drv)->userAdd($user, $pass);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, $pass);
|
|
}
|
|
|
|
public function testAddADuplicateUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userAdd->thenThrow(new ExceptionConflict("alreadyExists"));
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertException("alreadyExists", "User", "ExceptionConflict");
|
|
try {
|
|
$u->add($user, $pass);
|
|
} finally {
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify($this->drv)->userAdd($user, $pass);
|
|
}
|
|
}
|
|
|
|
public function testAddADuplicateUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userAdd->thenThrow(new ExceptionConflict("alreadyExists"));
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertException("alreadyExists", "User", "ExceptionConflict");
|
|
try {
|
|
$u->add($user, $pass);
|
|
} finally {
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, null);
|
|
\Phake::verify($this->drv)->userAdd($user, $pass);
|
|
}
|
|
}
|
|
|
|
/** @dataProvider provideInvalidUserNames */
|
|
public function testAddAnInvalidUser(string $user): void {
|
|
$u = new User($this->drv);
|
|
$this->assertException("invalidUsername", "User", "ExceptionInput");
|
|
$u->add($user, "secret");
|
|
}
|
|
|
|
public function provideInvalidUserNames(): iterable {
|
|
// output names with control characters
|
|
foreach (array_merge(range(0x00, 0x1F), [0x7F]) as $ord) {
|
|
yield [chr($ord)];
|
|
yield ["john".chr($ord)."doe@example.com"];
|
|
}
|
|
// also handle colons
|
|
yield [":"];
|
|
yield ["john:doe@example.com"];
|
|
}
|
|
|
|
public function testAddAUserWithARandomPassword(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "random password";
|
|
$u = \Phake::partialMock(User::class, $this->drv);
|
|
\Phake::when($u)->generatePassword->thenReturn($pass);
|
|
\Phake::when($this->drv)->userAdd->thenReturn(null)->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertSame($pass, $u->add($user));
|
|
\Phake::verify($this->drv)->userAdd($user, null);
|
|
\Phake::verify($this->drv)->userAdd($user, $pass);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testRenameAUser(): void {
|
|
$tr = \Phake::mock(Transaction::class);
|
|
\Phake::when(Arsse::$db)->begin->thenReturn($tr);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userAdd->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userRename->thenReturn(true);
|
|
\Phake::when($this->drv)->userRename->thenReturn(true);
|
|
$u = new User($this->drv);
|
|
$old = "john.doe@example.com";
|
|
$new = "jane.doe@example.com";
|
|
$this->assertTrue($u->rename($old, $new));
|
|
\Phake::inOrder(
|
|
\Phake::verify($this->drv)->userRename($old, $new),
|
|
\Phake::verify(Arsse::$db)->begin(),
|
|
\Phake::verify(Arsse::$db)->userExists($old),
|
|
\Phake::verify(Arsse::$db)->userRename($old, $new),
|
|
\Phake::verify(Arsse::$db)->sessionDestroy($new),
|
|
\Phake::verify(Arsse::$db)->tokenRevoke($new, "fever.login"),
|
|
\Phake::verify($tr)->commit()
|
|
);
|
|
}
|
|
|
|
public function testRenameAUserWeDoNotKnow(): void {
|
|
$tr = \Phake::mock(Transaction::class);
|
|
\Phake::when(Arsse::$db)->begin->thenReturn($tr);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
\Phake::when(Arsse::$db)->userAdd->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userRename->thenReturn(true);
|
|
\Phake::when($this->drv)->userRename->thenReturn(true);
|
|
$u = new User($this->drv);
|
|
$old = "john.doe@example.com";
|
|
$new = "jane.doe@example.com";
|
|
$this->assertTrue($u->rename($old, $new));
|
|
\Phake::inOrder(
|
|
\Phake::verify($this->drv)->userRename($old, $new),
|
|
\Phake::verify(Arsse::$db)->begin(),
|
|
\Phake::verify(Arsse::$db)->userExists($old),
|
|
\Phake::verify(Arsse::$db)->userAdd($new, null),
|
|
\Phake::verify($tr)->commit()
|
|
);
|
|
}
|
|
|
|
public function testRenameAUserWithoutEffect(): void {
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
\Phake::when(Arsse::$db)->userAdd->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userRename->thenReturn(true);
|
|
\Phake::when($this->drv)->userRename->thenReturn(false);
|
|
$u = new User($this->drv);
|
|
$old = "john.doe@example.com";
|
|
$this->assertFalse($u->rename($old, $old));
|
|
\Phake::verify($this->drv)->userRename($old, $old);
|
|
}
|
|
|
|
/** @dataProvider provideInvalidUserNames */
|
|
public function testRenameAUserToAnInvalidName(string $new): void {
|
|
$u = new User($this->drv);
|
|
$this->assertException("invalidUsername", "User", "ExceptionInput");
|
|
$u->rename("john.doe@example.com", $new);
|
|
}
|
|
|
|
public function testRemoveAUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userRemove->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertTrue($u->remove($user));
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify(Arsse::$db)->userRemove($user);
|
|
\Phake::verify($this->drv)->userRemove($user);
|
|
}
|
|
|
|
public function testRemoveAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userRemove->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertTrue($u->remove($user));
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify($this->drv)->userRemove($user);
|
|
}
|
|
|
|
public function testRemoveAMissingUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userRemove->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->remove($user);
|
|
} finally {
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify(Arsse::$db)->userRemove($user);
|
|
\Phake::verify($this->drv)->userRemove($user);
|
|
}
|
|
}
|
|
|
|
public function testRemoveAMissingUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userRemove->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->remove($user);
|
|
} finally {
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify($this->drv)->userRemove($user);
|
|
}
|
|
}
|
|
|
|
public function testSetAPassword(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertSame($pass, $u->passwordSet($user, $pass));
|
|
\Phake::verify($this->drv)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->sessionDestroy($user);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testSetARandomPassword(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "random password";
|
|
$u = \Phake::partialMock(User::class, $this->drv);
|
|
\Phake::when($u)->generatePassword->thenReturn($pass);
|
|
\Phake::when($this->drv)->userPasswordSet->thenReturn(null)->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertSame($pass, $u->passwordSet($user, null));
|
|
\Phake::verify($this->drv)->userPasswordSet($user, null, null);
|
|
\Phake::verify($this->drv)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->sessionDestroy($user);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testSetAPasswordForAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "secret";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertSame($pass, $u->passwordSet($user, $pass));
|
|
\Phake::verify($this->drv)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, $pass);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testSetARandomPasswordForAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "random password";
|
|
$u = \Phake::partialMock(User::class, $this->drv);
|
|
\Phake::when($u)->generatePassword->thenReturn($pass);
|
|
\Phake::when($this->drv)->userPasswordSet->thenReturn(null)->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userPasswordSet->thenReturn($pass);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertSame($pass, $u->passwordSet($user, null));
|
|
\Phake::verify($this->drv)->userPasswordSet($user, null, null);
|
|
\Phake::verify($this->drv)->userPasswordSet($user, $pass, null);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, $pass);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testSetARandomPasswordForAMissingUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$pass = "random password";
|
|
$u = \Phake::partialMock(User::class, $this->drv);
|
|
\Phake::when($u)->generatePassword->thenReturn($pass);
|
|
\Phake::when($this->drv)->userPasswordSet->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->passwordSet($user, null);
|
|
} finally {
|
|
\Phake::verify($this->drv)->userPasswordSet($user, null, null);
|
|
}
|
|
}
|
|
|
|
public function testUnsetAPassword(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPasswordUnset->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userPasswordUnset->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertTrue($u->passwordUnset($user));
|
|
\Phake::verify($this->drv)->userPasswordUnset($user, null);
|
|
\Phake::verify(Arsse::$db)->userPasswordSet($user, null);
|
|
\Phake::verify(Arsse::$db)->sessionDestroy($user);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testUnsetAPasswordForAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPasswordUnset->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userPasswordUnset->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertTrue($u->passwordUnset($user));
|
|
\Phake::verify($this->drv)->userPasswordUnset($user, null);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testUnsetAPasswordForAMissingUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPasswordUnset->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->passwordUnset($user);
|
|
} finally {
|
|
\Phake::verify($this->drv)->userPasswordUnset($user, null);
|
|
}
|
|
}
|
|
|
|
/** @dataProvider provideProperties */
|
|
public function testGetThePropertiesOfAUser(array $exp, array $base, array $extra): void {
|
|
$user = "john.doe@example.com";
|
|
$exp = array_merge(['num' => null], array_combine(array_keys(User::PROPERTIES), array_fill(0, sizeof(User::PROPERTIES), null)), $exp);
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPropertiesGet->thenReturn($extra);
|
|
\Phake::when(Arsse::$db)->userPropertiesGet->thenReturn($base);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
$this->assertSame($exp, $u->propertiesGet($user));
|
|
\Phake::verify($this->drv)->userPropertiesGet($user, true);
|
|
\Phake::verify(Arsse::$db)->userPropertiesGet($user, true);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function provideProperties(): iterable {
|
|
$defaults = ['num' => 1, 'admin' => false, 'lang' => null, 'tz' => "Etc/UTC", 'sort_asc' => false];
|
|
return [
|
|
[$defaults, $defaults, []],
|
|
[$defaults, $defaults, ['num' => 2112, 'blah' => "bloo"]],
|
|
[['num' => 1, 'admin' => true, 'lang' => "fr", 'tz' => "America/Toronto", 'sort_asc' => true], $defaults, ['admin' => true, 'lang' => "fr", 'tz' => "America/Toronto", 'sort_asc' => true]],
|
|
[['num' => 1, 'admin' => true, 'lang' => null, 'tz' => "America/Toronto", 'sort_asc' => true], ['num' => 1, 'admin' => true, 'lang' => "fr", 'tz' => "America/Toronto", 'sort_asc' => true], ['lang' => null]],
|
|
];
|
|
}
|
|
|
|
public function testGetThePropertiesOfAUserWeDoNotKnow(): void {
|
|
$user = "john.doe@example.com";
|
|
$extra = ['tz' => "Europe/Istanbul"];
|
|
$base = ['num' => 47, 'admin' => false, 'lang' => null, 'tz' => "Etc/UTC", 'sort_asc' => false];
|
|
$exp = ['num' => 47, 'admin' => false, 'lang' => null, 'tz' => "Europe/Istanbul", 'sort_asc' => false];
|
|
$exp = array_merge(['num' => null], array_combine(array_keys(User::PROPERTIES), array_fill(0, sizeof(User::PROPERTIES), null)), $exp);
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPropertiesGet->thenReturn($extra);
|
|
\Phake::when(Arsse::$db)->userPropertiesGet->thenReturn($base);
|
|
\Phake::when(Arsse::$db)->userAdd->thenReturn(true);
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
$this->assertSame($exp, $u->propertiesGet($user));
|
|
\Phake::verify($this->drv)->userPropertiesGet($user, true);
|
|
\Phake::verify(Arsse::$db)->userPropertiesGet($user, true);
|
|
\Phake::verify(Arsse::$db)->userPropertiesSet($user, $extra);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, null);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
|
|
public function testGetThePropertiesOfAMissingUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPropertiesGet->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->propertiesGet($user);
|
|
} finally {
|
|
\Phake::verify($this->drv)->userPropertiesGet($user, true);
|
|
}
|
|
}
|
|
|
|
/** @dataProvider providePropertyChanges */
|
|
public function testSetThePropertiesOfAUser(array $in, $out): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
if ($out instanceof \Exception) {
|
|
$this->assertException($out);
|
|
$u->propertiesSet($user, $in);
|
|
} else {
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(true);
|
|
\Phake::when($this->drv)->userPropertiesSet->thenReturn($out);
|
|
\Phake::when(Arsse::$db)->userPropertiesSet->thenReturn(true);
|
|
$this->assertSame($out, $u->propertiesSet($user, $in));
|
|
\Phake::verify($this->drv)->userPropertiesSet($user, $in);
|
|
\Phake::verify(Arsse::$db)->userPropertiesSet($user, $out);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
}
|
|
}
|
|
|
|
/** @dataProvider providePropertyChanges */
|
|
public function testSetThePropertiesOfAUserWeDoNotKnow(array $in, $out): void {
|
|
$user = "john.doe@example.com";
|
|
$u = new User($this->drv);
|
|
if ($out instanceof \Exception) {
|
|
$this->assertException($out);
|
|
$u->propertiesSet($user, $in);
|
|
} else {
|
|
\Phake::when(Arsse::$db)->userExists->thenReturn(false);
|
|
\Phake::when($this->drv)->userPropertiesSet->thenReturn($out);
|
|
\Phake::when(Arsse::$db)->userPropertiesSet->thenReturn(true);
|
|
$this->assertSame($out, $u->propertiesSet($user, $in));
|
|
\Phake::verify($this->drv)->userPropertiesSet($user, $in);
|
|
\Phake::verify(Arsse::$db)->userPropertiesSet($user, $out);
|
|
\Phake::verify(Arsse::$db)->userExists($user);
|
|
\Phake::verify(Arsse::$db)->userAdd($user, null);
|
|
}
|
|
}
|
|
|
|
public function providePropertyChanges(): iterable {
|
|
return [
|
|
[['admin' => true], ['admin' => true]],
|
|
[['admin' => 2], new ExceptionInput("invalidValue")],
|
|
[['sort_asc' => 2], new ExceptionInput("invalidValue")],
|
|
[['tz' => "Etc/UTC"], ['tz' => "Etc/UTC"]],
|
|
[['tz' => "Etc/blah"], new ExceptionInput("invalidTimezone")],
|
|
[['tz' => false], new ExceptionInput("invalidValue")],
|
|
[['lang' => "en-ca"], ['lang' => "en-CA"]],
|
|
[['lang' => null], ['lang' => null]],
|
|
[['page_size' => 0], new ExceptionInput("invalidNonZeroInteger")],
|
|
];
|
|
}
|
|
|
|
public function testSetThePropertiesOfAMissingUser(): void {
|
|
$user = "john.doe@example.com";
|
|
$in = ['admin' => true];
|
|
$u = new User($this->drv);
|
|
\Phake::when($this->drv)->userPropertiesSet->thenThrow(new ExceptionConflict("doesNotExist"));
|
|
$this->assertException("doesNotExist", "User", "ExceptionConflict");
|
|
try {
|
|
$u->propertiesSet($user, $in);
|
|
} finally {
|
|
\Phake::verify($this->drv)->userPropertiesSet($user, $in);
|
|
}
|
|
}
|
|
}
|