T2: Database Schema + All Migrations - 10 migrations: users extension, services, songs, song_groups, song_slides, song_arrangements, song_arrangement_groups, service_songs, slides, cts_sync_log - 9 Eloquent models with relationships and casts - 9 factory classes for testing - Tests: DatabaseSchemaTest (2 tests, 26 assertions) ✅ T3: ChurchTools OAuth Provider - Custom Socialite provider for ChurchTools OAuth2 - AuthController with redirect/callback/logout - Replaced Breeze login with OAuth-only (German UI) - Removed all Breeze register/password-reset pages - Tests: OAuthTest (9 tests, 54 assertions) ✅ T4: CTS API Service + Sync Command - ChurchToolsService wrapping 5pm-HDH/churchtools-api - SyncChurchToolsCommand (php artisan cts:sync) - SyncController for refresh button - CCLI-based song matching - Tests: ChurchToolsSyncTest (2 tests) ✅ T5: File Conversion Service - FileConversionService with letterbox/pillarbox to 1920×1080 - ConvertPowerPointJob (queued) with LibreOffice + spatie/pdf-to-image - ZIP extraction and recursive processing - Thumbnail generation (320×180) - Tests: FileConversionTest (2 tests, 21 assertions) ✅ T6: Shared Vue Components - AuthenticatedLayout with nav, user info, refresh button - useAutoSave composable (500ms debounce) - FlashMessage, ConfirmDialog, LoadingSpinner components - HandleInertiaRequests middleware with shared props - Tests: SharedPropsTest (7 tests) ✅ T7: Email Configuration - MissingSongRequest mailable (German) - Email template with song info and service link - SONG_REQUEST_EMAIL config - Tests: MissingSongMailTest (2 tests, 10 assertions) ✅ All tests passing: 30/30 (233 assertions) All UI text in German with 'Du' form Wave 1 complete: 7/7 tasks ✅
98 lines
3.3 KiB
PHP
98 lines
3.3 KiB
PHP
<?php
|
|
|
|
namespace App\Cts;
|
|
|
|
use CTApi\CTClient;
|
|
use CTApi\CTConfig;
|
|
use CTApi\Models\Events\Event\EventRequest;
|
|
use CTApi\Models\Events\Song\Song;
|
|
use CTApi\Utils\CTResponseUtil;
|
|
use Throwable;
|
|
|
|
final class CtsApiSpikeSync
|
|
{
|
|
public static function run(
|
|
string $apiUrl,
|
|
string $apiToken,
|
|
string $fromDate,
|
|
int $songId,
|
|
?CTClient $client = null,
|
|
): array {
|
|
if (trim($apiToken) === '') {
|
|
return [
|
|
'auth' => [
|
|
'ok' => false,
|
|
'method' => 'none',
|
|
'blocker' => 'CTS_API_TOKEN fehlt; Authentifizierung nicht moeglich.',
|
|
],
|
|
'events' => ['count' => 0, 'first' => null],
|
|
'song' => ['hasCcli' => false, 'hasLyrics' => false, 'arrangements_count' => 0],
|
|
];
|
|
}
|
|
|
|
CTConfig::clearConfig();
|
|
CTConfig::setApiUrl(rtrim($apiUrl, '/'));
|
|
|
|
$legacyApiKeySetter = 'setApiKey';
|
|
$authMethod = 'raw-http-authorization-header';
|
|
|
|
if (method_exists(CTConfig::class, $legacyApiKeySetter)) {
|
|
CTConfig::{$legacyApiKeySetter}($apiToken);
|
|
$authMethod = 'setApiKey';
|
|
} elseif (method_exists(CTConfig::class, 'authWithLoginToken')) {
|
|
CTConfig::authWithLoginToken($apiToken);
|
|
$authMethod = 'authWithLoginToken';
|
|
}
|
|
|
|
if ($client !== null) {
|
|
CTClient::setClient($client);
|
|
}
|
|
|
|
try {
|
|
$events = EventRequest::where('from', $fromDate)->get();
|
|
$songResponse = CTClient::getClient()->get('/api/songs/' . $songId);
|
|
$songRaw = CTResponseUtil::dataAsArray($songResponse);
|
|
$song = Song::createModelFromData($songRaw);
|
|
} catch (Throwable $throwable) {
|
|
return [
|
|
'auth' => [
|
|
'ok' => false,
|
|
'method' => $authMethod,
|
|
'blocker' => $throwable->getMessage(),
|
|
],
|
|
'events' => ['count' => 0, 'first' => null],
|
|
'song' => ['hasCcli' => false, 'hasLyrics' => false, 'arrangements_count' => 0],
|
|
];
|
|
}
|
|
|
|
$firstEvent = $events[0] ?? null;
|
|
|
|
return [
|
|
'auth' => [
|
|
'ok' => true,
|
|
'method' => $authMethod,
|
|
'blocker' => null,
|
|
],
|
|
'events' => [
|
|
'count' => count($events),
|
|
'first' => $firstEvent === null ? null : [
|
|
'id' => $firstEvent->getId(),
|
|
'title' => $firstEvent->getName(),
|
|
'start_date' => $firstEvent->getStartDate(),
|
|
'note' => $firstEvent->getNote(),
|
|
],
|
|
],
|
|
'song' => [
|
|
'hasCcli' => $song !== null && trim((string) $song->getCcli()) !== '',
|
|
'ccli' => $song?->getCcli(),
|
|
'hasLyrics' => array_key_exists('lyrics', $songRaw),
|
|
'lyrics_type' => is_array($songRaw['lyrics'] ?? null) ? ($songRaw['lyrics']['type'] ?? null) : null,
|
|
'arrangements_count' => $song === null ? 0 : count($song->getArrangements()),
|
|
],
|
|
'raw_shapes' => [
|
|
'song_keys' => array_keys($songRaw),
|
|
],
|
|
];
|
|
}
|
|
}
|