[1.7.x] switch out recursive dfs for stack based approach, to avoid possibly hitting the recursion limit
This commit is contained in:
parent
e5cdfb1510
commit
4ca44d5e10
|
@ -94,31 +94,26 @@ class MigrationGraph(object):
|
||||||
"""
|
"""
|
||||||
Dynamic programming based depth first search, for finding dependencies.
|
Dynamic programming based depth first search, for finding dependencies.
|
||||||
"""
|
"""
|
||||||
cache = {}
|
visited = []
|
||||||
|
visited.append(start)
|
||||||
|
path = [start]
|
||||||
|
stack = sorted(get_children(start))
|
||||||
|
while stack:
|
||||||
|
node = stack.pop(0)
|
||||||
|
|
||||||
def _dfs(start, get_children, path):
|
if node in path:
|
||||||
# If we already computed this, use that (dynamic programming)
|
raise CircularDependencyError()
|
||||||
if (start, get_children) in cache:
|
path.append(node)
|
||||||
return cache[(start, get_children)]
|
|
||||||
# If we've traversed here before, that's a circular dep
|
visited.insert(0, node)
|
||||||
if start in path:
|
children = sorted(get_children(node))
|
||||||
raise CircularDependencyError(path[path.index(start):] + [start])
|
|
||||||
# Build our own results list, starting with us
|
if not children:
|
||||||
results = []
|
path = []
|
||||||
results.append(start)
|
|
||||||
# We need to add to results all the migrations this one depends on
|
stack = children + stack
|
||||||
children = sorted(get_children(start))
|
|
||||||
path.append(start)
|
return list(OrderedSet(visited))
|
||||||
for n in children:
|
|
||||||
results = _dfs(n, get_children, path) + results
|
|
||||||
path.pop()
|
|
||||||
# Use OrderedSet to ensure only one instance of each result
|
|
||||||
results = list(OrderedSet(results))
|
|
||||||
# Populate DP cache
|
|
||||||
cache[(start, get_children)] = results
|
|
||||||
# Done!
|
|
||||||
return results
|
|
||||||
return _dfs(start, get_children, [])
|
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "Graph: %s nodes, %s edges" % (len(self.nodes), sum(len(x) for x in self.dependencies.values()))
|
return "Graph: %s nodes, %s edges" % (len(self.nodes), sum(len(x) for x in self.dependencies.values()))
|
||||||
|
|
|
@ -134,6 +134,21 @@ class GraphTests(TestCase):
|
||||||
graph.forwards_plan, ("app_a", "0003"),
|
graph.forwards_plan, ("app_a", "0003"),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_dfs(self):
|
||||||
|
graph = MigrationGraph()
|
||||||
|
root = ("app_a", "1")
|
||||||
|
graph.add_node(root, None)
|
||||||
|
expected = [root]
|
||||||
|
for i in xrange(2, 1000):
|
||||||
|
parent = ("app_a", str(i - 1))
|
||||||
|
child = ("app_a", str(i))
|
||||||
|
graph.add_node(child, None)
|
||||||
|
graph.add_dependency(str(i), child, parent)
|
||||||
|
expected.append(child)
|
||||||
|
|
||||||
|
actual = graph.dfs(root, lambda x: graph.dependents.get(x, set()))
|
||||||
|
self.assertEqual(expected[::-1], actual)
|
||||||
|
|
||||||
def test_plan_invalid_node(self):
|
def test_plan_invalid_node(self):
|
||||||
"""
|
"""
|
||||||
Tests for forwards/backwards_plan of nonexistent node.
|
Tests for forwards/backwards_plan of nonexistent node.
|
||||||
|
|
Loading…
Reference in New Issue