3.0 KiB
3.0 KiB
| name | description |
|---|---|
| swift-protocol-di-testing | 使用基於協定 (Protocol) 的依賴注入 (DI) 來編寫可測試的 Swift 程式碼 — 透過聚焦的協定與 Swift Testing 模擬檔案系統、網路與外部 API。 |
Swift 基於協定的依賴注入 (DI) 與測試實踐
這是透過將外部依賴(檔案系統、網路、iCloud)抽象化為小型、聚焦的協定,使 Swift 程式碼具備高測試性的模式。這能實現無 I/O 的確定性測試。
何時啟用
- 撰寫需存取檔案系統、網路或外部 API 的 Swift 程式碼時。
- 需要測試在不觸發真實失敗的情況下,驗證錯誤處理路徑。
- 建構需跨環境(App、測試、SwiftUI Preview)運行的模組。
- 使用 Swift Concurrency(Actors, Sendable)設計可測試架構。
核心模式
1. 定義小型且聚焦的協定
每個協定僅處理一個外部關注點。
// 檔案存取行為抽象化
public protocol FileAccessorProviding: Sendable {
func read(from url: URL) throws -> Data
func write(_ data: Data, to url: URL) throws
func fileExists(at url: URL) -> Bool
}
2. 實作生產環境版本與測試模擬版本
- 生產版本 (Default):封裝真實的
FileManager或URLSession操作。 - 測試版本 (Mock):內部使用字典或記憶體狀態,模擬檔案操作與注入預期的
Error。
3. 利用預設參數進行依賴注入
在建構子中使用預設參數,讓生產環境保持簡潔,同時允許測試案例傳入 Mock:
public actor SyncManager {
private let fileAccessor: FileAccessorProviding
public init(fileAccessor: FileAccessorProviding = DefaultFileAccessor()) {
self.fileAccessor = fileAccessor
}
public func load() async throws -> Data {
try fileAccessor.read(from: someURL)
}
}
4. 使用 Swift Testing 進行驗證
利用 Swift Testing 的 @Test 巨集與 #expect 語法驗證邏輯與例外:
@Test("驗證資料讀取錯誤處理")
func testReadError() async {
let mock = MockFileAccessor()
mock.readError = CocoaError(.fileReadCorruptFile)
let manager = SyncManager(fileAccessor: mock)
await #expect(throws: Error.self) {
try await manager.load()
}
}
實踐之最佳實踐
- 單一職責原則:協定應儘可能微小,避免開發「上帝協定」。
- Sendable 順應性:由於 Actor 運算,跨邊界傳遞的協定必須標註為
Sendable。 - 僅針對邊界進行 Mock:僅模擬外部資源(檔案、網路),內部的邏輯類型不需過度抽象化。
- 模擬錯誤路徑:這是 DI 最大的價值,確保系統在真實故障發生前已經過充分測試。
應避免的反模式
- 使用
#if DEBUG條件編譯來切換邏輯,這會破壞程式碼的純粹性。 - 過度設計 (Over-engineering):若該類型沒有外部依賴或不具備副作用,不需強行加上協定層。
- 忘記 Actor 呼叫是非同步的,漏掉
await關鍵字。