檔案儲存
簡介
得益於 Frank de Jonge 所開發的優秀 Flysystem PHP 套件,Laravel 提供了一個強大的檔案系統抽象層。Laravel 與 Flysystem 的整合提供了簡單的驅動程式,讓您能輕鬆操作本地檔案系統、SFTP 以及 Amazon S3。更棒的是,由於每個系統的 API 皆相同,在本地開發環境與正式伺服器之間切換這些儲存選項變得極其簡單。
設定
Laravel 的檔案系統設定檔位於 config/filesystems.php。在此檔案中,您可以設定所有的檔案系統「磁碟 (Disk)」。每個磁碟代表一個特定的儲存驅動程式與儲存位置。設定檔中包含了每個支援的驅動程式範例設定,因此您可以修改這些設定以符合您的儲存偏好與憑證。
local 驅動程式與儲存在運行 Laravel 應用程式伺服器上的本地檔案進行互動,而 sftp 儲存驅動程式則用於透過 SSH 金鑰進行 FTP 傳輸。s3 驅動程式則用於寫入 Amazon 的 S3 雲端儲存服務。
📌 備註
您可以根據需求設定多個磁碟,甚至可以有多個磁碟使用相同的驅動程式。
Local 驅動
當使用 local 驅動程式時,所有檔案操作都是相對於 filesystems 設定檔中定義的 root 目錄。預設情況下,此值設定為 storage/app/private 目錄。因此,以下方法會寫入到 storage/app/private/example.txt:
use Illuminate\Support\Facades\Storage;
Storage::disk('local')->put('example.txt', 'Contents');公開磁碟
應用程式 filesystems 設定檔中包含的 public 磁碟是用於存放要公開存取的檔案。預設情況下,public 磁碟使用 local 驅動程式,並將檔案儲存在 storage/app/public 中。
若您的 public 磁碟使用 local 驅動程式,且您希望讓這些檔案可以從網路存取,您應該建立一個從來源目錄 storage/app/public 到目標目錄 public/storage 的符號連結 (Symbolic Link):
若要建立符號連結,您可以使用 storage:link Artisan 指令:
php artisan storage:link一旦檔案儲存完成且符號連結建立後,您就可以使用 asset 輔助函式來建立檔案的 URL:
echo asset('storage/file.txt');您可以在 filesystems 設定檔中設定額外的符號連結。當您執行 storage:link 指令時,會建立每個已設定的連結:
'links' => [
public_path('storage') => storage_path('app/public'),
public_path('images') => storage_path('app/images'),
],可以使用 storage:unlink 指令來解除已設定的符號連結:
php artisan storage:unlink驅動程式前置需求
S3 驅動設定
在使用 S3 驅動程式之前,您需要透過 Composer 套件管理員安裝 Flysystem S3 套件:
composer require league/flysystem-aws-s3-v3 "^3.0" --with-all-dependenciesS3 磁碟設定陣列位於您的 config/filesystems.php 設定檔中。通常,您應該使用以下由 config/filesystems.php 設定檔所參照的環境變數來設定您的 S3 資訊與憑證:
AWS_ACCESS_KEY_ID=<your-key-id>
AWS_SECRET_ACCESS_KEY=<your-secret-access-key>
AWS_DEFAULT_REGION=us-east-1
AWS_BUCKET=<your-bucket-name>
AWS_USE_PATH_STYLE_ENDPOINT=false為了方便起見,這些環境變數符合 AWS CLI 所使用的命名慣例。
FTP 驅動設定
在使用 FTP 驅動程式之前,您需要透過 Composer 套件管理員安裝 Flysystem FTP 套件:
composer require league/flysystem-ftp "^3.0"Laravel 的 Flysystem 整合在 FTP 上運作良好;然而,框架預設的 config/filesystems.php 設定檔中並未包含範例設定。如果您需要設定 FTP 檔案系統,可以使用下方的設定範例:
'ftp' => [
'driver' => 'ftp',
'host' => env('FTP_HOST'),
'username' => env('FTP_USERNAME'),
'password' => env('FTP_PASSWORD'),
// Optional FTP Settings...
// 'port' => env('FTP_PORT', 21),
// 'root' => env('FTP_ROOT'),
// 'passive' => true,
// 'ssl' => true,
// 'timeout' => 30,
],SFTP 驅動設定
在使用 SFTP 驅動程式之前,您需要透過 Composer 套件管理員安裝 Flysystem SFTP 套件:
composer require league/flysystem-sftp-v3 "^3.0"Laravel 的 Flysystem 整合在 SFTP 上運作良好;然而,框架預設的 config/filesystems.php 設定檔中並未包含範例設定。如果您需要設定 SFTP 檔案系統,可以使用下方的設定範例:
'sftp' => [
'driver' => 'sftp',
'host' => env('SFTP_HOST'),
// Settings for basic authentication...
'username' => env('SFTP_USERNAME'),
'password' => env('SFTP_PASSWORD'),
// Settings for SSH key-based authentication with encryption password...
'privateKey' => env('SFTP_PRIVATE_KEY'),
'passphrase' => env('SFTP_PASSPHRASE'),
// Settings for file / directory permissions...
'visibility' => 'private', // `private` = 0600, `public` = 0644
'directory_visibility' => 'private', // `private` = 0700, `public` = 0755
// Optional SFTP Settings...
// 'hostFingerprint' => env('SFTP_HOST_FINGERPRINT'),
// 'maxTries' => 4,
// 'passphrase' => env('SFTP_PASSPHRASE'),
// 'port' => env('SFTP_PORT', 22),
// 'root' => env('SFTP_ROOT', ''),
// 'timeout' => 30,
// 'useAgent' => true,
],範圍限制與唯讀檔案系統
範圍限制 (Scoped) 磁碟允許您定義一個檔案系統,其中所有路徑都會自動加上指定的路徑前綴。在建立範圍限制檔案系統磁碟之前,您需要透過 Composer 套件管理員安裝額外的 Flysystem 套件:
composer require league/flysystem-path-prefixing "^3.0"您可以藉由定義一個使用 scoped 驅動程式的磁碟,來為任何現有的檔案系統磁碟建立路徑範圍限制的執行個體。例如,您可以建立一個磁碟,將現有的 s3 磁碟限制在特定的路徑前綴下,隨後使用該範圍限制磁碟進行的每個檔案操作都會使用指定的前綴:
's3-videos' => [
'driver' => 'scoped',
'disk' => 's3',
'prefix' => 'path/to/videos',
],「唯讀 (Read-only)」磁碟允許您建立不允許寫入操作的檔案系統磁碟。在使用 read-only 設定選項之前,您需要透過 Composer 套件管理員安裝額外的 Flysystem 套件:
composer require league/flysystem-read-only "^3.0"接著,您可以在一個或多個磁碟設定陣列中加入 read-only 設定選項:
's3-videos' => [
'driver' => 's3',
// ...
'read-only' => true,
],Amazon S3 相容檔案系統
預設情況下,應用程式的 filesystems 設定檔包含一個 s3 磁碟設定。除了使用此磁碟與 Amazon S3 互動外,您還可以用它與任何相容 S3 的檔案儲存服務進行互動,例如 RustFS、DigitalOcean Spaces、Vultr Object Storage、Cloudflare R2 或 Hetzner Cloud Storage。
通常在將磁碟憑證更新為符合您計畫使用的服務憑證後,您只需要更新 endpoint 設定選項的值即可。此選項的值通常透過 AWS_ENDPOINT 環境變數定義:
'endpoint' => env('AWS_ENDPOINT', 'https://rustfs:9000'),取得磁碟執行個體
Storage Facade 可以用來與任何已設定的磁碟進行互動。例如,您可以使用該 Facade 上的 put 方法將頭像儲存到預設磁碟。如果您在呼叫 Storage Facade 上的方法前沒有先呼叫 disk 方法,該方法將自動傳遞給預設磁碟:
use Illuminate\Support\Facades\Storage;
Storage::put('avatars/1', $content);如果您的應用程式與多個磁碟互動,您可以使用 Storage Facade 上的 disk 方法來處理特定磁碟上的檔案:
Storage::disk('s3')->put('avatars/1', $content);按需建立的磁碟
有時您可能希望在執行時使用指定的設定建立磁碟,而該設定實際上並不存在於應用程式的 filesystems 設定檔中。為此,您可以將設定陣列傳遞給 Storage Facade 的 build 方法:
use Illuminate\Support\Facades\Storage;
$disk = Storage::build([
'driver' => 'local',
'root' => '/path/to/root',
]);
$disk->put('image.jpg', $content);擷取檔案
get 方法可用於擷取檔案的內容。該方法將會回傳檔案的原始字串內容。請記住,所有的檔案路徑都應相對於磁碟的「根 (root)」位置來指定:
$contents = Storage::get('file.jpg');如果您要擷取的檔案包含 JSON,可以使用 json 方法來擷取檔案並解碼其內容:
$orders = Storage::json('orders.json');exists 方法可以用於判斷磁碟上是否存在某個檔案:
if (Storage::disk('s3')->exists('file.jpg')) {
// ...
}missing 方法可以用於判斷磁碟上是否缺少某個檔案:
if (Storage::disk('s3')->missing('file.jpg')) {
// ...
}下載檔案
download 方法可用於產生一個回應,強制使用者瀏覽器下載指定路徑的檔案。download 方法接受檔名作為第二個參數,這將決定使用者下載檔案時看到的檔名。最後,您可以將 HTTP 標頭陣列作為第三個參數傳遞給該方法:
return Storage::download('file.jpg');
return Storage::download('file.jpg', $name, $headers);檔案 URL
您可以使用 url 方法來取得給定檔案的 URL。如果您使用的是 local 驅動,這通常只是在指定路徑前加上 /storage 並回傳檔案的相對 URL。如果您使用的是 s3 驅動,則會回傳完整的遠端 URL:
use Illuminate\Support\Facades\Storage;
$url = Storage::url('file.jpg');使用 local 驅動時,所有應公開存取的檔案都應放置在 storage/app/public 目錄中。此外,您應該在 public/storage 建立一個符號連結 (Symbolic Link),指向 storage/app/public 目錄。
⚠️ 警告
使用 local 驅動時,url 的回傳值不會經過 URL 編碼。因此,我們建議在儲存檔案時,始終使用能產生有效 URL 的名稱。
URL 主機自定義
如果您想修改使用 Storage Facade 產生的 URL 的主機 (Host),可以增加或更改磁碟設定陣列中的 url 選項:
'public' => [
'driver' => 'local',
'root' => storage_path('app/public'),
'url' => env('APP_URL').'/storage',
'visibility' => 'public',
'throw' => false,
],暫時性 URL
使用 temporaryUrl 方法,您可以為儲存在 local 與 s3 驅動中的檔案建立暫時性 URL。此方法接受一個路徑與一個 DateTime 執行個體,用以指定 URL 何時過期:
use Illuminate\Support\Facades\Storage;
$url = Storage::temporaryUrl(
'file.jpg', now()->plus(minutes: 5)
);啟用 Local 暫時性 URL
如果您在 local 驅動支援暫時性 URL 之前就開始開發應用程式,您可能需要啟用 local 暫時性 URL。為此,請在 config/filesystems.php 設定檔中的 local 磁碟設定陣列裡加入 serve 選項:
'local' => [
'driver' => 'local',
'root' => storage_path('app/private'),
'serve' => true,
'throw' => false,
],S3 請求參數
如果您需要指定額外的 S3 請求參數,可以將參數陣列作為第三個參數傳遞給 temporaryUrl 方法:
$url = Storage::temporaryUrl(
'file.jpg',
now()->plus(minutes: 5),
[
'ResponseContentType' => 'application/octet-stream',
'ResponseContentDisposition' => 'attachment; filename=file2.jpg',
]
);自定義暫時性 URL
如果您需要自定義特定儲存磁碟產生暫時性 URL 的方式,可以使用 buildTemporaryUrlsUsing 方法。例如,當您有一個 Controller 允許下載存放在通常不支援暫時性 URL 的磁碟中的檔案時,這會非常有用。通常,此方法應從服務提供者的 boot 方法中呼叫:
<?php
namespace App\Providers;
use DateTime;
use Illuminate\Support\Facades\Storage;
use Illuminate\Support\Facades\URL;
use Illuminate\Support\ServiceProvider;
class AppServiceProvider extends ServiceProvider
{
/**
* Bootstrap any application services.
*/
public function boot(): void
{
Storage::disk('local')->buildTemporaryUrlsUsing(
function (string $path, DateTime $expiration, array $options) {
return URL::temporarySignedRoute(
'files.download',
$expiration,
array_merge($options, ['path' => $path])
);
}
);
}
}暫時性上傳 URL
⚠️ 警告
產生暫時性上傳 URL 的功能僅由 s3 與 local 驅動支援。
如果您需要產生一個可用於直接從用戶端應用程式上傳檔案的暫時性 URL,可以使用 temporaryUploadUrl 方法。此方法接受一個路徑與一個 DateTime 執行個體,用以指定 URL 何時過期。temporaryUploadUrl 方法會回傳一個關聯陣列,該陣列可以解構為上傳 URL 以及應隨上傳請求包含的標頭:
use Illuminate\Support\Facades\Storage;
['url' => $url, 'headers' => $headers] = Storage::temporaryUploadUrl(
'file.jpg', now()->plus(minutes: 5)
);此方法主要用於無伺服器 (Serverless) 環境,這些環境要求用戶端應用程式直接將檔案上傳到雲端儲存系統(如 Amazon S3)。
檔案中繼資料
除了讀取與寫入檔案外,Laravel 還可以提供檔案本身的資訊。例如,size 方法可以用於取得檔案的大小(以位元組為單位):
use Illuminate\Support\Facades\Storage;
$size = Storage::size('file.jpg');lastModified 方法會回傳檔案上次修改時的 UNIX 時間戳記:
$time = Storage::lastModified('file.jpg');給定檔案的 MIME 類型可以透過 mimeType 方法取得:
$mime = Storage::mimeType('file.jpg');檔案路徑
您可以使用 path 方法來取得給定檔案的路徑。如果您使用的是 local 驅動,這將回傳檔案的絕對路徑。如果您使用的是 s3 驅動,此方法將回傳 S3 儲存貯體中檔案的相對路徑:
use Illuminate\Support\Facades\Storage;
$path = Storage::path('file.jpg');儲存檔案
put 方法可用於將檔案內容儲存到磁碟上。您也可以將 PHP resource 傳遞給 put 方法,這將使用 Flysystem 底層的串流支援。請記住,所有檔案路徑都應相對於為磁碟設定的「根 (root)」位置:
use Illuminate\Support\Facades\Storage;
Storage::put('file.jpg', $contents);
Storage::put('file.jpg', $resource);失敗的寫入
如果 put 方法(或其他「寫入」操作)無法將檔案寫入磁碟,則會回傳 false:
if (! Storage::put('file.jpg', $contents)) {
// The file could not be written to disk...
}如果您願意,可以在檔案系統磁碟的設定陣列中定義 throw 選項。當此選項定義為 true 時,當寫入操作失敗時,如 put 之類的「寫入」方法將拋出 League\Flysystem\UnableToWriteFile 的執行個體:
'public' => [
'driver' => 'local',
// ...
'throw' => true,
],在檔案開頭與結尾附加內容
prepend 與 append 方法允許您在檔案的開頭或結尾寫入內容:
Storage::prepend('file.log', 'Prepended Text');
Storage::append('file.log', 'Appended Text');複製與移動檔案
copy 方法可用於將現有檔案複製到磁碟上的新位置,而 move 方法可用於重新命名現有檔案或將其移動到新位置:
Storage::copy('old/file.jpg', 'new/file.jpg');
Storage::move('old/file.jpg', 'new/file.jpg');自動串流
將檔案串流到儲存空間可以顯著減少記憶體使用量。如果您希望 Laravel 自動管理將指定檔案串流到您的儲存位置,您可以使用 putFile 或 putFileAs 方法。此方法接受 Illuminate\Http\File 或 Illuminate\Http\UploadedFile 執行個體,並會自動將檔案串流到您所需的位置:
use Illuminate\Http\File;
use Illuminate\Support\Facades\Storage;
// Automatically generate a unique ID for filename...
$path = Storage::putFile('photos', new File('/path/to/photo'));
// Manually specify a filename...
$path = Storage::putFileAs('photos', new File('/path/to/photo'), 'photo.jpg');關於 putFile 方法,有幾點重要事項需要注意。請注意,我們只指定了目錄名稱而不是檔案名稱。預設情況下,putFile 方法會產生一個唯一的 ID 作為檔案名稱。檔案的副檔名將透過檢查檔案的 MIME 類型來確定。putFile 方法會回傳檔案的路徑,包含產生的檔案名稱,以便您可以將路徑儲存在資料庫中。
putFile 與 putFileAs 方法還接受一個參數來指定儲存檔案的「可見性 (visibility)」。如果您將檔案儲存在 Amazon S3 等雲端磁碟上,並希望該檔案可以透過產生的 URL 公開存取,這將特別有用:
Storage::putFile('photos', new File('/path/to/photo'), 'public');檔案上傳
在網頁應用程式中,儲存檔案最常見的使用案例之一是儲存使用者上傳的檔案,例如照片和文件。Laravel 使用上傳檔案執行個體上的 store 方法,可以非常輕鬆地儲存上傳的檔案。請使用您希望儲存上傳檔案的路徑來呼叫 store 方法:
<?php
namespace App\Http\Controllers;
use Illuminate\Http\Request;
class UserAvatarController extends Controller
{
/**
* Update the avatar for the user.
*/
public function update(Request $request): string
{
$path = $request->file('avatar')->store('avatars');
return $path;
}
}關於這個範例,有幾點重要事項需要注意。請注意,我們只指定了目錄名稱,而不是檔案名稱。預設情況下,store 方法會產生一個唯一的 ID 作為檔案名稱。檔案的副檔名將透過檢查檔案的 MIME 類型來確定。store 方法會回傳檔案的路徑,包含產生的檔案名稱,以便您可以將路徑儲存在資料庫中。
您也可以呼叫 Storage Facade 上的 putFile 方法,執行與上述範例相同的檔案儲存操作:
$path = Storage::putFile('avatars', $request->file('avatar'));指定檔案名稱
如果您不希望自動為儲存的檔案分配檔案名稱,您可以使用 storeAs 方法,該方法接收路徑、檔案名稱和(選填的)磁碟作為其參數:
$path = $request->file('avatar')->storeAs(
'avatars', $request->user()->id
);您也可以使用 Storage Facade 上的 putFileAs 方法,這將執行與上述範例相同的檔案儲存操作:
$path = Storage::putFileAs(
'avatars', $request->file('avatar'), $request->user()->id
);⚠️ 警告
不可列印且無效的 Unicode 字元將自動從檔案路徑中移除。因此,在將檔案路徑傳遞給 Laravel 的檔案儲存方法之前,您可能希望先淨化您的檔案路徑。檔案路徑使用 League\Flysystem\WhitespacePathNormalizer::normalizePath 方法進行標準化。
指定磁碟
預設情況下,此上傳檔案的 store 方法將使用您的預設磁碟。如果您想指定另一個磁碟,請將磁碟名稱作為第二個參數傳遞給 store 方法:
$path = $request->file('avatar')->store(
'avatars/'.$request->user()->id, 's3'
);如果您使用的是 storeAs 方法,您可以將磁碟名稱作為第三個參數傳遞給該方法:
$path = $request->file('avatar')->storeAs(
'avatars',
$request->user()->id,
's3'
);其他上傳檔案資訊
如果您想取得上傳檔案的原始名稱和副檔名,可以使用 getClientOriginalName 與 getClientOriginalExtension 方法:
$file = $request->file('avatar');
$name = $file->getClientOriginalName();
$extension = $file->getClientOriginalExtension();然而,請記住 getClientOriginalName 與 getClientOriginalExtension 方法被視為是不安全的,因為檔案名稱與副檔名可能會被惡意使用者竄改。出於這個原因,您通常應該優先使用 hashName 與 extension 方法來取得指定上傳檔案的名稱與副檔名:
$file = $request->file('avatar');
$name = $file->hashName(); // Generate a unique, random name...
$extension = $file->extension(); // Determine the file's extension based on the file's MIME type...檔案可見性
在 Laravel 的 Flysystem 整合中,「可見性 (visibility)」是跨多個平台檔案權限的抽象化。檔案可以被宣告為 public 或 private。當檔案被宣告為 public 時,表示您希望該檔案通常可以被其他人存取。例如,使用 S3 驅動時,您可以取得 public 檔案的 URL。
您可以在透過 put 方法寫入檔案時設定可見性:
use Illuminate\Support\Facades\Storage;
Storage::put('file.jpg', $contents, 'public');如果檔案已經儲存,則可以透過 getVisibility 和 setVisibility 方法來取得或設定其可見性:
$visibility = Storage::getVisibility('file.jpg');
Storage::setVisibility('file.jpg', 'public');當與上傳的檔案互動時,您可以使用 storePublicly 和 storePubliclyAs 方法來以 public 可見性儲存上傳的檔案:
$path = $request->file('avatar')->storePublicly('avatars', 's3');
$path = $request->file('avatar')->storePubliclyAs(
'avatars',
$request->user()->id,
's3'
);Local 檔案與可見性
使用 local 驅動時,public 可見性會轉換為目錄的 0755 權限和檔案的 0644 權限。您可以在應用程式的 filesystems 設定檔中修改權限映射:
'local' => [
'driver' => 'local',
'root' => storage_path('app'),
'permissions' => [
'file' => [
'public' => 0644,
'private' => 0600,
],
'dir' => [
'public' => 0755,
'private' => 0700,
],
],
'throw' => false,
],刪除檔案
delete 方法接受單個檔名或一個檔名陣列來進行刪除:
use Illuminate\Support\Facades\Storage;
Storage::delete('file.jpg');
Storage::delete(['file.jpg', 'file2.jpg']);如有需要,您可以指定要從中刪除檔案的磁碟:
use Illuminate\Support\Facades\Storage;
Storage::disk('s3')->delete('path/file.jpg');目錄
取得目錄內的所有檔案
files 方法會回傳指定目錄中所有檔案的陣列。如果您想取得指定目錄(包含子目錄)中所有檔案的清單,可以使用 allFiles 方法:
use Illuminate\Support\Facades\Storage;
$files = Storage::files($directory);
$files = Storage::allFiles($directory);取得目錄內的所有目錄
directories 方法會回傳指定目錄中所有目錄的陣列。如果您想取得指定目錄(包含子目錄)中所有目錄的清單,可以使用 allDirectories 方法:
$directories = Storage::directories($directory);
$directories = Storage::allDirectories($directory);建立目錄
makeDirectory 方法將建立指定的目錄,包含任何必要的子目錄:
Storage::makeDirectory($directory);刪除目錄
最後,deleteDirectory 方法可用於移除目錄及其所有檔案:
Storage::deleteDirectory($directory);測試
Storage Facade 的 fake 方法讓您可以輕鬆地生成一個虛擬磁碟,結合 Illuminate\Http\UploadedFile 類別的檔案生成工具,極大地簡化了檔案上傳的測試。例如:
<?php
use Illuminate\Http\UploadedFile;
use Illuminate\Support\Facades\Storage;
test('albums can be uploaded', function () {
Storage::fake('photos');
$response = $this->json('POST', '/photos', [
UploadedFile::fake()->image('photo1.jpg'),
UploadedFile::fake()->image('photo2.jpg')
]);
// Assert one or more files were stored...
Storage::disk('photos')->assertExists('photo1.jpg');
Storage::disk('photos')->assertExists(['photo1.jpg', 'photo2.jpg']);
// Assert one or more files were not stored...
Storage::disk('photos')->assertMissing('missing.jpg');
Storage::disk('photos')->assertMissing(['missing.jpg', 'non-existing.jpg']);
// Assert that the number of files in a given directory matches the expected count...
Storage::disk('photos')->assertCount('/wallpapers', 2);
// Assert that a given directory is empty...
Storage::disk('photos')->assertDirectoryEmpty('/wallpapers');
});<?php
namespace Tests\Feature;
use Illuminate\Http\UploadedFile;
use Illuminate\Support\Facades\Storage;
use Tests\TestCase;
class ExampleTest extends TestCase
{
public function test_albums_can_be_uploaded(): void
{
Storage::fake('photos');
$response = $this->json('POST', '/photos', [
UploadedFile::fake()->image('photo1.jpg'),
UploadedFile::fake()->image('photo2.jpg')
]);
// Assert one or more files were stored...
Storage::disk('photos')->assertExists('photo1.jpg');
Storage::disk('photos')->assertExists(['photo1.jpg', 'photo2.jpg']);
// Assert one or more files were not stored...
Storage::disk('photos')->assertMissing('missing.jpg');
Storage::disk('photos')->assertMissing(['missing.jpg', 'non-existing.jpg']);
// Assert that the number of files in a given directory matches the expected count...
Storage::disk('photos')->assertCount('/wallpapers', 2);
// Assert that a given directory is empty...
Storage::disk('photos')->assertDirectoryEmpty('/wallpapers');
}
}預設情況下,fake 方法會刪除其暫存目錄中的所有檔案。如果您想保留這些檔案,可以改用 "persistentFake" 方法。有關測試檔案上傳的更多資訊,您可以查閱 HTTP 測試文件的檔案上傳相關資訊。
⚠️ 警告
image 方法需要 GD 擴充套件。
自定義檔案系統
Laravel 的 Flysystem 整合提供了對多種「驅動程式」的開箱即用支援;然而,Flysystem 並不侷限於這些,它還有許多其他儲存系統的轉接器 (Adapter)。如果您想在 Laravel 應用程式中使用這些額外的轉接器,可以建立自定義驅動程式。
為了定義自定義檔案系統,您需要一個 Flysystem 轉接器。讓我們將社群維護的 Dropbox 轉接器新增到專案中:
composer require spatie/flysystem-dropbox接下來,您可以在應用程式的其中一個 服務提供者 (Service Providers) 的 boot 方法中註冊該驅動程式。為此,您應該使用 Storage Facade 的 extend 方法:
<?php
namespace App\Providers;
use Illuminate\Contracts\Foundation\Application;
use Illuminate\Filesystem\FilesystemAdapter;
use Illuminate\Support\Facades\Storage;
use Illuminate\Support\ServiceProvider;
use League\Flysystem\Filesystem;
use Spatie\Dropbox\Client as DropboxClient;
use Spatie\FlysystemDropbox\DropboxAdapter;
class AppServiceProvider extends ServiceProvider
{
/**
* Register any application services.
*/
public function register(): void
{
// ...
}
/**
* Bootstrap any application services.
*/
public function boot(): void
{
Storage::extend('dropbox', function (Application $app, array $config) {
$adapter = new DropboxAdapter(new DropboxClient(
$config['authorization_token']
));
return new FilesystemAdapter(
new Filesystem($adapter, $config),
$adapter,
$config
);
});
}
}extend 方法的第一個參數是驅動程式的名稱,第二個參數是接收 $app 和 $config 變數的閉包 (Closure)。該閉包必須回傳 Illuminate\Filesystem\FilesystemAdapter 的執行個體。$config 變數包含在 config/filesystems.php 中為指定磁碟定義的值。
建立並註冊擴充功能的服務提供者後,您就可以在 config/filesystems.php 設定檔中使用 dropbox 驅動程式了。