testing-swift
4
总安装量
3
周安装量
#52964
全站排名
安装命令
npx skills add https://github.com/makgunay/claude-swift-skills --skill testing-swift
Agent 安装分布
opencode
3
claude-code
3
codex
3
cursor
3
gemini-cli
2
github-copilot
2
Skill 文档
Swift Testing Framework
Modern testing with @Test, #expect, and @Suite. Replaces XCTest for new projects.
Critical Constraints
- â DO NOT use
XCTAssertEqualin Swift Testing â â Use#expect(a == b) - â DO NOT use
func testSomething()naming convention â â Use@Test func something() - â DO NOT subclass
XCTestCaseâ â Use@Suite structor plain@Testfunctions - â DO NOT use
XCTAssertThrowsErrorâ â Use#expect(throws:) { try something() } - â DO NOT use
setUp/tearDownâ â Useinit/deiniton@Suitestruct or actor
Basic Test
import Testing
@Test("Adding items increases count")
func addItem() {
var list = ShoppingList()
list.add("Milk")
#expect(list.items.count == 1)
#expect(list.items.first == "Milk")
}
@Suite for Organization
@Suite("Shopping List Tests")
struct ShoppingListTests {
var list: ShoppingList
init() {
list = ShoppingList() // Replaces setUp()
}
@Test func addItem() {
list.add("Bread")
#expect(list.items.contains("Bread"))
}
@Test func removeItem() {
list.add("Bread")
list.remove("Bread")
#expect(list.items.isEmpty)
}
}
Parameterized Tests
@Test("Validates email format", arguments: [
("user@example.com", true),
("invalid", false),
("a@b.c", true),
("@missing.com", false),
])
func emailValidation(email: String, isValid: Bool) {
#expect(EmailValidator.isValid(email) == isValid)
}
// With enums
enum Priority: CaseIterable { case low, medium, high }
@Test("All priorities have colors", arguments: Priority.allCases)
func priorityColor(priority: Priority) {
#expect(priority.color != nil)
}
Expectations
// Equality
#expect(result == expected)
// Boolean
#expect(user.isActive)
#expect(!list.isEmpty)
// Optional
#expect(user.name != nil)
let name = try #require(user.name) // Unwrap or fail
// Throws
#expect(throws: ValidationError.self) {
try validator.validate(invalidInput)
}
// Specific error
#expect {
try parser.parse("")
} throws: { error in
guard let parseError = error as? ParseError else { return false }
return parseError.code == .emptyInput
}
// No throw
#expect(throws: Never.self) {
try safeOperation()
}
Testing Async Code
@Test func asyncFetch() async throws {
let service = DataService()
let items = try await service.fetchItems()
#expect(!items.isEmpty)
}
@Test(.timeLimit(.minutes(1)))
func longRunningOperation() async throws {
let result = try await processor.processLargeFile()
#expect(result.isComplete)
}
Testing @Observable Models
import Testing
import Observation
@Test func modelUpdates() {
let model = CounterModel()
#expect(model.count == 0)
model.increment()
#expect(model.count == 1)
model.reset()
#expect(model.count == 0)
}
@Test func asyncModelLoading() async {
let model = ItemListModel()
await model.loadItems()
#expect(!model.items.isEmpty)
#expect(!model.isLoading)
}
Confirmation (for Events/Callbacks)
@Test func notificationFires() async {
await confirmation("Callback received") { confirm in
let observer = EventObserver { event in
#expect(event.type == .update)
confirm()
}
observer.startListening()
EventEmitter.emit(.update)
}
}
// Expected count
await confirmation("Multiple events", expectedCount: 3) { confirm in
for _ in 0..<3 {
EventEmitter.emit(.tick)
confirm()
}
}
Test Traits
@Test(.disabled("Flaky on CI"))
func unreliableTest() { }
@Test(.bug("https://github.com/org/repo/issues/42", "Crashes on empty input"))
func knownBug() { }
@Test(.timeLimit(.seconds(5)))
func mustBeFast() async { }
@Test(.tags(.performance))
func benchmarkSort() { }
extension Tag {
@Tag static var performance: Self
@Tag static var integration: Self
}
Migration from XCTest
| XCTest | Swift Testing |
|---|---|
class FooTests: XCTestCase |
@Suite struct FooTests |
func testBar() |
@Test func bar() |
XCTAssertEqual(a, b) |
#expect(a == b) |
XCTAssertTrue(x) |
#expect(x) |
XCTAssertNil(x) |
#expect(x == nil) |
XCTUnwrap(x) |
try #require(x) |
XCTAssertThrowsError |
#expect(throws:) { } |
setUp() |
init() |
tearDown() |
deinit |
XCTSkip |
try #require(condition) |