Merge stdout and stderr in Popen

2019-07-06 09:08发布

In Ruby's popen/spawn, how do I merge both STDOUT and STDERR as a single stream wihthout resorting to using >2&1?

In Python, this would be:

>>> import subprocess
>>> subprocess.check_output('my_prog args', stderr=subprocess.STDOUT, shell=True)

Note the stderr argument.

I use Open3 - as I don't want just stdout - but it already separates them into two streams.

2条回答
迷人小祖宗
2楼-- · 2019-07-06 09:39

Using the code from your other question, here you go:

cmd = 'a_prog --arg ... --arg2 ...'
Open3.popen2({"MYVAR" => "a_value"}, "#{cmd}", {:err => [:child, :out]}) { |i,o|
  # This output should include stderr as well
  output = o.read()
  repr = "$ #{cmd}\n#{output}"
}

A couple changes:

  1. The third parameter to popen2 will redirect stderr to stdoutl. Note that it needs to be the spawned process's stdout, not the system-wide stdout, so you need to specify :child's :out
  2. You need to use .popen2 instead of .popen3 as it seems the redirection is ignored if you include the 3rd e option for stderr
  3. Because you're using .popen2, you only pass |i,o| to the block:
查看更多
爷的心禁止访问
3楼-- · 2019-07-06 09:41

A bit late, but take a look at Open3.popen2e - docs.

This behaves exactly as popen3, but merges stderr stdout as the second argument to the block.

So you can simply do

cmd = 'a_prog --arg ... --arg2 ...'
Open3.popen2e(cmd) { |input,output|
 # Process as desired, with output containing stdout and stderr
}
查看更多
登录 后发表回答