module Syntropy::SideRun

def call(machine, &block)

def call(machine, &block)
  setup if !@queue
  # TODO: share mailboxes, acquire them with e.g. with_mailbox { |mbox| ... }
  mailbox = Thread.current[:fiber_mailbox] ||= UM::Queue.new
  machine.push(@queue, [mailbox, block])
  result = machine.shift(mailbox)
  result.is_a?(Exception) ? (raise result) : result
end

def handle_request(machine, queue)

def handle_request(machine, queue)
  response_mailbox, closure = machine.shift(queue)
  result = closure.call
  machine.push(response_mailbox, result)
rescue Exception => e
  machine.push(response_mailbox, e)
end

def setup

def setup
  @queue = UM::Queue.new
  count = (Etc.nprocessors - 1).clamp(2..6)
  @workers = count.times.map {
    Thread.new { side_run_worker(@queue) }
  }
end

def side_run_worker(queue)

def side_run_worker(queue)
  machine = UM.new
  loop { handle_request(machine, queue) }
rescue UM::Terminate
  # # We can also add a timeout here
  # t0 = Time.now
  # while !queue.empty? && (Time.now - t0) < 10
  #   handle_request(machine, queue)
  # end
end