| Module | Bio::Command |
| In: |
lib/bio/command.rb
(CVS)
|
Bio::Command is a collection of useful methods for execution of external commands or web applications. Any wrapper class for applications shall use this class.
Library internal use only. Users should not directly use it.
| UNSAFE_CHARS_UNIX | = | /[^A-Za-z0-9\_\-\.\:\,\/\@\x1b\x80-\xfe]/n |
| QUOTE_CHARS_WINDOWS | = | /[^A-Za-z0-9\_\-\.\:\,\/\@\\]/n |
| UNESCAPABLE_CHARS | = | /[\x00-\x08\x10-\x1a\x1c-\x1f\x7f\xff]/n |
Executes the program. Automatically select popen for Windows environment and fork for the others. A block must be given. An IO object is passed to the block.
# File lib/bio/command.rb, line 87 def call_command(cmd, &block) case RUBY_PLATFORM when /mswin32|bccwin32/ call_command_popen(cmd, &block) else call_command_fork(cmd, &block) end end
Executes the program via fork (by using IO.popen("-")) and exec. A block must be given. An IO object is passed to the block.
From the view point of security, this method is recommended rather than call_command_popen.
# File lib/bio/command.rb, line 111 def call_command_fork(cmd) IO.popen("-", "r+") do |io| if io then # parent yield io else # child begin Kernel.exec(*cmd) rescue Errno::ENOENT, Errno::EACCES Process.exit!(127) rescue Exception end Process.exit!(1) end end end
Executes the program via Open3.popen3 A block must be given. IO objects are passed to the block.
You would use this method only when you really need to get stderr.
# File lib/bio/command.rb, line 133 def call_command_open3(cmd) cmd = cmd.collect { |x| x.to_s } Open3.popen3(*cmd) do |pin, pout, perr| yield pin, pout, perr end end
Executes the program via IO.popen for OS which doesn‘t support fork. A block must be given. An IO object is passed to the block.
# File lib/bio/command.rb, line 98 def call_command_popen(cmd) str = make_command_line(cmd) IO.popen(str, "w+") do |io| io.sync = true yield io end end
Escape special characters in command line string.
# File lib/bio/command.rb, line 53 def escape_shell(str) case RUBY_PLATFORM when /mswin32|bccwin32/ escape_shell_windows(str) else escape_shell_unix(str) end end
Escape special characters in command line string for UNIX shells.
# File lib/bio/command.rb, line 46 def escape_shell_unix(str) str = str.to_s raise 'cannot escape control characters' if UNESCAPABLE_CHARS =~ str str.gsub(UNSAFE_CHARS_UNIX) { |x| "\\#{x}" } end
Escape special characters in command line string for cmd.exe on Windows.
# File lib/bio/command.rb, line 35 def escape_shell_windows(str) str = str.to_s raise 'cannot escape control characters' if UNESCAPABLE_CHARS =~ str if QUOTE_CHARS_WINDOWS =~ str then '"' + str.gsub(/\"/, '""') + '"' else String.new(str) end end
# File lib/bio/command.rb, line 292 def make_cgi_params(params) data = "" case params when Hash data = params.map do |key, val| make_cgi_params_key_value(key, val) end.join('&') when Array case params.first when Hash data = params.map do |hash| hash.map do |key, val| make_cgi_params_key_value(key, val) end end.join('&') when Array data = params.map do |key, val| make_cgi_params_key_value(key, val) end.join('&') when String data = params.map do |str| URI.escape(str.strip) end.join('&') end when String data = URI.escape(params.strip) end return data end
# File lib/bio/command.rb, line 322 def make_cgi_params_key_value(key, value) result = [] case value when Array value.each do |val| result << [key, val].map {|x| URI.escape(x.to_s) }.join('=') end else result << [key, value].map {|x| URI.escape(x.to_s) }.join('=') end return result end
Generate command line string with special characters escaped.
# File lib/bio/command.rb, line 63 def make_command_line(ary) case RUBY_PLATFORM when /mswin32|bccwin32/ make_command_line_windows(ary) else make_command_line_unix(ary) end end
Generate command line string with special characters escaped for UNIX shells.
# File lib/bio/command.rb, line 80 def make_command_line_unix(ary) ary.collect { |str| escape_shell_unix(str) }.join(" ") end
Generate command line string with special characters escaped for cmd.exe on Windows.
# File lib/bio/command.rb, line 74 def make_command_line_windows(ary) ary.collect { |str| escape_shell_windows(str) }.join(" ") end
Same as:
Net::HTTP.new(address, port)
and it uses proxy if an environment variable (same as OpenURI.open_uri) is set.
# File lib/bio/command.rb, line 251 def new_http(address, port = 80) uri = URI.parse("http://#{address}:#{port}") # Note: URI#find_proxy is an unofficial method defined in open-uri.rb. # If the spec of open-uri.rb would be changed, we should change below. if proxyuri = uri.find_proxy then raise 'Non-HTTP proxy' if proxyuri.class != URI::HTTP Net::HTTP.new(address, port, proxyuri.host, proxyuri.port) else Net::HTTP.new(address, port) end end
Same as: Net::HTTP.post_form(uri, params) and it uses proxy if an environment variable (same as OpenURI.open_uri) is set. In addition, header can be set. (Note that Content-Type and Content-Length are automatically set by default.) uri must be a URI object, params must be a hash, and header must be a hash.
# File lib/bio/command.rb, line 274 def post_form(uri, params = nil, header = {}) unless uri.is_a?(URI) uri = URI.parse(uri) end data = make_cgi_params(params) hash = { 'Content-Type' => 'application/x-www-form-urlencoded', 'Content-Length' => data.length.to_s } hash.update(header) start_http(uri.host, uri.port) do |http| http.post(uri.path, data, hash) end end
Executes the program with the query (String) given to the standard input, waits the program termination, and returns the output data printed to the standard output as a string.
Automatically select popen for Windows environment and fork for the others.
# File lib/bio/command.rb, line 145 def query_command(cmd, query = nil) case RUBY_PLATFORM when /mswin32|bccwin32/ query_command_popen(cmd, query) else query_command_fork(cmd, query) end end
Executes the program with the query (String) given to the standard input, waits the program termination, and returns the output data printed to the standard output as a string.
Fork (by using IO.popen("-")) and exec is used to execute the program.
From the view point of security, this method is recommended rather than query_popen.
# File lib/bio/command.rb, line 177 def query_command_fork(cmd, query = nil) IO.popen("-", "r+") do |io| if io then # parent io.sync = true io.print query if query io.close_write io.read else # child begin Kernel.exec(*cmd) rescue Errno::ENOENT, Errno::EACCES Process.exit!(127) rescue Exception end Process.exit!(1) end end end
Executes the program via Open3.popen3 with the query (String) given to the stain, waits the program termination, and returns the data from stdout and stderr as an array of the strings.
From the view point of security, this method is recommended rather than exec_local_popen.
# File lib/bio/command.rb, line 204 def query_command_open3(cmd, query = nil) errorlog = nil cmd = cmd.collect { |x| x.to_s } Open3.popen3(*cmd) do |pin, pout, perr| perr.sync = true t = Thread.start { errorlog = perr.read } begin pin.print query if query pin.close output = pout.read ensure t.join end [ output, errorlog ] end end
Executes the program with the query (String) given to the standard input, waits the program termination, and returns the output data printed to the standard output as a string.
IO.popen is used for OS which doesn‘t support fork.
# File lib/bio/command.rb, line 159 def query_command_popen(cmd, query = nil) str = make_command_line(cmd) IO.popen(str, "w+") do |io| io.sync = true io.print query if query io.close_write io.read end end
Same as OpenURI.open_uri(uri).read.
# File lib/bio/command.rb, line 222 def read_uri(uri) OpenURI.open_uri(uri).read end
Same as:
Net::HTTP.start(address, port)
and it uses proxy if an environment variable (same as OpenURI.open_uri) is set.
# File lib/bio/command.rb, line 232 def start_http(address, port = 80, &block) uri = URI.parse("http://#{address}:#{port}") # Note: URI#find_proxy is an unofficial method defined in open-uri.rb. # If the spec of open-uri.rb would be changed, we should change below. if proxyuri = uri.find_proxy then raise 'Non-HTTP proxy' if proxyuri.class != URI::HTTP http = Net::HTTP.Proxy(proxyuri.host, proxyuri.port) else http = Net::HTTP end http.start(address, port, &block) end