Capture output of running a shell command in Ruby

Revision history
Tags: jekyll ruby

Preface

I am creating a plugin for Jekyll to perform extra processing of some of my posts. Specifically, for some of my Python posts, I want to include the actual output of the Python snippets I’ve embedded in the Markdown.

I need to

This is only about the Ruby command execution.

Run shell commands from Ruby and capture output

The simplest way to do this is to enclose the command in backticks (`), and it will return a string with the outputs.

msg = `echo hello`
puts msg
# Prints: hello

Execute a program from shell with string input to stdin in Ruby

There are some different functions we can use for this purpose in Open3, depending on what control over the input and ouput you need.

I wanted do pass in a string to STDIN to simplify my program. That can be done using the capture* variants.

cmd = "cat > file.txt"
post_content = "hello world"
stdout_str, exit_code = Open3.capture2(cmd, :stdin_data=>post_content)

Capture stdout and stderr of shell command in Ruby

In the code above I’m only caring about the stdout. But if you need to capture STDERR as well, either forward STDERR to STDIN in your shell command

stdout_str, exit_code = Open3.capture2("my_program 2>&1")

or let Ruby take care of exactly the same thing

stdout_and_stderr, status = Open3.capture2e("my_program")

or use capture3 if you want the outputs separated.

post_content = "this input is piped to stdin"
stdout_str, stderr_str, exit_code = Open3.capture3(cmd, :stdin_data=>post_content)

Note that the capture* functions operates with strings, while popen* operates with pipes. See the documentation for Open3 for a complete reference and list of available functions.

References

If you have any comments or feedback, please send me an e-mail. (stig at stigok dotcom).

Did you find any typos, incorrect information, or have something to add? Then please propose a change to this post.

Creative Commons License This work is licensed under a Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International License.