isEnabled()) { return 0; // NAS down or disabled — nothing to do } // Videos whose file OR thumbnail/slides are still on local disk $synced = 0; Video::with(['user', 'slides']) ->where('status', 'ready') ->where('path', 'like', 'users/%') ->get() ->each(function (Video $video) use ($nas, &$synced) { $hasLocalVideo = file_exists(storage_path('app/' . $video->path)); // Check for locally-stored thumbnail (NAS push failed during edit) $hasLocalThumb = $video->thumbnail && str_starts_with($video->thumbnail, 'users/') && file_exists(storage_path('app/' . $video->thumbnail)); // Check for locally-stored slides $hasLocalSlide = false; foreach ($video->slides as $slide) { if (str_starts_with($slide->filename, 'users/') && file_exists(storage_path('app/' . $slide->filename))) { $hasLocalSlide = true; break; } } if ($hasLocalVideo || $hasLocalThumb || $hasLocalSlide) { NasSyncVideoJob::dispatch($video); $synced++; } }); // Avatars and banners stuck locally User::whereNotNull('avatar')->orWhereNotNull('banner')->get() ->each(function (User $user) use ($nas) { if ($user->avatar && str_starts_with($user->avatar, 'users/') && file_exists(storage_path('app/' . $user->avatar))) { $nas->syncAvatar($user, storage_path('app/' . $user->avatar)); $nas->deleteLocalAvatar($user); } if ($user->banner && str_starts_with($user->banner, 'users/') && file_exists(storage_path('app/' . $user->banner))) { $nas->syncCover($user, storage_path('app/' . $user->banner)); $nas->deleteLocalBanner($user); } }); if ($synced > 0) { $this->info("Queued {$synced} video(s) for NAS sync."); } return 0; } }