Files
Classeo/backend/tests/Unit/Scolarite/Domain/Model/Grade/GradeTest.php
Mathias STRASSER b70d5ec2ad
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
feat: Permettre à l'enseignant de saisir les notes dans une grille inline
L'enseignant avait besoin d'un moyen rapide de saisir les notes après
une évaluation. La grille inline permet de compléter 30 élèves en moins
de 3 minutes grâce à la navigation clavier (Tab/Enter/Shift+Tab),
la validation temps réel, l'auto-save debounced (500ms) et les
raccourcis /abs et /disp pour marquer absents/dispensés.

Les notes restent en brouillon jusqu'à publication explicite (avec
confirmation modale). Une fois publiées, les élèves les voient
immédiatement ; les parents après un délai de 24h (VisibiliteNotesPolicy).
Le mode offline stocke les notes en IndexedDB et synchronise
automatiquement au retour de la connexion.

Chaque modification est auditée dans grade_events via un event
subscriber qui écoute NoteSaisie/NoteModifiee sur le bus d'événements.
2026-03-29 10:02:03 +02:00

286 lines
10 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Domain\Model\Grade;
use App\Administration\Domain\Model\User\UserId;
use App\Scolarite\Domain\Event\NoteModifiee;
use App\Scolarite\Domain\Event\NoteSaisie;
use App\Scolarite\Domain\Exception\NoteRequiseException;
use App\Scolarite\Domain\Exception\ValeurNoteInvalideException;
use App\Scolarite\Domain\Model\Evaluation\EvaluationId;
use App\Scolarite\Domain\Model\Evaluation\GradeScale;
use App\Scolarite\Domain\Model\Grade\Grade;
use App\Scolarite\Domain\Model\Grade\GradeId;
use App\Scolarite\Domain\Model\Grade\GradeStatus;
use App\Scolarite\Domain\Model\Grade\GradeValue;
use App\Shared\Domain\Tenant\TenantId;
use DateTimeImmutable;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class GradeTest extends TestCase
{
private const string TENANT_ID = '550e8400-e29b-41d4-a716-446655440001';
private const string EVALUATION_ID = '550e8400-e29b-41d4-a716-446655440040';
private const string STUDENT_ID = '550e8400-e29b-41d4-a716-446655440050';
private const string TEACHER_ID = '550e8400-e29b-41d4-a716-446655440010';
#[Test]
public function saisirCreatesGradedGrade(): void
{
$grade = $this->createGrade();
self::assertSame(GradeStatus::GRADED, $grade->status);
self::assertNotNull($grade->value);
self::assertSame(15.5, $grade->value->value);
}
#[Test]
public function saisirRecordsNoteSaisieEvent(): void
{
$grade = $this->createGrade();
$events = $grade->pullDomainEvents();
self::assertCount(1, $events);
self::assertInstanceOf(NoteSaisie::class, $events[0]);
self::assertSame($grade->id, $events[0]->gradeId);
self::assertSame(15.5, $events[0]->value);
self::assertSame('graded', $events[0]->status);
}
#[Test]
public function saisirSetsAllProperties(): void
{
$tenantId = TenantId::fromString(self::TENANT_ID);
$evaluationId = EvaluationId::fromString(self::EVALUATION_ID);
$studentId = UserId::fromString(self::STUDENT_ID);
$teacherId = UserId::fromString(self::TEACHER_ID);
$now = new DateTimeImmutable('2026-03-27 10:00:00');
$value = new GradeValue(15.5);
$gradeScale = new GradeScale(20);
$grade = Grade::saisir(
tenantId: $tenantId,
evaluationId: $evaluationId,
studentId: $studentId,
value: $value,
status: GradeStatus::GRADED,
gradeScale: $gradeScale,
createdBy: $teacherId,
now: $now,
);
self::assertTrue($grade->tenantId->equals($tenantId));
self::assertTrue($grade->evaluationId->equals($evaluationId));
self::assertTrue($grade->studentId->equals($studentId));
self::assertTrue($grade->createdBy->equals($teacherId));
self::assertSame(15.5, $grade->value->value);
self::assertSame(GradeStatus::GRADED, $grade->status);
self::assertEquals($now, $grade->createdAt);
self::assertEquals($now, $grade->updatedAt);
}
#[Test]
public function saisirCreatesAbsentGrade(): void
{
$grade = Grade::saisir(
tenantId: TenantId::fromString(self::TENANT_ID),
evaluationId: EvaluationId::fromString(self::EVALUATION_ID),
studentId: UserId::fromString(self::STUDENT_ID),
value: null,
status: GradeStatus::ABSENT,
gradeScale: new GradeScale(20),
createdBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 10:00:00'),
);
self::assertSame(GradeStatus::ABSENT, $grade->status);
self::assertNull($grade->value);
}
#[Test]
public function saisirCreatesDispensedGrade(): void
{
$grade = Grade::saisir(
tenantId: TenantId::fromString(self::TENANT_ID),
evaluationId: EvaluationId::fromString(self::EVALUATION_ID),
studentId: UserId::fromString(self::STUDENT_ID),
value: null,
status: GradeStatus::DISPENSED,
gradeScale: new GradeScale(20),
createdBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 10:00:00'),
);
self::assertSame(GradeStatus::DISPENSED, $grade->status);
self::assertNull($grade->value);
}
#[Test]
public function saisirThrowsWhenGradedWithoutValue(): void
{
$this->expectException(NoteRequiseException::class);
Grade::saisir(
tenantId: TenantId::fromString(self::TENANT_ID),
evaluationId: EvaluationId::fromString(self::EVALUATION_ID),
studentId: UserId::fromString(self::STUDENT_ID),
value: null,
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
createdBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 10:00:00'),
);
}
#[Test]
public function saisirThrowsWhenValueExceedsGradeScale(): void
{
$this->expectException(ValeurNoteInvalideException::class);
Grade::saisir(
tenantId: TenantId::fromString(self::TENANT_ID),
evaluationId: EvaluationId::fromString(self::EVALUATION_ID),
studentId: UserId::fromString(self::STUDENT_ID),
value: new GradeValue(25.0),
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
createdBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 10:00:00'),
);
}
#[Test]
public function modifierUpdatesValueAndRecordsEvent(): void
{
$grade = $this->createGrade();
$grade->pullDomainEvents();
$modifiedAt = new DateTimeImmutable('2026-03-27 14:00:00');
$modifierId = UserId::fromString(self::TEACHER_ID);
$grade->modifier(
value: new GradeValue(18.0),
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
modifiedBy: $modifierId,
now: $modifiedAt,
);
self::assertSame(18.0, $grade->value->value);
self::assertEquals($modifiedAt, $grade->updatedAt);
$events = $grade->pullDomainEvents();
self::assertCount(1, $events);
self::assertInstanceOf(NoteModifiee::class, $events[0]);
self::assertSame(15.5, $events[0]->oldValue);
self::assertSame(18.0, $events[0]->newValue);
self::assertSame('graded', $events[0]->oldStatus);
self::assertSame('graded', $events[0]->newStatus);
self::assertSame(self::TEACHER_ID, $events[0]->modifiedBy);
}
#[Test]
public function modifierChangesToAbsent(): void
{
$grade = $this->createGrade();
$grade->pullDomainEvents();
$grade->modifier(
value: null,
status: GradeStatus::ABSENT,
gradeScale: new GradeScale(20),
modifiedBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 14:00:00'),
);
self::assertSame(GradeStatus::ABSENT, $grade->status);
self::assertNull($grade->value);
}
#[Test]
public function modifierThrowsWhenGradedWithoutValue(): void
{
$grade = $this->createGrade();
$this->expectException(NoteRequiseException::class);
$grade->modifier(
value: null,
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
modifiedBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 14:00:00'),
);
}
#[Test]
public function modifierThrowsWhenValueExceedsGradeScale(): void
{
$grade = $this->createGrade();
$this->expectException(ValeurNoteInvalideException::class);
$grade->modifier(
value: new GradeValue(25.0),
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
modifiedBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 14:00:00'),
);
}
#[Test]
public function reconstituteRestoresAllPropertiesWithoutEvents(): void
{
$gradeId = GradeId::generate();
$tenantId = TenantId::fromString(self::TENANT_ID);
$evaluationId = EvaluationId::fromString(self::EVALUATION_ID);
$studentId = UserId::fromString(self::STUDENT_ID);
$teacherId = UserId::fromString(self::TEACHER_ID);
$createdAt = new DateTimeImmutable('2026-03-27 10:00:00');
$updatedAt = new DateTimeImmutable('2026-03-27 14:00:00');
$value = new GradeValue(15.5);
$grade = Grade::reconstitute(
id: $gradeId,
tenantId: $tenantId,
evaluationId: $evaluationId,
studentId: $studentId,
value: $value,
status: GradeStatus::GRADED,
createdBy: $teacherId,
createdAt: $createdAt,
updatedAt: $updatedAt,
);
self::assertTrue($grade->id->equals($gradeId));
self::assertTrue($grade->tenantId->equals($tenantId));
self::assertTrue($grade->evaluationId->equals($evaluationId));
self::assertTrue($grade->studentId->equals($studentId));
self::assertTrue($grade->createdBy->equals($teacherId));
self::assertSame(15.5, $grade->value->value);
self::assertSame(GradeStatus::GRADED, $grade->status);
self::assertEquals($createdAt, $grade->createdAt);
self::assertEquals($updatedAt, $grade->updatedAt);
self::assertEmpty($grade->pullDomainEvents());
}
private function createGrade(): Grade
{
return Grade::saisir(
tenantId: TenantId::fromString(self::TENANT_ID),
evaluationId: EvaluationId::fromString(self::EVALUATION_ID),
studentId: UserId::fromString(self::STUDENT_ID),
value: new GradeValue(15.5),
status: GradeStatus::GRADED,
gradeScale: new GradeScale(20),
createdBy: UserId::fromString(self::TEACHER_ID),
now: new DateTimeImmutable('2026-03-27 10:00:00'),
);
}
}