1
1
mirror of https://github.com/NixOS/mobile-nixos.git synced 2024-12-18 05:21:47 +03:00
mobile-nixos/boot/init/lib/task.rb

121 lines
2.9 KiB
Ruby
Raw Normal View History

2019-12-23 06:32:34 +03:00
# Namespace where tasks can be defined, and hosting methods harmonizing a run.
module Tasks
# Register a singleton task to be instantiated and ran.
# @internal
def self.register_singleton(klass)
$logger.debug("Task #{klass.name} registered...")
@singletons_to_be_instantiated ||= []
@singletons_to_be_instantiated << klass
end
# Register one task to be ran.
def self.register(task)
$logger.debug("Task #{task.name} registered...")
@tasks ||= []
@tasks << task
end
# Tries to run *all* tasks.
def self.go()
# Registers tasks that still needs to be instantiated
@singletons_to_be_instantiated.each do |klass|
# Their mere existence registers them.
klass.instance
2019-12-23 06:32:34 +03:00
end
@singletons_to_be_instantiated = []
# Sort tasks to reduce the amount of loops it needs to fulfill them all.
# It's only a reduction due to files, mounts and devices being
# unpredictable!
@tasks.sort!
2019-12-23 06:32:34 +03:00
until @tasks.all?(&:ran) do
2019-12-23 22:46:39 +03:00
$logger.debug("Tasks resolution loop start")
2019-12-23 06:32:34 +03:00
@tasks
.reject(&:ran)
.each do |task|
task._try_run_task
end
# Don't burn the CPU
sleep(0.1)
end
end
end
# Basic task class.
class Task
attr_reader :ran
def self.new(*args)
$logger.debug("New instance of #{self.name}...")
$logger.debug(" -> #{args.inspect}")
instance = super(*args)
Tasks.register(instance)
instance
end
def name
self.class.name
end
def self.inherited(subclass)
$logger.debug("#{subclass.name} created...")
end
# Sort first by dependencies, then by name, then by object_id
# (for stable sort order)
def <=>(other)
return -1 if other.depends_on?(self)
return 1 if depends_on?(other)
by_name = name <=> other.name
return by_name unless by_name == 0
object_id <=> other.object_id
end
def depends_on?(other)
dependencies.any? do |dependency|
dependency.depends_on?(other)
end
end
2019-12-23 06:32:34 +03:00
def add_dependency(kind, *args)
raise NameError.new("No dependency named #{kind}") unless Dependencies.constants.include?(kind.to_sym)
2019-12-23 06:32:34 +03:00
dependencies << Dependencies.const_get(kind.to_sym).new(*args)
end
2019-12-23 22:46:06 +03:00
def dependencies_fulfilled?()
dependencies.all?(&:fulfilled?)
2019-12-23 06:32:34 +03:00
end
# Internal actual way to run the task
# This runs the `#run` method.
def _try_run_task()
$logger.debug("Looking to run task #{name}...")
2019-12-23 22:46:06 +03:00
return unless dependencies_fulfilled?
2019-12-23 06:32:34 +03:00
unless @ran
$logger.info("Running #{name}...")
run()
2019-12-23 22:46:39 +03:00
$logger.debug("Finished #{name}...")
2019-12-23 06:32:34 +03:00
@ran = true
end
end
def dependencies()
@dependencies ||= []
@dependencies
end
end
# A task that can only have one instance.
class SingletonTask < Task
include Singleton
def self.inherited(subclass)
super
# Delay initializing, as right now we have an fresh new empty class.
Tasks.register_singleton(subclass)
end
end