How do I export an environment variable from within a Ruby script to the parent shell? For example, implementing a naïve implementation of the read
Bash builtin
What about in ruby printing out standard export code :
puts "export MYVAR=value"
and then using shell backtick to get it executed at shell comands:
$ `./myscript.rb`
this will take the output of the script and execute it, works in modern shells like bash and zsh
You can't export environment variables to the shell the ruby script runs in, but you could write a ruby script that creates a source-able bash file.
For example
% echo set_var.rb
#!/usr/bin/env ruby
varname = ARGV[0]
puts "#{varname}=#{STDIN.gets.chomp}"
% set_var.rb FOO
1
FOO=1
% set_var.rb BAR > temp.sh ; . temp.sh
2
% echo $BAR
2
%
Another alternative is that using ENV[]=
does set environment variables for subshells opened from within the ruby process. For example:
outer-bash% echo pass_var.rb
#!/usr/bin/env ruby
varname = ARGV[0]
ENV[varname] = STDIN.gets.chomp
exec '/usr/bin/env bash'
outer-bash% pass_var.rb BAZ
quux
inner-bash% echo $BAZ
quux
This can be quite potent if you combine it with the shell's exec
command, which will replace the outer-shell with the ruby process (so that when you exit the inner shell, the outer shell auto-exits as well, preventing any "I thought I set that variable in this shell" confusion).
# open terminal
% exec pass_var.rb BAZ
3
% echo $BAZ
3
% exit
# terminal closes
I just tried this and it looks good.
cmd = "echo \"FOO is \\\"$FOO\\\"\"";
system(cmd);
# Run some Ruby code (same program) in the child process
fork do
puts "In child process. parent pid is #$$"
ENV['FOO']='foo in sub process';
system(cmd);
exit 99
end
child_pid = Process.wait
puts "Child (pid #{child_pid}) terminated with status #{$?.exitstatus}"
system(cmd);
This seems to work well - at least on MacOSX
I get
FOO is ""
In child process. parent pid is 1388
FOO is "foo in sub process"
Child (pid 1388) terminated with status 99
FOO is ""
Seems nice in it restores prior state automatically
Ok - now tried a different one as this doesn't spawn 2 subprocesses
Use Process.spawn(env,command)
pid = Process.spawn({ 'FOO'=>'foo in spawned process'}, cmd );
pid = Process.wait();
This acts like the C system call and allows you to specify pipes and all that other stuff too.
Simple answer: You can't.
Longer answer: You can't, unless the operating environment provides hooks to do so. Most do not. The best you can usually do is print out the assignments you want done and have the parent execute them.