2013-05-10 15:18:19 +00:00
|
|
|
from django.test import TransactionTestCase, TestCase
|
2013-05-10 15:00:55 +00:00
|
|
|
from django.db import connection
|
|
|
|
from django.db.migrations.graph import MigrationGraph, CircularDependencyError
|
|
|
|
from django.db.migrations.loader import MigrationLoader
|
2013-05-10 15:18:19 +00:00
|
|
|
from django.db.migrations.recorder import MigrationRecorder
|
2013-05-10 11:52:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
class GraphTests(TransactionTestCase):
|
|
|
|
"""
|
|
|
|
Tests the digraph structure.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def test_simple_graph(self):
|
|
|
|
"""
|
|
|
|
Tests a basic dependency graph:
|
|
|
|
|
|
|
|
app_a: 0001 <-- 0002 <--- 0003 <-- 0004
|
|
|
|
/
|
|
|
|
app_b: 0001 <-- 0002 <-/
|
|
|
|
"""
|
|
|
|
# Build graph
|
2013-05-10 15:00:55 +00:00
|
|
|
graph = MigrationGraph()
|
2013-05-10 11:52:04 +00:00
|
|
|
graph.add_dependency(("app_a", "0004"), ("app_a", "0003"))
|
|
|
|
graph.add_dependency(("app_a", "0003"), ("app_a", "0002"))
|
|
|
|
graph.add_dependency(("app_a", "0002"), ("app_a", "0001"))
|
|
|
|
graph.add_dependency(("app_a", "0003"), ("app_b", "0002"))
|
|
|
|
graph.add_dependency(("app_b", "0002"), ("app_b", "0001"))
|
|
|
|
# Test root migration case
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("app_a", "0001")),
|
|
|
|
[('app_a', '0001')],
|
|
|
|
)
|
|
|
|
# Test branch B only
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("app_b", "0002")),
|
|
|
|
[("app_b", "0001"), ("app_b", "0002")],
|
|
|
|
)
|
|
|
|
# Test whole graph
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("app_a", "0004")),
|
|
|
|
[('app_b', '0001'), ('app_b', '0002'), ('app_a', '0001'), ('app_a', '0002'), ('app_a', '0003'), ('app_a', '0004')],
|
|
|
|
)
|
|
|
|
# Test reverse to b:0002
|
|
|
|
self.assertEqual(
|
|
|
|
graph.backwards_plan(("app_b", "0002")),
|
|
|
|
[('app_a', '0004'), ('app_a', '0003'), ('app_b', '0002')],
|
|
|
|
)
|
2013-05-10 15:09:57 +00:00
|
|
|
# Test roots and leaves
|
|
|
|
self.assertEqual(
|
|
|
|
graph.root_nodes(),
|
|
|
|
set([('app_a', '0001'), ('app_b', '0001')]),
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
graph.leaf_nodes(),
|
|
|
|
set([('app_a', '0004'), ('app_b', '0002')]),
|
|
|
|
)
|
2013-05-10 11:52:04 +00:00
|
|
|
|
|
|
|
def test_complex_graph(self):
|
|
|
|
"""
|
|
|
|
Tests a complex dependency graph:
|
|
|
|
|
|
|
|
app_a: 0001 <-- 0002 <--- 0003 <-- 0004
|
|
|
|
\ \ / /
|
|
|
|
app_b: 0001 <-\ 0002 <-X /
|
|
|
|
\ \ /
|
|
|
|
app_c: \ 0001 <-- 0002 <-
|
|
|
|
"""
|
|
|
|
# Build graph
|
2013-05-10 15:00:55 +00:00
|
|
|
graph = MigrationGraph()
|
2013-05-10 11:52:04 +00:00
|
|
|
graph.add_dependency(("app_a", "0004"), ("app_a", "0003"))
|
|
|
|
graph.add_dependency(("app_a", "0003"), ("app_a", "0002"))
|
|
|
|
graph.add_dependency(("app_a", "0002"), ("app_a", "0001"))
|
|
|
|
graph.add_dependency(("app_a", "0003"), ("app_b", "0002"))
|
|
|
|
graph.add_dependency(("app_b", "0002"), ("app_b", "0001"))
|
|
|
|
graph.add_dependency(("app_a", "0004"), ("app_c", "0002"))
|
|
|
|
graph.add_dependency(("app_c", "0002"), ("app_c", "0001"))
|
|
|
|
graph.add_dependency(("app_c", "0001"), ("app_b", "0001"))
|
|
|
|
graph.add_dependency(("app_c", "0002"), ("app_a", "0002"))
|
|
|
|
# Test branch C only
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("app_c", "0002")),
|
|
|
|
[('app_b', '0001'), ('app_c', '0001'), ('app_a', '0001'), ('app_a', '0002'), ('app_c', '0002')],
|
|
|
|
)
|
|
|
|
# Test whole graph
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("app_a", "0004")),
|
|
|
|
[('app_b', '0001'), ('app_c', '0001'), ('app_a', '0001'), ('app_a', '0002'), ('app_c', '0002'), ('app_b', '0002'), ('app_a', '0003'), ('app_a', '0004')],
|
|
|
|
)
|
|
|
|
# Test reverse to b:0001
|
|
|
|
self.assertEqual(
|
|
|
|
graph.backwards_plan(("app_b", "0001")),
|
|
|
|
[('app_a', '0004'), ('app_c', '0002'), ('app_c', '0001'), ('app_a', '0003'), ('app_b', '0002'), ('app_b', '0001')],
|
|
|
|
)
|
2013-05-10 15:09:57 +00:00
|
|
|
# Test roots and leaves
|
|
|
|
self.assertEqual(
|
|
|
|
graph.root_nodes(),
|
|
|
|
set([('app_a', '0001'), ('app_b', '0001'), ('app_c', '0001')]),
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
graph.leaf_nodes(),
|
|
|
|
set([('app_a', '0004'), ('app_b', '0002'), ('app_c', '0002')]),
|
|
|
|
)
|
2013-05-10 11:52:04 +00:00
|
|
|
|
|
|
|
def test_circular_graph(self):
|
|
|
|
"""
|
|
|
|
Tests a circular dependency graph.
|
|
|
|
"""
|
|
|
|
# Build graph
|
2013-05-10 15:00:55 +00:00
|
|
|
graph = MigrationGraph()
|
2013-05-10 11:52:04 +00:00
|
|
|
graph.add_dependency(("app_a", "0003"), ("app_a", "0002"))
|
|
|
|
graph.add_dependency(("app_a", "0002"), ("app_a", "0001"))
|
|
|
|
graph.add_dependency(("app_a", "0001"), ("app_b", "0002"))
|
|
|
|
graph.add_dependency(("app_b", "0002"), ("app_b", "0001"))
|
|
|
|
graph.add_dependency(("app_b", "0001"), ("app_a", "0003"))
|
|
|
|
# Test whole graph
|
|
|
|
self.assertRaises(
|
2013-05-10 15:00:55 +00:00
|
|
|
CircularDependencyError,
|
2013-05-10 11:52:04 +00:00
|
|
|
graph.forwards_plan, ("app_a", "0003"),
|
|
|
|
)
|
2013-05-10 15:00:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
class LoaderTests(TransactionTestCase):
|
|
|
|
"""
|
|
|
|
Tests the disk and database loader.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def test_load(self):
|
|
|
|
"""
|
|
|
|
Makes sure the loader can load the migrations for the test apps.
|
|
|
|
"""
|
|
|
|
migration_loader = MigrationLoader(connection)
|
|
|
|
graph = migration_loader.build_graph()
|
|
|
|
self.assertEqual(
|
|
|
|
graph.forwards_plan(("migrations", "0002_second")),
|
|
|
|
[("migrations", "0001_initial"), ("migrations", "0002_second")],
|
|
|
|
)
|
2013-05-10 15:18:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RecorderTests(TestCase):
|
|
|
|
"""
|
|
|
|
Tests the disk and database loader.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def test_apply(self):
|
|
|
|
"""
|
|
|
|
Tests marking migrations as applied/unapplied.
|
|
|
|
"""
|
|
|
|
recorder = MigrationRecorder(connection)
|
|
|
|
self.assertEqual(
|
|
|
|
recorder.applied_migrations(),
|
|
|
|
set(),
|
|
|
|
)
|
|
|
|
recorder.record_applied("myapp", "0432_ponies")
|
|
|
|
self.assertEqual(
|
|
|
|
recorder.applied_migrations(),
|
|
|
|
set([("myapp", "0432_ponies")]),
|
|
|
|
)
|
|
|
|
recorder.record_unapplied("myapp", "0432_ponies")
|
|
|
|
self.assertEqual(
|
|
|
|
recorder.applied_migrations(),
|
|
|
|
set(),
|
|
|
|
)
|