Le domaine Scraping ne génère plus d'archives CBZ ni ne modifie les
entités du domaine Manga directement. Il scrape, stocke les images
individuellement, et émet un événement partagé.
## Changements principaux
### Domaine Scraping
- Suppression : CbzGeneratorInterface, CbzGenerator, CbzGenerationRequest,
CbzPath, CbzGenerationException
- Suppression : save() de ChapterRepositoryInterface (Scraping)
- Suppression : cbzPath du modèle Chapter (Scraping)
- Ajout : ImageStorageInterface + LocalImageStorage
(stockage dans {MANGA_DATA_PATH}/pages/{chapterId}/)
- ScrapeChapterHandler utilise ImageStorage au lieu du générateur CBZ
### Événement partagé
- ChapterScraped déplacé dans Domain/Shared/Domain/Event/
avec jobId, chapterId, pagesDirectory, pageCount
- Routing Messenger ajouté
### Domaine Manga
- Ajout : ChapterScrapedEventListener + ChapterScrapedMessageHandler
pour mettre à jour Chapter.pagesDirectory via le Repository Manga
### Domaine Reader
- LegacyChapterRepository en dual-mode :
pagesDirectory en priorité, fallback cbzPath (backward compat)
- Requêtes prev/next : filtrent pagesDirectory IS NOT NULL OR cbzPath IS NOT NULL
- ChapterContext expose pagesDirectory
### Architecture
- phparkitect.php : App\Domain\Shared\Domain\Event autorisé dans
les couches Application (correction violations pré-existantes
ChapterImported/VolumeImported + nouvelle ChapterScraped)
## Tests
- 218/218 tests passent (+3 nouveaux)
- InMemoryImageStorage créé pour les tests unitaires
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
41 lines
1011 B
PHP
41 lines
1011 B
PHP
<?php
|
|
|
|
namespace App\Tests\Domain\Scraping\Adapter;
|
|
|
|
use App\Domain\Scraping\Domain\Contract\Repository\ChapterRepositoryInterface;
|
|
use App\Domain\Scraping\Domain\Model\Chapter;
|
|
use App\Domain\Scraping\Domain\Exception\ChapterNotFoundException;
|
|
|
|
class InMemoryChapterRepository implements ChapterRepositoryInterface
|
|
{
|
|
private array $chapters = [];
|
|
|
|
public function getById(string $id): ?Chapter
|
|
{
|
|
return $this->chapters[$id] ?? null;
|
|
}
|
|
|
|
public function getByMangaIdAndChapterNumber(string $mangaId, float $chapterNumber): Chapter
|
|
{
|
|
foreach ($this->chapters as $chapter) {
|
|
if ($chapter->mangaId === $mangaId && $chapter->chapterNumber === $chapterNumber) {
|
|
return $chapter;
|
|
}
|
|
}
|
|
|
|
throw new ChapterNotFoundException();
|
|
}
|
|
|
|
public function save(Chapter $chapter): void
|
|
{
|
|
$this->chapters[$chapter->id] = $chapter;
|
|
}
|
|
|
|
|
|
|
|
public function clear(): void
|
|
{
|
|
$this->chapters = [];
|
|
}
|
|
}
|