if RUBY_PLATFORM == 'java'
require 'java'
require 'ostruct'
elsif RUBY_VERSION =~ /^1.8/
begin
require 'open4'
rescue LoadError
warn "For Ruby #{RUBY_VERSION}, the open4 library must be installed or SH won't work"
end
else
require 'open3'
end
require 'methadone/process_status'
module Methadone
# Module with various helper methods for executing external commands.
# In most cases, you can use #sh to run commands and have decent logging
# done. You will likely use this in a class that also mixes-in
# Methadone::CLILogging (remembering that Methadone::Main mixes this in for you).
# If you don't, you must provide a logger via #set_sh_logger.
#
# == Examples
#
# include Methadone::SH
#
# sh 'cp foo.txt /tmp'
# # => logs the command to DEBUG, executes the command, logs its output to DEBUG and its
# # error output to WARN, returns 0
#
# sh 'cp non_existent_file.txt /nowhere_good'
# # => logs the command to DEBUG, executes the command, logs its output to INFO and
# # its error output to WARN, returns the nonzero exit status of the underlying command
#
# sh! 'cp non_existent_file.txt /nowhere_good'
# # => same as above, EXCEPT, raises a Methadone::FailedCommandError
#
# sh 'cp foo.txt /tmp' do
# # Behaves exactly as before, but this block is called after
# end
#
# sh 'cp non_existent_file.txt /nowhere_good' do
# # This block isn't called, since the command failed
# end
#
# sh 'ls -l /tmp/' do |stdout|
# # stdout contains the output of the command
# end
# sh 'ls -l /tmp/ /non_existent_dir' do |stdout,stderr|
# # stdout contains the output of the command,
# # stderr contains the standard error output.
# end
#
# == Handling process execution
#
# In order to work on as many Rubies as possible, this class defers the actual execution
# to an execution strategy. See #set_execution_strategy if you think you'd like to override
# that, or just want to know how it works.
#
# == More complex execution and subprocess management
#
# This is not intended to be a complete replacement for Open3 or an enhanced means of managing subprocesses.
# This is to make it easy for you to shell-out to external commands and have your app be robust and
# easy to maintain.
module SH
def self.included(k)
k.extend(self)
end
# Run a shell command, capturing and logging its output.
# If the command completed successfully, it's output is logged at DEBUG.
# If not, its output as logged at INFO. In either case, its
# error output is logged at WARN.
#
# command:: the command to run as a String or Array of String. The String form is simplest, but
# is open to injection. If you need to execute a command that is assembled from some portion
# of user input, consider using an Array of String. This form prevents tokenization that occurs
# in the String form. The first element is the command to execute,
# and the remainder are the arguments. See Methadone::ExecutionStrategy::Base for more info.
# options:: options to control the call. Currently responds to:
# +:expected+:: an Int or Array of Int representing error codes, in addition to 0, that are
# expected and therefore constitute success. Useful for commands that don't use
# exit codes the way you'd like
# block:: if provided, will be called if the command exited nonzero. The block may take 0, 1, 2, or 3 arguments.
# The arguments provided are the standard output as a string, standard error as a string, and
# the exitstatus as an Int.
# You should be safe to pass in a lambda instead of a block, as long as your
# lambda doesn't take more than three arguments
#
# Example
#
# sh "cp foo /tmp"
# sh "ls /tmp" do |stdout|
# # stdout contains the output of ls /tmp
# end
# sh "ls -l /tmp foobar" do |stdout,stderr|
# # ...
# end
#
# Returns the exit status of the command. Note that if the command doesn't exist, this returns 127.
def sh(command,options={},&block)
sh_logger.debug("Executing '#{command}'")
stdout,stderr,status = execution_strategy.run_command(command)
process_status = Methadone::ProcessStatus.new(status,options[:expected])
sh_logger.warn("stderr output of '#{command}': #{stderr}") unless stderr.strip.length == 0
if process_status.success?
sh_logger.debug("stdout output of '#{command}': #{stdout}") unless stdout.strip.length == 0
call_block(block,stdout,stderr,process_status.exitstatus) unless block.nil?
else
sh_logger.info("stdout output of '#{command}': #{stdout}") unless stdout.strip.length == 0
sh_logger.warn("Error running '#{command}'")
end
process_status.exitstatus
rescue *exception_meaning_command_not_found => ex
sh_logger.error("Error running '#{command}': #{ex.message}")
127
end
# Run a command, throwing an exception if the command exited nonzero.
# Otherwise, behaves exactly like #sh.
#
# options:: options hash, responding to:
# :expected:: same as for #sh
# :on_fail:: a custom error message. This allows you to have your
# app exit on shell command failures, but customize the error
# message that they see.
#
# Raises Methadone::FailedCommandError if the command exited nonzero.
#
# Examples:
#
# sh!("rsync foo bar")
# # => if command fails, app exits and user sees: "error: Command 'rsync foo bar' exited 12"
# sh!("rsync foo bar", :on_fail => "Couldn't rsync, check log for details")
# # => if command fails, app exits and user sees: "error: Couldn't rsync, check log for details
def sh!(command,options={},&block)
sh(command,options,&block).tap do |exitstatus|
process_status = Methadone::ProcessStatus.new(exitstatus,options[:expected])
unless process_status.success?
raise Methadone::FailedCommandError.new(exitstatus,command,options[:on_fail])
end
end
end
# Override the default logger (which is the one provided by CLILogging).
# You would do this if you want a custom logger or you aren't mixing-in
# CLILogging.
#
# Note that this method is *not* called sh_logger= to avoid annoying situations
# where Ruby thinks you are setting a local variable
def set_sh_logger(logger)
@sh_logger = logger
end
# Set the strategy to use for executing commands. In general, you don't need to set this
# since this module chooses an appropriate implementation based on your Ruby platform:
#
# 1.8 Rubies, including 1.8, and REE:: Open4 is used via Methadone::ExecutionStrategy::Open_4. open4 will not be
# installed as a dependency. RubyGems doesn't allow conditional dependencies,
# so make sure that your app declares it as a dependency if you think you'll be
# running on 1.8 or REE.
# Rubinius:: Open4 is used, but we handle things a bit differently; see Methadone::ExecutionStrategy::RBXOpen_4.
# Same warning on dependencies applies.
# JRuby:: Use JVM calls to +Runtime+ via Methadone::ExecutionStrategy::JVM
# Windows:: Currently no support for Windows
# All others:: we use Open3 from the standard library, via Methadone::ExecutionStrategy::Open_3
#
# See Methadone::ExecutionStrategy::Base for how to implement your own.
def set_execution_strategy(strategy)
@execution_strategy = strategy
end
private
def exception_meaning_command_not_found
execution_strategy.exception_meaning_command_not_found
end
def self.default_execution_strategy_class
if RUBY_PLATFORM == 'java'
Methadone::ExecutionStrategy::JVM
elsif defined?(RUBY_ENGINE) && RUBY_ENGINE == 'rbx'
Methadone::ExecutionStrategy::RBXOpen_4
elsif RUBY_VERSION =~ /^1.8/
Methadone::ExecutionStrategy::Open_4
else
Methadone::ExecutionStrategy::Open_3
end
end
def execution_strategy
@execution_strategy ||= SH.default_execution_strategy_class.new
end
def sh_logger
@sh_logger ||= begin
raise StandardError, "No logger set! Please include Methadone::CLILogging or provide your own via #set_sh_logger." unless self.respond_to?(:logger)
self.logger
end
end
# Safely call our block, even if the user passed in a lambda
def call_block(block,stdout,stderr,exitstatus)
# blocks that take no arguments have arity -1. Or 0. Ugh.
if block.arity > 0
case block.arity
when 1
block.call(stdout)
when 2
block.call(stdout,stderr)
else
# Let it fail for lambdas
block.call(stdout,stderr,exitstatus)
end
else
block.call
end
end
end
end