feat: migrer vers Symfony 8, PHP 8.4 et les dépendances majeures associées

- PHP 8.3 → 8.4 (Dockerfile + composer.json)
- Symfony 7.0 → 8.0 (tous les composants symfony/*)
- API Platform 3.x → 4.x : migration openapiContext → openapi: new Operation(...)
- Doctrine DBAL 3 → 4 : suppression use_savepoints, replace prepare/executeQuery
- Doctrine ORM 2.x → 3.x : ClassMetadataInfo → ClassMetadata, setParameters → setParameter
- Doctrine Bundle 2.x → 3.x, Fixtures Bundle 3.x → 4.x
- zenstruck/foundry 1.x → 2.x : ModelFactory → PersistentObjectFactory, getDefaults → defaults
- phpmd/phpmd 2.x → 3.x-dev (seule version supportant Symfony 8)
- phparkitect 0.3 → 0.8 : NotDependsOnTheseNamespaces prend un array
- symfony/mercure-bundle 0.3 → 0.4, symfony/monolog-bundle 3 → 4
- Suppression de runtime/frankenphp-symfony (intégré nativement dans symfony/runtime 8)
- worker.Caddyfile : suppression de APP_RUNTIME (détection automatique Symfony 8)
- Routes errors.xml/wdt.xml/profiler.xml → .php (Symfony 8 supprime le XML)
- Types::ARRAY → Types::JSON dans Entity/Manga.php (DBAL 4 retire array type)
- Suppression de src/Schedule.php (doublon vide avec MonitoringSchedule)
- Tests : hydra:Collection → Collection, hydra:member → member (API Platform 4)
This commit is contained in:
ext.jeremy.guillot@maxicoffee.domains
2026-03-26 17:55:12 +01:00
parent 5a0888eb28
commit 5ed303612a
371 changed files with 6194 additions and 4160 deletions

View File

@@ -26,13 +26,14 @@ class MangadexClientTest extends TestCase
);
}
private function mockAuthenticationResponse(): MockObject&ResponseInterface
private function mockAuthenticationResponse(): MockObject&ResponseInterface
{
$authResponse = $this->createMock(ResponseInterface::class);
$authResponse->method('toArray')->willReturn([
'access_token' => 'access_token',
'refresh_token' => 'refresh_token'
'refresh_token' => 'refresh_token',
]);
return $authResponse;
}
@@ -46,11 +47,11 @@ class MangadexClientTest extends TestCase
'POST',
'https://auth.mangadex.org/realms/mangadex/protocol/openid-connect/token',
$this->callback(function ($options) {
return $options['body']['grant_type'] === 'password'
&& $options['body']['username'] === 'username'
&& $options['body']['password'] === 'password'
&& $options['body']['client_id'] === 'client_id'
&& $options['body']['client_secret'] === 'client_secret';
return 'password' === $options['body']['grant_type']
&& 'username' === $options['body']['username']
&& 'password' === $options['body']['password']
&& 'client_id' === $options['body']['client_id']
&& 'client_secret' === $options['body']['client_secret'];
})
)
->willReturn($response);
@@ -76,10 +77,10 @@ class MangadexClientTest extends TestCase
[
'id' => '123',
'attributes' => [
'title' => ['en' => 'Test Manga']
]
]
]
'title' => ['en' => 'Test Manga'],
],
],
],
];
$authResponse = $this->mockAuthenticationResponse();
@@ -93,6 +94,7 @@ class MangadexClientTest extends TestCase
if (str_contains($url, 'auth.mangadex.org')) {
return $authResponse;
}
return $searchResponse;
});
@@ -105,9 +107,9 @@ class MangadexClientTest extends TestCase
$expectedResponse = [
'statistics' => [
'123' => [
'rating' => ['average' => 4.5]
]
]
'rating' => ['average' => 4.5],
],
],
];
$authResponse = $this->mockAuthenticationResponse();
@@ -121,10 +123,11 @@ class MangadexClientTest extends TestCase
if (str_contains($url, 'auth.mangadex.org')) {
return $authResponse;
}
return $ratingsResponse;
});
$result = $this->client->getMangaRatings(['123']);
$this->assertEquals($expectedResponse, $result);
}
}
}

View File

@@ -43,30 +43,30 @@ class MangadexProviderTest extends TestCase
'year' => 2020,
'status' => 'ongoing',
'tags' => [
['attributes' => ['name' => ['en' => 'Action']]]
]
['attributes' => ['name' => ['en' => 'Action']]],
],
],
'relationships' => [
[
'type' => 'author',
'attributes' => ['name' => 'Test Author']
'attributes' => ['name' => 'Test Author'],
],
[
'type' => 'cover_art',
'attributes' => ['fileName' => 'cover.jpg']
]
]
]
]
'attributes' => ['fileName' => 'cover.jpg'],
],
],
],
],
]);
$this->client->method('getMangaRatings')
->willReturn([
'statistics' => [
'123' => [
'rating' => ['average' => 4.5]
]
]
'rating' => ['average' => 4.5],
],
],
]);
$result = $this->provider->search('test');
@@ -74,7 +74,7 @@ class MangadexProviderTest extends TestCase
$this->assertCount(1, $mangas);
$manga = $mangas[0];
$this->assertEquals('Test Manga', $manga->getTitle()->getValue());
$this->assertEquals('test-manga', $manga->getSlug()->getValue());
$this->assertEquals('Test description', $manga->getDescription());
@@ -95,14 +95,14 @@ class MangadexProviderTest extends TestCase
'id' => '123',
'attributes' => [
// Missing required 'title' field
'description' => ['en' => 'Test description']
'description' => ['en' => 'Test description'],
],
'relationships' => []
]
]
'relationships' => [],
],
],
]);
$result = $this->provider->search('test');
$this->assertCount(0, $result->getItems());
}
}
}

View File

@@ -16,7 +16,7 @@ class FilenameAnalyzerTest extends TestCase
$this->analyzer = new FilenameAnalyzer();
}
public function test_it_analyzes_one_piece_filename_correctly(): void
public function testItAnalyzesOnePieceFilenameCorrectly(): void
{
// Given
$filename = 'one-piece_vol108_ch1094.cbz';
@@ -32,7 +32,7 @@ class FilenameAnalyzerTest extends TestCase
$this->assertTrue($result->hasVolumeNumber());
}
public function test_it_handles_different_filename_formats(): void
public function testItHandlesDifferentFilenameFormats(): void
{
$testCases = [
// Format underscore
@@ -77,7 +77,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_extracts_and_cleans_title(): void
public function testItExtractsAndCleansTitle(): void
{
// Given
$filename = 'one-piece_vol108_ch1094.cbz';
@@ -90,7 +90,7 @@ class FilenameAnalyzerTest extends TestCase
$this->assertNotEmpty($result->getTitle()->getValue(), 'Title should not be empty');
}
public function test_it_handles_files_without_volume_or_chapter(): void
public function testItHandlesFilesWithoutVolumeOrChapter(): void
{
$testCases = [
[
@@ -114,7 +114,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_handles_cbz_and_cbr_extensions(): void
public function testItHandlesCbzAndCbrExtensions(): void
{
// Given
$testCases = [
@@ -133,7 +133,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_cleans_common_patterns(): void
public function testItCleansCommonPatterns(): void
{
$testCases = [
[
@@ -160,7 +160,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_handles_filename_with_only_volume(): void
public function testItHandlesFilenameWithOnlyVolume(): void
{
$testCases = [
[
@@ -197,7 +197,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_handles_filename_with_only_chapter(): void
public function testItHandlesFilenameWithOnlyChapter(): void
{
$testCases = [
[
@@ -234,7 +234,7 @@ class FilenameAnalyzerTest extends TestCase
}
}
public function test_it_handles_full_dash_patterns(): void
public function testItHandlesFullDashPatterns(): void
{
$testCases = [
[

View File

@@ -57,7 +57,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
}
/**
* Chapitres sans volume entre deux volumes différents → assignés au volume précédent
* Chapitres sans volume entre deux volumes différents → assignés au volume précédent.
*
* Ch1→Vol1, Ch2→null, Ch3→null, Ch4→Vol2
* Après sync : Ch2 et Ch3 doivent avoir Vol1
@@ -83,7 +83,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
}
/**
* Chapitres en début de série sans volume → assignés au premier volume trouvé
* Chapitres en début de série sans volume → assignés au premier volume trouvé.
*
* Ch1→null, Ch2→null, Ch3→Vol1
* Après sync : Ch1 et Ch2 doivent avoir Vol1
@@ -107,7 +107,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
}
/**
* Chapitres avec volumes explicites ne sont pas modifiés
* Chapitres avec volumes explicites ne sont pas modifiés.
*
* Ch1→Vol1, Ch2→Vol1, Ch3→Vol2 → inchangé
*/
@@ -130,7 +130,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
}
/**
* La version française est prioritaire sur l'anglaise
* La version française est prioritaire sur l'anglaise.
*
* Même chapitre disponible EN (volume 1) et FR (volume 2) → FR gagne
*/
@@ -151,7 +151,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
}
/**
* Seuls les nouveaux chapitres sont sauvegardés (pas les doublons)
* Seuls les nouveaux chapitres sont sauvegardés (pas les doublons).
*
* Ch1 déjà en DB + Ch2 nouveau → seul Ch2 est retourné
*/
@@ -188,6 +188,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
/**
* @param Chapter[] $chapters
*
* @return array<float, Chapter>
*/
private function indexedByNumber(array $chapters): array
@@ -196,6 +197,7 @@ class MangadxChapterSynchronizationServiceTest extends TestCase
foreach ($chapters as $chapter) {
$result[$chapter->getNumber()] = $chapter;
}
return $result;
}
}