feat: Calculer automatiquement les moyennes après chaque saisie de notes
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 ont besoin de moyennes à jour immédiatement après la
publication ou modification des notes, sans attendre un batch nocturne.

Le système recalcule via Domain Events synchrones : statistiques
d'évaluation (min/max/moyenne/médiane), moyennes matières pondérées
(normalisation /20), et moyenne générale par élève. Les résultats sont
stockés dans des tables dénormalisées avec cache Redis (TTL 5 min).

Trois endpoints API exposent les données avec contrôle d'accès par rôle.
Une commande console permet le backfill des données historiques au
déploiement.
This commit is contained in:
2026-03-30 06:22:03 +02:00
parent b70d5ec2ad
commit b7dc27f2a5
786 changed files with 118783 additions and 316 deletions

View File

@@ -0,0 +1,53 @@
<?php
declare(strict_types=1);
namespace App\Tests\Unit\Scolarite\Domain\Model\Competency;
use App\Scolarite\Domain\Model\Competency\CompetencyLevel;
use PHPUnit\Framework\Attributes\Test;
use PHPUnit\Framework\TestCase;
final class CompetencyLevelTest extends TestCase
{
#[Test]
public function hasFourLevels(): void
{
self::assertCount(4, CompetencyLevel::cases());
}
#[Test]
public function levelsHaveCorrectValues(): void
{
self::assertSame('not_acquired', CompetencyLevel::NOT_ACQUIRED->value);
self::assertSame('in_progress', CompetencyLevel::IN_PROGRESS->value);
self::assertSame('acquired', CompetencyLevel::ACQUIRED->value);
self::assertSame('exceeded', CompetencyLevel::EXCEEDED->value);
}
#[Test]
public function labelsAreInFrench(): void
{
self::assertSame('Non acquis', CompetencyLevel::NOT_ACQUIRED->label());
self::assertSame('En cours d\'acquisition', CompetencyLevel::IN_PROGRESS->label());
self::assertSame('Acquis', CompetencyLevel::ACQUIRED->label());
self::assertSame('Dépassé', CompetencyLevel::EXCEEDED->label());
}
#[Test]
public function sortOrderIsAscending(): void
{
self::assertSame(1, CompetencyLevel::NOT_ACQUIRED->sortOrder());
self::assertSame(2, CompetencyLevel::IN_PROGRESS->sortOrder());
self::assertSame(3, CompetencyLevel::ACQUIRED->sortOrder());
self::assertSame(4, CompetencyLevel::EXCEEDED->sortOrder());
}
#[Test]
public function colorsAreValid(): void
{
foreach (CompetencyLevel::cases() as $level) {
self::assertMatchesRegularExpression('/^#[0-9a-f]{6}$/', $level->color());
}
}
}