Files
SIRH/tests/Service/WorkHours/OvertimeContingentExportBuilderTest.php
T
tristan 327c10fda4
Auto Tag Develop / tag (push) Successful in 7s
feat(overtime-contingent) : contingent d'heures supplémentaires payées (#29)
## Résumé
Suivi par **année civile** (Janv–Déc) des heures supplémentaires payées des employés non-forfait (chauffeurs inclus) face au plafond légal (**350 h** chauffeurs / **220 h** autres).

- **Fiche employé** : encart header `Total H.payés {année} : X h / plafond h` (année civile courante, rouge si dépassement), via `GET /employees/{id}/overtime-contingent`.
- **Export PDF** `GET /overtime-contingent/print?year=&siteIds=` (ROLE_USER, périmètre `findScoped`) : groupé par site, colonnes Janv–Déc + colonne `Total payé / payable`. Drawer liste employés (année + sites).
- Heures payées = `base25 + base50` (hors majoration). Mapping exercice→civil : `mois ≥ 6 ? exercice−1 : exercice`.
- Cœur partagé pur `OvertimePaidContingentCalculator`.
- Ajout « Année civile » dans le titre des deux exports PDF (contingent H.supp. et heures de nuit).

## Tests
- 214 tests PHPUnit verts (calculateur : mapping civil, base-only, plafond ; builder : ventilation mensuelle, ligne à zéro).

## Hors périmètre (consigné)
- Bug latent `SalaryRecapPrintProvider` : rattachement des paiements RTT des mois Juin–Déc par année civile sur un stockage par exercice. À traiter séparément.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Reviewed-on: #29
Co-authored-by: tristan <tristan@yuno.malio.fr>
Co-committed-by: tristan <tristan@yuno.malio.fr>
2026-06-11 15:47:19 +00:00

76 lines
2.7 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Tests\Service\WorkHours;
use App\Entity\Employee;
use App\Entity\EmployeeRttPayment;
use App\Repository\EmployeeRttPaymentRepository;
use App\Service\WorkHours\OvertimeContingentExportBuilder;
use App\Service\WorkHours\OvertimePaidContingentCalculator;
use PHPUnit\Framework\TestCase;
use ReflectionProperty;
/**
* @internal
*/
final class OvertimeContingentExportBuilderTest extends TestCase
{
public function testBuildsRowsWithMonthlyTotalsAndCap(): void
{
// isDriver est résolu via le contrat courant : on le force par une
// sous-classe anonyme pour rester en test unitaire (sans BDD).
$driverEmp = new class extends Employee {
public function getIsDriver(): bool
{
return true;
}
};
$driverEmp->setLastName('Martin')->setFirstName('Luc');
$idRef = new ReflectionProperty(Employee::class, 'id');
$idRef->setValue($driverEmp, 7);
// Paiement : exercice 2027, mois 9 -> civil 2026, mois 9 ; base 100+20.
$payment = new EmployeeRttPayment()
->setEmployee($driverEmp)
->setYear(2027)->setMonth(9)
->setBase25Minutes(100)->setBase50Minutes(20)
;
$repo = $this->createStub(EmployeeRttPaymentRepository::class);
$repo->method('findByEmployeesAndYears')->willReturn([$payment]);
$builder = new OvertimeContingentExportBuilder($repo, new OvertimePaidContingentCalculator());
$rows = $builder->buildRows([$driverEmp], 2026);
self::assertCount(1, $rows);
self::assertSame(7, $rows[0]->employeeId);
self::assertSame('Martin Luc', $rows[0]->employeeName);
self::assertSame(120, $rows[0]->months[9]);
self::assertSame(0, $rows[0]->months[1]);
self::assertSame(120, $rows[0]->totalMinutes);
self::assertSame(350, $rows[0]->capHours); // chauffeur
}
public function testEmployeeWithNoPaymentsYieldsZeroRow(): void
{
$emp = new Employee();
$emp->setLastName('Durand')->setFirstName('Alice');
$idRef = new ReflectionProperty(Employee::class, 'id');
$idRef->setValue($emp, 99);
$repo = $this->createStub(EmployeeRttPaymentRepository::class);
$repo->method('findByEmployeesAndYears')->willReturn([]);
$builder = new OvertimeContingentExportBuilder($repo, new OvertimePaidContingentCalculator());
$rows = $builder->buildRows([$emp], 2026);
self::assertCount(1, $rows);
self::assertSame(0, $rows[0]->totalMinutes);
self::assertSame(0, $rows[0]->months[6]);
self::assertSame(220, $rows[0]->capHours); // non-driver
}
}