pp-planer/tests/Feature/TranslationServiceTest.php
Thorsten Bus bdbf0c65e3 refactor(php): rename SongGroup references throughout controllers/services/tests
Replace all SongGroup/SongArrangementGroup model usages with Label/SongArrangementLabel
after the schema migration to global labels. Updates 12 app files and 11 test files:

- SongService: createDefaultGroups now finds-or-creates global Labels by name
- ArrangementController: validates label_id (labels are global, no song-ownership)
- ProImportService: imports groups as Labels (firstOrCreate by name); does not
  overwrite existing label colors per spec
- ProExportService/SongPdfController/TranslationService/etc: traverse via
  arrangements -> arrangementLabels -> label -> songSlides chain
- All test factories and assertions adapted to label-based schema
2026-05-03 22:55:02 +02:00

357 lines
11 KiB
PHP

<?php
use App\Models\Label;
use App\Models\Song;
use App\Models\SongArrangement;
use App\Models\SongArrangementLabel;
use App\Models\SongSlide;
use App\Models\User;
use App\Services\TranslationService;
use Illuminate\Support\Facades\Http;
beforeEach(function () {
$this->user = User::factory()->create();
$this->service = app(TranslationService::class);
});
test('fetchFromUrl returns text from successful HTTP response', function () {
Http::fake([
'https://example.com/lyrics' => Http::response('<html><body><p>Zeile 1</p><p>Zeile 2</p></body></html>', 200),
]);
$result = $this->service->fetchFromUrl('https://example.com/lyrics');
expect($result)->not->toBeNull();
expect($result)->toContain('Zeile 1');
expect($result)->toContain('Zeile 2');
expect($result)->not->toContain('<p>');
expect($result)->not->toContain('<html>');
});
test('fetchFromUrl returns null on HTTP failure', function () {
Http::fake([
'https://example.com/broken' => Http::response('Not Found', 404),
]);
$result = $this->service->fetchFromUrl('https://example.com/broken');
expect($result)->toBeNull();
});
test('fetchFromUrl returns null on connection error', function () {
Http::fake([
'https://timeout.example.com/*' => fn () => throw new \Illuminate\Http\Client\ConnectionException('Timeout'),
]);
$result = $this->service->fetchFromUrl('https://timeout.example.com/lyrics');
expect($result)->toBeNull();
});
test('fetchFromUrl returns null for empty response body', function () {
Http::fake([
'https://example.com/empty' => Http::response('', 200),
]);
$result = $this->service->fetchFromUrl('https://example.com/empty');
expect($result)->toBeNull();
});
function makeSongWithDefaultArrangement(): array
{
$song = Song::factory()->create(['has_translation' => false]);
$arrangement = SongArrangement::factory()->create([
'song_id' => $song->id,
'name' => 'Normal',
'is_default' => true,
]);
return [$song, $arrangement];
}
function attachLabelWithSlides(SongArrangement $arrangement, string $labelName, array $slides, int $arrangementOrder): Label
{
$label = Label::firstOrCreate(['name' => $labelName]);
SongArrangementLabel::factory()->create([
'song_arrangement_id' => $arrangement->id,
'label_id' => $label->id,
'order' => $arrangementOrder,
]);
foreach ($slides as $slide) {
SongSlide::factory()->create(array_merge(
['label_id' => $label->id],
$slide,
));
}
return $label;
}
test('importTranslation distributes lines by slide line counts', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - dist', [], 1);
$slide1 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => "Original 1\nOriginal 2\nOriginal 3\nOriginal 4",
]);
$slide2 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 2,
'text_content' => "Original 5\nOriginal 6",
]);
$slide3 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 3,
'text_content' => "Original 7\nOriginal 8\nOriginal 9\nOriginal 10",
]);
$translatedText = "Zeile 1\nZeile 2\nZeile 3\nZeile 4\nZeile 5\nZeile 6\nZeile 7\nZeile 8\nZeile 9\nZeile 10";
$this->service->importTranslation($song, $translatedText);
$slide1->refresh();
$slide2->refresh();
$slide3->refresh();
expect($slide1->text_content_translated)->toBe("Zeile 1\nZeile 2\nZeile 3\nZeile 4");
expect($slide2->text_content_translated)->toBe("Zeile 5\nZeile 6");
expect($slide3->text_content_translated)->toBe("Zeile 7\nZeile 8\nZeile 9\nZeile 10");
});
test('importTranslation distributes across multiple groups', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$label1 = attachLabelWithSlides($arrangement, 'Strophe 1 - multi', [], 1);
$label2 = attachLabelWithSlides($arrangement, 'Refrain - multi', [], 2);
$slide1 = SongSlide::factory()->create([
'label_id' => $label1->id,
'order' => 1,
'text_content' => "Line A\nLine B",
]);
$slide2 = SongSlide::factory()->create([
'label_id' => $label2->id,
'order' => 1,
'text_content' => "Line C\nLine D\nLine E",
]);
$translatedText = "Über A\nÜber B\nÜber C\nÜber D\nÜber E";
$this->service->importTranslation($song, $translatedText);
$slide1->refresh();
$slide2->refresh();
expect($slide1->text_content_translated)->toBe("Über A\nÜber B");
expect($slide2->text_content_translated)->toBe("Über C\nÜber D\nÜber E");
});
test('importTranslation handles fewer translation lines than original', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - fewer', [], 1);
$slide1 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => "Line 1\nLine 2\nLine 3",
]);
$slide2 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 2,
'text_content' => "Line 4\nLine 5",
]);
$translatedText = "Zeile 1\nZeile 2\nZeile 3";
$this->service->importTranslation($song, $translatedText);
$slide1->refresh();
$slide2->refresh();
expect($slide1->text_content_translated)->toBe("Zeile 1\nZeile 2\nZeile 3");
expect($slide2->text_content_translated)->toBe('');
});
test('importTranslation marks song as translated', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - mark', [], 1);
SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => 'Line 1',
]);
$this->service->importTranslation($song, 'Zeile 1');
$song->refresh();
expect($song->has_translation)->toBeTrue();
});
test('markAsTranslated sets has_translation to true', function () {
$song = Song::factory()->create(['has_translation' => false]);
$this->service->markAsTranslated($song);
$song->refresh();
expect($song->has_translation)->toBeTrue();
});
test('removeTranslation clears all translated text and sets flag to false', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$song->update(['has_translation' => true]);
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - remove', [], 1);
$slide1 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => 'Original',
'text_content_translated' => 'Übersetzt',
]);
$slide2 = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 2,
'text_content' => 'Original 2',
'text_content_translated' => 'Übersetzt 2',
]);
$this->service->removeTranslation($song);
$song->refresh();
$slide1->refresh();
$slide2->refresh();
expect($song->has_translation)->toBeFalse();
expect($slide1->text_content_translated)->toBeNull();
expect($slide2->text_content_translated)->toBeNull();
});
test('POST translation/fetch-url returns scraped text', function () {
Http::fake([
'https://lyrics.example.com/song' => Http::response('<div>Liedtext Zeile 1</div>', 200),
]);
$response = $this->actingAs($this->user)
->postJson('/api/translation/fetch-url', [
'url' => 'https://lyrics.example.com/song',
]);
$response->assertOk()
->assertJsonStructure(['text']);
expect($response->json('text'))->toContain('Liedtext Zeile 1');
});
test('POST translation/fetch-url returns error on failure', function () {
Http::fake([
'https://broken.example.com/*' => Http::response('', 500),
]);
$response = $this->actingAs($this->user)
->postJson('/api/translation/fetch-url', [
'url' => 'https://broken.example.com/song',
]);
$response->assertStatus(422)
->assertJsonFragment(['message' => 'Konnte Text nicht abrufen']);
});
test('POST translation/fetch-url validates url field', function () {
$response = $this->actingAs($this->user)
->postJson('/api/translation/fetch-url', []);
$response->assertUnprocessable()
->assertJsonValidationErrors(['url']);
});
test('POST songs/{song}/translation/import distributes and saves translation', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - controller', [], 1);
$slide = SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => "Line 1\nLine 2",
]);
$response = $this->actingAs($this->user)
->postJson("/api/songs/{$song->id}/translation/import", [
'text' => "Zeile 1\nZeile 2",
]);
$response->assertOk()
->assertJsonFragment(['message' => 'Übersetzung erfolgreich importiert']);
$slide->refresh();
$song->refresh();
expect($slide->text_content_translated)->toBe("Zeile 1\nZeile 2");
expect($song->has_translation)->toBeTrue();
});
test('POST songs/{song}/translation/import validates text field', function () {
$song = Song::factory()->create();
$response = $this->actingAs($this->user)
->postJson("/api/songs/{$song->id}/translation/import", []);
$response->assertUnprocessable()
->assertJsonValidationErrors(['text']);
});
test('POST songs/{song}/translation/import returns 404 for missing song', function () {
$response = $this->actingAs($this->user)
->postJson('/api/songs/99999/translation/import', [
'text' => 'Some text',
]);
$response->assertNotFound();
});
test('DELETE songs/{song}/translation removes translation', function () {
[$song, $arrangement] = makeSongWithDefaultArrangement();
$song->update(['has_translation' => true]);
$label = attachLabelWithSlides($arrangement, 'Strophe 1 - delete', [], 1);
SongSlide::factory()->create([
'label_id' => $label->id,
'order' => 1,
'text_content' => 'Original',
'text_content_translated' => 'Übersetzt',
]);
$response = $this->actingAs($this->user)
->deleteJson("/api/songs/{$song->id}/translation");
$response->assertOk()
->assertJsonFragment(['message' => 'Übersetzung entfernt']);
$song->refresh();
expect($song->has_translation)->toBeFalse();
});
test('translation endpoints require authentication', function () {
$this->postJson('/api/translation/fetch-url', ['url' => 'https://example.com'])
->assertUnauthorized();
$this->postJson('/api/songs/1/translation/import', ['text' => 'test'])
->assertUnauthorized();
$this->deleteJson('/api/songs/1/translation')
->assertUnauthorized();
});