diff --git a/Library/Homebrew/extend/os/mac/system_config.rb b/Library/Homebrew/extend/os/mac/system_config.rb index 0b63b3434f30f8776f2b2b3ad56117eeb7df51b3..721cc871a90fc12f29e3d24ce7a263e855eb4a96 100644 --- a/Library/Homebrew/extend/os/mac/system_config.rb +++ b/Library/Homebrew/extend/os/mac/system_config.rb @@ -6,12 +6,11 @@ class SystemConfig # java_home doesn't exist on all macOSs; it might be missing on older versions. return "N/A" unless File.executable? "/usr/libexec/java_home" - java_xml = Utils.popen_read("/usr/libexec/java_home", "--xml", "--failfast", err: :close) - return "N/A" unless $CHILD_STATUS.success? + out, _, status = system_command("/usr/libexec/java_home", args: ["--xml", "--failfast"], print_stderr: false) + return "N/A" unless status.success? javas = [] - REXML::XPath.each( - REXML::Document.new(java_xml), "//key[text()='JVMVersion']/following-sibling::string" - ) do |item| + xml = REXML::Document.new(out) + REXML::XPath.each(xml, "//key[text()='JVMVersion']/following-sibling::string") do |item| javas << item.text end javas.uniq.join(", ") diff --git a/Library/Homebrew/readall.rb b/Library/Homebrew/readall.rb index c1a79ab702beff19c0d4919b58180f1e1f752a88..d5d3add6e25940696dffdd0836cc27b5b300e613 100644 --- a/Library/Homebrew/readall.rb +++ b/Library/Homebrew/readall.rb @@ -73,21 +73,20 @@ module Readall private def syntax_errors_or_warnings?(rb) - # Retrieve messages about syntax errors/warnings printed to `$stderr`, but - # discard a `Syntax OK` printed to `$stdout` (in absence of syntax errors). - messages = Utils.popen_read("#{RUBY_PATH} -c -w #{rb} 2>&1 >/dev/null") + # Retrieve messages about syntax errors/warnings printed to `$stderr`. + _, err, status = system_command(RUBY_PATH, args: ["-c", "-w", rb], print_stderr: false) # Ignore unnecessary warning about named capture conflicts. # See https://bugs.ruby-lang.org/issues/12359. - messages = messages.lines - .grep_v(/named capture conflicts a local variable/) - .join + messages = err.lines + .grep_v(/named capture conflicts a local variable/) + .join $stderr.print messages # Only syntax errors result in a non-zero status code. To detect syntax # warnings we also need to inspect the output to `$stderr`. - !$CHILD_STATUS.success? || !messages.chomp.empty? + !status.success? || !messages.chomp.empty? end end end diff --git a/Library/Homebrew/requirements/java_requirement.rb b/Library/Homebrew/requirements/java_requirement.rb index 06f5f32b36ce5c213943cc8d8a6ee5b5a06b1e6e..726ecc3f95256e1d1a21795fa54b636075bf478e 100644 --- a/Library/Homebrew/requirements/java_requirement.rb +++ b/Library/Homebrew/requirements/java_requirement.rb @@ -122,7 +122,7 @@ class JavaRequirement < Requirement end def satisfies_version(java) - java_version_s = Utils.popen_read(java, "-version", err: :out)[/\d+.\d/] + java_version_s = system_command(java, args: ["-version"], print_stderr: false).stderr[/\d+.\d/] return false unless java_version_s java_version = Version.create(java_version_s) needed_version = Version.create(version_without_plus) diff --git a/Library/Homebrew/system_command.rb b/Library/Homebrew/system_command.rb index 4eef3e664419379129595a5d7e2b07c56f88f389..571b3540be8474dd6307382c15eea66fa61ea67d 100644 --- a/Library/Homebrew/system_command.rb +++ b/Library/Homebrew/system_command.rb @@ -30,16 +30,16 @@ class SystemCommand def run! puts command.shelljoin.gsub(/\\=/, "=") if verbose? || ARGV.debug? - @merged_output = [] + @output = [] each_output_line do |type, line| case type when :stdout $stdout << line if print_stdout? - @merged_output << [:stdout, line] + @output << [:stdout, line] when :stderr $stderr << line if print_stderr? - @merged_output << [:stderr, line] + @output << [:stderr, line] end end @@ -100,7 +100,7 @@ class SystemCommand return if @status.success? raise ErrorDuringExecution.new(command, status: @status, - output: @merged_output) + output: @output) end def expanded_args @@ -128,7 +128,7 @@ class SystemCommand @status = raw_wait_thr.value rescue SystemCallError => e @status = $CHILD_STATUS - @merged_output << [:stderr, e.message] + @output << [:stderr, e.message] end def write_input_to(raw_stdin) @@ -158,22 +158,29 @@ class SystemCommand end def result - output = @merged_output.each_with_object(stdout: "", stderr: "") do |(type, line), hash| - hash[type] << line - end - - Result.new(command, output[:stdout], output[:stderr], @status) + Result.new(command, @output, @status) end class Result - attr_accessor :command, :stdout, :stderr, :status, :exit_status - - def initialize(command, stdout, stderr, status) - @command = command - @stdout = stdout - @stderr = stderr - @status = status - @exit_status = status.exitstatus + attr_accessor :command, :status, :exit_status + + def initialize(command, output, status) + @command = command + @output = output + @status = status + @exit_status = status.exitstatus + end + + def stdout + @stdout ||= @output.select { |type,| type == :stdout } + .map { |_, line| line } + .join + end + + def stderr + @stderr ||= @output.select { |type,| type == :stderr } + .map { |_, line| line } + .join end def success? diff --git a/Library/Homebrew/system_config.rb b/Library/Homebrew/system_config.rb index 0f7c56b1be27c0edd1f868dd3a408e9b371e3ca9..a4bd6aa8b63f2475e3e55d352575e8cbc27e53cc 100644 --- a/Library/Homebrew/system_config.rb +++ b/Library/Homebrew/system_config.rb @@ -79,9 +79,9 @@ class SystemConfig def describe_java return "N/A" unless which "java" - java_version = Utils.popen_read("java", "-version") - return "N/A" unless $CHILD_STATUS.success? - java_version[/java version "([\d\._]+)"/, 1] || "N/A" + _, err, status = system_command("java", args: ["-version"], print_stderr: false) + return "N/A" unless status.success? + err[/java version "([\d\._]+)"/, 1] || "N/A" end def describe_git @@ -90,12 +90,13 @@ class SystemConfig end def describe_curl - curl_version_output = Utils.popen_read("#{curl_executable} --version", err: :close) - curl_version_output =~ /^curl ([\d\.]+)/ - curl_version = Regexp.last_match(1) - "#{curl_version} => #{curl_executable}" - rescue - "N/A" + out, = system_command(curl_executable, args: ["--version"]) + + if /^curl (?<curl_version>[\d\.]+)/ =~ out + "#{curl_version} => #{curl_executable}" + else + "N/A" + end end def dump_verbose_config(f = $stdout) diff --git a/Library/Homebrew/test/cask/pkg_spec.rb b/Library/Homebrew/test/cask/pkg_spec.rb index 47257812a1eb2efb840aa77f103a1bf693bd38ff..cde09e01ecfe9720e1aa083abd91e61ea6a3c3ef 100644 --- a/Library/Homebrew/test/cask/pkg_spec.rb +++ b/Library/Homebrew/test/cask/pkg_spec.rb @@ -150,7 +150,7 @@ describe Hbc::Pkg, :cask do "/usr/sbin/pkgutil", args: ["--pkg-info-plist", pkg_id], ).and_return( - SystemCommand::Result.new(nil, pkg_info_plist, nil, instance_double(Process::Status, exitstatus: 0)), + SystemCommand::Result.new(nil, [[:stdout, pkg_info_plist]], instance_double(Process::Status, exitstatus: 0)), ) info = pkg.info diff --git a/Library/Homebrew/test/java_requirement_spec.rb b/Library/Homebrew/test/java_requirement_spec.rb index 3d6b8cebc94ece8053f86ef166ebe5b1dd2e7907..b47593ab129981828299b58e5daac650c17f4f70 100644 --- a/Library/Homebrew/test/java_requirement_spec.rb +++ b/Library/Homebrew/test/java_requirement_spec.rb @@ -55,7 +55,7 @@ describe JavaRequirement do def setup_java_with_version(version) IO.write java, <<~SH #!/bin/sh - echo 'java version "#{version}"' + echo 'java version "#{version}"' 1>&2 SH FileUtils.chmod "+x", java end diff --git a/Library/Homebrew/test/support/helper/cask/fake_system_command.rb b/Library/Homebrew/test/support/helper/cask/fake_system_command.rb index e740e73539ad1c7660c645720245c38f410c2b62..2769683d84399e14095037b707a745e8985eaeac 100644 --- a/Library/Homebrew/test/support/helper/cask/fake_system_command.rb +++ b/Library/Homebrew/test/support/helper/cask/fake_system_command.rb @@ -52,7 +52,7 @@ class FakeSystemCommand if response.respond_to?(:call) response.call(command_string, options) else - SystemCommand::Result.new(command, response, "", OpenStruct.new(exitstatus: 0)) + SystemCommand::Result.new(command, [[:stdout, response]], OpenStruct.new(exitstatus: 0)) end end diff --git a/Library/Homebrew/test/system_command_result_spec.rb b/Library/Homebrew/test/system_command_result_spec.rb index d65339a133d8f8aab817751d173327fcb3343bd0..508dad2212dea6c47c9859b451543bb7ef793bc8 100644 --- a/Library/Homebrew/test/system_command_result_spec.rb +++ b/Library/Homebrew/test/system_command_result_spec.rb @@ -2,8 +2,14 @@ require "system_command" describe SystemCommand::Result do describe "#to_ary" do + let(:output) { + [ + [:stdout, "output"], + [:stderr, "error"], + ] + } subject(:result) { - described_class.new([], "output", "error", instance_double(Process::Status, exitstatus: 0, success?: true)) + described_class.new([], output, instance_double(Process::Status, exitstatus: 0, success?: true)) } it "can be destructed like `Open3.capture3`" do @@ -16,7 +22,9 @@ describe SystemCommand::Result do end describe "#plist" do - subject { described_class.new(command, stdout, "", instance_double(Process::Status, exitstatus: 0)).plist } + subject { + described_class.new(command, [[:stdout, stdout]], instance_double(Process::Status, exitstatus: 0)).plist + } let(:command) { ["true"] } let(:garbage) {