mirror of
https://github.com/apple/sourcekit-lsp.git
synced 2026-03-02 18:23:24 +01:00
Otherwise, we infer the SourceKit plugin paths from the toolchain when creating a `SourceKitLSPServer` during testing because we don’t override the plugin paths in the SourceKitLSPOptions. But when running `SourceKitDTests`, we pass `pluginPaths: nil`, which would not load any plugins. If both of the tests run in the same process, this causes a fault to get logged because sourcekitd can only loaded once per process and we can’t modify which plugins are loaded after the fact.
176 lines
5.9 KiB
Swift
176 lines
5.9 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2022 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#if compiler(>=6)
|
|
@_spi(Testing) import BuildSystemIntegration
|
|
package import Foundation
|
|
package import LanguageServerProtocol
|
|
import SKOptions
|
|
import SourceKitLSP
|
|
import SwiftExtensions
|
|
import TSCBasic
|
|
import ToolchainRegistry
|
|
#else
|
|
@_spi(Testing) import BuildSystemIntegration
|
|
import Foundation
|
|
import LanguageServerProtocol
|
|
import SKOptions
|
|
import SourceKitLSP
|
|
import SwiftExtensions
|
|
import TSCBasic
|
|
import ToolchainRegistry
|
|
#endif
|
|
|
|
package struct IndexedSingleSwiftFileTestProject {
|
|
enum Error: Swift.Error {
|
|
case swiftcNotFound
|
|
}
|
|
|
|
package let testClient: TestSourceKitLSPClient
|
|
package let fileURI: DocumentURI
|
|
package let positions: DocumentPositions
|
|
package let indexDBURL: URL
|
|
|
|
/// Writes a single file to a temporary directory on disk and compiles it to index it.
|
|
///
|
|
/// - Parameters:
|
|
/// - markedText: The contents of the source file including location markers.
|
|
/// - allowBuildFailure: Whether to fail if the input source file fails to build or whether to continue the test
|
|
/// even if the input source is invalid.
|
|
/// - workspaceDirectory: If specified, the temporary files will be put in this directory. If `nil` a temporary
|
|
/// scratch directory will be created based on `testName`.
|
|
/// - cleanUp: Whether to remove the temporary directory when the SourceKit-LSP server shuts down.
|
|
package init(
|
|
_ markedText: String,
|
|
capabilities: ClientCapabilities = ClientCapabilities(),
|
|
indexSystemModules: Bool = false,
|
|
allowBuildFailure: Bool = false,
|
|
workspaceDirectory: URL? = nil,
|
|
cleanUp: Bool = cleanScratchDirectories,
|
|
testName: String = #function
|
|
) async throws {
|
|
let testWorkspaceDirectory = try workspaceDirectory ?? testScratchDir(testName: testName)
|
|
|
|
let testFileURL = testWorkspaceDirectory.appendingPathComponent("test.swift")
|
|
let indexURL = testWorkspaceDirectory.appendingPathComponent("index")
|
|
self.indexDBURL = testWorkspaceDirectory.appendingPathComponent("index-db")
|
|
guard let swiftc = await ToolchainRegistry.forTesting.default?.swiftc else {
|
|
throw Error.swiftcNotFound
|
|
}
|
|
|
|
// Create workspace with source file and compile_commands.json
|
|
|
|
try extractMarkers(markedText).textWithoutMarkers.write(to: testFileURL, atomically: false, encoding: .utf8)
|
|
|
|
var compilerArguments: [String] = [
|
|
try testFileURL.filePath,
|
|
"-index-store-path", try indexURL.filePath,
|
|
"-typecheck",
|
|
]
|
|
if let globalModuleCache = try globalModuleCache {
|
|
compilerArguments += [
|
|
"-module-cache-path", try globalModuleCache.filePath,
|
|
]
|
|
}
|
|
if !indexSystemModules {
|
|
compilerArguments.append("-index-ignore-system-modules")
|
|
}
|
|
|
|
if let sdk = defaultSDKPath {
|
|
compilerArguments += ["-sdk", sdk]
|
|
|
|
// The following are needed so we can import XCTest
|
|
let sdkUrl = URL(fileURLWithPath: sdk)
|
|
#if os(Windows)
|
|
let xctestModuleDir =
|
|
sdkUrl
|
|
.deletingLastPathComponent()
|
|
.deletingLastPathComponent()
|
|
.appendingPathComponent("Library")
|
|
.appendingPathComponent("XCTest-development")
|
|
.appendingPathComponent("usr")
|
|
.appendingPathComponent("lib")
|
|
.appendingPathComponent("swift")
|
|
.appendingPathComponent("windows")
|
|
compilerArguments += ["-I", try xctestModuleDir.filePath]
|
|
#else
|
|
let usrLibDir =
|
|
sdkUrl
|
|
.deletingLastPathComponent()
|
|
.deletingLastPathComponent()
|
|
.appendingPathComponent("usr")
|
|
.appendingPathComponent("lib")
|
|
let frameworksDir =
|
|
sdkUrl
|
|
.deletingLastPathComponent()
|
|
.deletingLastPathComponent()
|
|
.appendingPathComponent("Library")
|
|
.appendingPathComponent("Frameworks")
|
|
compilerArguments += [
|
|
"-I", try usrLibDir.filePath,
|
|
"-F", try frameworksDir.filePath,
|
|
]
|
|
#endif
|
|
}
|
|
|
|
let compilationDatabase = JSONCompilationDatabase(
|
|
[
|
|
CompilationDatabaseCompileCommand(
|
|
directory: try testWorkspaceDirectory.filePath,
|
|
filename: try testFileURL.filePath,
|
|
commandLine: [try swiftc.filePath] + compilerArguments
|
|
)
|
|
]
|
|
)
|
|
let encoder = JSONEncoder()
|
|
encoder.outputFormatting = .prettyPrinted
|
|
try encoder.encode(compilationDatabase).write(
|
|
to: testWorkspaceDirectory.appendingPathComponent(JSONCompilationDatabaseBuildSystem.dbName)
|
|
)
|
|
|
|
// Run swiftc to build the index store
|
|
do {
|
|
try await Process.checkNonZeroExit(arguments: [swiftc.filePath] + compilerArguments)
|
|
} catch {
|
|
if !allowBuildFailure {
|
|
throw error
|
|
}
|
|
}
|
|
|
|
// Create the test client
|
|
var options = try await SourceKitLSPOptions.testDefault()
|
|
options.indexOrDefault = SourceKitLSPOptions.IndexOptions(
|
|
indexStorePath: try indexURL.filePath,
|
|
indexDatabasePath: try indexDBURL.filePath
|
|
)
|
|
self.testClient = try await TestSourceKitLSPClient(
|
|
options: options,
|
|
capabilities: capabilities,
|
|
workspaceFolders: [
|
|
WorkspaceFolder(uri: DocumentURI(testWorkspaceDirectory))
|
|
],
|
|
cleanUp: {
|
|
if cleanUp {
|
|
try? FileManager.default.removeItem(at: testWorkspaceDirectory)
|
|
}
|
|
}
|
|
)
|
|
|
|
// Wait for the indexstore-db to finish indexing
|
|
try await testClient.send(PollIndexRequest())
|
|
|
|
// Open the document
|
|
self.fileURI = DocumentURI(testFileURL)
|
|
self.positions = testClient.openDocument(markedText, uri: fileURI)
|
|
}
|
|
}
|