Files
sourcekit-lsp/Sources/SwiftLanguageService/SwiftCodeLensScanner.swift
Adam Ward ddcddded67 Support swift.play in textDocument/codelens request
- New `swift.play` CodeLens support that is an experimental feature while [swift play](https://github.com/apple/swift-play-experimental/) is still experimental
- Add #Playground macro visitor to parse the macro expansions
- File must `import Playgrounds` to record the macro expansion
- The `swift-play` binary must exist in the toolchain to
- TextDocumentPlayground will record the id and optionally label to match detail you get from
```
$ swift play --list
Building for debugging...
Found 1 Playground
* Fibonacci/Fibonacci.swift:23 "Fibonacci"
```
- Add LSP extension documentation for designing pending `workspace/playground` request
- Add new parsing test cases
- Update CMake files

Issue: #2339 #2343

Add column to unnamed label

Update Sources/SwiftLanguageService/SwiftCodeLensScanner.swift

Co-authored-by: Alex Hoppen <alex@alexhoppen.de>

Update Sources/SwiftLanguageService/SwiftPlaygroundsScanner.swift

Co-authored-by: Alex Hoppen <alex@alexhoppen.de>

Update Sources/SwiftLanguageService/SwiftPlaygroundsScanner.swift

Co-authored-by: Alex Hoppen <alex@alexhoppen.de>

Update Tests/SourceKitLSPTests/CodeLensTests.swift

Co-authored-by: Alex Hoppen <alex@alexhoppen.de>

Address review comments

Fix test failures

Fix more review comments

Update for swift-tools-core
2025-11-07 15:51:17 -05:00

143 lines
4.9 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2020 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
//
//===----------------------------------------------------------------------===//
internal import BuildServerIntegration
import BuildServerProtocol
@_spi(SourceKitLSP) import LanguageServerProtocol
import SourceKitLSP
import SwiftSyntax
import ToolchainRegistry
/// Scans a source file for classes or structs annotated with `@main` and returns a code lens for them.
final class SwiftCodeLensScanner: SyntaxVisitor {
/// The document snapshot of the syntax tree that is being walked.
private let snapshot: DocumentSnapshot
/// The collection of CodeLenses found in the document.
private var result: [CodeLens] = []
private let targetName: String?
/// The map of supported commands and their client side command names
private let supportedCommands: [SupportedCodeLensCommand: String]
private init(
snapshot: DocumentSnapshot,
targetName: String?,
supportedCommands: [SupportedCodeLensCommand: String]
) {
self.snapshot = snapshot
self.targetName = targetName
self.supportedCommands = supportedCommands
super.init(viewMode: .fixedUp)
}
/// Public entry point. Scans the syntax tree of the given snapshot for an `@main` annotation
/// and returns CodeLens's with Commands to run/debug the application.
public static func findCodeLenses(
in snapshot: DocumentSnapshot,
workspace: Workspace?,
syntaxTreeManager: SyntaxTreeManager,
supportedCommands: [SupportedCodeLensCommand: String],
toolchain: Toolchain
) async -> [CodeLens] {
guard !supportedCommands.isEmpty else {
return []
}
var targetDisplayName: String? = nil
if let workspace,
let target = await workspace.buildServerManager.canonicalTarget(for: snapshot.uri),
let buildTarget = await workspace.buildServerManager.buildTarget(named: target)
{
targetDisplayName = buildTarget.displayName
}
var codeLenses: [CodeLens] = []
let syntaxTree = await syntaxTreeManager.syntaxTree(for: snapshot)
if snapshot.text.contains("@main") {
let visitor = SwiftCodeLensScanner(
snapshot: snapshot,
targetName: targetDisplayName,
supportedCommands: supportedCommands
)
visitor.walk(syntaxTree)
codeLenses += visitor.result
}
// "swift.play" CodeLens should be ignored if "swift-play" is not in the toolchain as the client has no way of running
if toolchain.swiftPlay != nil, let workspace, let playCommand = supportedCommands[SupportedCodeLensCommand.play],
snapshot.text.contains("#Playground")
{
let playgrounds = await SwiftPlaygroundsScanner.findDocumentPlaygrounds(
in: syntaxTree,
workspace: workspace,
snapshot: snapshot
)
codeLenses += playgrounds.map({
CodeLens(
range: $0.range,
command: Command(
title: "Play \"\($0.label ?? $0.id)\"",
command: playCommand,
arguments: [$0.encodeToLSPAny()]
)
)
})
}
return codeLenses
}
override func visit(_ node: ClassDeclSyntax) -> SyntaxVisitorContinueKind {
node.attributes.forEach(self.captureLensFromAttribute)
return .skipChildren
}
override func visit(_ node: StructDeclSyntax) -> SyntaxVisitorContinueKind {
node.attributes.forEach(self.captureLensFromAttribute)
return .skipChildren
}
private func captureLensFromAttribute(attribute: AttributeListSyntax.Element) {
if attribute.trimmedDescription == "@main" {
let range = self.snapshot.absolutePositionRange(of: attribute.trimmedRange)
var targetNameToAppend: String = ""
var arguments: [LSPAny] = []
if let targetName {
targetNameToAppend = " \(targetName)"
arguments.append(.string(targetName))
}
if let runCommand = supportedCommands[SupportedCodeLensCommand.run] {
// Return commands for running/debugging the executable.
// These command names must be recognized by the client and so should not be chosen arbitrarily.
self.result.append(
CodeLens(
range: range,
command: Command(title: "Run" + targetNameToAppend, command: runCommand, arguments: arguments)
)
)
}
if let debugCommand = supportedCommands[SupportedCodeLensCommand.debug] {
self.result.append(
CodeLens(
range: range,
command: Command(title: "Debug" + targetNameToAppend, command: debugCommand, arguments: arguments)
)
)
}
}
}
}