Admin Menü neu geordnet.

Introduced a two-tier media pipeline with dynamic disks, asset tracking, admin controls, and alerting around
  upload/archival workflows.
  - Added storage metadata + asset tables and models so every photo/variant knows where it lives
 (database/migrations/2025_10_20_090000_create_media_storage_targets_table.php, database/  migrations/2025_10_20_090200_create_event_media_assets_table.php, app/Models/MediaStorageTarget.php:1, app/
    Models/EventMediaAsset.php:1, app/Models/EventStorageAssignment.php:1, app/Models/Event.php:27).
  - Rewired guest and tenant uploads to pick the event’s hot disk, persist EventMediaAsset records, compute
    checksums, and clean up on delete (app/Http/Controllers/Api/EventPublicController.php:243, app/Http/
Controllers/Api/Tenant/PhotoController.php:25, app/Models/Photo.php:25).
  - Implemented storage services, archival job scaffolding, monitoring config, and queue-failure notifications for upload issues (app/Services/Storage/EventStorageManager.php:16, app/Services/Storage/
    StorageHealthService.php:9, app/Jobs/ArchiveEventMediaAssets.php:16, app/Providers/AppServiceProvider.php:39, app/Notifications/UploadPipelineFailed.php:8, config/storage-monitor.php:1).
  - Seeded default hot/cold targets and exposed super-admin tooling via a Filament resource and capacity widget (database/seeders/MediaStorageTargetSeeder.php:13, database/seeders/DatabaseSeeder.php:17, app/Filament/Resources/MediaStorageTargetResource.php:1, app/Filament/Widgets/StorageCapacityWidget.php:12, app/Providers/Filament/SuperAdminPanelProvider.php:47).
- Dropped cron skeletons and artisan placeholders to schedule storage monitoring, archival dispatch, and upload queue health checks (cron/storage_monitor.sh, cron/archive_dispatcher.sh, cron/upload_queue_health.sh, routes/console.php:9).
This commit is contained in:
Codex Agent
2025-10-17 22:26:13 +02:00
parent 48a2974152
commit 5817270c35
44 changed files with 1336 additions and 72 deletions

View File

@@ -0,0 +1,36 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('media_storage_targets', function (Blueprint $table) {
$table->id();
$table->string('key')->unique();
$table->string('name');
$table->string('driver')->default('local');
$table->json('config')->nullable();
$table->boolean('is_hot')->default(false);
$table->boolean('is_default')->default(false);
$table->boolean('is_active')->default(true);
$table->unsignedInteger('priority')->default(0);
$table->timestamps();
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('media_storage_targets');
}
};

View File

@@ -0,0 +1,37 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('event_storage_assignments', function (Blueprint $table) {
$table->id();
$table->foreignId('event_id')->constrained()->cascadeOnDelete();
$table->foreignId('media_storage_target_id')->constrained()->cascadeOnDelete();
$table->string('role')->default('hot'); // hot, archive
$table->string('status')->default('active'); // active, pending, archived, restoring
$table->timestamp('assigned_at')->nullable();
$table->timestamp('released_at')->nullable();
$table->json('meta')->nullable();
$table->timestamps();
$table->index(['event_id', 'role', 'status']);
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('event_storage_assignments');
}
};

View File

@@ -0,0 +1,46 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('event_media_assets', function (Blueprint $table) {
$table->id();
$table->foreignId('event_id')->constrained()->cascadeOnDelete();
$table->foreignId('media_storage_target_id')->constrained()->cascadeOnDelete();
$table->foreignId('photo_id')->nullable()->constrained('photos')->nullOnDelete();
$table->string('variant')->default('original'); // original, thumbnail, etc.
$table->string('disk');
$table->string('path');
$table->unsignedBigInteger('size_bytes')->nullable();
$table->string('checksum')->nullable();
$table->string('mime_type')->nullable();
$table->string('status')->default('pending'); // pending, hot, archived, restoring, failed
$table->timestamp('processed_at')->nullable();
$table->timestamp('archived_at')->nullable();
$table->timestamp('restored_at')->nullable();
$table->text('error_message')->nullable();
$table->json('meta')->nullable();
$table->timestamps();
$table->index(['event_id', 'variant', 'status']);
$table->index(['media_storage_target_id', 'status']);
});
}
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('event_media_assets');
}
};

View File

@@ -0,0 +1,27 @@
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
return new class extends Migration
{
public function up(): void
{
Schema::table('photos', function (Blueprint $table) {
$table->foreignId('media_asset_id')
->nullable()
->after('file_path')
->constrained('event_media_assets')
->nullOnDelete();
});
}
public function down(): void
{
Schema::table('photos', function (Blueprint $table) {
$table->dropConstrainedForeignId('media_asset_id');
});
}
};