From e00a816838896d81382b33ef9de58aebe370703d Mon Sep 17 00:00:00 2001 From: Ian Leitch Date: Wed, 18 Dec 2024 12:44:13 +0000 Subject: [PATCH] Output stderr for failed commands. Closes #629 #840 (#847) --- MODULE.bazel.lock | 2 +- .../ProjectDrivers/BazelProjectDriver.swift | 2 +- Sources/ProjectDrivers/SPM.swift | 2 +- Sources/Shared/Shell.swift | 59 +++++++++++-------- Sources/XcodeSupport/Xcodebuild.swift | 2 +- Tests/XcodeTests/ShellMock.swift | 2 +- 6 files changed, 38 insertions(+), 31 deletions(-) diff --git a/MODULE.bazel.lock b/MODULE.bazel.lock index 679c0601b..0ff47ee68 100644 --- a/MODULE.bazel.lock +++ b/MODULE.bazel.lock @@ -177,7 +177,7 @@ "//bazel:generated.bzl%generated": { "general": { "bzlTransitiveDigest": "nMR2FBcoRPImVocN9DNOnm2NQWyTbJPu7SHJgAXsLFw=", - "usagesDigest": "wpzfE+FqjMSvrEYHa2419pepTym02qH7AT1nBCOagKY=", + "usagesDigest": "8mdYbdjeyWhYUhUViUkt2T2IwtnrN+PHXdKUhqo0aqk=", "recordedFileInputs": {}, "recordedDirentsInputs": {}, "envVariables": {}, diff --git a/Sources/ProjectDrivers/BazelProjectDriver.swift b/Sources/ProjectDrivers/BazelProjectDriver.swift index 169a12f2e..1c15d1587 100644 --- a/Sources/ProjectDrivers/BazelProjectDriver.swift +++ b/Sources/ProjectDrivers/BazelProjectDriver.swift @@ -123,7 +123,7 @@ public class BazelProjectDriver: ProjectDriver { private func queryTargets() throws -> [String] { try shell - .exec(["bazel", "query", query], stderr: false) + .exec(["bazel", "query", query]) .split(separator: "\n") .map { "\"@@\($0)\"" } } diff --git a/Sources/ProjectDrivers/SPM.swift b/Sources/ProjectDrivers/SPM.swift index 3afb262fe..cfd465c6f 100644 --- a/Sources/ProjectDrivers/SPM.swift +++ b/Sources/ProjectDrivers/SPM.swift @@ -46,7 +46,7 @@ public enum SPM { if let path = configuration.jsonPackageManifestPath { jsonData = try Data(contentsOf: path.url) } else { - let jsonString = try shell.exec(["swift", "package", "describe", "--type", "json"], stderr: false) + let jsonString = try shell.exec(["swift", "package", "describe", "--type", "json"]) guard let data = jsonString.data(using: .utf8) else { throw PeripheryError.packageError(message: "Failed to read swift package description.") diff --git a/Sources/Shared/Shell.swift b/Sources/Shared/Shell.swift index 7be7eea45..d9e834732 100644 --- a/Sources/Shared/Shell.swift +++ b/Sources/Shared/Shell.swift @@ -39,29 +39,23 @@ open class Shell { } @discardableResult - open func exec( - _ args: [String], - stderr: Bool = true - ) throws -> String { - let (status, output) = try exec(args, environment: environment, stderr: stderr) + open func exec(_ args: [String]) throws -> String { + let (status, stdout, stderr) = try exec(args) if status == 0 { - return output + return stdout } throw PeripheryError.shellCommandFailed( cmd: args, status: status, - output: output + output: [stdout, stderr].filter { !$0.isEmpty }.joined(separator: "\n").trimmed ) } @discardableResult - open func execStatus( - _ args: [String], - stderr: Bool = true - ) throws -> Int32 { - let (status, _) = try exec(args, environment: environment, stderr: stderr, captureOutput: false) + open func execStatus(_ args: [String]) throws -> Int32 { + let (status, _, _) = try exec(args, captureOutput: false) return status } @@ -69,10 +63,8 @@ open class Shell { private func exec( _ args: [String], - environment: [String: String], - stderr: Bool = false, captureOutput: Bool = true - ) throws -> (Int32, String) { + ) throws -> (Int32, String, String) { let launchPath: String let newArgs: [String] @@ -92,22 +84,37 @@ open class Shell { logger.debug("\(launchPath) \(newArgs.joined(separator: " "))") ShellProcessStore.shared.add(process) - var outputPipe: Pipe? + var stdoutPipe: Pipe? + var stderrPipe: Pipe? if captureOutput { - outputPipe = Pipe() - process.standardOutput = outputPipe - process.standardError = stderr ? outputPipe : nil + stdoutPipe = Pipe() + stderrPipe = Pipe() + process.standardOutput = stdoutPipe + process.standardError = stderrPipe } process.launch() - var output = "" + var stdout = "" + var stderr = "" + + if let stdoutData = try stdoutPipe?.fileHandleForReading.readToEnd() { + guard let stdoutStr = String(data: stdoutData, encoding: .utf8) + else { + ShellProcessStore.shared.remove(process) + throw PeripheryError.shellOutputEncodingFailed( + cmd: launchPath, + args: newArgs, + encoding: .utf8 + ) + } + stdout = stdoutStr + } - if let outputPipe, - let outputData = try outputPipe.fileHandleForReading.readToEnd() - { - guard let str = String(data: outputData, encoding: .utf8) else { + if let stderrData = try stderrPipe?.fileHandleForReading.readToEnd() { + guard let stderrStr = String(data: stderrData, encoding: .utf8) + else { ShellProcessStore.shared.remove(process) throw PeripheryError.shellOutputEncodingFailed( cmd: launchPath, @@ -115,11 +122,11 @@ open class Shell { encoding: .utf8 ) } - output = str + stderr = stderrStr } process.waitUntilExit() ShellProcessStore.shared.remove(process) - return (process.terminationStatus, output) + return (process.terminationStatus, stdout, stderr) } } diff --git a/Sources/XcodeSupport/Xcodebuild.swift b/Sources/XcodeSupport/Xcodebuild.swift index 92ed1d310..fb59a86a8 100644 --- a/Sources/XcodeSupport/Xcodebuild.swift +++ b/Sources/XcodeSupport/Xcodebuild.swift @@ -86,7 +86,7 @@ public final class Xcodebuild { let quotedArguments = quote(arguments: additionalArguments) let xcodebuild = "xcodebuild \((args + quotedArguments).joined(separator: " "))" - let lines = try shell.exec(["/bin/sh", "-c", xcodebuild], stderr: false).split(separator: "\n").map { String($0).trimmed } + let lines = try shell.exec(["/bin/sh", "-c", xcodebuild]).split(separator: "\n").map { String($0).trimmed } // xcodebuild may output unrelated warnings, we need to strip them out otherwise // JSON parsing will fail. diff --git a/Tests/XcodeTests/ShellMock.swift b/Tests/XcodeTests/ShellMock.swift index 0ae7918d6..8a99a60b7 100644 --- a/Tests/XcodeTests/ShellMock.swift +++ b/Tests/XcodeTests/ShellMock.swift @@ -10,7 +10,7 @@ class ShellMock: Shell { self.init(environment: ProcessInfo.processInfo.environment, logger: logger) } - override func exec(_: [String], stderr _: Bool = true) throws -> String { + override func exec(_: [String]) throws -> String { output } }