Method: PTY.spawn
- Defined in:
- pty.c
.spawn([env,]) {|r, w, pid| ... } ⇒ Object .spawn([env,]) ⇒ Array .spawn([env,], arguments, ...) {|r, w, pid| ... } ⇒ Object .spawn([env,], arguments, ...) ⇒ Array
Spawns the specified command on a newly allocated pty. You can also use the alias ::getpty.
The command’s controlling tty is set to the slave device of the pty and its standard input/output/error is redirected to the slave device.
env is an optional hash that provides additional environment variables to the spawned pty.
# sets FOO to "bar"
PTY.spawn({"FOO"=>"bar"}, "printenv", "FOO") do |r, w, pid|
p r.read #=> "bar\r\n"
ensure
r.close; w.close; Process.wait(pid)
end
# unsets FOO
PTY.spawn({"FOO"=>nil}, "printenv", "FOO") do |r, w, pid|
p r.read #=> ""
ensure
r.close; w.close; Process.wait(pid)
end
command and command_line are the full commands to run, given a String. Any additional arguments will be passed to the command.
Return values
In the non-block form this returns an array of size three, [r, w, pid].
In the block form these same values will be yielded to the block:
r-
A readable IO that contains the command’s standard output and standard error
w-
A writable IO that is the command’s standard input
pid-
The process identifier for the command.
Clean up
This method does not clean up like closing IOs or waiting for child process, except that the process is detached in the block form to prevent it from becoming a zombie (see Process.detach). Any other cleanup is the responsibility of the caller. If waiting for pid, be sure to close both r and w before doing so; doing it in the reverse order may cause deadlock on some OSes.
649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 |
# File 'pty.c', line 649 static VALUE pty_getpty(int argc, VALUE *argv, VALUE self) { VALUE res; struct pty_info info; char SlaveName[DEVICELEN]; establishShell(argc, argv, &info, SlaveName); VALUE pty_path = rb_obj_freeze(rb_str_new_cstr(SlaveName)); VALUE rport = rb_io_open_descriptor( rb_cFile, info.fd, FMODE_READABLE, pty_path, RUBY_IO_TIMEOUT_DEFAULT, NULL ); int wpty_fd = rb_cloexec_dup(info.fd); if (wpty_fd == -1) { rb_sys_fail("dup()"); } VALUE wport = rb_io_open_descriptor( rb_cFile, wpty_fd, FMODE_WRITABLE | FMODE_TRUNC | FMODE_CREATE | FMODE_SYNC, pty_path, RUBY_IO_TIMEOUT_DEFAULT, NULL ); res = rb_ary_new2(3); rb_ary_store(res, 0, rport); rb_ary_store(res, 1, wport); rb_ary_store(res,2,PIDT2NUM(info.child_pid)); if (rb_block_given_p()) { rb_ensure(rb_yield, res, pty_detach_process, (VALUE)&info); return Qnil; } return res; } |