merge extend-user-entity

This commit is contained in:
Yisroel Baum 2026-05-06 22:03:56 +03:00
commit 03a1f02843
Signed by: yisroelbaum
GPG key ID: 0FA60884F75520A9
15 changed files with 296 additions and 15 deletions

View file

@ -3,12 +3,15 @@
namespace App\User; namespace App\User;
use App\Shared\ValueObject\EmailAddress; use App\Shared\ValueObject\EmailAddress;
use DateTimeImmutable;
readonly class CreateUserDto readonly class CreateUserDto
{ {
public function __construct( public function __construct(
public EmailAddress $email, public EmailAddress $email,
public string $displayName,
public string $passwordHash, public string $passwordHash,
public bool $isAdmin, public bool $isAdmin,
public ?DateTimeImmutable $emailConfirmedAt,
) {} ) {}
} }

View file

@ -3,6 +3,9 @@
namespace App\User; namespace App\User;
use App\Shared\ValueObject\EmailAddress; use App\Shared\ValueObject\EmailAddress;
use DateTimeImmutable;
use DateTimeZone;
use RuntimeException;
class EloquentUserRepository implements UserRepository class EloquentUserRepository implements UserRepository
{ {
@ -10,8 +13,10 @@ class EloquentUserRepository implements UserRepository
{ {
$model = UserModel::create([ $model = UserModel::create([
'email' => $dto->email->value(), 'email' => $dto->email->value(),
'display_name' => $dto->displayName,
'password_hash' => $dto->passwordHash, 'password_hash' => $dto->passwordHash,
'is_admin' => $dto->isAdmin, 'is_admin' => $dto->isAdmin,
'email_confirmed_at' => $dto->emailConfirmedAt,
]); ]);
return $this->toDomain($model); return $this->toDomain($model);
@ -31,13 +36,51 @@ class EloquentUserRepository implements UserRepository
return $model === null ? null : $this->toDomain($model); return $model === null ? null : $this->toDomain($model);
} }
public function findByDisplayName(string $displayName): ?User
{
$model = UserModel::where('display_name', $displayName)->first();
return $model === null ? null : $this->toDomain($model);
}
/**
* @throws RuntimeException
*/
public function update(User $user): User
{
$model = UserModel::find($user->getId());
if ($model === null) {
throw new RuntimeException(
"User with id: {$user->getId()} does not exist"
);
}
$model->email = $user->getEmail()->value();
$model->display_name = $user->getDisplayName();
$model->password_hash = $user->getPasswordHash();
$model->is_admin = $user->isAdmin();
$model->email_confirmed_at = $user->getEmailConfirmedAt();
$model->save();
return $this->toDomain($model);
}
private function toDomain(UserModel $model): User private function toDomain(UserModel $model): User
{ {
$confirmedAt = null;
if ($model->email_confirmed_at !== null) {
$confirmedAt = new DateTimeImmutable(
$model->email_confirmed_at->toDateTimeString(),
new DateTimeZone('UTC'),
);
}
return new User( return new User(
id: $model->id, id: $model->id,
email: new EmailAddress($model->email), email: new EmailAddress($model->email),
displayName: $model->display_name,
passwordHash: $model->password_hash, passwordHash: $model->password_hash,
isAdmin: $model->is_admin, isAdmin: $model->is_admin,
emailConfirmedAt: $confirmedAt,
); );
} }
} }

View file

@ -15,6 +15,8 @@ class SignupUser
{ {
private const MIN_PASSWORD_LENGTH = 8; private const MIN_PASSWORD_LENGTH = 8;
private const DISPLAY_NAME_PATTERN = '/^[a-z0-9_-]{3,30}$/';
public function __construct( public function __construct(
private UserRepository $userRepo, private UserRepository $userRepo,
private PasswordHasher $hasher, private PasswordHasher $hasher,
@ -29,6 +31,19 @@ class SignupUser
if ($request->email === null || $request->email === '') { if ($request->email === null || $request->email === '') {
throw new BadRequestException('email is required'); throw new BadRequestException('email is required');
} }
if ($request->displayName === null || $request->displayName === '') {
throw new BadRequestException('displayName is required');
}
if (
preg_match(
self::DISPLAY_NAME_PATTERN,
$request->displayName,
) !== 1
) {
throw new BadRequestException(
'displayName must be 3-30 chars of [a-z0-9_-]'
);
}
if ($request->password === null || $request->password === '') { if ($request->password === null || $request->password === '') {
throw new BadRequestException('password is required'); throw new BadRequestException('password is required');
} }
@ -47,11 +62,16 @@ class SignupUser
if ($this->userRepo->findByEmail($email) !== null) { if ($this->userRepo->findByEmail($email) !== null) {
throw new DomainException('email already registered'); throw new DomainException('email already registered');
} }
if ($this->userRepo->findByDisplayName($request->displayName) !== null) {
throw new DomainException('displayName already taken');
}
return $this->userRepo->create(new CreateUserDto( return $this->userRepo->create(new CreateUserDto(
email: $email, email: $email,
displayName: $request->displayName,
passwordHash: $this->hasher->hash($request->password), passwordHash: $this->hasher->hash($request->password),
isAdmin: false, isAdmin: false,
emailConfirmedAt: null,
)); ));
} }
} }

View file

@ -6,6 +6,7 @@ class SignupUserRequest
{ {
public function __construct( public function __construct(
public ?string $email, public ?string $email,
public ?string $displayName,
public ?string $password, public ?string $password,
) {} ) {}
} }

View file

@ -3,14 +3,17 @@
namespace App\User; namespace App\User;
use App\Shared\ValueObject\EmailAddress; use App\Shared\ValueObject\EmailAddress;
use DateTimeImmutable;
class User class User
{ {
public function __construct( public function __construct(
private int $id, private int $id,
private EmailAddress $email, private EmailAddress $email,
private string $displayName,
private string $passwordHash, private string $passwordHash,
private bool $isAdmin, private bool $isAdmin,
private ?DateTimeImmutable $emailConfirmedAt,
) {} ) {}
public function getId(): int public function getId(): int
@ -23,6 +26,11 @@ class User
return $this->email; return $this->email;
} }
public function getDisplayName(): string
{
return $this->displayName;
}
public function getPasswordHash(): string public function getPasswordHash(): string
{ {
return $this->passwordHash; return $this->passwordHash;
@ -32,4 +40,14 @@ class User
{ {
return $this->isAdmin; return $this->isAdmin;
} }
public function getEmailConfirmedAt(): ?DateTimeImmutable
{
return $this->emailConfirmedAt;
}
public function isEmailConfirmed(): bool
{
return $this->emailConfirmedAt !== null;
}
} }

View file

@ -2,21 +2,26 @@
namespace App\User; namespace App\User;
use DateTimeImmutable;
use Illuminate\Database\Eloquent\Builder; use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
/** /**
* @property int $id * @property int $id
* @property string $email * @property string $email
* @property string $display_name
* @property string $password_hash * @property string $password_hash
* @property bool $is_admin * @property bool $is_admin
* @property ?DateTimeImmutable $email_confirmed_at
* *
* @method static Builder<static>|UserModel newModelQuery() * @method static Builder<static>|UserModel newModelQuery()
* @method static Builder<static>|UserModel newQuery() * @method static Builder<static>|UserModel newQuery()
* @method static Builder<static>|UserModel query() * @method static Builder<static>|UserModel query()
* @method static Builder<static>|UserModel whereId($value) * @method static Builder<static>|UserModel whereId($value)
* @method static Builder<static>|UserModel whereEmail($value) * @method static Builder<static>|UserModel whereEmail($value)
* @method static Builder<static>|UserModel whereDisplayName($value)
* @method static Builder<static>|UserModel whereIsAdmin($value) * @method static Builder<static>|UserModel whereIsAdmin($value)
* @method static Builder<static>|UserModel whereEmailConfirmedAt($value)
* *
* @mixin \Eloquent * @mixin \Eloquent
*/ */
@ -28,11 +33,14 @@ class UserModel extends Model
protected $fillable = [ protected $fillable = [
'email', 'email',
'display_name',
'password_hash', 'password_hash',
'is_admin', 'is_admin',
'email_confirmed_at',
]; ];
protected $casts = [ protected $casts = [
'is_admin' => 'boolean', 'is_admin' => 'boolean',
'email_confirmed_at' => 'immutable_datetime',
]; ];
} }

View file

@ -11,4 +11,8 @@ interface UserRepository
public function find(int $id): ?User; public function find(int $id): ?User;
public function findByEmail(EmailAddress $email): ?User; public function findByEmail(EmailAddress $email): ?User;
public function findByDisplayName(string $displayName): ?User;
public function update(User $user): User;
} }

View file

@ -11,8 +11,10 @@ return new class extends Migration
Schema::create('users', function (Blueprint $table) { Schema::create('users', function (Blueprint $table) {
$table->id(); $table->id();
$table->string('email')->unique(); $table->string('email')->unique();
$table->string('display_name')->unique();
$table->string('password_hash'); $table->string('password_hash');
$table->boolean('is_admin')->default(false); $table->boolean('is_admin')->default(false);
$table->dateTime('email_confirmed_at')->nullable();
}); });
} }

View file

@ -6,6 +6,7 @@ use App\Shared\ValueObject\EmailAddress;
use App\User\CreateUserDto; use App\User\CreateUserDto;
use App\User\User; use App\User\User;
use App\User\UserRepository; use App\User\UserRepository;
use RuntimeException;
class FakeUserRepository implements UserRepository class FakeUserRepository implements UserRepository
{ {
@ -20,12 +21,14 @@ class FakeUserRepository implements UserRepository
$user = new User( $user = new User(
id: $id, id: $id,
email: $dto->email, email: $dto->email,
displayName: $dto->displayName,
passwordHash: $dto->passwordHash, passwordHash: $dto->passwordHash,
isAdmin: $dto->isAdmin, isAdmin: $dto->isAdmin,
emailConfirmedAt: $dto->emailConfirmedAt,
); );
$this->existingUsers[$id] = $user; $this->existingUsers[$id] = $user;
return $user; return $this->copy($user);
} }
public function find(int $id): ?User public function find(int $id): ?User
@ -35,30 +38,59 @@ class FakeUserRepository implements UserRepository
return null; return null;
} }
return new User( return $this->copy($user);
id: $user->getId(),
email: $user->getEmail(),
passwordHash: $user->getPasswordHash(),
isAdmin: $user->isAdmin(),
);
} }
public function findByEmail(EmailAddress $email): ?User public function findByEmail(EmailAddress $email): ?User
{ {
foreach ($this->existingUsers as $user) { foreach ($this->existingUsers as $user) {
if ($user->getEmail()->equals($email)) { if ($user->getEmail()->equals($email)) {
return new User( return $this->copy($user);
id: $user->getId(),
email: $user->getEmail(),
passwordHash: $user->getPasswordHash(),
isAdmin: $user->isAdmin(),
);
} }
} }
return null; return null;
} }
public function findByDisplayName(string $displayName): ?User
{
foreach ($this->existingUsers as $user) {
if ($user->getDisplayName() === $displayName) {
return $this->copy($user);
}
}
return null;
}
/**
* @throws RuntimeException
*/
public function update(User $user): User
{
$id = $user->getId();
if (! isset($this->existingUsers[$id])) {
throw new RuntimeException(
"User with id: $id does not exist"
);
}
$this->existingUsers[$id] = $user;
return $this->copy($user);
}
private function copy(User $user): User
{
return new User(
id: $user->getId(),
email: $user->getEmail(),
displayName: $user->getDisplayName(),
passwordHash: $user->getPasswordHash(),
isAdmin: $user->isAdmin(),
emailConfirmedAt: $user->getEmailConfirmedAt(),
);
}
private function getNextId(): int private function getNextId(): int
{ {
return count($this->existingUsers) + 1; return count($this->existingUsers) + 1;

View file

@ -58,13 +58,15 @@ class AuthMiddlewareTest extends TestCase
}; };
} }
private function makeUser(int $id = 7): User private function makeUser(int $id): User
{ {
return new User( return new User(
id: $id, id: $id,
email: new EmailAddress('user@example.com'), email: new EmailAddress('user@example.com'),
displayName: 'user',
passwordHash: 'hashed:irrelevant', passwordHash: 'hashed:irrelevant',
isAdmin: false, isAdmin: false,
emailConfirmedAt: null,
); );
} }
@ -100,7 +102,7 @@ class AuthMiddlewareTest extends TestCase
{ {
$this->sessionRepo->create(new CreateSessionDto( $this->sessionRepo->create(new CreateSessionDto(
token: 'expired-token', token: 'expired-token',
user: $this->makeUser(), user: $this->makeUser(7),
createdAt: $this->now->modify('-8 days'), createdAt: $this->now->modify('-8 days'),
expiresAt: $this->now->modify('-1 day'), expiresAt: $this->now->modify('-1 day'),
)); ));

View file

@ -33,13 +33,16 @@ class AuthenticateUserTest extends TestCase
private function seedUser( private function seedUser(
string $email, string $email,
string $displayName,
string $password, string $password,
bool $isAdmin, bool $isAdmin,
): User { ): User {
return $this->userRepo->create(new CreateUserDto( return $this->userRepo->create(new CreateUserDto(
email: new EmailAddress($email), email: new EmailAddress($email),
displayName: $displayName,
passwordHash: $this->hasher->hash($password), passwordHash: $this->hasher->hash($password),
isAdmin: $isAdmin, isAdmin: $isAdmin,
emailConfirmedAt: null,
)); ));
} }
@ -101,6 +104,7 @@ class AuthenticateUserTest extends TestCase
{ {
$this->seedUser( $this->seedUser(
email: 'user@example.com', email: 'user@example.com',
displayName: 'user',
password: 'correctpassword', password: 'correctpassword',
isAdmin: false, isAdmin: false,
); );
@ -116,6 +120,7 @@ class AuthenticateUserTest extends TestCase
{ {
$seeded = $this->seedUser( $seeded = $this->seedUser(
email: 'user@example.com', email: 'user@example.com',
displayName: 'user',
password: 'correctpassword', password: 'correctpassword',
isAdmin: false, isAdmin: false,
); );
@ -140,6 +145,7 @@ class AuthenticateUserTest extends TestCase
{ {
$this->seedUser( $this->seedUser(
email: 'admin@example.com', email: 'admin@example.com',
displayName: 'admin',
password: 'adminpassword', password: 'adminpassword',
isAdmin: true, isAdmin: true,
); );

View file

@ -46,8 +46,10 @@ class CreateSessionTest extends TestCase
return new User( return new User(
id: 7, id: 7,
email: new EmailAddress('user@example.com'), email: new EmailAddress('user@example.com'),
displayName: 'user',
passwordHash: 'hashed:irrelevant', passwordHash: 'hashed:irrelevant',
isAdmin: false, isAdmin: false,
emailConfirmedAt: null,
); );
} }

View file

@ -34,8 +34,10 @@ class LogoutTest extends TestCase
$user = new User( $user = new User(
id: 7, id: 7,
email: new EmailAddress('user@example.com'), email: new EmailAddress('user@example.com'),
displayName: 'user',
passwordHash: 'hashed:irrelevant', passwordHash: 'hashed:irrelevant',
isAdmin: false, isAdmin: false,
emailConfirmedAt: null,
); );
$this->sessionRepo->create(new CreateSessionDto( $this->sessionRepo->create(new CreateSessionDto(
token: 'token-abc', token: 'token-abc',

View file

@ -36,6 +36,7 @@ class SignupUserTest extends TestCase
$this->expectException(BadRequestException::class); $this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: null, email: null,
displayName: 'alice',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));
} }
@ -45,6 +46,7 @@ class SignupUserTest extends TestCase
$this->expectException(BadRequestException::class); $this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: '', email: '',
displayName: 'alice',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));
} }
@ -54,6 +56,37 @@ class SignupUserTest extends TestCase
$this->expectException(BadRequestException::class); $this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: 'not-an-email', email: 'not-an-email',
displayName: 'alice',
password: 'longenoughpassword',
));
}
public function test_null_display_name_throws_bad_request(): void
{
$this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest(
email: 'user@example.com',
displayName: null,
password: 'longenoughpassword',
));
}
public function test_short_display_name_throws_bad_request(): void
{
$this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest(
email: 'user@example.com',
displayName: 'ab',
password: 'longenoughpassword',
));
}
public function test_display_name_with_invalid_chars_throws(): void
{
$this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest(
email: 'user@example.com',
displayName: 'Has Spaces',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));
} }
@ -63,6 +96,7 @@ class SignupUserTest extends TestCase
$this->expectException(BadRequestException::class); $this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: 'user@example.com', email: 'user@example.com',
displayName: 'alice',
password: null, password: null,
)); ));
} }
@ -72,6 +106,7 @@ class SignupUserTest extends TestCase
$this->expectException(BadRequestException::class); $this->expectException(BadRequestException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: 'user@example.com', email: 'user@example.com',
displayName: 'alice',
password: 'short', password: 'short',
)); ));
} }
@ -80,13 +115,34 @@ class SignupUserTest extends TestCase
{ {
$this->userRepo->create(new CreateUserDto( $this->userRepo->create(new CreateUserDto(
email: new EmailAddress('user@example.com'), email: new EmailAddress('user@example.com'),
displayName: 'first',
passwordHash: $this->hasher->hash('original-password'), passwordHash: $this->hasher->hash('original-password'),
isAdmin: false, isAdmin: false,
emailConfirmedAt: null,
)); ));
$this->expectException(DomainException::class); $this->expectException(DomainException::class);
$this->useCase->execute(new SignupUserRequest( $this->useCase->execute(new SignupUserRequest(
email: 'user@example.com', email: 'user@example.com',
displayName: 'second',
password: 'second-attempt-password',
));
}
public function test_duplicate_display_name_throws_domain_exception(): void
{
$this->userRepo->create(new CreateUserDto(
email: new EmailAddress('first@example.com'),
displayName: 'taken',
passwordHash: $this->hasher->hash('original-password'),
isAdmin: false,
emailConfirmedAt: null,
));
$this->expectException(DomainException::class);
$this->useCase->execute(new SignupUserRequest(
email: 'second@example.com',
displayName: 'taken',
password: 'second-attempt-password', password: 'second-attempt-password',
)); ));
} }
@ -95,22 +151,26 @@ class SignupUserTest extends TestCase
{ {
$created = $this->useCase->execute(new SignupUserRequest( $created = $this->useCase->execute(new SignupUserRequest(
email: 'new@example.com', email: 'new@example.com',
displayName: 'newuser',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));
$this->assertInstanceOf(User::class, $created); $this->assertInstanceOf(User::class, $created);
$this->assertSame('new@example.com', $created->getEmail()->value()); $this->assertSame('new@example.com', $created->getEmail()->value());
$this->assertSame('newuser', $created->getDisplayName());
$this->assertSame( $this->assertSame(
$this->hasher->hash('longenoughpassword'), $this->hasher->hash('longenoughpassword'),
$created->getPasswordHash(), $created->getPasswordHash(),
); );
$this->assertFalse($created->isAdmin()); $this->assertFalse($created->isAdmin());
$this->assertFalse($created->isEmailConfirmed());
} }
public function test_created_user_is_findable_by_email(): void public function test_created_user_is_findable_by_email(): void
{ {
$created = $this->useCase->execute(new SignupUserRequest( $created = $this->useCase->execute(new SignupUserRequest(
email: 'lookup@example.com', email: 'lookup@example.com',
displayName: 'lookup',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));
@ -121,10 +181,24 @@ class SignupUserTest extends TestCase
$this->assertSame($created->getId(), $found->getId()); $this->assertSame($created->getId(), $found->getId());
} }
public function test_created_user_is_findable_by_display_name(): void
{
$created = $this->useCase->execute(new SignupUserRequest(
email: 'lookup@example.com',
displayName: 'lookupbyname',
password: 'longenoughpassword',
));
$found = $this->userRepo->findByDisplayName('lookupbyname');
$this->assertNotNull($found);
$this->assertSame($created->getId(), $found->getId());
}
public function test_signup_normalizes_email_domain(): void public function test_signup_normalizes_email_domain(): void
{ {
$created = $this->useCase->execute(new SignupUserRequest( $created = $this->useCase->execute(new SignupUserRequest(
email: 'Mixed@CASE.com', email: 'Mixed@CASE.com',
displayName: 'mixed',
password: 'longenoughpassword', password: 'longenoughpassword',
)); ));

View file

@ -0,0 +1,64 @@
<?php
namespace Tests\Unit\User;
use App\Shared\ValueObject\EmailAddress;
use App\User\User;
use DateTimeImmutable;
use DateTimeZone;
use Tests\TestCase;
class UserTest extends TestCase
{
public function test_user_exposes_all_properties(): void
{
$confirmedAt = new DateTimeImmutable(
'2026-05-06T12:00:00',
new DateTimeZone('UTC'),
);
$user = new User(
id: 7,
email: new EmailAddress('alice@example.com'),
displayName: 'alice',
passwordHash: 'hash',
isAdmin: true,
emailConfirmedAt: $confirmedAt,
);
$this->assertSame(7, $user->getId());
$this->assertSame('alice@example.com', $user->getEmail()->value());
$this->assertSame('alice', $user->getDisplayName());
$this->assertSame('hash', $user->getPasswordHash());
$this->assertTrue($user->isAdmin());
$this->assertSame($confirmedAt, $user->getEmailConfirmedAt());
}
public function test_user_email_confirmed_at_can_be_null(): void
{
$user = new User(
id: 1,
email: new EmailAddress('bob@example.com'),
displayName: 'bob',
passwordHash: '',
isAdmin: false,
emailConfirmedAt: null,
);
$this->assertNull($user->getEmailConfirmedAt());
$this->assertFalse($user->isEmailConfirmed());
}
public function test_email_confirmed_when_timestamp_present(): void
{
$user = new User(
id: 1,
email: new EmailAddress('bob@example.com'),
displayName: 'bob',
passwordHash: 'hash',
isAdmin: false,
emailConfirmedAt: new DateTimeImmutable('2026-05-01T00:00:00Z'),
);
$this->assertTrue($user->isEmailConfirmed());
}
}