mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Add RunPython migration operation and tests
This commit is contained in:
		| @@ -32,6 +32,10 @@ class Migration(object): | |||||||
|     # are not applied. |     # are not applied. | ||||||
|     replaces = [] |     replaces = [] | ||||||
|  |  | ||||||
|  |     # Error class which is raised when a migration is irreversible | ||||||
|  |     class IrreversibleError(RuntimeError): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|     def __init__(self, name, app_label): |     def __init__(self, name, app_label): | ||||||
|         self.name = name |         self.name = name | ||||||
|         self.app_label = app_label |         self.app_label = app_label | ||||||
| @@ -91,6 +95,8 @@ class Migration(object): | |||||||
|         # We need to pre-calculate the stack of project states |         # We need to pre-calculate the stack of project states | ||||||
|         to_run = [] |         to_run = [] | ||||||
|         for operation in self.operations: |         for operation in self.operations: | ||||||
|  |             if not operation.reversible: | ||||||
|  |                 raise Migration.IrreversibleError("Operation %s in %s is not reversible" % (operation, sekf)) | ||||||
|             new_state = project_state.clone() |             new_state = project_state.clone() | ||||||
|             operation.state_forwards(self.app_label, new_state) |             operation.state_forwards(self.app_label, new_state) | ||||||
|             to_run.append((operation, project_state, new_state)) |             to_run.append((operation, project_state, new_state)) | ||||||
|   | |||||||
| @@ -1,3 +1,3 @@ | |||||||
| from .models import CreateModel, DeleteModel, AlterModelTable, AlterUniqueTogether, AlterIndexTogether | from .models import CreateModel, DeleteModel, AlterModelTable, AlterUniqueTogether, AlterIndexTogether | ||||||
| from .fields import AddField, RemoveField, AlterField, RenameField | from .fields import AddField, RemoveField, AlterField, RenameField | ||||||
| from .special import SeparateDatabaseAndState, RunSQL | from .special import SeparateDatabaseAndState, RunSQL, RunPython | ||||||
|   | |||||||
| @@ -15,6 +15,9 @@ class Operation(object): | |||||||
|     # Some operations are impossible to reverse, like deleting data. |     # Some operations are impossible to reverse, like deleting data. | ||||||
|     reversible = True |     reversible = True | ||||||
|  |  | ||||||
|  |     # Can this migration be represented as SQL? (things like RunPython cannot) | ||||||
|  |     reduces_to_sql = True | ||||||
|  |  | ||||||
|     def __new__(cls, *args, **kwargs): |     def __new__(cls, *args, **kwargs): | ||||||
|         # We capture the arguments to make returning them trivial |         # We capture the arguments to make returning them trivial | ||||||
|         self = object.__new__(cls) |         self = object.__new__(cls) | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import re | import re | ||||||
|  | import textwrap | ||||||
| from .base import Operation | from .base import Operation | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -59,6 +59,10 @@ class RunSQL(Operation): | |||||||
|         self.state_operations = state_operations or [] |         self.state_operations = state_operations or [] | ||||||
|         self.multiple = multiple |         self.multiple = multiple | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def reversible(self): | ||||||
|  |         return self.reverse_sql is not None | ||||||
|  |  | ||||||
|     def state_forwards(self, app_label, state): |     def state_forwards(self, app_label, state): | ||||||
|         for state_operation in self.state_operations: |         for state_operation in self.state_operations: | ||||||
|             state_operation.state_forwards(app_label, state) |             state_operation.state_forwards(app_label, state) | ||||||
| @@ -92,3 +96,39 @@ class RunSQL(Operation): | |||||||
|  |  | ||||||
|     def describe(self): |     def describe(self): | ||||||
|         return "Raw SQL operation" |         return "Raw SQL operation" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RunPython(Operation): | ||||||
|  |     """ | ||||||
|  |     Runs Python code in a context suitable for doing versioned ORM operations. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     reduces_to_sql = False | ||||||
|  |     reversible = False | ||||||
|  |  | ||||||
|  |     def __init__(self, code): | ||||||
|  |         # Trim any leading whitespace that is at the start of all code lines | ||||||
|  |         # so users can nicely indent code in migration files | ||||||
|  |         code = textwrap.dedent(code) | ||||||
|  |         # Run the code through a parser first to make sure it's at least | ||||||
|  |         # syntactically correct | ||||||
|  |         self.code = compile(code, "<string>", "exec") | ||||||
|  |  | ||||||
|  |     def state_forwards(self, app_label, state): | ||||||
|  |         # RunPython objects have no state effect. To add some, combine this | ||||||
|  |         # with SeparateDatabaseAndState. | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|  |         # We now execute the Python code in a context that contains a 'models' | ||||||
|  |         # object, representing the versioned models as an AppCache. | ||||||
|  |         # We could try to override the global cache, but then people will still | ||||||
|  |         # use direct imports, so we go with a documentation approach instead. | ||||||
|  |         context = { | ||||||
|  |             "models": from_state.render(), | ||||||
|  |             "schema_editor": schema_editor, | ||||||
|  |         } | ||||||
|  |         eval(self.code, context) | ||||||
|  |  | ||||||
|  |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|  |         raise NotImplementedError("You cannot reverse this operation") | ||||||
|   | |||||||
| @@ -282,7 +282,7 @@ class OperationTests(MigrationTestBase): | |||||||
|  |  | ||||||
|     def test_run_sql(self): |     def test_run_sql(self): | ||||||
|         """ |         """ | ||||||
|         Tests the AlterIndexTogether operation. |         Tests the RunSQL operation. | ||||||
|         """ |         """ | ||||||
|         project_state = self.set_up_test_model("test_runsql") |         project_state = self.set_up_test_model("test_runsql") | ||||||
|         # Create the operation |         # Create the operation | ||||||
| @@ -306,6 +306,33 @@ class OperationTests(MigrationTestBase): | |||||||
|             operation.database_backwards("test_runsql", editor, new_state, project_state) |             operation.database_backwards("test_runsql", editor, new_state, project_state) | ||||||
|         self.assertTableNotExists("i_love_ponies") |         self.assertTableNotExists("i_love_ponies") | ||||||
|  |  | ||||||
|  |     def test_run_python(self): | ||||||
|  |         """ | ||||||
|  |         Tests the RunPython operation | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         project_state = self.set_up_test_model("test_runpython") | ||||||
|  |         # Create the operation | ||||||
|  |         operation = migrations.RunPython( | ||||||
|  |             """ | ||||||
|  |             Pony = models.get_model("test_runpython", "Pony") | ||||||
|  |             Pony.objects.create(pink=2, weight=4.55) | ||||||
|  |             Pony.objects.create(weight=1) | ||||||
|  |             """, | ||||||
|  |         ) | ||||||
|  |         # Test the state alteration does nothing | ||||||
|  |         new_state = project_state.clone() | ||||||
|  |         operation.state_forwards("test_runpython", new_state) | ||||||
|  |         self.assertEqual(new_state, project_state) | ||||||
|  |         # Test the database alteration | ||||||
|  |         self.assertEqual(project_state.render().get_model("test_runpython", "Pony").objects.count(), 0) | ||||||
|  |         with connection.schema_editor() as editor: | ||||||
|  |             operation.database_forwards("test_runpython", editor, project_state, new_state) | ||||||
|  |         self.assertEqual(project_state.render().get_model("test_runpython", "Pony").objects.count(), 2) | ||||||
|  |         # And test reversal fails | ||||||
|  |         with self.assertRaises(NotImplementedError): | ||||||
|  |             operation.database_backwards("test_runpython", None, new_state, project_state) | ||||||
|  |  | ||||||
|  |  | ||||||
| class MigrateNothingRouter(object): | class MigrateNothingRouter(object): | ||||||
|     """ |     """ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user