Files
sourcekit-lsp/Tests/BuildSystemIntegrationTests/SwiftPMBuildSystemTests.swift
Alex Hoppen 4c8916ac60 Merge SKSwiftPMWorkspace into BuildSystemIntegration
The SwiftPM build system integration was the only one in its own modules. Merge it into the `BuildSystemIntegration` modules next to eg. the compilation database build system.
2024-07-25 09:13:46 -07:00

892 lines
32 KiB
Swift

//===----------------------------------------------------------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2019 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 Build
@_spi(Testing) import BuildSystemIntegration
import LanguageServerProtocol
import PackageModel
import SKOptions
import SKTestSupport
import SourceKitLSP
import TSCBasic
import ToolchainRegistry
import XCTest
import struct PackageModel.BuildFlags
#if canImport(SPMBuildCore)
@preconcurrency import SPMBuildCore
#endif
fileprivate extension SwiftPMBuildSystem {
func buildSettings(for uri: DocumentURI, language: Language) async throws -> FileBuildSettings? {
guard let target = self.configuredTargets(for: uri).only else {
return nil
}
return try await buildSettings(for: uri, in: target, language: language)
}
}
final class SwiftPMBuildSystemTests: XCTestCase {
func testNoPackage() async throws {
let fs = InMemoryFileSystem()
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": ""
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
await assertThrowsError(
try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
)
}
}
func testUnparsablePackage() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let pack
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
let buildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
await assertThrowsError(try await buildSystem.generateBuildGraph(allowFileSystemWrites: false))
}
}
func testNoToolchain() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
await assertThrowsError(
try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: ToolchainRegistry(toolchains: []),
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
)
}
}
func testBasicSwiftArgs() async throws {
try await SkipUnless.swiftpmStoresModulesInSubdirectory()
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "lib", "a.swift")
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
let build = buildPath(root: packageRoot, platform: hostTriple.platformBuildPathComponent)
assertEqual(await swiftpmBuildSystem.buildPath, build)
assertNotNil(await swiftpmBuildSystem.indexStorePath)
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain("-module-name", "lib", arguments: arguments)
assertArgumentsContain("-emit-dependencies", arguments: arguments)
assertArgumentsContain("-emit-module", arguments: arguments)
assertArgumentsContain("-emit-module-path", arguments: arguments)
assertArgumentsContain("-incremental", arguments: arguments)
assertArgumentsContain("-parse-as-library", arguments: arguments)
assertArgumentsContain("-c", arguments: arguments)
assertArgumentsContain("-target", arguments: arguments) // Only one!
#if os(macOS)
let versionString = PackageModel.Platform.macOS.oldestSupportedVersion.versionString
assertArgumentsContain(
"-target",
hostTriple.tripleString(forPlatformVersion: versionString),
arguments: arguments
)
assertArgumentsContain("-sdk", arguments: arguments)
assertArgumentsContain("-F", arguments: arguments, allowMultiple: true)
#else
assertArgumentsContain("-target", hostTriple.tripleString, arguments: arguments)
#endif
assertArgumentsContain("-I", build.appending(component: "Modules").pathString, arguments: arguments)
assertArgumentsContain(aswift.pathString, arguments: arguments)
}
}
func testCompilerArgumentsForFileThatContainsPlusCharacterURLEncoded() async throws {
try await withTestScratchDir { tempDir in
try localFileSystem.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Sources/lib/a+something.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: localFileSystem,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aPlusSomething = packageRoot.appending(components: "Sources", "lib", "a+something.swift")
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
let build = buildPath(root: packageRoot, platform: hostTriple.platformBuildPathComponent)
assertEqual(await swiftpmBuildSystem.buildPath, build)
assertNotNil(await swiftpmBuildSystem.indexStorePath)
let arguments = try await unwrap(
swiftpmBuildSystem.buildSettings(
for: DocumentURI(URL(string: "file://\(aPlusSomething.asURL.path.replacing("+", with: "%2B"))")!),
language: .swift
)
)
.compilerArguments
// Check that we have both source files in the compiler arguments, which means that we didn't compute the compiler
// arguments for a+something.swift using substitute arguments from a.swift.
XCTAssert(
arguments.contains(aPlusSomething.pathString),
"Compiler arguments do not contain a+something.swift: \(arguments)"
)
XCTAssert(
arguments.contains(packageRoot.appending(components: "Sources", "lib", "a.swift").pathString),
"Compiler arguments do not contain a.swift: \(arguments)"
)
}
}
func testBuildSetup() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
let options = SourceKitLSPOptions.SwiftPMOptions(
configuration: .release,
scratchPath: packageRoot.appending(component: "non_default_build_path").pathString,
cCompilerFlags: ["-m32"],
swiftCompilerFlags: ["-typecheck"]
)
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(swiftPM: options),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "lib", "a.swift")
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
let build = buildPath(root: packageRoot, options: options, platform: hostTriple.platformBuildPathComponent)
assertEqual(await swiftpmBuildSystem.buildPath, build)
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain("-typecheck", arguments: arguments)
assertArgumentsContain("-Xcc", "-m32", arguments: arguments)
assertArgumentsContain("-O", arguments: arguments)
}
}
func testManifestArgs() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let source = try resolveSymlinks(packageRoot.appending(component: "Package.swift"))
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: source.asURI, language: .swift))
.compilerArguments
assertArgumentsContain("-swift-version", "4.2", arguments: arguments)
assertArgumentsContain(source.pathString, arguments: arguments)
}
}
func testMultiFileSwift() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Sources/lib/b.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "lib", "a.swift")
let bswift = packageRoot.appending(components: "Sources", "lib", "b.swift")
let argumentsA = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain(aswift.pathString, arguments: argumentsA)
assertArgumentsContain(bswift.pathString, arguments: argumentsA)
let argumentsB = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain(aswift.pathString, arguments: argumentsB)
assertArgumentsContain(bswift.pathString, arguments: argumentsB)
}
}
func testMultiTargetSwift() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/libA/a.swift": "",
"pkg/Sources/libB/b.swift": "",
"pkg/Sources/libC/include/libC.h": "",
"pkg/Sources/libC/libC.c": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [
.target(name: "libA", dependencies: ["libB", "libC"]),
.target(name: "libB"),
.target(name: "libC"),
]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "libA", "a.swift")
let bswift = packageRoot.appending(components: "Sources", "libB", "b.swift")
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain(aswift.pathString, arguments: arguments)
assertArgumentsDoNotContain(bswift.pathString, arguments: arguments)
// Temporary conditional to work around revlock between SourceKit-LSP and SwiftPM
// as a result of fix for SR-12050. Can be removed when that fix has been merged.
if arguments.joined(separator: " ").contains("-Xcc -I -Xcc") {
assertArgumentsContain(
"-Xcc",
"-I",
"-Xcc",
packageRoot.appending(components: "Sources", "libC", "include").pathString,
arguments: arguments
)
} else {
assertArgumentsContain(
"-I",
packageRoot.appending(components: "Sources", "libC", "include").pathString,
arguments: arguments
)
}
let argumentsB = try await unwrap(swiftpmBuildSystem.buildSettings(for: bswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain(bswift.pathString, arguments: argumentsB)
assertArgumentsDoNotContain(aswift.pathString, arguments: argumentsB)
assertArgumentsDoNotContain(
"-I",
packageRoot.appending(components: "Sources", "libC", "include").pathString,
arguments: argumentsB
)
}
}
func testUnknownFile() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/libA/a.swift": "",
"pkg/Sources/libB/b.swift": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "libA")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "libA", "a.swift")
let bswift = packageRoot.appending(components: "Sources", "libB", "b.swift")
assertNotNil(try await swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
assertNil(try await swiftpmBuildSystem.buildSettings(for: bswift.asURI, language: .swift))
assertNil(
try await swiftpmBuildSystem.buildSettings(
for: DocumentURI(URL(string: "https://www.apple.com")!),
language: .swift
)
)
}
}
func testBasicCXXArgs() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.cpp": "",
"pkg/Sources/lib/b.cpp": "",
"pkg/Sources/lib/include/a.h": "",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")],
cxxLanguageStandard: .cxx14
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let acxx = packageRoot.appending(components: "Sources", "lib", "a.cpp")
let bcxx = packageRoot.appending(components: "Sources", "lib", "b.cpp")
let header = packageRoot.appending(components: "Sources", "lib", "include", "a.h")
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
let build = buildPath(root: packageRoot, platform: hostTriple.platformBuildPathComponent)
assertEqual(await swiftpmBuildSystem.buildPath, build)
assertNotNil(await swiftpmBuildSystem.indexStorePath)
for file in [acxx, header] {
let args = try await unwrap(swiftpmBuildSystem.buildSettings(for: file.asURI, language: .cpp)).compilerArguments
assertArgumentsContain("-std=c++14", arguments: args)
assertArgumentsDoNotContain("-arch", arguments: args)
assertArgumentsContain("-target", arguments: args) // Only one!
#if os(macOS)
let versionString = PackageModel.Platform.macOS.oldestSupportedVersion.versionString
assertArgumentsContain(
"-target",
hostTriple.tripleString(forPlatformVersion: versionString),
arguments: args
)
assertArgumentsContain("-isysroot", arguments: args)
assertArgumentsContain("-F", arguments: args, allowMultiple: true)
#else
assertArgumentsContain("-target", hostTriple.tripleString, arguments: args)
#endif
assertArgumentsContain(
"-I",
packageRoot.appending(components: "Sources", "lib", "include").pathString,
arguments: args
)
assertArgumentsDoNotContain("-I", build.pathString, arguments: args)
assertArgumentsDoNotContain(bcxx.pathString, arguments: args)
URL(fileURLWithPath: build.appending(components: "lib.build", "a.cpp.d").pathString)
.withUnsafeFileSystemRepresentation {
assertArgumentsContain("-MD", "-MT", "dependencies", "-MF", String(cString: $0!), arguments: args)
}
URL(fileURLWithPath: file.pathString).withUnsafeFileSystemRepresentation {
assertArgumentsContain("-c", String(cString: $0!), arguments: args)
}
URL(fileURLWithPath: build.appending(components: "lib.build", "a.cpp.o").pathString)
.withUnsafeFileSystemRepresentation {
assertArgumentsContain("-o", String(cString: $0!), arguments: args)
}
}
}
}
func testDeploymentTargetSwift() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Package.swift": """
// swift-tools-version:5.0
import PackageDescription
let package = Package(name: "a",
platforms: [.macOS(.v10_13)],
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: ToolchainRegistry.forTesting,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "lib", "a.swift")
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain("-target", arguments: arguments) // Only one!
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
#if os(macOS)
assertArgumentsContain(
"-target",
hostTriple.tripleString(forPlatformVersion: "10.13"),
arguments: arguments
)
#else
assertArgumentsContain("-target", hostTriple.tripleString, arguments: arguments)
#endif
}
}
func testSymlinkInWorkspaceSwift() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg_real/Sources/lib/a.swift": "",
"pkg_real/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
try FileManager.default.createSymbolicLink(
at: URL(fileURLWithPath: packageRoot.pathString),
withDestinationURL: URL(fileURLWithPath: tempDir.appending(component: "pkg_real").pathString)
)
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift1 = packageRoot.appending(components: "Sources", "lib", "a.swift")
let aswift2 =
tempDir
.appending(component: "pkg_real")
.appending(components: "Sources", "lib", "a.swift")
let manifest = packageRoot.appending(components: "Package.swift")
let arguments1 = try await swiftpmBuildSystem.buildSettings(for: aswift1.asURI, language: .swift)?
.compilerArguments
let arguments2 = try await swiftpmBuildSystem.buildSettings(for: aswift2.asURI, language: .swift)?
.compilerArguments
XCTAssertNotNil(arguments1)
XCTAssertNotNil(arguments2)
XCTAssertEqual(arguments1, arguments2)
assertArgumentsDoNotContain(aswift1.pathString, arguments: arguments1 ?? [])
assertArgumentsContain(try resolveSymlinks(aswift1).pathString, arguments: arguments1 ?? [])
let argsManifest = try await swiftpmBuildSystem.buildSettings(for: manifest.asURI, language: .swift)?
.compilerArguments
XCTAssertNotNil(argsManifest)
assertArgumentsDoNotContain(manifest.pathString, arguments: argsManifest ?? [])
assertArgumentsContain(try resolveSymlinks(manifest).pathString, arguments: argsManifest ?? [])
}
}
func testSymlinkInWorkspaceCXX() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg_real/Sources/lib/a.cpp": "",
"pkg_real/Sources/lib/b.cpp": "",
"pkg_real/Sources/lib/include/a.h": "",
"pkg_real/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")],
cxxLanguageStandard: .cxx14
)
""",
]
)
let acpp = ["Sources", "lib", "a.cpp"]
let ah = ["Sources", "lib", "include", "a.h"]
let realRoot = tempDir.appending(component: "pkg_real")
let symlinkRoot = tempDir.appending(component: "pkg")
try FileManager.default.createSymbolicLink(
at: URL(fileURLWithPath: symlinkRoot.pathString),
withDestinationURL: URL(fileURLWithPath: tempDir.appending(component: "pkg_real").pathString)
)
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: symlinkRoot,
toolchainRegistry: ToolchainRegistry.forTesting,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
for file in [acpp, ah] {
let args = try unwrap(
await swiftpmBuildSystem.buildSettings(for: symlinkRoot.appending(components: file).asURI, language: .cpp)?
.compilerArguments
)
assertArgumentsContain(realRoot.appending(components: file).pathString, arguments: args)
assertArgumentsDoNotContain(symlinkRoot.appending(components: file).pathString, arguments: args)
}
}
}
func testSwiftDerivedSources() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/a.swift": "",
"pkg/Sources/lib/a.txt": "",
"pkg/Package.swift": """
// swift-tools-version:5.3
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib", resources: [.copy("a.txt")])]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(component: "pkg"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Sources", "lib", "a.swift")
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
assertArgumentsContain(aswift.pathString, arguments: arguments)
XCTAssertNotNil(
arguments.firstIndex(where: {
$0.hasSuffix(".swift") && $0.contains("DerivedSources")
}),
"missing resource_bundle_accessor.swift from \(arguments)"
)
}
}
func testNestedInvalidPackageSwift() async throws {
let fs = InMemoryFileSystem()
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Sources/lib/Package.swift": "// not a valid package",
"pkg/Package.swift": """
// swift-tools-version:4.2
import PackageDescription
let package = Package(
name: "a",
targets: [.target(name: "lib")]
)
""",
]
)
let packageRoot = try resolveSymlinks(tempDir.appending(components: "pkg", "Sources", "lib"))
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
assertEqual(await swiftpmBuildSystem.projectRoot, try resolveSymlinks(tempDir.appending(component: "pkg")))
}
}
func testPluginArgs() async throws {
let fs = localFileSystem
try await withTestScratchDir { tempDir in
try fs.createFiles(
root: tempDir,
files: [
"pkg/Plugins/MyPlugin/a.swift": "",
"pkg/Sources/lib/lib.swift": "",
"pkg/Package.swift": """
// swift-tools-version:5.7
import PackageDescription
let package = Package(
name: "a",
targets: [
.target(name: "lib"),
.plugin(name: "MyPlugin", capability: .buildTool)
]
)
""",
]
)
let packageRoot = tempDir.appending(component: "pkg")
let tr = ToolchainRegistry.forTesting
let swiftpmBuildSystem = try await SwiftPMBuildSystem(
workspacePath: packageRoot,
toolchainRegistry: tr,
fileSystem: fs,
options: SourceKitLSPOptions(),
testHooks: SwiftPMTestHooks()
)
try await swiftpmBuildSystem.generateBuildGraph(allowFileSystemWrites: false)
let aswift = packageRoot.appending(components: "Plugins", "MyPlugin", "a.swift")
let hostTriple = await swiftpmBuildSystem.destinationBuildParameters.triple
let build = buildPath(root: packageRoot, platform: hostTriple.platformBuildPathComponent)
assertEqual(await swiftpmBuildSystem.buildPath, build)
assertNotNil(await swiftpmBuildSystem.indexStorePath)
let arguments = try await unwrap(swiftpmBuildSystem.buildSettings(for: aswift.asURI, language: .swift))
.compilerArguments
// Plugins get compiled with the same compiler arguments as the package manifest
assertArgumentsContain("-package-description-version", "5.7.0", arguments: arguments)
assertArgumentsContain(aswift.pathString, arguments: arguments)
}
}
}
private func assertArgumentsDoNotContain(
_ pattern: String...,
arguments: [String],
file: StaticString = #filePath,
line: UInt = #line
) {
if let index = arguments.firstIndex(of: pattern) {
XCTFail(
"not-pattern \(pattern) unexpectedly found at \(index) in arguments \(arguments)",
file: file,
line: line
)
return
}
}
private func assertArgumentsContain(
_ pattern: String...,
arguments: [String],
allowMultiple: Bool = false,
file: StaticString = #filePath,
line: UInt = #line
) {
guard let index = arguments.firstIndex(of: pattern) else {
XCTFail("pattern \(pattern) not found in arguments \(arguments)", file: file, line: line)
return
}
if !allowMultiple, let index2 = arguments[(index + 1)...].firstIndex(of: pattern) {
XCTFail(
"pattern \(pattern) found twice (\(index), \(index2)) in \(arguments)",
file: file,
line: line
)
}
}
private func buildPath(
root: AbsolutePath,
options: SourceKitLSPOptions.SwiftPMOptions = SourceKitLSPOptions.SwiftPMOptions(),
platform: String
) -> AbsolutePath {
let buildPath = AbsolutePath(validatingOrNil: options.scratchPath) ?? root.appending(component: ".build")
return buildPath.appending(components: platform, "\(options.configuration ?? .debug)")
}