Merge pull request #8541 from claui/fix-patch-stdout
Return standard output in `popen_write`
This commit is contained in:
commit
8604cdb15a
@ -26,11 +26,49 @@ describe Utils do
|
|||||||
end
|
end
|
||||||
|
|
||||||
describe "::popen_write" do
|
describe "::popen_write" do
|
||||||
it "with supports writing to a command's standard input" do
|
let(:foo) { mktmpdir/"foo" }
|
||||||
|
|
||||||
|
before { foo.write "Foo\n" }
|
||||||
|
|
||||||
|
it "supports writing to a command's standard input" do
|
||||||
subject.popen_write("grep", "-q", "success") do |pipe|
|
subject.popen_write("grep", "-q", "success") do |pipe|
|
||||||
pipe.write("success\n")
|
pipe.write "success\n"
|
||||||
end
|
end
|
||||||
expect($CHILD_STATUS).to be_a_success
|
expect($CHILD_STATUS).to be_a_success
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it "returns the command's standard output before writing" do
|
||||||
|
child_stdout = subject.popen_write("cat", foo, "-") do |pipe|
|
||||||
|
pipe.write "Bar\n"
|
||||||
|
end
|
||||||
|
expect($CHILD_STATUS).to be_a_success
|
||||||
|
expect(child_stdout).to eq <<~EOS
|
||||||
|
Foo
|
||||||
|
Bar
|
||||||
|
EOS
|
||||||
|
end
|
||||||
|
|
||||||
|
it "returns the command's standard output after writing" do
|
||||||
|
child_stdout = subject.popen_write("cat", "-", foo) do |pipe|
|
||||||
|
pipe.write "Bar\n"
|
||||||
|
end
|
||||||
|
expect($CHILD_STATUS).to be_a_success
|
||||||
|
expect(child_stdout).to eq <<~EOS
|
||||||
|
Bar
|
||||||
|
Foo
|
||||||
|
EOS
|
||||||
|
end
|
||||||
|
|
||||||
|
it "supports interleaved writing between two reads" do
|
||||||
|
child_stdout = subject.popen_write("cat", foo, "-", foo) do |pipe|
|
||||||
|
pipe.write "Bar\n"
|
||||||
|
end
|
||||||
|
expect($CHILD_STATUS).to be_a_success
|
||||||
|
expect(child_stdout).to eq <<~EOS
|
||||||
|
Foo
|
||||||
|
Bar
|
||||||
|
Foo
|
||||||
|
EOS
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@ -1,6 +1,9 @@
|
|||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
module Utils
|
module Utils
|
||||||
|
IO_DEFAULT_BUFFER_SIZE = 4096
|
||||||
|
private_constant :IO_DEFAULT_BUFFER_SIZE
|
||||||
|
|
||||||
def self.popen_read(*args, **options, &block)
|
def self.popen_read(*args, **options, &block)
|
||||||
popen(args, "rb", options, &block)
|
popen(args, "rb", options, &block)
|
||||||
end
|
end
|
||||||
@ -12,8 +15,25 @@ module Utils
|
|||||||
raise ErrorDuringExecution.new(args, status: $CHILD_STATUS, output: [[:stdout, output]])
|
raise ErrorDuringExecution.new(args, status: $CHILD_STATUS, output: [[:stdout, output]])
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.popen_write(*args, **options, &block)
|
def self.popen_write(*args, **options)
|
||||||
popen(args, "wb", options, &block)
|
popen(args, "w+b", options) do |pipe|
|
||||||
|
output = ""
|
||||||
|
|
||||||
|
# Before we yield to the block, capture as much output as we can
|
||||||
|
loop do
|
||||||
|
output += pipe.read_nonblock(IO_DEFAULT_BUFFER_SIZE)
|
||||||
|
rescue IO::WaitReadable, EOFError
|
||||||
|
break
|
||||||
|
end
|
||||||
|
|
||||||
|
yield pipe
|
||||||
|
pipe.close_write
|
||||||
|
IO.select([pipe])
|
||||||
|
|
||||||
|
# Capture the rest of the output
|
||||||
|
output += pipe.read
|
||||||
|
output.freeze
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.safe_popen_write(*args, **options, &block)
|
def self.safe_popen_write(*args, **options, &block)
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user