| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 |
- <?php
- declare(strict_types=1);
- namespace App\Tests\Services\Import;
- use App\Domain\Import\ParsedSheet;
- use App\Domain\TaskAssignment;
- use App\Services\Import\XlsxSprintImporter;
- use App\Tests\TestCase;
- /**
- * Phase 20 — parser smoke test against the real sample workbook
- * (doc/Tool_Sprint Planning.xlsx). Skipped when the host PHP lacks the
- * extensions PhpSpreadsheet needs (ext-dom, ext-zip etc.); the production
- * Docker image and standard CI runners both have them.
- */
- final class XlsxSprintImporterTest extends TestCase
- {
- private const FIXTURE = __DIR__ . '/../../../doc/Tool_Sprint Planning.xlsx';
- protected function setUp(): void
- {
- foreach (['dom', 'zip', 'xmlreader', 'simplexml', 'gd'] as $ext) {
- if (!extension_loaded($ext)) {
- $this->markTestSkipped("ext-{$ext} not loaded; PhpSpreadsheet cannot run on this host.");
- }
- }
- if (!is_file(self::FIXTURE)) {
- $this->markTestSkipped('Sample workbook not present at ' . self::FIXTURE);
- }
- }
- public function testParsesEverySheet(): void
- {
- $parser = new XlsxSprintImporter();
- $sheets = $parser->parse(self::FIXTURE);
- $this->assertCount(3, $sheets);
- $this->assertSame(['Sprint 1', 'Sprint 2', 'Sprint 3'], array_map(fn(ParsedSheet $s) => $s->sheetName, $sheets));
- }
- public function testSprint1ShapeAndCounts(): void
- {
- $parser = new XlsxSprintImporter();
- $sheets = $parser->parse(self::FIXTURE);
- $s = $sheets[0];
- $this->assertSame(5, count($s->weeks), '5 weeks');
- $this->assertSame(15, count($s->workers), '15 workers');
- $this->assertGreaterThan(20, count($s->tasks), 'more than 20 tasks');
- $this->assertEqualsWithDelta(0.2, $s->reserveFraction, 1e-9, 'reserve fraction = 0.2');
- $kws = array_map(fn($w) => $w->kw, $s->weeks);
- $this->assertSame([13, 14, 15, 16, 17], $kws, 'KWs 13..17 in order');
- $maxDays = array_map(fn($w) => $w->maxWorkingDays, $s->weeks);
- $this->assertSame([2, 4, 4, 5, 2], $maxDays);
- }
- public function testSprint2ColourMappingMatchesSpreadsheet(): void
- {
- $parser = new XlsxSprintImporter();
- $sheets = $parser->parse(self::FIXTURE);
- $s2 = $sheets[1];
- $this->assertSame('Sprint 2', $s2->sheetName);
- $statusCounts = [
- TaskAssignment::STATUS_ZUGEWIESEN => 0,
- TaskAssignment::STATUS_GESTARTET => 0,
- TaskAssignment::STATUS_ABGESCHLOSSEN => 0,
- TaskAssignment::STATUS_ABGEBROCHEN => 0,
- ];
- foreach ($s2->tasks as $t) {
- foreach ($t->assignments as $a) {
- $statusCounts[$a->status]++;
- }
- }
- // From the openpyxl colour audit on the sample:
- // 17× FFFFFF00 + 6× FFFFEB9C + 4× FFFFC000 = 27 yellow/orange -> gestartet
- // 4× FF00B050 + 1× FFC6EFCE = 5 green -> abgeschlossen
- // the only red-coded cells in the workbook are zero.
- $this->assertSame(27, $statusCounts[TaskAssignment::STATUS_GESTARTET], 'yellow + orange cells = 27');
- $this->assertSame(5, $statusCounts[TaskAssignment::STATUS_ABGESCHLOSSEN], 'green cells = 5');
- $this->assertSame(0, $statusCounts[TaskAssignment::STATUS_ABGEBROCHEN], 'no red cells in sample');
- }
- public function testSprint2SkipsArbeitstageGapAndDefinesSixteenWorkers(): void
- {
- // Sprint 2's Arbeitstage block has a blank row at C13 between Titus and
- // Suzan; the parser should resume past the gap and end up with 16 workers.
- $parser = new XlsxSprintImporter();
- $sheets = $parser->parse(self::FIXTURE);
- $s2 = $sheets[1];
- $this->assertCount(16, $s2->workers, 'Sprint 2 has 16 workers (gap row tolerated)');
- $names = array_map(fn($w) => $w->name, $s2->workers);
- $this->assertContains('Suzan', $names);
- $this->assertContains('Nicole', $names);
- }
- public function testRoundTripsViaToArrayFromArray(): void
- {
- $parser = new XlsxSprintImporter();
- $sheets = $parser->parse(self::FIXTURE);
- foreach ($sheets as $orig) {
- $clone = ParsedSheet::fromArray($orig->toArray());
- $this->assertSame($orig->sheetName, $clone->sheetName);
- $this->assertSame(count($orig->weeks), count($clone->weeks));
- $this->assertSame(count($orig->workers), count($clone->workers));
- $this->assertSame(count($orig->tasks), count($clone->tasks));
- $this->assertEqualsWithDelta($orig->reserveFraction, $clone->reserveFraction, 1e-9);
- }
- }
- }
|