From a5ea68f80aace8b132377f18c01c8d8f3990bdd1 Mon Sep 17 00:00:00 2001 From: Teppei Shintani Date: Tue, 17 Sep 2024 21:11:02 +0900 Subject: [PATCH 1/2] stdlib: Add types for Open3 ref. https://docs.ruby-lang.org/ja/latest/class/Open3.html#M_CAPTURE2E --- stdlib/open3/0/open3.rbs | 155 ++++++++++++++++++++++++++++++++++++++ test/stdlib/Open3_test.rb | 17 +++++ 2 files changed, 172 insertions(+) create mode 100644 stdlib/open3/0/open3.rbs create mode 100644 test/stdlib/Open3_test.rb diff --git a/stdlib/open3/0/open3.rbs b/stdlib/open3/0/open3.rbs new file mode 100644 index 000000000..5b082393f --- /dev/null +++ b/stdlib/open3/0/open3.rbs @@ -0,0 +1,155 @@ +# +# Module Open3 supports creating child processes with access to their $stdin, +# $stdout, and $stderr streams. +# +# ## What's Here +# +# Each of these methods executes a given command in a new process or subshell, +# or multiple commands in new processes and/or subshells: +# +# * Each of these methods executes a single command in a process or subshell, +# accepts a string for input to $stdin, and returns string output from +# $stdout, $stderr, or both: +# +# * Open3.capture2: Executes the command; returns the string from $stdout. +# * Open3.capture2e: Executes the command; returns the string from merged +# $stdout and $stderr. +# * Open3.capture3: Executes the command; returns strings from $stdout and +# $stderr. +# +# +# * Each of these methods executes a single command in a process or subshell, +# and returns pipes for $stdin, $stdout, and/or $stderr: +# +# * Open3.popen2: Executes the command; returns pipes for $stdin and +# $stdout. +# * Open3.popen2e: Executes the command; returns pipes for $stdin and +# merged $stdout and $stderr. +# * Open3.popen3: Executes the command; returns pipes for $stdin, $stdout, +# and $stderr. +# +# +# * Each of these methods executes one or more commands in processes and/or +# subshells, returns pipes for the first $stdin, the last $stdout, or both: +# +# * Open3.pipeline_r: Returns a pipe for the last $stdout. +# * Open3.pipeline_rw: Returns pipes for the first $stdin and the last +# $stdout. +# * Open3.pipeline_w: Returns a pipe for the first $stdin. +# * Open3.pipeline_start: Does not wait for processes to complete. +# * Open3.pipeline: Waits for processes to complete. +# +# +# +# Each of the methods above accepts: +# +# * An optional hash of environment variable names and values; see [Execution +# Environment](rdoc-ref:Process@Execution+Environment). +# * A required string argument that is a `command_line` or `exe_path`; see +# [Argument command_line or +# exe_path](rdoc-ref:Process@Argument+command_line+or+exe_path). +# * An optional hash of execution options; see [Execution +# Options](rdoc-ref:Process@Execution+Options). +# +module Open3 + # + # Basically a wrapper for Open3.popen3 that: + # + # * Creates a child process, by calling Open3.popen3 with the given arguments + # (except for certain entries in hash `options`; see below). + # * Returns as string `stdout_and_stderr_s` the merged standard output and + # standard error of the child process. + # * Returns as `status` a `Process::Status` object that represents the exit + # status of the child process. + # + # + # Returns the array `[stdout_and_stderr_s, status]`: + # + # stdout_and_stderr_s, status = Open3.capture2e('echo "Foo"') + # # => ["Foo\n", #] + # + # Like Process.spawn, this method has potential security vulnerabilities if + # called with untrusted input; see [Command + # Injection](rdoc-ref:command_injection.rdoc@Command+Injection). + # + # Unlike Process.spawn, this method waits for the child process to exit before + # returning, so the caller need not do so. + # + # If the first argument is a hash, it becomes leading argument `env` in the call + # to Open3.popen3; see [Execution + # Environment](rdoc-ref:Process@Execution+Environment). + # + # If the last argument is a hash, it becomes trailing argument `options` in the + # call to Open3.popen3; see [Execution + # Options](rdoc-ref:Process@Execution+Options). + # + # The hash `options` is given; two options have local effect in method + # Open3.capture2e: + # + # * If entry `options[:stdin_data]` exists, the entry is removed and its + # string value is sent to the command's standard input: + # + # Open3.capture2e('tee', stdin_data: 'Foo') + # # => ["Foo", #] + # + # * If entry `options[:binmode]` exists, the entry is removed and the internal + # streams are set to binary mode. + # + # + # The single required argument is one of the following: + # + # * `command_line` if it is a string, and if it begins with a shell reserved + # word or special built-in, or if it contains one or more metacharacters. + # * `exe_path` otherwise. + # + # + # **Argument `command_line`** + # + # String argument `command_line` is a command line to be passed to a shell; it + # must begin with a shell reserved word, begin with a special built-in, or + # contain meta characters: + # + # Open3.capture2e('if true; then echo "Foo"; fi') # Shell reserved word. + # # => ["Foo\n", #] + # Open3.capture2e('echo') # Built-in. + # # => ["\n", #] + # Open3.capture2e('date > date.tmp') # Contains meta character. + # # => ["", #] + # + # The command line may also contain arguments and options for the command: + # + # Open3.capture2e('echo "Foo"') + # # => ["Foo\n", #] + # + # **Argument `exe_path`** + # + # Argument `exe_path` is one of the following: + # + # * The string path to an executable to be called. + # * A 2-element array containing the path to an executable and the string to + # be used as the name of the executing process. + # + # + # Example: + # + # Open3.capture2e('/usr/bin/date') + # # => ["Sat Sep 30 09:01:46 AM CDT 2023\n", #] + # + # Ruby invokes the executable directly, with no shell and no shell expansion: + # + # Open3.capture2e('doesnt_exist') # Raises Errno::ENOENT + # + # If one or more `args` is given, each is an argument or option to be passed to + # the executable: + # + # Open3.capture2e('echo', 'C #') + # # => ["C #\n", #] + # Open3.capture2e('echo', 'hello', 'world') + # # => ["hello world\n", #] + # + def self.capture2e: (*String, ?stdin_data: String, ?binmode: boolish) -> [String, Process::Status] +end diff --git a/test/stdlib/Open3_test.rb b/test/stdlib/Open3_test.rb new file mode 100644 index 000000000..f1578eabc --- /dev/null +++ b/test/stdlib/Open3_test.rb @@ -0,0 +1,17 @@ +require_relative "test_helper" + +class Open3SingletonTest < Test::Unit::TestCase + include TestHelper + + library "open3" + testing "singleton(::Open3)" + + def test_capture2e + assert_send_type "(*::String) -> [ ::String, ::Process::Status ]", + Open3, :capture2e, 'echo "Foo"' + assert_send_type "(*::String, binmode: boolish) -> [ ::String, ::Process::Status ]", + Open3, :capture2e, 'echo "Foo"', binmode: true + assert_send_type "(*::String, stdin_data: ::String) -> [ ::String, ::Process::Status ]", + Open3, :capture2e, 'ruby -e "puts STDIN.read"', stdin_data: 'Foo' + end +end From 68f734366a4735dd6e4c18d72cc89dcc2dc23b00 Mon Sep 17 00:00:00 2001 From: Soutaro Matsumoto Date: Mon, 30 Sep 2024 16:23:30 +0900 Subject: [PATCH 2/2] Use `RUBY_EXECUTABLE` for Ruby CI --- test/stdlib/Open3_test.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/stdlib/Open3_test.rb b/test/stdlib/Open3_test.rb index f1578eabc..d9e79efec 100644 --- a/test/stdlib/Open3_test.rb +++ b/test/stdlib/Open3_test.rb @@ -12,6 +12,6 @@ def test_capture2e assert_send_type "(*::String, binmode: boolish) -> [ ::String, ::Process::Status ]", Open3, :capture2e, 'echo "Foo"', binmode: true assert_send_type "(*::String, stdin_data: ::String) -> [ ::String, ::Process::Status ]", - Open3, :capture2e, 'ruby -e "puts STDIN.read"', stdin_data: 'Foo' + Open3, :capture2e, "#{RUBY_EXECUTABLE} -e 'puts STDIN.read'", stdin_data: 'Foo' end end