Skip to content

Commit 5b7bbac

Browse files
committed
Run build commands without declared outputs
It is possible to declare build commands without outputs and the expectation is that those would still run. Currently, that is not the case, since the only condition that makes llbuild run build commands is that there's a client for the output files (either compilation or copying of resources). This change adds a phony output to any command that has no declared outputs and declares these phony outputs as input for any targets that are asking for the plugin to be applied. This will lead to these commands running unconditionally, fixing the current silent failure to run these. rdar://100415491
1 parent 3ce3894 commit 5b7bbac

File tree

6 files changed

+99
-7
lines changed

6 files changed

+99
-7
lines changed

Sources/Build/BuildDescription/TargetBuildDescription.swift

+10
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
import Basics
1414
import struct PackageGraph.ResolvedTarget
1515
import struct PackageModel.Resource
16+
import struct PackageModel.ToolsVersion
1617
import struct SPMBuildCore.BuildToolPluginInvocationResult
1718
import struct SPMBuildCore.BuildParameters
1819

@@ -105,4 +106,13 @@ public enum TargetBuildDescription {
105106
return clangTargetBuildDescription.buildParameters
106107
}
107108
}
109+
110+
var toolsVersion: ToolsVersion {
111+
switch self {
112+
case .swift(let swiftTargetBuildDescription):
113+
return swiftTargetBuildDescription.toolsVersion
114+
case .clang(let clangTargetBuildDescription):
115+
return clangTargetBuildDescription.toolsVersion
116+
}
117+
}
108118
}

Sources/Build/BuildManifest/LLBuildManifestBuilder+Clang.swift

+3-2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
import struct LLBuildManifest.Node
1414
import struct Basics.AbsolutePath
1515
import struct Basics.InternalError
16+
import class Basics.ObservabilityScope
1617
import struct PackageGraph.ResolvedTarget
1718
import PackageModel
1819

@@ -89,7 +90,7 @@ extension LLBuildManifestBuilder {
8990
)
9091
}
9192

92-
try addBuildToolPlugins(.clang(target))
93+
let additionalInputs = try addBuildToolPlugins(.clang(target))
9394

9495
// Create a phony node to represent the entire target.
9596
let targetName = target.target.getLLBuildTargetName(config: target.buildParameters.buildConfig)
@@ -98,7 +99,7 @@ extension LLBuildManifestBuilder {
9899
self.manifest.addNode(output, toTarget: targetName)
99100
self.manifest.addPhonyCmd(
100101
name: output.name,
101-
inputs: objectFileNodes,
102+
inputs: objectFileNodes + additionalInputs,
102103
outputs: [output]
103104
)
104105

Sources/Build/BuildManifest/LLBuildManifestBuilder+Swift.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -484,14 +484,14 @@ extension LLBuildManifestBuilder {
484484
}
485485
}
486486

487-
try self.addBuildToolPlugins(.swift(target))
487+
let additionalInputs = try self.addBuildToolPlugins(.swift(target))
488488

489489
// Depend on any required macro product's output.
490490
try target.requiredMacroProducts.forEach { macro in
491491
try inputs.append(.virtual(macro.getLLBuildTargetName(config: target.buildParameters.buildConfig)))
492492
}
493493

494-
return inputs
494+
return inputs + additionalInputs
495495
}
496496

497497
/// Adds a top-level phony command that builds the entire target.

Sources/Build/BuildManifest/LLBuildManifestBuilder.swift

+22-2
Original file line numberDiff line numberDiff line change
@@ -194,7 +194,12 @@ extension LLBuildManifestBuilder {
194194
// MARK: - Compilation
195195

196196
extension LLBuildManifestBuilder {
197-
func addBuildToolPlugins(_ target: TargetBuildDescription) throws {
197+
func addBuildToolPlugins(_ target: TargetBuildDescription) throws -> [Node] {
198+
// For any build command that doesn't declare any outputs, we need to create a phony output to ensure they will still be run by the build system.
199+
var phonyOutputs = [Node]()
200+
// If we have multiple commands with no output files and no display name, this serves as a way to disambiguate the virtual nodes being created.
201+
var pluginNumber = 1
202+
198203
// Add any regular build commands created by plugins for the target.
199204
for result in target.buildToolPluginInvocationResults {
200205
// Only go through the regular build commands — prebuild commands are handled separately.
@@ -213,17 +218,32 @@ extension LLBuildManifestBuilder {
213218
writableDirectories: [result.pluginOutputDirectory]
214219
)
215220
}
221+
let additionalOutputs: [Node]
222+
if command.outputFiles.isEmpty {
223+
if target.toolsVersion >= .v5_11 {
224+
additionalOutputs = [.virtual("\(target.target.c99name)-\(command.configuration.displayName ?? "\(pluginNumber)")")]
225+
phonyOutputs += additionalOutputs
226+
} else {
227+
additionalOutputs = []
228+
observabilityScope.emit(warning: "Build tool command '\(displayName)' (applied to target '\(target.target.name)') does not declare any output files and therefore will not run. You may want to consider updating the given package to tools-version 5.11 (or higher) which would run such a build tool command even without declared outputs.")
229+
}
230+
pluginNumber += 1
231+
} else {
232+
additionalOutputs = []
233+
}
216234
self.manifest.addShellCmd(
217235
name: displayName + "-" + ByteString(encodingAsUTF8: uniquedName).sha256Checksum,
218236
description: displayName,
219237
inputs: command.inputFiles.map { .file($0) },
220-
outputs: command.outputFiles.map { .file($0) },
238+
outputs: command.outputFiles.map { .file($0) } + additionalOutputs,
221239
arguments: commandLine,
222240
environment: command.configuration.environment,
223241
workingDirectory: command.configuration.workingDirectory?.pathString
224242
)
225243
}
226244
}
245+
246+
return phonyOutputs
227247
}
228248
}
229249

Sources/PackageModel/ToolsVersion.swift

+1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ public struct ToolsVersion: Equatable, Hashable, Codable, Sendable {
3131
public static let v5_8 = ToolsVersion(version: "5.8.0")
3232
public static let v5_9 = ToolsVersion(version: "5.9.0")
3333
public static let v5_10 = ToolsVersion(version: "5.10.0")
34+
public static let v5_11 = ToolsVersion(version: "5.11.0")
3435
public static let vNext = ToolsVersion(version: "999.0.0")
3536

3637
/// The current tools version in use.

Tests/FunctionalTests/PluginTests.swift

+61-1
Original file line numberDiff line numberDiff line change
@@ -167,7 +167,67 @@ class PluginTests: XCTestCase {
167167
XCTAssert(stdout.contains("Build of product 'MyLocalTool' complete!"), "stdout:\n\(stdout)")
168168
}
169169
}
170-
170+
171+
func testBuildToolWithoutOutputs() throws {
172+
// Only run the test if the environment in which we're running actually supports Swift concurrency (which the plugin APIs require).
173+
try XCTSkipIf(!UserToolchain.default.supportsSwiftConcurrency(), "skipping because test environment doesn't support concurrency")
174+
175+
func createPackageUnderTest(packageDir: AbsolutePath, toolsVersion: ToolsVersion) throws {
176+
let manifestFile = packageDir.appending("Package.swift")
177+
try localFileSystem.createDirectory(manifestFile.parentDirectory, recursive: true)
178+
try localFileSystem.writeFileContents(
179+
manifestFile,
180+
string: """
181+
// swift-tools-version: \(toolsVersion.description)
182+
import PackageDescription
183+
let package = Package(name: "MyPackage",
184+
targets: [
185+
.target(name: "SomeTarget", plugins: ["Plugin"]),
186+
.plugin(name: "Plugin", capability: .buildTool),
187+
])
188+
""")
189+
190+
let targetSourceFile = packageDir.appending(components: "Sources", "SomeTarget", "dummy.swift")
191+
try localFileSystem.createDirectory(targetSourceFile.parentDirectory, recursive: true)
192+
try localFileSystem.writeFileContents(targetSourceFile, string: "")
193+
194+
let pluginSourceFile = packageDir.appending(components: "Plugins", "Plugin", "plugin.swift")
195+
try localFileSystem.createDirectory(pluginSourceFile.parentDirectory, recursive: true)
196+
try localFileSystem.writeFileContents(pluginSourceFile, string: """
197+
import PackagePlugin
198+
@main
199+
struct Plugin: BuildToolPlugin {
200+
func createBuildCommands(context: PluginContext, target: Target) async throws -> [Command] {
201+
return [
202+
.buildCommand(
203+
displayName: "empty",
204+
executable: .init("/usr/bin/touch"),
205+
arguments: [context.pluginWorkDirectory.appending("best.txt")],
206+
inputFiles: [],
207+
outputFiles: []
208+
)
209+
]
210+
}
211+
}
212+
""")
213+
}
214+
215+
try testWithTemporaryDirectory { tmpPath in
216+
let packageDir = tmpPath.appending(components: "MyPackage")
217+
let pathOfGeneratedFile = packageDir.appending(components: [".build", "plugins", "outputs", "mypackage", "SomeTarget", "Plugin", "best.txt"])
218+
219+
try createPackageUnderTest(packageDir: packageDir, toolsVersion: .v5_9)
220+
let (_, stderr) = try executeSwiftBuild(packageDir)
221+
XCTAssertTrue(stderr.contains("warning: Build tool command 'empty' (applied to target 'SomeTarget') does not declare any output files"), "expected warning not emitted")
222+
XCTAssertFalse(localFileSystem.exists(pathOfGeneratedFile), "plugin generated file unexpectedly exists at \(pathOfGeneratedFile.pathString)")
223+
224+
try createPackageUnderTest(packageDir: packageDir, toolsVersion: .v5_11)
225+
let (_, stderr2) = try executeSwiftBuild(packageDir)
226+
XCTAssertEqual("", stderr2)
227+
XCTAssertTrue(localFileSystem.exists(pathOfGeneratedFile), "plugin did not run, generated file does not exist at \(pathOfGeneratedFile.pathString)")
228+
}
229+
}
230+
171231
func testCommandPluginInvocation() async throws {
172232
try XCTSkipIf(true, "test is disabled because it isn't stable, see rdar://117870608")
173233

0 commit comments

Comments
 (0)