mirror of
https://github.com/django/django.git
synced 2025-01-19 14:52:54 +00:00
Use topological sort for migration operation dependency resolution
rather than an ad-hoc algorithm
This commit is contained in:
parent
4f90c99635
commit
13d613f800
1
AUTHORS
1
AUTHORS
@ -377,6 +377,7 @@ answer newbie questions, and generally made Django that much better:
|
||||
Kevin McConnell <kevin.mcconnell@gmail.com>
|
||||
Kieran Holland <http://www.kieranholland.com>
|
||||
kilian <kilian.cavalotti@lip6.fr>
|
||||
Klaas van Schelven <klaas@vanschelven.com>
|
||||
knox <christobzr@gmail.com>
|
||||
konrad@gwu.edu
|
||||
Kowito Charoenratchatabhan <kowito@felspar.com>
|
||||
|
@ -14,6 +14,8 @@ from django.db.migrations.questioner import MigrationQuestioner
|
||||
from django.db.migrations.optimizer import MigrationOptimizer
|
||||
from django.db.migrations.operations.models import AlterModelOptions
|
||||
|
||||
from .topological_sort import stable_topological_sort
|
||||
|
||||
|
||||
class MigrationAutodetector(object):
|
||||
"""
|
||||
@ -191,28 +193,19 @@ class MigrationAutodetector(object):
|
||||
# isn't bad, but we need to pull a few things around so FKs work nicely
|
||||
# inside the same app
|
||||
for app_label, ops in sorted(self.generated_operations.items()):
|
||||
for i in range(10000):
|
||||
found = False
|
||||
for i, op in enumerate(ops):
|
||||
for dep in op._auto_deps:
|
||||
if dep[0] == app_label:
|
||||
# Alright, there's a dependency on the same app.
|
||||
for j, op2 in enumerate(ops):
|
||||
if j > i and self.check_dependency(op2, dep):
|
||||
# shift the operation from position i after
|
||||
# the operation at position j
|
||||
ops = ops[:i] + ops[i + 1:j + 1] + [op] + ops[j + 1:]
|
||||
found = True
|
||||
break
|
||||
if found:
|
||||
break
|
||||
if found:
|
||||
break
|
||||
if not found:
|
||||
break
|
||||
else:
|
||||
raise ValueError("Infinite loop caught in operation dependency resolution")
|
||||
self.generated_operations[app_label] = ops
|
||||
|
||||
# construct a dependency-graph for in-app dependencies
|
||||
dependency_graph = dict((op, set()) for op in ops)
|
||||
for op in ops:
|
||||
for dep in op._auto_deps:
|
||||
if dep[0] == app_label:
|
||||
for op2 in ops:
|
||||
if self.check_dependency(op2, dep):
|
||||
dependency_graph[op].add(op2)
|
||||
|
||||
# we use a stable sort for deterministic tests & general behavior
|
||||
self.generated_operations[app_label] = stable_topological_sort(
|
||||
ops, dependency_graph)
|
||||
|
||||
# Now, we need to chop the lists of operations up into migrations with
|
||||
# dependencies on each other.
|
||||
|
34
django/db/migrations/topological_sort.py
Normal file
34
django/db/migrations/topological_sort.py
Normal file
@ -0,0 +1,34 @@
|
||||
def topological_sort_as_sets(dependency_graph):
|
||||
"""Variation of Kahn's algorithm (1962) that returns sets.
|
||||
|
||||
Takes a dependency graph as a dictionary of node => dependencies.
|
||||
|
||||
Yields sets of items in topological order, where the first set contains
|
||||
all nodes without dependencies, and each following set contains all
|
||||
nodes that depend on the nodes in the previously yielded sets.
|
||||
"""
|
||||
|
||||
todo = dependency_graph.copy()
|
||||
|
||||
while todo:
|
||||
current = set(node for node, deps in todo.items() if len(deps) == 0)
|
||||
|
||||
if not current:
|
||||
raise ValueError('Cyclic dependency in graph: {}'.format(
|
||||
', '.join(repr(x) for x in todo.items())))
|
||||
|
||||
yield current
|
||||
|
||||
# remove current from todo's nodes & dependencies
|
||||
todo = {node: (dependencies - current) for node, dependencies in
|
||||
todo.items() if node not in current}
|
||||
|
||||
|
||||
def stable_topological_sort(l, dependency_graph):
|
||||
result = []
|
||||
for layer in topological_sort_as_sets(dependency_graph):
|
||||
for node in l:
|
||||
if node in layer:
|
||||
result.append(node)
|
||||
|
||||
return result
|
@ -1107,12 +1107,12 @@ class AutodetectorTests(TestCase):
|
||||
# Right number of migrations?
|
||||
self.assertNumberMigrations(changes, "testapp", 1)
|
||||
# Right actions in right order?
|
||||
self.assertOperationTypes(changes, "testapp", 0, ["RemoveField", "RemoveField", "DeleteModel", "RemoveField", "DeleteModel"])
|
||||
self.assertOperationTypes(changes, "testapp", 0, ["RemoveField", "RemoveField", "RemoveField", "DeleteModel", "DeleteModel"])
|
||||
# Actions touching the right stuff?
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 0, name="publishers")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 1, name="author")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 2, name="Author")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 3, name="publisher")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 2, name="publisher")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 3, name="Author")
|
||||
self.assertOperationAttributes(changes, "testapp", 0, 4, name="Contract")
|
||||
|
||||
def test_non_circular_foreignkey_dependency_removal(self):
|
||||
|
Loading…
x
Reference in New Issue
Block a user