lbry-sdk/lbry/extras/daemon/componentmanager.py

172 lines
5.9 KiB
Python
Raw Normal View History

import logging
2019-01-22 17:44:17 -05:00
import asyncio
2019-06-20 20:55:47 -04:00
from lbry.conf import Config
from lbry.error import ComponentStartConditionNotMetError
2019-06-20 20:55:47 -04:00
from lbry.dht.peer import PeerManager
log = logging.getLogger(__name__)
2018-07-25 01:23:02 -04:00
class RegisteredConditions:
conditions = {}
class RequiredConditionType(type):
def __new__(mcs, name, bases, newattrs):
klass = type.__new__(mcs, name, bases, newattrs)
if name != "RequiredCondition":
if klass.name in RegisteredConditions.conditions:
raise SyntaxError("already have a component registered for \"%s\"" % klass.name)
RegisteredConditions.conditions[klass.name] = klass
return klass
class RequiredCondition(metaclass=RequiredConditionType):
name = ""
component = ""
message = ""
@staticmethod
def evaluate(component):
raise NotImplementedError()
2018-07-25 01:23:02 -04:00
class ComponentManager:
default_component_classes = {}
2019-01-22 17:44:17 -05:00
def __init__(self, conf: Config, analytics_manager=None, skip_components=None,
peer_manager=None, **override_components):
2019-01-21 15:55:50 -05:00
self.conf = conf
self.skip_components = skip_components or []
2019-01-22 17:44:17 -05:00
self.loop = asyncio.get_event_loop()
self.analytics_manager = analytics_manager
self.component_classes = {}
self.components = set()
2019-01-22 17:44:17 -05:00
self.started = asyncio.Event(loop=self.loop)
self.peer_manager = peer_manager or PeerManager(asyncio.get_event_loop_policy().get_event_loop())
2018-07-06 01:17:20 -04:00
for component_name, component_class in self.default_component_classes.items():
if component_name in override_components:
component_class = override_components.pop(component_name)
if component_name not in self.skip_components:
self.component_classes[component_name] = component_class
if override_components:
raise SyntaxError("unexpected components: %s" % override_components)
2018-07-06 01:17:20 -04:00
for component_class in self.component_classes.values():
self.components.add(component_class(self))
def evaluate_condition(self, condition_name):
if condition_name not in RegisteredConditions.conditions:
raise NameError(condition_name)
condition = RegisteredConditions.conditions[condition_name]
try:
component = self.get_component(condition.component)
2018-10-16 15:04:20 -04:00
result = condition.evaluate(component)
2018-11-04 01:24:41 -05:00
except Exception:
log.exception('failed to evaluate condition:')
result = False
2018-10-16 15:04:20 -04:00
return result, "" if result else condition.message
def sort_components(self, reverse=False):
"""
Sort components by requirements
"""
steps = []
staged = set()
components = set(self.components)
# components with no requirements
step = []
for component in set(components):
if not component.depends_on:
step.append(component)
staged.add(component.component_name)
components.remove(component)
if step:
step.sort()
steps.append(step)
while components:
step = []
to_stage = set()
for component in set(components):
reqs_met = 0
for needed in component.depends_on:
if needed in staged:
reqs_met += 1
if reqs_met == len(component.depends_on):
step.append(component)
to_stage.add(component.component_name)
components.remove(component)
if step:
step.sort()
staged.update(to_stage)
steps.append(step)
elif components:
raise ComponentStartConditionNotMetError(components)
if reverse:
steps.reverse()
return steps
2019-01-23 20:26:29 -05:00
async def start(self):
2018-12-15 15:31:02 -05:00
""" Start Components in sequence sorted by requirements """
2019-01-23 20:26:29 -05:00
for stage in self.sort_components():
2018-12-15 15:31:02 -05:00
needing_start = [
2019-01-23 20:26:29 -05:00
component._setup() for component in stage if not component.running
2018-12-15 15:31:02 -05:00
]
if needing_start:
await asyncio.wait(needing_start)
2019-01-22 17:44:17 -05:00
self.started.set()
2018-12-15 15:31:02 -05:00
async def stop(self):
"""
Stop Components in reversed startup order
"""
stages = self.sort_components(reverse=True)
for stage in stages:
2018-12-15 15:31:02 -05:00
needing_stop = [
component._stop() for component in stage if component.running
]
if needing_stop:
2019-01-23 20:26:29 -05:00
await asyncio.wait(needing_stop)
def all_components_running(self, *component_names):
"""
2018-04-02 16:49:48 -04:00
Check if components are running
:return: (bool) True if all specified components are running
"""
components = {component.component_name: component for component in self.components}
for component in component_names:
2018-04-02 16:49:48 -04:00
if component not in components:
raise NameError("%s is not a known Component" % component)
2018-04-02 16:49:48 -04:00
if not components[component].running:
return False
return True
def get_components_status(self):
"""
List status of all the components, whether they are running or not
:return: (dict) {(str) component_name: (bool) True is running else False}
"""
return {
component.component_name: component.running
for component in self.components
}
def get_actual_component(self, component_name):
for component in self.components:
if component.component_name == component_name:
return component
raise NameError(component_name)
2019-07-12 19:54:04 -03:00
def get_component(self, component_name):
return self.get_actual_component(component_name).component
2019-07-12 19:54:04 -03:00
def has_component(self, component_name):
2019-07-13 01:58:57 -03:00
return any(component for component in self.components if component_name == component.component_name)