[jhbuild/desrt/packagedb: 3/21] core: rewrite the dependency solver, again



commit ab9e40fdd56a9c007c095a7cbb9331f0d661703b
Author: Ryan Lortie <desrt desrt ca>
Date:   Sat Jan 10 13:08:20 2015 -0500

    core: rewrite the dependency solver, again
    
    Although this was already done a couple of years ago (bug 669554), there
    is room for more simplicity in the algorithm, with only small and
    desirable functionality changes.
    
    The new algorithm offers much more consistent handling of 'after'
    depends (if they are enabled): it simply installs them after the module
    in question, as if the user had requested that.  It also removes
    warnings about cyclic dependencies when 'A' depends on 'B' and 'B' is
    after 'A'.  This is a perfectly reasonable situation.
    
    The change in handling of 'after' depends will produce some surprising
    results when we enable 'after' by default.  For example, 'dconf' is an
    'after' of 'glib', but depends on 'gtk+' (for the editor) so building
    'glib' will pull in 'gtk+'.  That's perfectly logical -- we'll just need
    to figure out a way to sort it all out.
    
    The new algorithm will also abort in case a dependency cycle is detected
    while attempting to satisfy the hard depends of a module requested for
    building which is the only reasonable thing to do, in my opinion.
    
    https://bugzilla.gnome.org/show_bug.cgi?id=693290

 jhbuild/moduleset.py |  142 +++++++++++++++++++++++--------------------------
 1 files changed, 67 insertions(+), 75 deletions(-)
---
diff --git a/jhbuild/moduleset.py b/jhbuild/moduleset.py
index 695b3fa..dbbcadb 100644
--- a/jhbuild/moduleset.py
+++ b/jhbuild/moduleset.py
@@ -109,90 +109,82 @@ class ModuleSet:
                                 include_suggests=True, include_afters=False,
                                 warn_about_circular_dependencies=True):
 
-        def dep_resolve(node, resolved, seen, after):
+        def add_module(to_build, name, seen = []):
             ''' Recursive depth-first search of the dependency tree. Creates
-            the build order into the list 'resolved'. <after/> modules are
-            added to the dependency tree but flagged. When search finished
-            <after/> modules not a real dependency are removed.
+            the build order into the list 'to_build'.  Returns True on success,
+            or False if a hard dependency problem was encountered.
             '''
-            circular = False
-            seen.append(node)
+            if name in skip:
+                return True
+
+            toplevel = not(seen)
+
+            try:
+                module = self.get_module(name, ignore_case = toplevel)
+            except KeyError:
+                if toplevel:
+                    raise UsageError(_("A module called '%s' could not be found.") % e)
+                else:
+                    self._warn(_('%(module)s has a dependency on unknown'
+                                 ' "%(invalid)s" module') % \
+                                     {'module'  : seen[-1].name,
+                                      'invalid' : name})
+                return False
+
+            # if this module is already in the build order, we're OK
+            if module in to_build:
+                return True
+
+            # detect circular dependencies
+            if module in seen:
+                if self.raise_exception_on_warning:
+                    # Translation of string not required - used in
+                    # unit tests only
+                    raise UsageError('Circular dependencies detected')
+                if warn_about_circular_dependencies:
+                    self._warn(_('Circular dependencies detected: %s') \
+                               % ' -> '.join([i.name for i in seen] + [module.name]))
+
+                return False
+
+            # construct a new list so that we can avoid unwinding
+            seen = seen + [module]
+
+            # hard dependencies: a failure here means we can't proceed
+            for dep in module.dependencies:
+                if not add_module(to_build, dep, seen):
+                    return False
+
+            # suggests: a failure here can be ignored (but will be warned about)
             if include_suggests:
-                edges = node.dependencies + node.suggests + node.after
-            else:
-                edges = node.dependencies + node.after
-            # do not include <after> modules because a previous visited <after>
-            # module may later be a hard dependency
-            resolved_deps = [module for module, after_module in resolved \
-                             if not after_module]
-            for edge_name in edges:
-                edge = self.modules.get(edge_name)
-                if edge == None:
-                    if node not in [i[0] for i in resolved]:
-                        self._warn(_('%(module)s has a dependency on unknown'
-                                     ' "%(invalid)s" module') % \
-                                         {'module'  : node.name,
-                                          'invalid' : edge_name})
-                elif edge_name not in skip and edge not in resolved_deps:
-                    if edge in seen:
-                        # circular dependency detected
-                        circular = True
-                        if self.raise_exception_on_warning:
-                            # Translation of string not required - used in
-                            # unit tests only
-                            raise UsageError('Circular dependencies detected')
-                        if warn_about_circular_dependencies:
-                            self._warn(_('Circular dependencies detected: %s') \
-                                       % ' -> '.join([i.name for i in seen] \
-                                                     + [edge.name]))
-                        break
-                    else:
-                        if edge_name in node.after:
-                            dep_resolve(edge, resolved, seen, True)
-                        elif edge_name in node.suggests:
-                            dep_resolve(edge, resolved, seen, after)
-                        elif edge_name in node.dependencies:
-                            dep_resolve(edge, resolved, seen, after)
-                            # hard dependency may be missed if a cyclic
-                            # dependency. Add it:
-                            if edge not in [i[0] for i in resolved]:
-                                resolved.append((edge, after))
-
-            seen.remove(node)
-
-            if not circular:
-                if node not in [i[0] for i in resolved]:
-                    resolved.append((node, after))
-                elif not after:
-                    # a dependency exists for an after, flag to keep
-                    for index, item in enumerate(resolved):
-                        if item[1] == True and item[0] == node:
-                            resolved[index] = (node, False)
+                for dep in module.suggests:
+                    add_module(to_build, dep, seen)
+
+            # add ourselves, but be careful: we may already have been added as
+            # an 'after' of a module that we depend on
+            if not module in to_build:
+                to_build.append(module)
+
+            # deal with 'after'. recursive <after> will be handled by
+            # the module finding itself already in 'to_build'.
+            if include_afters:
+                for dep in module.after:
+                    add_module(to_build, dep)
+
+            return True
 
         if module_names == 'all':
             module_names = self.modules.keys()
-        try:
-            # remove skip modules from module_name list
-            modules = [self.get_module(module, ignore_case = True) \
-                       for module in module_names if module not in skip]
-        except KeyError, e:
-            raise UsageError(_("A module called '%s' could not be found.") % e)
-
-        resolved = []
-        for module in modules:
-            dep_resolve(module, resolved, [], False)
 
-        if include_afters:
-            module_list = [module[0] for module in resolved]
-        else:
-            module_list = [module for module, after_module in resolved \
-                           if not after_module]
+        to_build = []
+        for name in module_names:
+            if not add_module(to_build, name):
+                raise DependencyCycleError(_("Could not resolve dependencies for module '%s'.") % 
module.name)
 
         if '*' in skip:
-            module_list = [module for module in module_list \
-                           if module.name in self.config.modules]
-        
-        return module_list
+            to_build = [module for module in to_build if module.name in self.config.modules]
+
+        return to_build
 
     def get_test_module_list (self, seed, skip=[]):
         test_modules = []


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]