feat: Permettre aux enseignants de créer et gérer les devoirs
Some checks failed
CI / Backend Tests (push) Has been cancelled
CI / Frontend Tests (push) Has been cancelled
CI / E2E Tests (push) Has been cancelled
CI / Naming Conventions (push) Has been cancelled
CI / Build Check (push) Has been cancelled

Les enseignants avaient besoin d'un outil pour créer des devoirs assignés
à leurs classes, avec filtrage automatique par matière selon la classe
sélectionnée. Le système valide que la date d'échéance tombe un jour
ouvrable (lundi-vendredi) et empêche les dates dans le passé.

Le domaine modélise le devoir comme un agrégat avec pièces jointes,
statut brouillon/publié, et événements métier (création, modification,
suppression). Les handlers de notification écoutent ces événements pour
les futurs envois aux parents et élèves.
This commit is contained in:
2026-03-12 10:11:06 +01:00
parent 56bc808d85
commit e9efb90f59
51 changed files with 4776 additions and 7 deletions

View File

@@ -0,0 +1,157 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Application\Command\CreateHomework;
use App\Administration\Domain\Model\SchoolCalendar\SchoolCalendar;
use App\Administration\Domain\Model\SchoolClass\AcademicYearId;
use App\Administration\Domain\Model\SchoolClass\ClassId;
use App\Administration\Domain\Model\Subject\SubjectId;
use App\Administration\Domain\Model\User\UserId;
use App\Scolarite\Application\Command\CreateHomework\CreateHomeworkCommand;
use App\Scolarite\Application\Command\CreateHomework\CreateHomeworkHandler;
use App\Scolarite\Application\Port\CurrentCalendarProvider;
use App\Scolarite\Application\Port\EnseignantAffectationChecker;
use App\Scolarite\Domain\Exception\DateEcheanceInvalideException;
use App\Scolarite\Domain\Exception\EnseignantNonAffecteException;
use App\Scolarite\Domain\Model\Homework\HomeworkId;
use App\Scolarite\Domain\Model\Homework\HomeworkStatus;
use App\Scolarite\Domain\Service\DueDateValidator;
use App\Scolarite\Infrastructure\Persistence\InMemory\InMemoryHomeworkRepository;
use App\Shared\Domain\Clock;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class CreateHomeworkHandlerTest extends TestCase
{
private const string TENANT_ID = '550e8400-e29b-41d4-a716-446655440001';
private const string CLASS_ID = '550e8400-e29b-41d4-a716-446655440020';
private const string SUBJECT_ID = '550e8400-e29b-41d4-a716-446655440030';
private const string TEACHER_ID = '550e8400-e29b-41d4-a716-446655440010';
private InMemoryHomeworkRepository $homeworkRepository;
private Clock $clock;
protected function setUp(): void
{
$this->homeworkRepository = new InMemoryHomeworkRepository();
$this->clock = new class implements Clock {
public function now(): DateTimeImmutable
{
return new DateTimeImmutable('2026-03-12 10:00:00');
}
};
}
#[Test]
public function itCreatesHomeworkSuccessfully(): void
{
$handler = $this->createHandler(affecte: true);
$command = $this->createCommand();
$homework = $handler($command);
self::assertNotEmpty((string) $homework->id);
self::assertSame(HomeworkStatus::PUBLISHED, $homework->status);
self::assertSame('Exercices chapitre 5', $homework->title);
}
#[Test]
public function itPersistsHomeworkInRepository(): void
{
$handler = $this->createHandler(affecte: true);
$command = $this->createCommand();
$created = $handler($command);
$homework = $this->homeworkRepository->get(
HomeworkId::fromString((string) $created->id),
TenantId::fromString(self::TENANT_ID),
);
self::assertSame('Exercices chapitre 5', $homework->title);
}
#[Test]
public function itThrowsWhenTeacherNotAffected(): void
{
$handler = $this->createHandler(affecte: false);
$this->expectException(EnseignantNonAffecteException::class);
$handler($this->createCommand());
}
#[Test]
public function itThrowsWhenDueDateIsInvalid(): void
{
$handler = $this->createHandler(affecte: true);
$this->expectException(DateEcheanceInvalideException::class);
$handler($this->createCommand(dueDate: '2026-03-11'));
}
#[Test]
public function itAllowsNullDescription(): void
{
$handler = $this->createHandler(affecte: true);
$command = $this->createCommand(description: null);
$homework = $handler($command);
self::assertNull($homework->description);
}
private function createHandler(bool $affecte): CreateHomeworkHandler
{
$affectationChecker = new class($affecte) implements EnseignantAffectationChecker {
public function __construct(private readonly bool $affecte)
{
}
public function estAffecte(UserId $teacherId, ClassId $classId, SubjectId $subjectId, TenantId $tenantId): bool
{
return $this->affecte;
}
};
$calendarProvider = new class implements CurrentCalendarProvider {
public function forCurrentYear(TenantId $tenantId): SchoolCalendar
{
return SchoolCalendar::reconstitute(
tenantId: $tenantId,
academicYearId: AcademicYearId::fromString('550e8400-e29b-41d4-a716-446655440002'),
zone: null,
entries: [],
);
}
};
return new CreateHomeworkHandler(
$this->homeworkRepository,
$affectationChecker,
$calendarProvider,
new DueDateValidator(),
$this->clock,
);
}
private function createCommand(
?string $dueDate = null,
mixed $description = 'Faire les exercices 1 à 10',
): CreateHomeworkCommand {
return new CreateHomeworkCommand(
tenantId: self::TENANT_ID,
classId: self::CLASS_ID,
subjectId: self::SUBJECT_ID,
teacherId: self::TEACHER_ID,
title: 'Exercices chapitre 5',
description: $description,
dueDate: $dueDate ?? '2026-04-15',
);
}
}

View File

@@ -0,0 +1,130 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Application\Command\DeleteHomework;
use App\Administration\Domain\Model\SchoolClass\ClassId;
use App\Administration\Domain\Model\Subject\SubjectId;
use App\Administration\Domain\Model\User\UserId;
use App\Scolarite\Application\Command\DeleteHomework\DeleteHomeworkCommand;
use App\Scolarite\Application\Command\DeleteHomework\DeleteHomeworkHandler;
use App\Scolarite\Domain\Exception\DevoirDejaSupprimeException;
use App\Scolarite\Domain\Exception\HomeworkNotFoundException;
use App\Scolarite\Domain\Exception\NonProprietaireDuDevoirException;
use App\Scolarite\Domain\Model\Homework\Homework;
use App\Scolarite\Domain\Model\Homework\HomeworkId;
use App\Scolarite\Domain\Model\Homework\HomeworkStatus;
use App\Scolarite\Infrastructure\Persistence\InMemory\InMemoryHomeworkRepository;
use App\Shared\Domain\Clock;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class DeleteHomeworkHandlerTest extends TestCase
{
private const string TENANT_ID = '550e8400-e29b-41d4-a716-446655440001';
private InMemoryHomeworkRepository $homeworkRepository;
private Clock $clock;
private HomeworkId $existingHomeworkId;
protected function setUp(): void
{
$this->homeworkRepository = new InMemoryHomeworkRepository();
$this->clock = new class implements Clock {
public function now(): DateTimeImmutable
{
return new DateTimeImmutable('2026-03-12 10:00:00');
}
};
$this->seedHomework();
}
#[Test]
public function itDeletesHomeworkSuccessfully(): void
{
$handler = $this->createHandler();
$homework = $handler(new DeleteHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440010',
));
self::assertSame(HomeworkStatus::DELETED, $homework->status);
}
#[Test]
public function itThrowsWhenHomeworkNotFound(): void
{
$handler = $this->createHandler();
$this->expectException(HomeworkNotFoundException::class);
$handler(new DeleteHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) HomeworkId::generate(),
teacherId: '550e8400-e29b-41d4-a716-446655440010',
));
}
#[Test]
public function itThrowsWhenHomeworkAlreadyDeleted(): void
{
$homework = $this->homeworkRepository->get($this->existingHomeworkId, TenantId::fromString(self::TENANT_ID));
$homework->supprimer(new DateTimeImmutable('2026-03-12'));
$this->homeworkRepository->save($homework);
$handler = $this->createHandler();
$this->expectException(DevoirDejaSupprimeException::class);
$handler(new DeleteHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440010',
));
}
#[Test]
public function itThrowsWhenNotOwner(): void
{
$handler = $this->createHandler();
$this->expectException(NonProprietaireDuDevoirException::class);
$handler(new DeleteHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440099',
));
}
private function seedHomework(): void
{
$homework = Homework::creer(
tenantId: TenantId::fromString(self::TENANT_ID),
classId: ClassId::fromString('550e8400-e29b-41d4-a716-446655440020'),
subjectId: SubjectId::fromString('550e8400-e29b-41d4-a716-446655440030'),
teacherId: UserId::fromString('550e8400-e29b-41d4-a716-446655440010'),
title: 'Exercices',
description: 'Description',
dueDate: new DateTimeImmutable('2026-04-15'),
now: new DateTimeImmutable('2026-03-10 10:00:00'),
);
$this->existingHomeworkId = $homework->id;
$this->homeworkRepository->save($homework);
}
private function createHandler(): DeleteHomeworkHandler
{
return new DeleteHomeworkHandler(
$this->homeworkRepository,
$this->clock,
);
}
}

View File

@@ -0,0 +1,179 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Application\Command\UpdateHomework;
use App\Administration\Domain\Model\SchoolCalendar\SchoolCalendar;
use App\Administration\Domain\Model\SchoolClass\AcademicYearId;
use App\Administration\Domain\Model\SchoolClass\ClassId;
use App\Administration\Domain\Model\Subject\SubjectId;
use App\Administration\Domain\Model\User\UserId;
use App\Scolarite\Application\Command\UpdateHomework\UpdateHomeworkCommand;
use App\Scolarite\Application\Command\UpdateHomework\UpdateHomeworkHandler;
use App\Scolarite\Application\Port\CurrentCalendarProvider;
use App\Scolarite\Domain\Exception\DateEcheanceInvalideException;
use App\Scolarite\Domain\Exception\DevoirDejaSupprimeException;
use App\Scolarite\Domain\Exception\HomeworkNotFoundException;
use App\Scolarite\Domain\Exception\NonProprietaireDuDevoirException;
use App\Scolarite\Domain\Model\Homework\Homework;
use App\Scolarite\Domain\Model\Homework\HomeworkId;
use App\Scolarite\Domain\Service\DueDateValidator;
use App\Scolarite\Infrastructure\Persistence\InMemory\InMemoryHomeworkRepository;
use App\Shared\Domain\Clock;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class UpdateHomeworkHandlerTest extends TestCase
{
private const string TENANT_ID = '550e8400-e29b-41d4-a716-446655440001';
private InMemoryHomeworkRepository $homeworkRepository;
private Clock $clock;
private HomeworkId $existingHomeworkId;
protected function setUp(): void
{
$this->homeworkRepository = new InMemoryHomeworkRepository();
$this->clock = new class implements Clock {
public function now(): DateTimeImmutable
{
return new DateTimeImmutable('2026-03-12 10:00:00');
}
};
$this->seedHomework();
}
#[Test]
public function itUpdatesHomeworkSuccessfully(): void
{
$handler = $this->createHandler();
$command = new UpdateHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440010',
title: 'Titre modifié',
description: 'Nouvelle description',
dueDate: '2026-04-20',
);
$homework = $handler($command);
self::assertSame('Titre modifié', $homework->title);
self::assertSame('Nouvelle description', $homework->description);
}
#[Test]
public function itThrowsWhenHomeworkNotFound(): void
{
$handler = $this->createHandler();
$this->expectException(HomeworkNotFoundException::class);
$handler(new UpdateHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) HomeworkId::generate(),
teacherId: '550e8400-e29b-41d4-a716-446655440010',
title: 'Test',
description: null,
dueDate: '2026-04-20',
));
}
#[Test]
public function itThrowsWhenDueDateInvalid(): void
{
$handler = $this->createHandler();
$this->expectException(DateEcheanceInvalideException::class);
$handler(new UpdateHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440010',
title: 'Test',
description: null,
dueDate: '2026-03-11',
));
}
#[Test]
public function itThrowsWhenHomeworkDeleted(): void
{
$homework = $this->homeworkRepository->get($this->existingHomeworkId, TenantId::fromString(self::TENANT_ID));
$homework->supprimer(new DateTimeImmutable('2026-03-12'));
$this->homeworkRepository->save($homework);
$handler = $this->createHandler();
$this->expectException(DevoirDejaSupprimeException::class);
$handler(new UpdateHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440010',
title: 'Test',
description: null,
dueDate: '2026-04-20',
));
}
#[Test]
public function itThrowsWhenNotOwner(): void
{
$handler = $this->createHandler();
$this->expectException(NonProprietaireDuDevoirException::class);
$handler(new UpdateHomeworkCommand(
tenantId: self::TENANT_ID,
homeworkId: (string) $this->existingHomeworkId,
teacherId: '550e8400-e29b-41d4-a716-446655440099',
title: 'Test',
description: null,
dueDate: '2026-04-20',
));
}
private function seedHomework(): void
{
$homework = Homework::creer(
tenantId: TenantId::fromString(self::TENANT_ID),
classId: ClassId::fromString('550e8400-e29b-41d4-a716-446655440020'),
subjectId: SubjectId::fromString('550e8400-e29b-41d4-a716-446655440030'),
teacherId: UserId::fromString('550e8400-e29b-41d4-a716-446655440010'),
title: 'Exercices',
description: 'Description',
dueDate: new DateTimeImmutable('2026-04-15'),
now: new DateTimeImmutable('2026-03-10 10:00:00'),
);
$this->existingHomeworkId = $homework->id;
$this->homeworkRepository->save($homework);
}
private function createHandler(): UpdateHomeworkHandler
{
$calendarProvider = new class implements CurrentCalendarProvider {
public function forCurrentYear(TenantId $tenantId): SchoolCalendar
{
return SchoolCalendar::reconstitute(
tenantId: $tenantId,
academicYearId: AcademicYearId::fromString('550e8400-e29b-41d4-a716-446655440002'),
zone: null,
entries: [],
);
}
};
return new UpdateHomeworkHandler(
$this->homeworkRepository,
$calendarProvider,
new DueDateValidator(),
$this->clock,
);
}
}

View File

@@ -0,0 +1,106 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Domain\Model\Homework;
use App\Scolarite\Domain\Exception\PieceJointeInvalideException;
use App\Scolarite\Domain\Model\Homework\HomeworkAttachment;
use App\Scolarite\Domain\Model\Homework\HomeworkAttachmentId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class HomeworkAttachmentTest extends TestCase
{
#[Test]
public function createsValidAttachment(): void
{
$id = HomeworkAttachmentId::generate();
$uploadedAt = new DateTimeImmutable('2026-03-12 10:00:00');
$attachment = new HomeworkAttachment(
id: $id,
filename: 'exercices.pdf',
filePath: 'homework/abc123/exercices.pdf',
fileSize: 500_000,
mimeType: 'application/pdf',
uploadedAt: $uploadedAt,
);
self::assertTrue($attachment->id->equals($id));
self::assertSame('exercices.pdf', $attachment->filename);
self::assertSame('homework/abc123/exercices.pdf', $attachment->filePath);
self::assertSame(500_000, $attachment->fileSize);
self::assertSame('application/pdf', $attachment->mimeType);
self::assertEquals($uploadedAt, $attachment->uploadedAt);
}
#[Test]
public function acceptsJpegMimeType(): void
{
$attachment = $this->createAttachmentWithMimeType('image/jpeg');
self::assertSame('image/jpeg', $attachment->mimeType);
}
#[Test]
public function acceptsPngMimeType(): void
{
$attachment = $this->createAttachmentWithMimeType('image/png');
self::assertSame('image/png', $attachment->mimeType);
}
#[Test]
public function rejectsInvalidMimeType(): void
{
$this->expectException(PieceJointeInvalideException::class);
$this->expectExceptionMessageMatches('/text\/plain/');
$this->createAttachmentWithMimeType('text/plain');
}
#[Test]
public function rejectsFileTooLarge(): void
{
$this->expectException(PieceJointeInvalideException::class);
$this->expectExceptionMessageMatches('/taille maximum/');
new HomeworkAttachment(
id: HomeworkAttachmentId::generate(),
filename: 'big.pdf',
filePath: 'homework/abc/big.pdf',
fileSize: 11 * 1024 * 1024, // 11 Mo
mimeType: 'application/pdf',
uploadedAt: new DateTimeImmutable(),
);
}
#[Test]
public function acceptsExactMaxSize(): void
{
$attachment = new HomeworkAttachment(
id: HomeworkAttachmentId::generate(),
filename: 'max.pdf',
filePath: 'homework/abc/max.pdf',
fileSize: 10 * 1024 * 1024, // Exactement 10 Mo
mimeType: 'application/pdf',
uploadedAt: new DateTimeImmutable(),
);
self::assertSame(10 * 1024 * 1024, $attachment->fileSize);
}
private function createAttachmentWithMimeType(string $mimeType): HomeworkAttachment
{
return new HomeworkAttachment(
id: HomeworkAttachmentId::generate(),
filename: 'test.file',
filePath: 'homework/abc/test.file',
fileSize: 1000,
mimeType: $mimeType,
uploadedAt: new DateTimeImmutable(),
);
}
}

View File

@@ -0,0 +1,223 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Domain\Model\Homework;
use App\Administration\Domain\Model\SchoolClass\ClassId;
use App\Administration\Domain\Model\Subject\SubjectId;
use App\Administration\Domain\Model\User\UserId;
use App\Scolarite\Domain\Event\DevoirCree;
use App\Scolarite\Domain\Event\DevoirModifie;
use App\Scolarite\Domain\Event\DevoirSupprime;
use App\Scolarite\Domain\Exception\DevoirDejaSupprimeException;
use App\Scolarite\Domain\Model\Homework\Homework;
use App\Scolarite\Domain\Model\Homework\HomeworkId;
use App\Scolarite\Domain\Model\Homework\HomeworkStatus;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class HomeworkTest extends TestCase
{
private const string TENANT_ID = '550e8400-e29b-41d4-a716-446655440001';
private const string CLASS_ID = '550e8400-e29b-41d4-a716-446655440020';
private const string SUBJECT_ID = '550e8400-e29b-41d4-a716-446655440030';
private const string TEACHER_ID = '550e8400-e29b-41d4-a716-446655440010';
#[Test]
public function creerCreatesPublishedHomework(): void
{
$homework = $this->createHomework();
self::assertSame(HomeworkStatus::PUBLISHED, $homework->status);
}
#[Test]
public function creerRecordsDevoirCreeEvent(): void
{
$homework = $this->createHomework();
$events = $homework->pullDomainEvents();
self::assertCount(1, $events);
self::assertInstanceOf(DevoirCree::class, $events[0]);
self::assertSame($homework->id, $events[0]->homeworkId);
}
#[Test]
public function creerSetsAllProperties(): void
{
$tenantId = TenantId::fromString(self::TENANT_ID);
$classId = ClassId::fromString(self::CLASS_ID);
$subjectId = SubjectId::fromString(self::SUBJECT_ID);
$teacherId = UserId::fromString(self::TEACHER_ID);
$dueDate = new DateTimeImmutable('2026-04-15');
$now = new DateTimeImmutable('2026-03-12 10:00:00');
$homework = Homework::creer(
tenantId: $tenantId,
classId: $classId,
subjectId: $subjectId,
teacherId: $teacherId,
title: 'Exercices chapitre 5',
description: 'Faire les exercices 1 à 10',
dueDate: $dueDate,
now: $now,
);
self::assertTrue($homework->tenantId->equals($tenantId));
self::assertTrue($homework->classId->equals($classId));
self::assertTrue($homework->subjectId->equals($subjectId));
self::assertTrue($homework->teacherId->equals($teacherId));
self::assertSame('Exercices chapitre 5', $homework->title);
self::assertSame('Faire les exercices 1 à 10', $homework->description);
self::assertEquals($dueDate, $homework->dueDate);
self::assertSame(HomeworkStatus::PUBLISHED, $homework->status);
self::assertEquals($now, $homework->createdAt);
self::assertEquals($now, $homework->updatedAt);
}
#[Test]
public function creerAllowsNullDescription(): void
{
$homework = Homework::creer(
tenantId: TenantId::fromString(self::TENANT_ID),
classId: ClassId::fromString(self::CLASS_ID),
subjectId: SubjectId::fromString(self::SUBJECT_ID),
teacherId: UserId::fromString(self::TEACHER_ID),
title: 'Devoir sans description',
description: null,
dueDate: new DateTimeImmutable('2026-04-15'),
now: new DateTimeImmutable('2026-03-12 10:00:00'),
);
self::assertNull($homework->description);
}
#[Test]
public function modifierUpdatesFieldsAndRecordsEvent(): void
{
$homework = $this->createHomework();
$homework->pullDomainEvents();
$modifiedAt = new DateTimeImmutable('2026-03-13 14:00:00');
$newDueDate = new DateTimeImmutable('2026-04-20');
$homework->modifier(
title: 'Titre modifié',
description: 'Nouvelle description',
dueDate: $newDueDate,
now: $modifiedAt,
);
self::assertSame('Titre modifié', $homework->title);
self::assertSame('Nouvelle description', $homework->description);
self::assertEquals($newDueDate, $homework->dueDate);
self::assertEquals($modifiedAt, $homework->updatedAt);
$events = $homework->pullDomainEvents();
self::assertCount(1, $events);
self::assertInstanceOf(DevoirModifie::class, $events[0]);
self::assertSame($homework->id, $events[0]->homeworkId);
}
#[Test]
public function modifierThrowsWhenDeleted(): void
{
$homework = $this->createHomework();
$homework->supprimer(new DateTimeImmutable('2026-03-13'));
$this->expectException(DevoirDejaSupprimeException::class);
$homework->modifier(
title: 'Titre',
description: null,
dueDate: new DateTimeImmutable('2026-04-20'),
now: new DateTimeImmutable('2026-03-14'),
);
}
#[Test]
public function supprimerChangesStatusAndRecordsEvent(): void
{
$homework = $this->createHomework();
$homework->pullDomainEvents();
$deletedAt = new DateTimeImmutable('2026-03-14 08:00:00');
$homework->supprimer($deletedAt);
self::assertSame(HomeworkStatus::DELETED, $homework->status);
self::assertEquals($deletedAt, $homework->updatedAt);
$events = $homework->pullDomainEvents();
self::assertCount(1, $events);
self::assertInstanceOf(DevoirSupprime::class, $events[0]);
self::assertSame($homework->id, $events[0]->homeworkId);
}
#[Test]
public function supprimerThrowsWhenAlreadyDeleted(): void
{
$homework = $this->createHomework();
$homework->supprimer(new DateTimeImmutable('2026-03-14'));
$this->expectException(DevoirDejaSupprimeException::class);
$homework->supprimer(new DateTimeImmutable('2026-03-15'));
}
#[Test]
public function reconstituteRestoresAllPropertiesWithoutEvents(): void
{
$id = HomeworkId::generate();
$tenantId = TenantId::fromString(self::TENANT_ID);
$classId = ClassId::fromString(self::CLASS_ID);
$subjectId = SubjectId::fromString(self::SUBJECT_ID);
$teacherId = UserId::fromString(self::TEACHER_ID);
$dueDate = new DateTimeImmutable('2026-04-15');
$createdAt = new DateTimeImmutable('2026-03-12 10:00:00');
$updatedAt = new DateTimeImmutable('2026-03-13 14:00:00');
$homework = Homework::reconstitute(
id: $id,
tenantId: $tenantId,
classId: $classId,
subjectId: $subjectId,
teacherId: $teacherId,
title: 'Exercices chapitre 5',
description: 'Faire les exercices',
dueDate: $dueDate,
status: HomeworkStatus::PUBLISHED,
createdAt: $createdAt,
updatedAt: $updatedAt,
);
self::assertTrue($homework->id->equals($id));
self::assertTrue($homework->tenantId->equals($tenantId));
self::assertTrue($homework->classId->equals($classId));
self::assertTrue($homework->subjectId->equals($subjectId));
self::assertTrue($homework->teacherId->equals($teacherId));
self::assertSame('Exercices chapitre 5', $homework->title);
self::assertSame('Faire les exercices', $homework->description);
self::assertEquals($dueDate, $homework->dueDate);
self::assertSame(HomeworkStatus::PUBLISHED, $homework->status);
self::assertEquals($createdAt, $homework->createdAt);
self::assertEquals($updatedAt, $homework->updatedAt);
self::assertEmpty($homework->pullDomainEvents());
}
private function createHomework(): Homework
{
return Homework::creer(
tenantId: TenantId::fromString(self::TENANT_ID),
classId: ClassId::fromString(self::CLASS_ID),
subjectId: SubjectId::fromString(self::SUBJECT_ID),
teacherId: UserId::fromString(self::TEACHER_ID),
title: 'Exercices chapitre 5',
description: 'Faire les exercices 1 à 10',
dueDate: new DateTimeImmutable('2026-04-15'),
now: new DateTimeImmutable('2026-03-12 10:00:00'),
);
}
}

View File

@@ -0,0 +1,138 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Domain\Service;
use App\Administration\Domain\Model\SchoolCalendar\CalendarEntry;
use App\Administration\Domain\Model\SchoolCalendar\CalendarEntryId;
use App\Administration\Domain\Model\SchoolCalendar\CalendarEntryType;
use App\Administration\Domain\Model\SchoolCalendar\SchoolCalendar;
use App\Administration\Domain\Model\SchoolClass\AcademicYearId;
use App\Scolarite\Domain\Exception\DateEcheanceInvalideException;
use App\Scolarite\Domain\Service\DueDateValidator;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class DueDateValidatorTest extends TestCase
{
private DueDateValidator $validator;
protected function setUp(): void
{
$this->validator = new DueDateValidator();
}
#[Test]
public function acceptsValidFutureSchoolDay(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00'); // Jeudi
$dueDate = new DateTimeImmutable('2026-03-16'); // Lundi
$calendar = $this->createEmptyCalendar();
$this->validator->valider($dueDate, $now, $calendar);
$this->addToAssertionCount(1);
}
#[Test]
public function rejectsPastDate(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00');
$dueDate = new DateTimeImmutable('2026-03-11');
$calendar = $this->createEmptyCalendar();
$this->expectException(DateEcheanceInvalideException::class);
$this->expectExceptionMessageMatches('/futur/');
$this->validator->valider($dueDate, $now, $calendar);
}
#[Test]
public function rejectsTodayDate(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00');
$dueDate = new DateTimeImmutable('2026-03-12');
$calendar = $this->createEmptyCalendar();
$this->expectException(DateEcheanceInvalideException::class);
$this->expectExceptionMessageMatches('/futur/');
$this->validator->valider($dueDate, $now, $calendar);
}
#[Test]
public function acceptsTomorrowDate(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00'); // Jeudi
$dueDate = new DateTimeImmutable('2026-03-13'); // Vendredi
$calendar = $this->createEmptyCalendar();
$this->validator->valider($dueDate, $now, $calendar);
$this->addToAssertionCount(1);
}
#[Test]
public function rejectsWeekendDate(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00');
$dueDate = new DateTimeImmutable('2026-03-14'); // Samedi
$calendar = $this->createEmptyCalendar();
$this->expectException(DateEcheanceInvalideException::class);
$this->expectExceptionMessageMatches('/weekend/');
$this->validator->valider($dueDate, $now, $calendar);
}
#[Test]
public function rejectsHolidayDate(): void
{
$now = new DateTimeImmutable('2026-03-12 10:00:00');
$dueDate = new DateTimeImmutable('2026-04-06'); // Lundi
$calendar = $this->createCalendarWithVacation(
new DateTimeImmutable('2026-04-04'),
new DateTimeImmutable('2026-04-19'),
'Vacances de printemps',
);
$this->expectException(DateEcheanceInvalideException::class);
$this->expectExceptionMessageMatches('/Vacances de printemps/');
$this->validator->valider($dueDate, $now, $calendar);
}
private function createEmptyCalendar(): SchoolCalendar
{
return SchoolCalendar::reconstitute(
tenantId: TenantId::fromString('550e8400-e29b-41d4-a716-446655440001'),
academicYearId: AcademicYearId::fromString('550e8400-e29b-41d4-a716-446655440002'),
zone: null,
entries: [],
);
}
private function createCalendarWithVacation(
DateTimeImmutable $start,
DateTimeImmutable $end,
string $label,
): SchoolCalendar {
return SchoolCalendar::reconstitute(
tenantId: TenantId::fromString('550e8400-e29b-41d4-a716-446655440001'),
academicYearId: AcademicYearId::fromString('550e8400-e29b-41d4-a716-446655440002'),
zone: null,
entries: [
new CalendarEntry(
id: CalendarEntryId::generate(),
type: CalendarEntryType::VACATION,
startDate: $start,
endDate: $end,
label: $label,
),
],
);
}
}

View File

@@ -0,0 +1,41 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Infrastructure\Storage;
use App\Scolarite\Application\Port\FileStorage;
use function is_string;
use Override;
final class InMemoryFileStorage implements FileStorage
{
/** @var array<string, string> */
private array $files = [];
#[Override]
public function upload(string $path, mixed $content, string $mimeType): string
{
$this->files[$path] = is_string($content) ? $content : '';
return $path;
}
#[Override]
public function delete(string $path): void
{
unset($this->files[$path]);
}
public function has(string $path): bool
{
return isset($this->files[$path]);
}
public function get(string $path): ?string
{
return $this->files[$path] ?? null;
}
}