Skip to content

Commit 0e3bdfd

Browse files
authored
Emit a diagnostic if an exit test's body closure includes a capture list. (#1046)
This PR adds a custom diagnostic for `#expect(exitsWith:)` if the passed closure visibly closes over any state (via a capture list). For example: ```swift await #expect(exitsWith: .failure) { [x] in // ... } ``` Produces: > 🛑 Cannot specify a capture clause in closure passed to '#expect(exitsWith:_:)' With a fix-it to remove the capture list. ### Checklist: - [x] Code and documentation should follow the style of the [Style Guide](https://github.com/apple/swift-testing/blob/main/Documentation/StyleGuide.md). - [x] If public symbols are renamed or modified, DocC references should be updated.
1 parent 1f49df9 commit 0e3bdfd

File tree

3 files changed

+74
-1
lines changed

3 files changed

+74
-1
lines changed

Sources/TestingMacros/ConditionMacro.swift

+9-1
Original file line numberDiff line numberDiff line change
@@ -436,7 +436,15 @@ extension ExitTestConditionMacro {
436436
fatalError("Could not find the body argument to this exit test. Please file a bug report at https://github.com/swiftlang/swift-testing/issues/new")
437437
}
438438

439-
let bodyArgumentExpr = arguments[trailingClosureIndex].expression
439+
// Extract the body argument and, if it's a closure with a capture list,
440+
// emit an appropriate diagnostic.
441+
var bodyArgumentExpr = arguments[trailingClosureIndex].expression
442+
bodyArgumentExpr = removeParentheses(from: bodyArgumentExpr) ?? bodyArgumentExpr
443+
if let closureExpr = bodyArgumentExpr.as(ClosureExprSyntax.self),
444+
let captureClause = closureExpr.signature?.capture,
445+
!captureClause.items.isEmpty {
446+
context.diagnose(.captureClauseUnsupported(captureClause, in: closureExpr, inExitTest: macro))
447+
}
440448

441449
// TODO: use UUID() here if we can link to Foundation
442450
let exitTestID = (UInt64.random(in: 0 ... .max), UInt64.random(in: 0 ... .max))

Sources/TestingMacros/Support/DiagnosticMessage.swift

+48
Original file line numberDiff line numberDiff line change
@@ -764,3 +764,51 @@ struct DiagnosticMessage: SwiftDiagnostics.DiagnosticMessage {
764764
var severity: DiagnosticSeverity
765765
var fixIts: [FixIt] = []
766766
}
767+
768+
// MARK: - Captured values
769+
770+
extension DiagnosticMessage {
771+
/// Create a diagnostic message stating that a capture clause cannot be used
772+
/// in an exit test.
773+
///
774+
/// - Parameters:
775+
/// - captureClause: The invalid capture clause.
776+
/// - closure: The closure containing `captureClause`.
777+
/// - exitTestMacro: The containing exit test macro invocation.
778+
///
779+
/// - Returns: A diagnostic message.
780+
static func captureClauseUnsupported(_ captureClause: ClosureCaptureClauseSyntax, in closure: ClosureExprSyntax, inExitTest exitTestMacro: some FreestandingMacroExpansionSyntax) -> Self {
781+
let changes: [FixIt.Change]
782+
if let signature = closure.signature,
783+
Array(signature.with(\.capture, nil).tokens(viewMode: .sourceAccurate)).count == 1 {
784+
// The only remaining token in the signature is `in`, so remove the whole
785+
// signature tree instead of just the capture clause.
786+
changes = [
787+
.replaceTrailingTrivia(token: closure.leftBrace, newTrivia: ""),
788+
.replace(
789+
oldNode: Syntax(signature),
790+
newNode: Syntax("" as ExprSyntax)
791+
)
792+
]
793+
} else {
794+
changes = [
795+
.replace(
796+
oldNode: Syntax(captureClause),
797+
newNode: Syntax("" as ExprSyntax)
798+
)
799+
]
800+
}
801+
802+
return Self(
803+
syntax: Syntax(captureClause),
804+
message: "Cannot specify a capture clause in closure passed to \(_macroName(exitTestMacro))",
805+
severity: .error,
806+
fixIts: [
807+
FixIt(
808+
message: MacroExpansionFixItMessage("Remove '\(captureClause.trimmed)'"),
809+
changes: changes
810+
),
811+
]
812+
)
813+
}
814+
}

Tests/TestingMacrosTests/ConditionMacroTests.swift

+17
Original file line numberDiff line numberDiff line change
@@ -383,6 +383,23 @@ struct ConditionMacroTests {
383383
#expect(diagnostic.message.contains("is redundant"))
384384
}
385385

386+
@Test(
387+
"Capture list on an exit test produces a diagnostic",
388+
arguments: [
389+
"#expectExitTest(exitsWith: x) { [a] in }":
390+
"Cannot specify a capture clause in closure passed to '#expectExitTest(exitsWith:_:)'"
391+
]
392+
)
393+
func exitTestCaptureListProducesDiagnostic(input: String, expectedMessage: String) throws {
394+
let (_, diagnostics) = try parse(input)
395+
396+
#expect(diagnostics.count > 0)
397+
for diagnostic in diagnostics {
398+
#expect(diagnostic.diagMessage.severity == .error)
399+
#expect(diagnostic.message == expectedMessage)
400+
}
401+
}
402+
386403
@Test("Macro expansion is performed within a test function")
387404
func macroExpansionInTestFunction() throws {
388405
let input = ##"""

0 commit comments

Comments
 (0)