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:
parent
5a0888eb28
commit
5ed303612a
@@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -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 = [
|
||||
[
|
||||
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user