MongoDB Ruby 驱动程序 - “同步”:无法从陷阱上下文调用
MongoDB Ruby driver - `synchronize': can't be called from trap context
我正在尝试使用 Kernel#trap 捕获信号,并且 运行 是该上下文中的数据库请求,但出现上述错误。以前有人遇到过吗?有什么解决办法吗?
示例代码:
trap('HUP') do
$db[:db_name].update({_id: 123}, {:$set => {a: 1}})
end
loop { sleep 1 }
会报错:
/usr/local/lib/ruby/gems/2.1.0/gems/mongo-1.11.1/lib/mongo/connection/pool.rb:266:in `synchronize': can't be called from trap context (ThreadError)
当脚本接收到 HUP 信号时,可以通过 运行ning kill -HUP {pid}
发送。 $db 必须是 MongoDB 对象。
Ruby 不允许从陷阱上下文中进行互斥锁同步,大概是因为它会导致死锁(即,您处于同步上下文中,然后向进程发送信号并尝试重新同步,你陷入僵局)。您可以通过以下方式轻松重现:
# trap.rb
require 'thread'
mutex = Mutex.new
trap('HUP') { mutex.synchronize {} }
gets
# pkill -HUP -f trap.rb
trap.rb:3:in `synchronize': can't be called from trap context (ThreadError)
from trap.rb:3:in `block in <main>'
from trap.rb:4:in `call'
from trap.rb:4:in `gets'
from trap.rb:4:in `gets'
from trap.rb:4:in `<main>'
要解决这个问题,您的信号处理程序可能应该将一个作业排队等待另一个线程处理,或者您可以生成一个新线程并在其中执行您的工作:
# trap.rb
require 'thread'
mutex = Mutex.new
trap('HUP') do
Thread.new { mutex.synchronize { puts "hi!" } }
end
gets
# pkill -HUP -f trap.rb
hi!
我正在尝试使用 Kernel#trap 捕获信号,并且 运行 是该上下文中的数据库请求,但出现上述错误。以前有人遇到过吗?有什么解决办法吗?
示例代码:
trap('HUP') do
$db[:db_name].update({_id: 123}, {:$set => {a: 1}})
end
loop { sleep 1 }
会报错:
/usr/local/lib/ruby/gems/2.1.0/gems/mongo-1.11.1/lib/mongo/connection/pool.rb:266:in `synchronize': can't be called from trap context (ThreadError)
当脚本接收到 HUP 信号时,可以通过 运行ning kill -HUP {pid}
发送。 $db 必须是 MongoDB 对象。
Ruby 不允许从陷阱上下文中进行互斥锁同步,大概是因为它会导致死锁(即,您处于同步上下文中,然后向进程发送信号并尝试重新同步,你陷入僵局)。您可以通过以下方式轻松重现:
# trap.rb
require 'thread'
mutex = Mutex.new
trap('HUP') { mutex.synchronize {} }
gets
# pkill -HUP -f trap.rb
trap.rb:3:in `synchronize': can't be called from trap context (ThreadError)
from trap.rb:3:in `block in <main>'
from trap.rb:4:in `call'
from trap.rb:4:in `gets'
from trap.rb:4:in `gets'
from trap.rb:4:in `<main>'
要解决这个问题,您的信号处理程序可能应该将一个作业排队等待另一个线程处理,或者您可以生成一个新线程并在其中执行您的工作:
# trap.rb
require 'thread'
mutex = Mutex.new
trap('HUP') do
Thread.new { mutex.synchronize { puts "hi!" } }
end
gets
# pkill -HUP -f trap.rb
hi!