90 lines
2.8 KiB
PHP
90 lines
2.8 KiB
PHP
<?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(' …其餘省略');
|
||
}
|
||
}
|