SUMMARY:
When a signal interrupt is trapped, we can no longer call #close on GServer without it throwing a ThreadError.
STEPS TO REPEAT:
Run the following code:
require 'gserver'
server = GServer.new 8080
server.start
trap("SIGINT") { server.stop }
gets # or any command that keeps the process running
Hit CTRL+C or whichever command will send the interrupt signal to this program.
WHAT I EXPECTED: In version 1.9.3, CTRL+C sends an interrupt signal and the program exits cleanly.
WHAT HAPPENED: When running the version from trunk the following stack trace is thrown.
^C/home/joe/.rvm/rubies/ruby-head/lib/ruby/2.0.0/gserver.rb:116:in synchronize': can't be called from trap context (ThreadError) from /home/joe/.rvm/rubies/ruby-head/lib/ruby/2.0.0/gserver.rb:116:in stop'
from gserver_bug.rb:5:in block in <main>' from gserver_bug.rb:6:in call'
from gserver_bug.rb:6:in gets' from gserver_bug.rb:6:in gets'
from gserver_bug.rb:6:in `'
/home/joe/lib/lib/ruby/2.0.0/gserver.rb:116:in synchronize': can't be called from trap context (ThreadError) from /home/joe/lib/lib/ruby/2.0.0/gserver.rb:116:in stop'
from /home/joe/dev/bane/lib/bane/launcher.rb:19:in block in stop' from /home/joe/dev/bane/lib/bane/launcher.rb:19:in each'
from /home/joe/dev/bane/lib/bane/launcher.rb:19:in stop' from ./bin/bane:22:in block in '
from /home/joe/lib/lib/ruby/2.0.0/gserver.rb:140:in call' from /home/joe/lib/lib/ruby/2.0.0/gserver.rb:140:in join'
from /home/joe/lib/lib/ruby/2.0.0/gserver.rb:140:in join' from /home/joe/dev/bane/lib/bane/launcher.rb:15:in block in join'
from /home/joe/dev/bane/lib/bane/launcher.rb:15:in each' from /home/joe/dev/bane/lib/bane/launcher.rb:15:in join'
from ./bin/bane:23:in `'
Holding mutex in trap is deadlockable. It is what ruby complained. The best workaround is to make new thread in trap handler and stop gserver asynchnorously, I think.