Skip to content

Commit

Permalink
Rewrite FixItApplier to be string based
Browse files Browse the repository at this point in the history
  • Loading branch information
kimdv committed Oct 9, 2023
1 parent f8ea4db commit 5c1043f
Show file tree
Hide file tree
Showing 6 changed files with 273 additions and 148 deletions.
40 changes: 26 additions & 14 deletions Sources/SwiftParserDiagnostics/DiagnosticExtensions.swift
Original file line number Diff line number Diff line change
Expand Up @@ -48,28 +48,21 @@ extension FixIt {

extension FixIt.MultiNodeChange {
/// Replaced a present token with a missing node.
///
/// If `transferTrivia` is `true`, the leading and trailing trivia of the
/// removed node will be transferred to the trailing trivia of the previous token.
static func makeMissing(_ token: TokenSyntax, transferTrivia: Bool = true) -> Self {
return makeMissing([token], transferTrivia: transferTrivia)
}

/// Replace present tokens with missing tokens.
/// If `transferTrivia` is `true`, the leading and trailing trivia of the
/// removed node will be transferred to the trailing trivia of the previous token.
///
/// If `transferTrivia` is `true`, the leading trivia of the first token and
/// the trailing trivia of the last token will be transferred to their adjecent
/// tokens.
static func makeMissing(_ tokens: [TokenSyntax], transferTrivia: Bool = true) -> Self {
precondition(!tokens.isEmpty)
precondition(tokens.allSatisfy({ $0.isPresent }))
var changes = tokens.map {
FixIt.Change.replace(
oldNode: Syntax($0),
newNode: Syntax($0.with(\.presence, .missing))
)
}
if transferTrivia {
changes += FixIt.MultiNodeChange.transferTriviaAtSides(from: tokens).primitiveChanges
}
return FixIt.MultiNodeChange(primitiveChanges: changes)
precondition(tokens.allSatisfy(\.isPresent))
return .makeMissing(tokens.map(Syntax.init), transferTrivia: transferTrivia)
}

/// If `transferTrivia` is `true`, the leading and trailing trivia of the
Expand Down Expand Up @@ -104,6 +97,25 @@ extension FixIt.MultiNodeChange {
return FixIt.MultiNodeChange()
}
}

/// Replace present nodes with their missing equivalents.
///
/// If `transferTrivia` is `true`, the leading trivia of the first node and
/// the trailing trivia of the last node will be transferred to their adjecent
/// tokens.
static func makeMissing(_ nodes: [Syntax], transferTrivia: Bool = true) -> Self {
precondition(!nodes.isEmpty)
var changes = nodes.map {
FixIt.Change.replace(
oldNode: $0,
newNode: MissingMaker().rewrite($0, detach: true)
)
}
if transferTrivia {
changes += FixIt.MultiNodeChange.transferTriviaAtSides(from: nodes).primitiveChanges
}
return FixIt.MultiNodeChange(primitiveChanges: changes)
}
}

// MARK: - Make present
Expand Down
26 changes: 12 additions & 14 deletions Sources/SwiftParserDiagnostics/ParseDiagnosticsGenerator.swift
Original file line number Diff line number Diff line change
Expand Up @@ -182,7 +182,7 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
correctToken.isMissing
{
// We are exchanging two adjacent tokens, transfer the trivia from the incorrect token to the corrected token.
changes += misplacedTokens.map { FixIt.MultiNodeChange.makeMissing($0, transferTrivia: false) }
changes.append(FixIt.MultiNodeChange.makeMissing(misplacedTokens, transferTrivia: false))
changes.append(
FixIt.MultiNodeChange.makePresent(
correctToken,
Expand Down Expand Up @@ -236,7 +236,7 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
exchangeTokens(
unexpected: misplacedSpecifiers,
unexpectedTokenCondition: { EffectSpecifier(token: $0) != nil },
correctTokens: [effectSpecifiers?.throwsSpecifier, effectSpecifiers?.asyncSpecifier],
correctTokens: [effectSpecifiers?.asyncSpecifier, effectSpecifiers?.throwsSpecifier],
message: { EffectsSpecifierAfterArrow(effectsSpecifiersAfterArrow: $0) },
moveFixIt: { MoveTokensInFrontOfFixIt(movedTokens: $0, inFrontOf: .arrow) },
removeRedundantFixIt: { RemoveRedundantFixIt(removeTokens: $0) }
Expand Down Expand Up @@ -764,20 +764,17 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
if let unexpected = node.unexpectedBetweenRequirementAndTrailingComma,
let token = unexpected.presentTokens(satisfying: { $0.tokenKind == .binaryOperator("&&") }).first,
let trailingComma = node.trailingComma,
trailingComma.isMissing,
let previous = node.unexpectedBetweenRequirementAndTrailingComma?.previousToken(viewMode: .sourceAccurate)
trailingComma.isMissing
{

addDiagnostic(
unexpected,
.expectedCommaInWhereClause,
fixIts: [
FixIt(
message: ReplaceTokensFixIt(replaceTokens: [token], replacements: [.commaToken()]),
changes: [
.makeMissing(token),
.makePresent(trailingComma),
FixIt.MultiNodeChange(.replaceTrailingTrivia(token: previous, newTrivia: [])),
.makeMissing(token, transferTrivia: false),
.makePresent(trailingComma, leadingTrivia: token.leadingTrivia, trailingTrivia: token.trailingTrivia),
]
)
],
Expand Down Expand Up @@ -818,7 +815,7 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
fixIts: [
FixIt(
message: RemoveNodesFixIt(nodes),
changes: nodes.map { .makeMissing($0) }
changes: .makeMissing(nodes)
)
],
handledNodes: nodes.map { $0.id }
Expand Down Expand Up @@ -1542,7 +1539,7 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
fixIts: [
FixIt(
message: RemoveNodesFixIt(rawDelimiters),
changes: rawDelimiters.map { .makeMissing($0) }
changes: .makeMissing(rawDelimiters)
)
],
handledNodes: rawDelimiters.map { $0.id }
Expand Down Expand Up @@ -1862,8 +1859,8 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
replacements: [node.colon]
),
changes: [
FixIt.MultiNodeChange.makeMissing(equalToken),
FixIt.MultiNodeChange.makePresent(node.colon),
.makeMissing(equalToken, transferTrivia: false),
.makePresent(node.colon, leadingTrivia: equalToken.leadingTrivia, trailingTrivia: equalToken.trailingTrivia),
]
)
],
Expand Down Expand Up @@ -1971,8 +1968,9 @@ public class ParseDiagnosticsGenerator: SyntaxAnyVisitor {
FixIt(
message: fixItMessage,
changes: [
FixIt.MultiNodeChange.makePresent(detail.detail)
] + unexpectedTokens.map { FixIt.MultiNodeChange.makeMissing($0) }
.makePresent(detail.detail),
.makeMissing(unexpectedTokens),
]
)
],
handledNodes: [detail.id] + unexpectedTokens.map(\.id)
Expand Down
118 changes: 118 additions & 0 deletions Sources/_SwiftSyntaxTestSupport/FixItApplier.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2023 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//

import SwiftDiagnostics
import SwiftSyntax

public enum FixItApplier {
fileprivate struct Edit: Equatable {
let startUtf8Offset: Int
let endUtf8Offset: Int
let replacement: String

var replacementLength: Int {
return replacement.utf8.count
}
}

/// Applies selected or all Fix-Its from the provided diagnostics to a given syntax tree.
///
/// - Parameters:
/// - diagnostics: An array of `Diagnostic` objects, each containing one or more Fix-Its.
/// - filterByMessages: An optional array of message strings to filter which Fix-Its to apply.
/// If `nil`, all Fix-Its in `diagnostics` are applied.
/// - tree: The syntax tree to which the Fix-Its will be applied.
///
/// - Returns: A ``String`` representation of the modified syntax tree after applying the Fix-Its.
public static func applyFixes(
from diagnostics: [Diagnostic],
filterByMessages messages: [String]?,
to tree: any SyntaxProtocol
) -> String {
let messages = messages ?? diagnostics.compactMap { $0.fixIts.first?.message.message }

let changes =
diagnostics
.flatMap(\.fixIts)
.filter { messages.contains($0.message.message) }
.flatMap(\.changes)

var edits: [Edit] = []

for change in changes {
switch change {
case .replace(let oldNode, let newNode):
edits.append(
Edit(
startUtf8Offset: oldNode.position.utf8Offset,
endUtf8Offset: oldNode.endPosition.utf8Offset,
replacement: newNode.description
)
)

case .replaceLeadingTrivia(let token, let newTrivia):
edits.append(
Edit(
startUtf8Offset: token.position.utf8Offset,
endUtf8Offset: token.endPosition.utf8Offset,
replacement: token.with(\.leadingTrivia, newTrivia).description
)
)

case .replaceTrailingTrivia(let token, let newTrivia):
edits.append(
Edit(
startUtf8Offset: token.position.utf8Offset,
endUtf8Offset: token.endPosition.utf8Offset,
replacement: token.with(\.trailingTrivia, newTrivia).description
)
)
}
}

var source = tree.description
var editedOffset = 0

// As we need to start apply the edits at the end of a source, start by reversing edit
// and then sort edits by decrementing start offset. If they are equal then descrementing end offset.
// edits = edits.reversed().sorted(by: { edit1, edit2 in
// if edit1.startUtf8Offset == edit2.startUtf8Offset {
// return edit1.endUtf8Offset > edit2.endUtf8Offset
// } else {
// return edit1.startUtf8Offset > edit2.startUtf8Offset
// }
// })

for edit in edits where edits.canInsert(editToApply: edit) {
let startUtf8Offset = edit.startUtf8Offset + editedOffset
let endUtf8Offset = edit.endUtf8Offset + editedOffset

let startIndex = source.utf8.index(source.utf8.startIndex, offsetBy: startUtf8Offset)
let endIndex = source.utf8.index(source.utf8.startIndex, offsetBy: endUtf8Offset)

source.replaceSubrange(startIndex..<endIndex, with: edit.replacement)
editedOffset -= (endUtf8Offset - startUtf8Offset)
editedOffset += edit.replacementLength
}

return source
}
}

extension Array where Element == FixItApplier.Edit {
fileprivate func canInsert(editToApply: Element) -> Bool {
return self.contains { edit in
return !(editToApply.startUtf8Offset >= edit.startUtf8Offset
&& editToApply.endUtf8Offset < edit.endUtf8Offset)
}
}
}
54 changes: 1 addition & 53 deletions Tests/SwiftParserTest/Assertions.swift
Original file line number Diff line number Diff line change
Expand Up @@ -276,58 +276,6 @@ struct DiagnosticSpec {
}
}

class FixItApplier: SyntaxRewriter {
var changes: [FixIt.Change]

init(diagnostics: [Diagnostic], withMessages messages: [String]?) {
let messages = messages ?? diagnostics.compactMap { $0.fixIts.first?.message.message }

self.changes =
diagnostics
.flatMap { $0.fixIts }
.filter {
return messages.contains($0.message.message)
}
.flatMap { $0.changes }

super.init(viewMode: .all)
}

public override func visitAny(_ node: Syntax) -> Syntax? {
for change in changes {
switch change {
case .replace(oldNode: let oldNode, newNode: let newNode) where oldNode.id == node.id:
return newNode
default:
break
}
}
return nil
}

override func visit(_ node: TokenSyntax) -> TokenSyntax {
var modifiedNode = node
for change in changes {
switch change {
case .replaceLeadingTrivia(token: let changedNode, newTrivia: let newTrivia) where changedNode.id == node.id:
modifiedNode = node.with(\.leadingTrivia, newTrivia)
case .replaceTrailingTrivia(token: let changedNode, newTrivia: let newTrivia) where changedNode.id == node.id:
modifiedNode = node.with(\.trailingTrivia, newTrivia)
default:
break
}
}
return modifiedNode
}

/// If `messages` is `nil`, applies all Fix-Its in `diagnostics` to `tree` and returns the fixed syntax tree.
/// If `messages` is not `nil`, applies only Fix-Its whose message is in `messages`.
public static func applyFixes<T: SyntaxProtocol>(in diagnostics: [Diagnostic], withMessages messages: [String]?, to tree: T) -> Syntax {
let applier = FixItApplier(diagnostics: diagnostics, withMessages: messages)
return applier.rewrite(tree)
}
}

/// Assert that `location` is the same as that of `locationMarker` in `tree`.
func assertLocation<T: SyntaxProtocol>(
_ location: SourceLocation,
Expand Down Expand Up @@ -679,7 +627,7 @@ extension ParserTestCase {
if expectedDiagnostics.contains(where: { !$0.fixIts.isEmpty }) && expectedFixedSource == nil {
XCTFail("Expected a fixed source if the test case produces diagnostics with Fix-Its", file: file, line: line)
} else if let expectedFixedSource = expectedFixedSource {
let fixedTree = FixItApplier.applyFixes(in: diags, withMessages: applyFixIts, to: tree)
let fixedTree = FixItApplier.applyFixes(from: diags, filterByMessages: applyFixIts, to: tree)
var fixedTreeDescription = fixedTree.description
if options.contains(.normalizeNewlinesInFixedSource) {
fixedTreeDescription =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -572,7 +572,7 @@ final class AvailabilityQueryUnavailabilityTests: ParserTestCase {
),
],
fixedSource: """
if #unavailable(*) , true {
if #unavailable(*), true {
}
"""
)
Expand Down
Loading

0 comments on commit 5c1043f

Please sign in to comment.