diff --git a/Contributor Documentation/LSP Extensions.md b/Contributor Documentation/LSP Extensions.md index d7b34bb58..d5a9be50e 100644 --- a/Contributor Documentation/LSP Extensions.md +++ b/Contributor Documentation/LSP Extensions.md @@ -596,19 +596,18 @@ export interface PeekDocumentsResult { } ``` -## `workspace/getReferenceDocument` +## `workspace/textDocumentContent` -Request from the client to the server asking for contents of a URI having a custom scheme. -For example: "sourcekit-lsp:" +Request from the client to the server for querying the contents of a document, potentially using a custom URI scheme (such as `sourcekit-lsp:`). This closely models the proposed LSP 3.18 request of the same name, thus can be removed once LSP 3.18 has been stabilized. -Enable the experimental client capability `"workspace/getReferenceDocument"` so that the server responds with reference document URLs for certain requests or commands whenever possible. +Currently requires enabling the experimental client capability `"workspace/textDocumentContent"` to have the server respond with custom `sourcekit-lsp:` URIs (e.g. in macro expansion requests). -- params: `GetReferenceDocumentParams` +- params: `TextDocumentContentParams` -- result: `GetReferenceDocumentResponse` +- result: `TextDocumentContentResponse` ```ts -export interface GetReferenceDocumentParams { +export interface TextDocumentContentParams { /** * The `DocumentUri` of the custom scheme url for which content is required */ @@ -616,9 +615,9 @@ export interface GetReferenceDocumentParams { } /** - * Response containing `content` of `GetReferenceDocumentRequest` + * Response containing the content of the requested document */ -export interface GetReferenceDocumentResult { - content: string; +export interface TextDocumentContentResult { + text: string; } ``` diff --git a/Sources/LanguageServerProtocol/CMakeLists.txt b/Sources/LanguageServerProtocol/CMakeLists.txt index 3b6483686..9d56c8599 100644 --- a/Sources/LanguageServerProtocol/CMakeLists.txt +++ b/Sources/LanguageServerProtocol/CMakeLists.txt @@ -55,7 +55,6 @@ add_library(LanguageServerProtocol STATIC Requests/ExecuteCommandRequest.swift Requests/FoldingRangeRequest.swift Requests/FormattingRequests.swift - Requests/GetReferenceDocumentRequest.swift Requests/HoverRequest.swift Requests/ImplementationRequest.swift Requests/IndexedRenameRequest.swift @@ -79,6 +78,7 @@ add_library(LanguageServerProtocol STATIC Requests/ShutdownRequest.swift Requests/SignatureHelpRequest.swift Requests/SymbolInfoRequest.swift + Requests/TextDocumentContentRequest.swift Requests/TriggerReindexRequest.swift Requests/TypeDefinitionRequest.swift Requests/TypeHierarchyPrepareRequest.swift diff --git a/Sources/LanguageServerProtocol/Messages.swift b/Sources/LanguageServerProtocol/Messages.swift index e0bd03a27..b32e370fa 100644 --- a/Sources/LanguageServerProtocol/Messages.swift +++ b/Sources/LanguageServerProtocol/Messages.swift @@ -48,7 +48,6 @@ public let builtinRequests: [_RequestType.Type] = [ DocumentTestsRequest.self, ExecuteCommandRequest.self, FoldingRangeRequest.self, - GetReferenceDocumentRequest.self, HoverRequest.self, ImplementationRequest.self, InitializeRequest.self, @@ -71,6 +70,7 @@ public let builtinRequests: [_RequestType.Type] = [ ShutdownRequest.self, SignatureHelpRequest.self, SymbolInfoRequest.self, + TextDocumentContentRequest.self, TriggerReindexRequest.self, TypeDefinitionRequest.self, TypeHierarchyPrepareRequest.self, diff --git a/Sources/LanguageServerProtocol/Requests/GetReferenceDocumentRequest.swift b/Sources/LanguageServerProtocol/Requests/GetReferenceDocumentRequest.swift deleted file mode 100644 index eb3a233f1..000000000 --- a/Sources/LanguageServerProtocol/Requests/GetReferenceDocumentRequest.swift +++ /dev/null @@ -1,44 +0,0 @@ -//===----------------------------------------------------------------------===// -// -// This source file is part of the Swift.org open source project -// -// Copyright (c) 2014 - 2024 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 -// -//===----------------------------------------------------------------------===// - -/// Request from the client to the server asking for contents of a URI having a custom scheme **(LSP Extension)** -/// For example: "sourcekit-lsp:" -/// -/// - Parameters: -/// - uri: The `DocumentUri` of the custom scheme url for which content is required -/// -/// - Returns: `GetReferenceDocumentResponse` which contains the `content` to be displayed. -/// -/// ### LSP Extension -/// -/// This request is an extension to LSP supported by SourceKit-LSP. -/// Enable the experimental client capability `"workspace/getReferenceDocument"` so that the server responds with -/// reference document URLs for certain requests or commands whenever possible. -public struct GetReferenceDocumentRequest: RequestType { - public static let method: String = "workspace/getReferenceDocument" - public typealias Response = GetReferenceDocumentResponse - - public var uri: DocumentURI - - public init(uri: DocumentURI) { - self.uri = uri - } -} - -/// Response containing `content` of `GetReferenceDocumentRequest` -public struct GetReferenceDocumentResponse: ResponseType { - public var content: String - - public init(content: String) { - self.content = content - } -} diff --git a/Sources/LanguageServerProtocol/Requests/TextDocumentContentRequest.swift b/Sources/LanguageServerProtocol/Requests/TextDocumentContentRequest.swift new file mode 100644 index 000000000..180cce2a2 --- /dev/null +++ b/Sources/LanguageServerProtocol/Requests/TextDocumentContentRequest.swift @@ -0,0 +1,42 @@ +//===----------------------------------------------------------------------===// +// +// This source file is part of the Swift.org open source project +// +// Copyright (c) 2014 - 2024 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 +// +//===----------------------------------------------------------------------===// + +/// Request from the client to the server asking for contents of a URI having a custom scheme +/// For example: "sourcekit-lsp:" +/// +/// - Parameters: +/// - uri: The `DocumentUri` of the custom scheme url for which content is required +/// +/// - Returns: `TextDocumentContentResponse` which contains the `content` to be displayed. +public struct TextDocumentContentRequest: RequestType { + public static let method: String = "workspace/textDocumentContent" + public typealias Response = TextDocumentContentResponse + + public var uri: DocumentURI + + public init(uri: DocumentURI) { + self.uri = uri + } +} + +/// Response containing the content of the requested text document. +/// +/// Please note, that the content of any subsequent open notifications for the +/// text document might differ from the returned content due to whitespace and +/// line ending normalizations done on the client. +public struct TextDocumentContentResponse: ResponseType { + public var text: String + + public init(text: String) { + self.text = text + } +} diff --git a/Sources/LanguageServerProtocol/SupportTypes/ClientCapabilities.swift b/Sources/LanguageServerProtocol/SupportTypes/ClientCapabilities.swift index 95c6af3ea..99159bd92 100644 --- a/Sources/LanguageServerProtocol/SupportTypes/ClientCapabilities.swift +++ b/Sources/LanguageServerProtocol/SupportTypes/ClientCapabilities.swift @@ -188,6 +188,8 @@ public struct WorkspaceClientCapabilities: Hashable, Codable, Sendable { public var diagnostics: RefreshRegistrationCapability? = nil + public var textDocumentContent: DynamicRegistrationCapability? = nil + public init( applyEdit: Bool? = nil, workspaceEdit: WorkspaceEdit? = nil, @@ -202,7 +204,8 @@ public struct WorkspaceClientCapabilities: Hashable, Codable, Sendable { fileOperations: FileOperations? = nil, inlineValue: RefreshRegistrationCapability? = nil, inlayHint: RefreshRegistrationCapability? = nil, - diagnostics: RefreshRegistrationCapability? = nil + diagnostics: RefreshRegistrationCapability? = nil, + textDocumentContent: DynamicRegistrationCapability? = nil ) { self.applyEdit = applyEdit self.workspaceEdit = workspaceEdit @@ -218,6 +221,7 @@ public struct WorkspaceClientCapabilities: Hashable, Codable, Sendable { self.inlineValue = inlineValue self.inlayHint = inlayHint self.diagnostics = diagnostics + self.textDocumentContent = textDocumentContent } } diff --git a/Sources/LanguageServerProtocol/SupportTypes/ServerCapabilities.swift b/Sources/LanguageServerProtocol/SupportTypes/ServerCapabilities.swift index 36e4e03d8..cc7112681 100644 --- a/Sources/LanguageServerProtocol/SupportTypes/ServerCapabilities.swift +++ b/Sources/LanguageServerProtocol/SupportTypes/ServerCapabilities.swift @@ -1219,10 +1219,27 @@ public struct WorkspaceServerCapabilities: Codable, Hashable, Sendable { public var willDelete: FileOperationRegistrationOptions? } + /// Text document content provider options. + public struct TextDocumentContentOptions: Codable, Hashable, Sendable { + /// The schemes for which the server provides content. + public var schemes: [String] + + public init(schemes: [String] = []) { + self.schemes = schemes + } + } + /// The server supports workspace folder. public var workspaceFolders: WorkspaceFolders? - public init(workspaceFolders: WorkspaceFolders? = nil) { + /// The server supports the `workspace/textDocumentContent` request. + public var textDocumentContent: TextDocumentContentOptions? + + public init( + workspaceFolders: WorkspaceFolders? = nil, + textDocumentContent: TextDocumentContentOptions? = nil + ) { self.workspaceFolders = workspaceFolders + self.textDocumentContent = textDocumentContent } } diff --git a/Sources/SourceKitLSP/Clang/ClangLanguageService.swift b/Sources/SourceKitLSP/Clang/ClangLanguageService.swift index 48dcd0249..4639744f2 100644 --- a/Sources/SourceKitLSP/Clang/ClangLanguageService.swift +++ b/Sources/SourceKitLSP/Clang/ClangLanguageService.swift @@ -648,7 +648,7 @@ extension ClangLanguageService { return try await forwardRequestToClangd(req) } - func getReferenceDocument(_ req: GetReferenceDocumentRequest) async throws -> GetReferenceDocumentResponse { + func textDocumentContent(_ req: TextDocumentContentRequest) async throws -> TextDocumentContentResponse { throw ResponseError.unknown("unsupported method") } } diff --git a/Sources/SourceKitLSP/LanguageService.swift b/Sources/SourceKitLSP/LanguageService.swift index bc32fce75..47a0e952e 100644 --- a/Sources/SourceKitLSP/LanguageService.swift +++ b/Sources/SourceKitLSP/LanguageService.swift @@ -252,7 +252,7 @@ package protocol LanguageService: AnyObject, Sendable { func executeCommand(_ req: ExecuteCommandRequest) async throws -> LSPAny? - func getReferenceDocument(_ req: GetReferenceDocumentRequest) async throws -> GetReferenceDocumentResponse + func textDocumentContent(_ req: TextDocumentContentRequest) async throws -> TextDocumentContentResponse /// Perform a syntactic scan of the file at the given URI for test cases and test classes. /// diff --git a/Sources/SourceKitLSP/MessageHandlingDependencyTracker.swift b/Sources/SourceKitLSP/MessageHandlingDependencyTracker.swift index 8262403f5..35a7a4dd4 100644 --- a/Sources/SourceKitLSP/MessageHandlingDependencyTracker.swift +++ b/Sources/SourceKitLSP/MessageHandlingDependencyTracker.swift @@ -179,7 +179,7 @@ enum MessageHandlingDependencyTracker: DependencyTracker { } else { self = .freestanding } - case let request as GetReferenceDocumentRequest: + case let request as TextDocumentContentRequest: self = .documentRequest(request.uri) case is InitializeRequest: self = .globalConfigurationChange diff --git a/Sources/SourceKitLSP/SourceKitLSPServer.swift b/Sources/SourceKitLSP/SourceKitLSPServer.swift index 7576802cc..18215f3a6 100644 --- a/Sources/SourceKitLSP/SourceKitLSPServer.swift +++ b/Sources/SourceKitLSP/SourceKitLSPServer.swift @@ -746,8 +746,8 @@ extension SourceKitLSPServer: MessageHandler { await request.reply { try await executeCommand(request.params) } case let request as RequestAndReply: await self.handleRequest(for: request, requestHandler: self.foldingRange) - case let request as RequestAndReply: - await request.reply { try await getReferenceDocument(request.params) } + case let request as RequestAndReply: + await request.reply { try await textDocumentContent(request.params) } case let request as RequestAndReply: await self.handleRequest(for: request, requestHandler: self.hover) case let request as RequestAndReply: @@ -987,7 +987,7 @@ extension SourceKitLSPServer { // The below is a workaround for the vscode-swift extension since it cannot set client capabilities. // It passes "workspace/peekDocuments" through the `initializationOptions`. // - // Similarly, for "workspace/getReferenceDocument". + // Similarly for "workspace/textDocumentContent". var clientCapabilities = req.capabilities if case .dictionary(let initializationOptions) = req.initializationOptions { if let peekDocuments = initializationOptions["workspace/peekDocuments"] { @@ -999,12 +999,12 @@ extension SourceKitLSPServer { } } - if let getReferenceDocument = initializationOptions["workspace/getReferenceDocument"] { + if let textDocumentContent = initializationOptions["workspace/textDocumentContent"] { if case .dictionary(var experimentalCapabilities) = clientCapabilities.experimental { - experimentalCapabilities["workspace/getReferenceDocument"] = getReferenceDocument + experimentalCapabilities["workspace/textDocumentContent"] = textDocumentContent clientCapabilities.experimental = .dictionary(experimentalCapabilities) } else { - clientCapabilities.experimental = .dictionary(["workspace/getReferenceDocument": getReferenceDocument]) + clientCapabilities.experimental = .dictionary(["workspace/textDocumentContent": textDocumentContent]) } } @@ -1166,6 +1166,9 @@ extension SourceKitLSPServer { workspaceFolders: .init( supported: true, changeNotifications: .bool(true) + ), + textDocumentContent: .init( + schemes: [ReferenceDocumentURL.scheme] ) ), callHierarchyProvider: .bool(true), @@ -1176,7 +1179,6 @@ extension SourceKitLSPServer { "workspace/tests": .dictionary(["version": .int(2)]), "textDocument/tests": .dictionary(["version": .int(2)]), "workspace/triggerReindex": .dictionary(["version": .int(1)]), - "workspace/getReferenceDocument": .dictionary(["version": .int(1)]), ]) ) } @@ -1744,7 +1746,7 @@ extension SourceKitLSPServer { return try await languageService.executeCommand(executeCommand) } - func getReferenceDocument(_ req: GetReferenceDocumentRequest) async throws -> GetReferenceDocumentResponse { + func textDocumentContent(_ req: TextDocumentContentRequest) async throws -> TextDocumentContentResponse { let primaryFileURI = try ReferenceDocumentURL(from: req.uri).primaryFile guard let workspace = await workspaceForDocument(uri: primaryFileURI) else { @@ -1755,7 +1757,7 @@ extension SourceKitLSPServer { throw ResponseError.unknown("No Language Service for URI: \(primaryFileURI)") } - return try await languageService.getReferenceDocument(req) + return try await languageService.textDocumentContent(req) } func codeAction( diff --git a/Sources/SourceKitLSP/Swift/MacroExpansion.swift b/Sources/SourceKitLSP/Swift/MacroExpansion.swift index 6ff5c9cac..d38a235ad 100644 --- a/Sources/SourceKitLSP/Swift/MacroExpansion.swift +++ b/Sources/SourceKitLSP/Swift/MacroExpansion.swift @@ -219,7 +219,7 @@ extension SwiftLanguageService { if case .dictionary(let experimentalCapabilities) = self.capabilityRegistry.clientCapabilities.experimental, case .bool(true) = experimentalCapabilities["workspace/peekDocuments"], - case .bool(true) = experimentalCapabilities["workspace/getReferenceDocument"] + case .bool(true) = experimentalCapabilities["workspace/textDocumentContent"] { let expansionURIs = try macroExpansionReferenceDocumentURLs.map { return DocumentURI(try $0.url) diff --git a/Sources/SourceKitLSP/Swift/ReferenceDocumentURL.swift b/Sources/SourceKitLSP/Swift/ReferenceDocumentURL.swift index 04d332322..924262613 100644 --- a/Sources/SourceKitLSP/Swift/ReferenceDocumentURL.swift +++ b/Sources/SourceKitLSP/Swift/ReferenceDocumentURL.swift @@ -14,7 +14,7 @@ import Foundation import LanguageServerProtocol /// A Reference Document is a document whose url scheme is `sourcekit-lsp:` and whose content can only be retrieved -/// using `GetReferenceDocumentRequest`. The enum represents a specific type of reference document and its +/// using `TextDocumentContentRequest`. The enum represents a specific type of reference document and its /// associated value represents the data necessary to generate the document's contents and its url /// /// The `url` will be of the form: `sourcekit-lsp:///?` diff --git a/Sources/SourceKitLSP/Swift/SwiftLanguageService.swift b/Sources/SourceKitLSP/Swift/SwiftLanguageService.swift index 0151fe01e..799903fb0 100644 --- a/Sources/SourceKitLSP/Swift/SwiftLanguageService.swift +++ b/Sources/SourceKitLSP/Swift/SwiftLanguageService.swift @@ -1002,13 +1002,13 @@ extension SwiftLanguageService { return nil } - package func getReferenceDocument(_ req: GetReferenceDocumentRequest) async throws -> GetReferenceDocumentResponse { + package func textDocumentContent(_ req: TextDocumentContentRequest) async throws -> TextDocumentContentResponse { let referenceDocumentURL = try ReferenceDocumentURL(from: req.uri) switch referenceDocumentURL { case let .macroExpansion(data): - return GetReferenceDocumentResponse( - content: try await macroExpansionManager.macroExpansion(for: data) + return TextDocumentContentResponse( + text: try await macroExpansionManager.macroExpansion(for: data) ) } } diff --git a/Tests/SourceKitLSPTests/ExpandMacroTests.swift b/Tests/SourceKitLSPTests/ExpandMacroTests.swift index d379a73f9..d53657d8e 100644 --- a/Tests/SourceKitLSPTests/ExpandMacroTests.swift +++ b/Tests/SourceKitLSPTests/ExpandMacroTests.swift @@ -59,13 +59,12 @@ final class ExpandMacroTests: XCTestCase { """, ] - for (getReferenceDocument, peekDocuments) in cartesianProduct([true], [true]) { + for peekDocuments in [true] { let project = try await SwiftPMTestProject( files: files, manifest: SwiftPMTestProject.macroPackageManifest, capabilities: ClientCapabilities(experimental: [ "workspace/peekDocuments": .bool(peekDocuments), - "workspace/getReferenceDocument": .bool(getReferenceDocument), ]), options: SourceKitLSPOptions.testDefault(), enableBackgroundIndexing: true @@ -93,7 +92,7 @@ final class ExpandMacroTests: XCTestCase { let request = ExecuteCommandRequest(command: command.command, arguments: command.arguments) - if peekDocuments && getReferenceDocument { + if peekDocuments { let expectation = self.expectation(description: "Handle Peek Documents Request") let peekDocumentsRequestURIs = ThreadSafeBox<[DocumentURI]?>(initialValue: nil) @@ -114,9 +113,9 @@ final class ExpandMacroTests: XCTestCase { var filesContents = [String]() for uri in uris { - let result = try await project.testClient.send(GetReferenceDocumentRequest(uri: uri)) + let result = try await project.testClient.send(TextDocumentContentRequest(uri: uri)) - filesContents.append(result.content) + filesContents.append(result.text) } XCTAssertEqual( @@ -234,13 +233,12 @@ final class ExpandMacroTests: XCTestCase { """#, ] - for (getReferenceDocument, peekDocuments) in cartesianProduct([true, false], [true, false]) { + for peekDocuments in [true, false] { let project = try await SwiftPMTestProject( files: files, manifest: SwiftPMTestProject.macroPackageManifest, capabilities: ClientCapabilities(experimental: [ "workspace/peekDocuments": .bool(peekDocuments), - "workspace/getReferenceDocument": .bool(getReferenceDocument), ]), options: SourceKitLSPOptions.testDefault(), enableBackgroundIndexing: true @@ -268,7 +266,7 @@ final class ExpandMacroTests: XCTestCase { let request = ExecuteCommandRequest(command: command.command, arguments: command.arguments) - if peekDocuments && getReferenceDocument { + if peekDocuments { let expectation = self.expectation(description: "Handle Peek Documents Request") let peekDocumentsRequestURIs = ThreadSafeBox<[DocumentURI]?>(initialValue: nil) @@ -290,9 +288,9 @@ final class ExpandMacroTests: XCTestCase { var filesContents = [String]() for uri in uris { - let result = try await project.testClient.send(GetReferenceDocumentRequest(uri: uri)) + let result = try await project.testClient.send(TextDocumentContentRequest(uri: uri)) - filesContents.append(result.content) + filesContents.append(result.text) } XCTAssertEqual( @@ -437,7 +435,6 @@ final class ExpandMacroTests: XCTestCase { manifest: SwiftPMTestProject.macroPackageManifest, capabilities: ClientCapabilities(experimental: [ "workspace/peekDocuments": .bool(true), - "workspace/getReferenceDocument": .bool(true), ]), options: SourceKitLSPOptions.testDefault(), enableBackgroundIndexing: true @@ -473,10 +470,10 @@ final class ExpandMacroTests: XCTestCase { try await fulfillmentOfOrThrow([outerPeekDocumentRequestReceived]) let outerPeekDocumentURI = try XCTUnwrap(outerPeekDocumentsRequestURIs.value?.only) - let outerMacroExpansion = try await project.testClient.send(GetReferenceDocumentRequest(uri: outerPeekDocumentURI)) + let outerMacroExpansion = try await project.testClient.send(TextDocumentContentRequest(uri: outerPeekDocumentURI)) - guard outerMacroExpansion.content == "/* padding */ #intermediate" else { - XCTFail("Received unexpected macro expansion content: \(outerMacroExpansion.content)") + guard outerMacroExpansion.text == "/* padding */ #intermediate" else { + XCTFail("Received unexpected macro expansion content: \(outerMacroExpansion.text)") return } @@ -509,11 +506,11 @@ final class ExpandMacroTests: XCTestCase { let intermediatePeekDocumentURI = try XCTUnwrap(intermediatePeekDocumentsRequestURIs.value?.only) let intermediateMacroExpansion = try await project.testClient.send( - GetReferenceDocumentRequest(uri: intermediatePeekDocumentURI) + TextDocumentContentRequest(uri: intermediatePeekDocumentURI) ) - guard intermediateMacroExpansion.content == "#stringify(1 + 2)" else { - XCTFail("Received unexpected macro expansion content: \(intermediateMacroExpansion.content)") + guard intermediateMacroExpansion.text == "#stringify(1 + 2)" else { + XCTFail("Received unexpected macro expansion content: \(intermediateMacroExpansion.text)") return } @@ -545,8 +542,8 @@ final class ExpandMacroTests: XCTestCase { try await fulfillmentOfOrThrow([innerPeekDocumentRequestReceived]) let innerPeekDocumentURI = try XCTUnwrap(innerPeekDocumentsRequestURIs.value?.only) - let innerMacroExpansion = try await project.testClient.send(GetReferenceDocumentRequest(uri: innerPeekDocumentURI)) + let innerMacroExpansion = try await project.testClient.send(TextDocumentContentRequest(uri: innerPeekDocumentURI)) - XCTAssertEqual(innerMacroExpansion.content, #"(1 + 2, "1 + 2")"#) + XCTAssertEqual(innerMacroExpansion.text, #"(1 + 2, "1 + 2")"#) } }