summaryrefslogtreecommitdiff
path: root/src/buildstream/_loader/loadelement.py
diff options
context:
space:
mode:
Diffstat (limited to 'src/buildstream/_loader/loadelement.py')
-rw-r--r--src/buildstream/_loader/loadelement.py181
1 files changed, 181 insertions, 0 deletions
diff --git a/src/buildstream/_loader/loadelement.py b/src/buildstream/_loader/loadelement.py
new file mode 100644
index 000000000..684c32554
--- /dev/null
+++ b/src/buildstream/_loader/loadelement.py
@@ -0,0 +1,181 @@
+#
+# Copyright (C) 2016 Codethink Limited
+#
+# This program is free software; you can redistribute it and/or
+# modify it under the terms of the GNU Lesser General Public
+# License as published by the Free Software Foundation; either
+# version 2 of the License, or (at your option) any later version.
+#
+# This library is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+# Lesser General Public License for more details.
+#
+# You should have received a copy of the GNU Lesser General Public
+# License along with this library. If not, see <http://www.gnu.org/licenses/>.
+#
+# Authors:
+# Tristan Van Berkom <tristan.vanberkom@codethink.co.uk>
+
+# System imports
+from itertools import count
+
+from pyroaring import BitMap, FrozenBitMap # pylint: disable=no-name-in-module
+
+# BuildStream toplevel imports
+from .. import _yaml
+
+# Local package imports
+from .types import Symbol, Dependency
+
+
+# LoadElement():
+#
+# A transient object breaking down what is loaded allowing us to
+# do complex operations in multiple passes.
+#
+# Args:
+# node (dict): A YAML loaded dictionary
+# name (str): The element name
+# loader (Loader): The Loader object for this element
+#
+class LoadElement():
+ # Dependency():
+ #
+ # A link from a LoadElement to its dependencies.
+ #
+ # Keeps a link to one of the current Element's dependencies, together with
+ # its dependency type.
+ #
+ # Args:
+ # element (LoadElement): a LoadElement on which there is a dependency
+ # dep_type (str): the type of dependency this dependency link is
+ class Dependency:
+ def __init__(self, element, dep_type):
+ self.element = element
+ self.dep_type = dep_type
+
+ _counter = count()
+
+ def __init__(self, node, filename, loader):
+
+ #
+ # Public members
+ #
+ self.node = node # The YAML node
+ self.name = filename # The element name
+ self.full_name = None # The element full name (with associated junction)
+ self.deps = None # The list of Dependency objects
+ self.node_id = next(self._counter)
+
+ #
+ # Private members
+ #
+ self._loader = loader # The Loader object
+ self._dep_cache = None # The dependency cache, to speed up depends()
+
+ #
+ # Initialization
+ #
+ if loader.project.junction:
+ # dependency is in subproject, qualify name
+ self.full_name = '{}:{}'.format(loader.project.junction.name, self.name)
+ else:
+ # dependency is in top-level project
+ self.full_name = self.name
+
+ # Ensure the root node is valid
+ _yaml.node_validate(self.node, [
+ 'kind', 'depends', 'sources', 'sandbox',
+ 'variables', 'environment', 'environment-nocache',
+ 'config', 'public', 'description',
+ 'build-depends', 'runtime-depends',
+ ])
+
+ self.dependencies = []
+
+ @property
+ def junction(self):
+ return self._loader.project.junction
+
+ # depends():
+ #
+ # Checks if this element depends on another element, directly
+ # or indirectly.
+ #
+ # Args:
+ # other (LoadElement): Another LoadElement
+ #
+ # Returns:
+ # (bool): True if this LoadElement depends on 'other'
+ #
+ def depends(self, other):
+ self._ensure_depends_cache()
+ return other.node_id in self._dep_cache
+
+ ###########################################
+ # Private Methods #
+ ###########################################
+ def _ensure_depends_cache(self):
+
+ if self._dep_cache:
+ return
+
+ self._dep_cache = BitMap()
+
+ for dep in self.dependencies:
+ elt = dep.element
+
+ # Ensure the cache of the element we depend on
+ elt._ensure_depends_cache()
+
+ # We depend on this element
+ self._dep_cache.add(elt.node_id)
+
+ # And we depend on everything this element depends on
+ self._dep_cache.update(elt._dep_cache)
+
+ self._dep_cache = FrozenBitMap(self._dep_cache)
+
+
+# _extract_depends_from_node():
+#
+# Creates an array of Dependency objects from a given dict node 'node',
+# allows both strings and dicts for expressing the dependency and
+# throws a comprehensive LoadError in the case that the node is malformed.
+#
+# After extracting depends, the symbol is deleted from the node
+#
+# Args:
+# node (dict): A YAML loaded dictionary
+#
+# Returns:
+# (list): a list of Dependency objects
+#
+def _extract_depends_from_node(node, *, key=None):
+ if key is None:
+ build_depends = _extract_depends_from_node(node, key=Symbol.BUILD_DEPENDS)
+ runtime_depends = _extract_depends_from_node(node, key=Symbol.RUNTIME_DEPENDS)
+ depends = _extract_depends_from_node(node, key=Symbol.DEPENDS)
+ return build_depends + runtime_depends + depends
+ elif key == Symbol.BUILD_DEPENDS:
+ default_dep_type = Symbol.BUILD
+ elif key == Symbol.RUNTIME_DEPENDS:
+ default_dep_type = Symbol.RUNTIME
+ elif key == Symbol.DEPENDS:
+ default_dep_type = None
+ else:
+ assert False, "Unexpected value of key '{}'".format(key)
+
+ depends = _yaml.node_get(node, list, key, default_value=[])
+ output_deps = []
+
+ for index, dep in enumerate(depends):
+ dep_provenance = _yaml.node_get_provenance(node, key=key, indices=[index])
+ dependency = Dependency(dep, dep_provenance, default_dep_type=default_dep_type)
+ output_deps.append(dependency)
+
+ # Now delete the field, we dont want it anymore
+ _yaml.node_del(node, key, safe=True)
+
+ return output_deps