Files
sourcekit-lsp/Tests/SourceKitLSPTests/TypeDefinitionTests.swift
T
femaref c4c552203a provide proper start and end positions in the LSP protocol
vscode provides an alternative action if the target of goto definition is in the same character range we requested.
this requires the start and end in the LSP response to be properly set, so vscode can see that the current character
is actually part of the request
2026-05-19 00:02:17 +02:00

207 lines
5.8 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2026 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
//
//===----------------------------------------------------------------------===//
@_spi(SourceKitLSP) import LanguageServerProtocol
import SKTestSupport
import SwiftExtensions
import XCTest
final class TypeDefinitionTests: SourceKitLSPTestCase {
func testTypeDefinitionLocalType() async throws {
let testClient = try await TestSourceKitLSPClient()
let uri = DocumentURI(for: .swift)
let positions = testClient.openDocument(
"""
struct 1️⃣MyType3️⃣ {}
let 2️⃣x = MyType()
""",
uri: uri
)
let response = try await testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["2️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.first else {
XCTFail("Expected location response")
return
}
XCTAssertEqual(location.uri, uri)
XCTAssertEqual(location.range, positions["1️⃣"]..<positions["3️⃣"])
}
func testTypeDefinitionCrossModule() async throws {
let project = try await SwiftPMTestProject(
files: [
"LibA/MyType.swift": """
public struct 1️⃣MyType {
public init() {}
}
""",
"LibB/UseType.swift": """
import LibA
let 2️⃣x = MyType()
""",
],
manifest: """
let package = Package(
name: "MyLibrary",
targets: [
.target(name: "LibA"),
.target(name: "LibB", dependencies: ["LibA"]),
]
)
""",
enableBackgroundIndexing: true
)
let (uri, positions) = try project.openDocument("UseType.swift")
let response = try await project.testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["2️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.first else {
XCTFail("Expected location response")
return
}
XCTAssertEqual(location.uri, try project.uri(for: "MyType.swift"))
XCTAssertEqual(location.range, try Range(project.position(of: "1️⃣", in: "MyType.swift")))
}
func testTypeDefinitionGenericType() async throws {
let testClient = try await TestSourceKitLSPClient()
let uri = DocumentURI(for: .swift)
let positions = testClient.openDocument(
"""
struct 1️⃣Container3️⃣<T> {
var value: T
}
let 2️⃣x = Container(value: 42)
""",
uri: uri
)
let response = try await testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["2️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.first else {
XCTFail("Expected location response")
return
}
XCTAssertEqual(location.uri, uri)
XCTAssertEqual(location.range, positions["1️⃣"]..<positions["3️⃣"])
}
func testTypeDefinitionOnTypeAnnotation() async throws {
let testClient = try await TestSourceKitLSPClient()
let uri = DocumentURI(for: .swift)
let positions = testClient.openDocument(
"""
struct 1️⃣MyType3️⃣ {}
let x: 2️⃣MyType = MyType()
""",
uri: uri
)
let response = try await testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["2️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.first else {
XCTFail("Expected location response")
return
}
XCTAssertEqual(location.uri, uri)
XCTAssertEqual(location.range, positions["1️⃣"]..<positions["3️⃣"])
}
func testTypeDefinitionFunctionParameter() async throws {
let testClient = try await TestSourceKitLSPClient()
let uri = DocumentURI(for: .swift)
let positions = testClient.openDocument(
"""
struct 1️⃣MyType3️⃣ {}
func process(_ 2️⃣value: MyType) {}
""",
uri: uri
)
let response = try await testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["2️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.first else {
XCTFail("Expected location response")
return
}
XCTAssertEqual(location.uri, uri)
XCTAssertEqual(location.range, positions["1️⃣"]..<positions["3️⃣"])
}
func testTypeDefinitionGeneratedInterface() async throws {
let testClient = try await TestSourceKitLSPClient()
let uri = DocumentURI(for: .swift)
let positions = testClient.openDocument(
"""
let 1️⃣x = "hello"
""",
uri: uri
)
let response = try await testClient.send(
TypeDefinitionRequest(
textDocument: TextDocumentIdentifier(uri),
position: positions["1️⃣"]
)
)
guard case .locations(let locations) = response, let location = locations.only else {
XCTFail("Expected single location response")
return
}
// Should jump to String in the generated Swift interface
XCTAssertTrue(
location.uri.pseudoPath.hasSuffix(".swiftinterface"),
"Expected swiftinterface file, got: \(location.uri.pseudoPath)"
)
assertContains(location.uri.pseudoPath, "String")
}
}