2014-09-03 22:28:28 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2007-10-20 22:26:43 +00:00
|
|
|
#
|
2010-03-20 19:11:32 -07:00
|
|
|
# Copyright (C) 2007-2010 Andrew Resch <andrewresch@gmail.com>
|
2008-11-17 00:57:32 +00:00
|
|
|
#
|
2014-09-03 22:28:28 +01:00
|
|
|
# This file is part of Deluge and is licensed under GNU General Public License 3.0, or later, with
|
|
|
|
# the additional special exception to link portions of this program with the OpenSSL library.
|
|
|
|
# See LICENSE for more details.
|
2007-10-20 22:26:43 +00:00
|
|
|
#
|
2008-11-23 04:58:01 +00:00
|
|
|
|
2010-12-06 11:20:22 +00:00
|
|
|
import logging
|
2012-12-07 19:00:43 -05:00
|
|
|
from collections import defaultdict
|
2014-09-03 18:18:29 +01:00
|
|
|
|
|
|
|
from twisted.internet.defer import DeferredList, fail, maybeDeferred, succeed
|
2009-01-28 17:59:27 +00:00
|
|
|
from twisted.internet.task import LoopingCall
|
2010-12-06 11:20:22 +00:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
2007-10-20 22:26:43 +00:00
|
|
|
|
2014-09-03 22:28:28 +01:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
class ComponentAlreadyRegistered(Exception):
|
|
|
|
pass
|
2007-10-20 22:26:43 +00:00
|
|
|
|
2014-09-03 22:28:28 +01:00
|
|
|
|
2008-12-20 04:27:59 +00:00
|
|
|
class Component(object):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
2010-03-25 12:01:40 -07:00
|
|
|
Component objects are singletons managed by the :class:`ComponentRegistry`.
|
2010-03-20 19:11:32 -07:00
|
|
|
When a new Component object is instantiated, it will be automatically
|
2010-03-25 12:01:40 -07:00
|
|
|
registered with the :class:`ComponentRegistry`.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
The ComponentRegistry has the ability to start, stop, pause and shutdown the
|
|
|
|
components registered with it.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-25 12:01:40 -07:00
|
|
|
**Events:**
|
2009-02-27 17:12:53 +00:00
|
|
|
|
2010-03-25 12:01:40 -07:00
|
|
|
**start()** - This method is called when the client has connected to a
|
2010-03-20 19:11:32 -07:00
|
|
|
Deluge core.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-25 12:01:40 -07:00
|
|
|
**stop()** - This method is called when the client has disconnected from a
|
2010-03-20 19:11:32 -07:00
|
|
|
Deluge core.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-25 12:01:40 -07:00
|
|
|
**update()** - This method is called every 1 second by default while the
|
2010-03-25 14:43:15 -07:00
|
|
|
Componented is in a *Started* state. The interval can be
|
2010-03-20 19:11:32 -07:00
|
|
|
specified during instantiation. The update() timer can be
|
2010-03-25 12:01:40 -07:00
|
|
|
paused by instructing the :class:`ComponentRegistry` to pause
|
2010-03-20 19:11:32 -07:00
|
|
|
this Component.
|
2007-10-20 22:26:43 +00:00
|
|
|
|
2010-03-25 12:01:40 -07:00
|
|
|
**shutdown()** - This method is called when the client is exiting. If the
|
2010-03-20 19:11:32 -07:00
|
|
|
Component is in a "Started" state when this is called, a
|
|
|
|
call to stop() will be issued prior to shutdown().
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
**States:**
|
2010-03-27 20:05:20 -07:00
|
|
|
|
|
|
|
A Component can be in one of these 5 states.
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
**Started** - The Component has been started by the :class:`ComponentRegistry`
|
|
|
|
and will have it's update timer started.
|
2010-03-27 20:05:20 -07:00
|
|
|
|
|
|
|
**Starting** - The Component has had it's start method called, but it hasn't
|
|
|
|
fully started yet.
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
**Stopped** - The Component has either been stopped or has yet to be started.
|
2010-03-27 20:05:20 -07:00
|
|
|
|
|
|
|
**Stopping** - The Component has had it's stop method called, but it hasn't
|
|
|
|
fully stopped yet.
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
**Paused** - The Component has had it's update timer stopped, but will
|
|
|
|
still be considered in a Started state.
|
2010-03-27 20:05:20 -07:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
def __init__(self, name, interval=1, depend=None):
|
|
|
|
self._component_name = name
|
|
|
|
self._component_interval = interval
|
|
|
|
self._component_depend = depend
|
|
|
|
self._component_state = "Stopped"
|
|
|
|
self._component_timer = None
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_starting_deferred = None
|
|
|
|
self._component_stopping_deferred = None
|
2010-03-20 19:11:32 -07:00
|
|
|
_ComponentRegistry.register(self)
|
|
|
|
|
2011-06-02 11:53:34 -07:00
|
|
|
def __del__(self):
|
2011-07-05 17:58:18 +10:00
|
|
|
if _ComponentRegistry:
|
|
|
|
_ComponentRegistry.deregister(self)
|
2012-02-28 13:34:51 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
def _component_start_timer(self):
|
|
|
|
if hasattr(self, "update"):
|
|
|
|
self._component_timer = LoopingCall(self.update)
|
|
|
|
self._component_timer.start(self._component_interval)
|
|
|
|
|
|
|
|
def _component_start(self):
|
|
|
|
def on_start(result):
|
|
|
|
self._component_state = "Started"
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_starting_deferred = None
|
2010-03-20 19:11:32 -07:00
|
|
|
self._component_start_timer()
|
|
|
|
return True
|
|
|
|
|
2010-04-08 17:16:28 -07:00
|
|
|
def on_start_fail(result):
|
|
|
|
self._component_state = "Stopped"
|
|
|
|
self._component_starting_deferred = None
|
|
|
|
log.error(result)
|
|
|
|
return result
|
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
if self._component_state == "Stopped":
|
|
|
|
if hasattr(self, "start"):
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_state = "Starting"
|
2010-03-20 19:11:32 -07:00
|
|
|
d = maybeDeferred(self.start)
|
|
|
|
d.addCallback(on_start)
|
2010-04-08 17:16:28 -07:00
|
|
|
d.addErrback(on_start_fail)
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_starting_deferred = d
|
2010-03-20 19:11:32 -07:00
|
|
|
else:
|
|
|
|
d = maybeDeferred(on_start, None)
|
2010-03-27 20:05:20 -07:00
|
|
|
elif self._component_state == "Starting":
|
|
|
|
return self._component_starting_deferred
|
2010-03-20 19:11:32 -07:00
|
|
|
elif self._component_state == "Started":
|
|
|
|
d = succeed(True)
|
|
|
|
else:
|
|
|
|
d = fail("Cannot start a component not in a Stopped state!")
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _component_stop(self):
|
|
|
|
def on_stop(result):
|
|
|
|
self._component_state = "Stopped"
|
|
|
|
if self._component_timer and self._component_timer.running:
|
|
|
|
self._component_timer.stop()
|
|
|
|
return True
|
|
|
|
|
2011-06-02 11:53:34 -07:00
|
|
|
def on_stop_fail(result):
|
|
|
|
self._component_state = "Started"
|
|
|
|
self._component_stopping_deferred = None
|
|
|
|
log.error(result)
|
|
|
|
return result
|
2012-02-28 13:34:51 +00:00
|
|
|
|
2010-03-27 20:05:20 -07:00
|
|
|
if self._component_state != "Stopped" and self._component_state != "Stopping":
|
2010-03-20 19:11:32 -07:00
|
|
|
if hasattr(self, "stop"):
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_state = "Stopping"
|
2010-03-20 19:11:32 -07:00
|
|
|
d = maybeDeferred(self.stop)
|
|
|
|
d.addCallback(on_stop)
|
2011-06-02 11:53:34 -07:00
|
|
|
d.addErrback(on_stop_fail)
|
2010-03-27 20:05:20 -07:00
|
|
|
self._component_stopping_deferred = d
|
2010-03-20 19:11:32 -07:00
|
|
|
else:
|
|
|
|
d = maybeDeferred(on_stop, None)
|
|
|
|
|
2010-03-27 20:05:20 -07:00
|
|
|
if self._component_state == "Stopping":
|
|
|
|
return self._component_stopping_deferred
|
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
return succeed(None)
|
|
|
|
|
|
|
|
def _component_pause(self):
|
|
|
|
def on_pause(result):
|
|
|
|
self._component_state = "Paused"
|
|
|
|
|
|
|
|
if self._component_state == "Started":
|
|
|
|
if self._component_timer and self._component_timer.running:
|
|
|
|
d = maybeDeferred(self._component_timer.stop)
|
|
|
|
d.addCallback(on_pause)
|
|
|
|
else:
|
|
|
|
d = succeed(None)
|
|
|
|
elif self._component_state == "Paused":
|
|
|
|
d = succeed(None)
|
|
|
|
else:
|
|
|
|
d = fail("Cannot pause a component in a non-Started state!")
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _component_resume(self):
|
|
|
|
def on_resume(result):
|
|
|
|
self._component_state = "Started"
|
|
|
|
|
|
|
|
if self._component_state == "Paused":
|
|
|
|
d = maybeDeferred(self._component_start_timer)
|
|
|
|
d.addCallback(on_resume)
|
|
|
|
else:
|
|
|
|
d = fail("Component cannot be resumed from a non-Paused state!")
|
|
|
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
def _component_shutdown(self):
|
|
|
|
def on_stop(result):
|
|
|
|
if hasattr(self, "shutdown"):
|
|
|
|
return maybeDeferred(self.shutdown)
|
|
|
|
return succeed(None)
|
|
|
|
|
|
|
|
d = self._component_stop()
|
|
|
|
d.addCallback(on_stop)
|
|
|
|
return d
|
|
|
|
|
2012-02-28 13:34:51 +00:00
|
|
|
def start(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def shutdown(self):
|
|
|
|
pass
|
|
|
|
|
2014-09-03 22:28:28 +01:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
class ComponentRegistry(object):
|
|
|
|
"""
|
|
|
|
The ComponentRegistry holds a list of currently registered
|
2010-03-25 12:01:40 -07:00
|
|
|
:class:`Component` objects. It is used to manage the Components by
|
2010-03-20 19:11:32 -07:00
|
|
|
starting, stopping, pausing and shutting them down.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.components = {}
|
2012-12-07 19:00:43 -05:00
|
|
|
# Stores all of the components that are dependent on a particular component
|
|
|
|
self.dependents = defaultdict(list)
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
def register(self, obj):
|
|
|
|
"""
|
|
|
|
Registers a component object with the registry. This is done
|
|
|
|
automatically when a Component object is instantiated.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
:param obj: the Component object
|
|
|
|
:type obj: object
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
:raises ComponentAlreadyRegistered: if a component with the same name is already registered.
|
2010-03-20 19:11:32 -07:00
|
|
|
|
|
|
|
"""
|
|
|
|
name = obj._component_name
|
|
|
|
if name in self.components:
|
|
|
|
raise ComponentAlreadyRegistered(
|
|
|
|
"Component already registered with name %s" % name)
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
self.components[obj._component_name] = obj
|
2012-12-07 19:00:43 -05:00
|
|
|
if obj._component_depend:
|
|
|
|
for depend in obj._component_depend:
|
|
|
|
self.dependents[depend].append(name)
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2011-06-06 14:19:51 -07:00
|
|
|
def deregister(self, obj):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Deregisters a component from the registry. A stop will be
|
|
|
|
issued to the component prior to deregistering it.
|
|
|
|
|
2011-06-06 14:19:51 -07:00
|
|
|
:param obj: the Component object
|
|
|
|
:type obj: object
|
2010-03-20 19:11:32 -07:00
|
|
|
|
|
|
|
"""
|
|
|
|
|
2011-06-06 14:19:51 -07:00
|
|
|
if obj in self.components.values():
|
|
|
|
log.debug("Deregistering Component: %s", obj._component_name)
|
|
|
|
d = self.stop([obj._component_name])
|
2014-09-03 22:28:28 +01:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
def on_stop(result, name):
|
|
|
|
del self.components[name]
|
2011-06-06 14:19:51 -07:00
|
|
|
return d.addCallback(on_stop, obj._component_name)
|
2010-03-20 19:11:32 -07:00
|
|
|
else:
|
|
|
|
return succeed(None)
|
|
|
|
|
2015-10-22 23:14:14 +01:00
|
|
|
def start(self, names=None):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Starts Components that are currently in a Stopped state and their
|
2010-03-25 14:43:15 -07:00
|
|
|
dependencies. If *names* is specified, will only start those
|
2010-03-20 19:11:32 -07:00
|
|
|
Components and their dependencies and if not it will start all
|
|
|
|
registered components.
|
|
|
|
|
|
|
|
:param names: a list of Components to start
|
|
|
|
:type names: list
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
:returns: a Deferred object that will fire once all Components have been sucessfully started
|
2010-03-20 19:11:32 -07:00
|
|
|
:rtype: twisted.internet.defer.Deferred
|
|
|
|
|
|
|
|
"""
|
|
|
|
# Start all the components if names is empty
|
|
|
|
if not names:
|
|
|
|
names = self.components.keys()
|
|
|
|
elif isinstance(names, str):
|
|
|
|
names = [names]
|
|
|
|
|
|
|
|
def on_depends_started(result, name):
|
|
|
|
return self.components[name]._component_start()
|
|
|
|
|
|
|
|
deferreds = []
|
|
|
|
|
|
|
|
for name in names:
|
|
|
|
if self.components[name]._component_depend:
|
|
|
|
# This component has depends, so we need to start them first.
|
|
|
|
d = self.start(self.components[name]._component_depend)
|
|
|
|
d.addCallback(on_depends_started, name)
|
|
|
|
deferreds.append(d)
|
|
|
|
else:
|
|
|
|
deferreds.append(self.components[name]._component_start())
|
|
|
|
|
|
|
|
return DeferredList(deferreds)
|
|
|
|
|
2015-10-22 23:14:14 +01:00
|
|
|
def stop(self, names=None):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Stops Components that are currently not in a Stopped state. If
|
2010-03-25 14:43:15 -07:00
|
|
|
*names* is specified, then it will only stop those Components,
|
2010-03-20 19:11:32 -07:00
|
|
|
and if not it will stop all the registered Components.
|
|
|
|
|
|
|
|
:param names: a list of Components to start
|
|
|
|
:type names: list
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
:returns: a Deferred object that will fire once all Components have been sucessfully stopped
|
2010-03-20 19:11:32 -07:00
|
|
|
:rtype: twisted.internet.defer.Deferred
|
|
|
|
|
|
|
|
"""
|
|
|
|
if not names:
|
|
|
|
names = self.components.keys()
|
|
|
|
elif isinstance(names, str):
|
2010-03-25 12:01:40 -07:00
|
|
|
names = [names]
|
2010-03-20 19:11:32 -07:00
|
|
|
|
2012-12-07 19:00:43 -05:00
|
|
|
def on_dependents_stopped(result, name):
|
|
|
|
return self.components[name]._component_stop()
|
|
|
|
|
|
|
|
stopped_in_deferred = set()
|
2010-03-20 19:11:32 -07:00
|
|
|
deferreds = []
|
|
|
|
|
|
|
|
for name in names:
|
2012-12-07 19:00:43 -05:00
|
|
|
if name in stopped_in_deferred:
|
|
|
|
continue
|
2010-03-22 19:59:11 -07:00
|
|
|
if name in self.components:
|
2012-12-07 19:00:43 -05:00
|
|
|
if name in self.dependents:
|
|
|
|
# If other components depend on this component, stop them first
|
|
|
|
d = self.stop(self.dependents[name]).addCallback(on_dependents_stopped, name)
|
|
|
|
deferreds.append(d)
|
|
|
|
stopped_in_deferred.update(self.dependents[name])
|
|
|
|
else:
|
|
|
|
deferreds.append(self.components[name]._component_stop())
|
2010-03-20 19:11:32 -07:00
|
|
|
|
|
|
|
return DeferredList(deferreds)
|
|
|
|
|
2015-10-22 23:14:14 +01:00
|
|
|
def pause(self, names=None):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Pauses Components that are currently in a Started state. If
|
2010-03-25 14:43:15 -07:00
|
|
|
*names* is specified, then it will only pause those Components,
|
2010-03-20 19:11:32 -07:00
|
|
|
and if not it will pause all the registered Components.
|
|
|
|
|
|
|
|
:param names: a list of Components to pause
|
|
|
|
:type names: list
|
|
|
|
|
2010-03-25 14:43:15 -07:00
|
|
|
:returns: a Deferred object that will fire once all Components have been sucessfully paused
|
2010-03-20 19:11:32 -07:00
|
|
|
:rtype: twisted.internet.defer.Deferred
|
|
|
|
|
|
|
|
"""
|
|
|
|
if not names:
|
|
|
|
names = self.components.keys()
|
|
|
|
elif isinstance(names, str):
|
|
|
|
names = [names]
|
|
|
|
|
|
|
|
deferreds = []
|
|
|
|
|
|
|
|
for name in names:
|
|
|
|
if self.components[name]._component_state == "Started":
|
|
|
|
deferreds.append(self.components[name]._component_pause())
|
|
|
|
|
|
|
|
return DeferredList(deferreds)
|
|
|
|
|
2015-10-22 23:14:14 +01:00
|
|
|
def resume(self, names=None):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Resumes Components that are currently in a Paused state. If
|
2010-03-25 14:43:15 -07:00
|
|
|
*names* is specified, then it will only resume those Components,
|
2010-03-20 19:11:32 -07:00
|
|
|
and if not it will resume all the registered Components.
|
2008-02-27 07:43:47 +00:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
:param names: a list of Components to resume
|
|
|
|
:type names: list
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2012-12-07 19:00:43 -05:00
|
|
|
:returns: a Deferred object that will fire once all Components have been successfully resumed
|
2010-03-20 19:11:32 -07:00
|
|
|
:rtype: twisted.internet.defer.Deferred
|
|
|
|
|
|
|
|
"""
|
|
|
|
if not names:
|
|
|
|
names = self.components.keys()
|
|
|
|
elif isinstance(names, str):
|
|
|
|
names = [names]
|
|
|
|
|
|
|
|
deferreds = []
|
|
|
|
|
|
|
|
for name in names:
|
|
|
|
if self.components[name]._component_state == "Paused":
|
|
|
|
deferreds.append(self.components[name]._component_resume())
|
|
|
|
|
|
|
|
return DeferredList(deferreds)
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2007-12-08 05:02:29 +00:00
|
|
|
def shutdown(self):
|
2010-03-20 19:11:32 -07:00
|
|
|
"""
|
|
|
|
Shutdowns all Components regardless of state. This will call
|
2010-03-25 12:01:40 -07:00
|
|
|
:meth:`stop` on call the components prior to shutting down. This should
|
2010-03-20 19:11:32 -07:00
|
|
|
be called when the program is exiting to ensure all Components have a
|
|
|
|
chance to properly shutdown.
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2012-12-07 19:00:43 -05:00
|
|
|
:returns: a Deferred object that will fire once all Components have been successfully shut down
|
2010-03-20 19:11:32 -07:00
|
|
|
:rtype: twisted.internet.defer.Deferred
|
|
|
|
|
|
|
|
"""
|
2012-12-07 19:00:43 -05:00
|
|
|
def on_stopped(result):
|
2015-10-22 23:14:14 +01:00
|
|
|
return DeferredList([comp._component_shutdown() for comp in self.components.values()])
|
2010-03-20 19:11:32 -07:00
|
|
|
|
2012-12-07 19:00:43 -05:00
|
|
|
return self.stop(self.components.keys()).addCallback(on_stopped)
|
2010-03-20 19:11:32 -07:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""
|
|
|
|
Updates all Components that are in a Started state.
|
|
|
|
|
|
|
|
"""
|
|
|
|
for component in self.components.items():
|
|
|
|
component.update()
|
2008-11-17 00:57:32 +00:00
|
|
|
|
2007-10-20 22:26:43 +00:00
|
|
|
_ComponentRegistry = ComponentRegistry()
|
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
deregister = _ComponentRegistry.deregister
|
|
|
|
start = _ComponentRegistry.start
|
|
|
|
stop = _ComponentRegistry.stop
|
|
|
|
pause = _ComponentRegistry.pause
|
|
|
|
resume = _ComponentRegistry.resume
|
|
|
|
update = _ComponentRegistry.update
|
|
|
|
shutdown = _ComponentRegistry.shutdown
|
|
|
|
|
2014-09-03 22:28:28 +01:00
|
|
|
|
2010-03-20 19:11:32 -07:00
|
|
|
def get(name):
|
|
|
|
"""
|
|
|
|
Return a reference to a component.
|
|
|
|
|
|
|
|
:param name: the Component name to get
|
|
|
|
:type name: string
|
|
|
|
|
|
|
|
:returns: the Component object
|
|
|
|
:rtype: object
|
|
|
|
|
|
|
|
:raises KeyError: if the Component does not exist
|
|
|
|
|
|
|
|
"""
|
|
|
|
return _ComponentRegistry.components[name]
|