class GraphQL::Dataloader::AsyncDataloader
def run
def run job_fibers = [] next_job_fibers = [] source_tasks = [] next_source_tasks = [] first_pass = true sources_condition = Async::Condition.new manager = spawn_fiber do while first_pass || job_fibers.any? first_pass = false while (f = (job_fibers.shift || spawn_job_fiber)) if f.alive? finished = run_fiber(f) if !finished next_job_fibers << f end end end job_fibers.concat(next_job_fibers) next_job_fibers.clear Sync do |root_task| while source_tasks.any? || @source_cache.each_value.any? { |group_sources| group_sources.each_value.any?(&:pending?) } while (task = source_tasks.shift || spawn_source_task(root_task, sources_condition)) if task.alive? root_task.yield # give the source task a chance to run next_source_tasks << task end end sources_condition.signal source_tasks.concat(next_source_tasks) next_source_tasks.clear end end end end manager.resume if manager.alive? raise "Invariant: Manager didn't terminate successfully: #{manager}" end rescue UncaughtThrowError => e throw e.tag, e.value end
def spawn_source_task(parent_task, condition)
def spawn_source_task(parent_task, condition) pending_sources = nil @source_cache.each_value do |source_by_batch_params| source_by_batch_params.each_value do |source| if source.pending? pending_sources ||= [] pending_sources << source end end end if pending_sources fiber_vars = get_fiber_variables parent_task.async do set_fiber_variables(fiber_vars) Thread.current[:graphql_dataloader_next_tick] = condition pending_sources.each(&:run_pending_keys) end end end
def yield
def yield if (condition = Thread.current[:graphql_dataloader_next_tick]) condition.wait else Fiber.yield end nil end