KTVSingle/app/Console/Commands/CheckFtpSongs.php

90 lines
2.8 KiB
PHP
Raw Normal View History

<?php
namespace App\Console\Commands;
use Illuminate\Console\Command;
use Illuminate\Support\Facades\Storage;
use Illuminate\Support\Str;
use App\Models\Song;
class CheckFtpSongs extends Command
{
/**
* The name and signature of the console command.
*
* @var string
*/
protected $signature = 'songs:check-ftp {disk : 如 DISK01或 ALL 依序處理多個 DISK}';
/**
* The console command description.
*
* @var string
*/
protected $description = '比對 songs.filename 與 FTP 檔案,找出缺失 / 多餘檔案';
/**
* Execute the console command.
*/
public function handle()
{
$inputDisk = strtoupper($this->argument('disk'));
$disks = ($inputDisk === 'ALL')
? ['DISK01', 'DISK02', 'DISK03', 'DISK04', 'DISK05', 'DISK09', 'DISK10']
: [$inputDisk];
foreach ($disks as $diskName) {
$this->info("========== 🔍 處理 {$diskName} ==========");
$folder = $diskName;
$prefix = $diskName;
$this->processDisk($folder, $prefix);
$this->newLine();
}
$this->info('🎉 所有比對完成!');
return self::SUCCESS;
}
protected function processDisk(string $folder, string $prefix): void
{
$disk = Storage::disk('ftp_test');
$this->info("📂 FTP 目錄: {$folder}");
$this->info("🎯 DB prefix: {$prefix}");
// 1) 取得 FTP 檔案清單
$this->info('🔍 取得 FTP 檔案列表…');
$ftpPaths = collect($disk->allFiles($folder))
->map(fn ($p) => strtolower(str_replace('\\', '/', ltrim($p, '/'))))
->filter(fn ($p) => Str::startsWith($p, strtolower($prefix)))
->unique()
->values();
$this->info("✅ FTP 檔案數:{$ftpPaths->count()}");
// 2) 資料庫 songs.filename
$this->info('🗃️ 讀取資料庫 songs.filename…');
$dbPaths = Song::query()
->when($prefix, fn ($q) => $q->where('filename', 'like', $prefix . '%'))
->pluck('filename')
->map(fn ($p) => strtolower(str_replace('\\', '/', $p)))
->unique()
->values();
$this->info("✅ DB 記錄數:{$dbPaths->count()}");
// 3) 比對
$missing = $dbPaths->diff($ftpPaths);
$extra = $ftpPaths->diff($dbPaths);
// 4) 結果
$this->warn("❌ 缺失檔案DB 有 / FTP 無):{$missing->count()}");
$missing->each(fn ($p) => $this->line(" - $p"));
$this->info("⚠️ 多餘檔案FTP 有 / DB 無):{$extra->count()}");
$extra->take(20)->each(fn ($p) => $this->line(" - $p"));
if ($extra->count() > 20) $this->line(' …其餘省略');
}
}