mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #27845 -- Allowed both right and left optimizations of operations.
Thanks Raphael Gaschignard for the suggestion.
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							0025dd5eb4
						
					
				
				
					commit
					37cafbfb79
				
			| @@ -42,21 +42,29 @@ class MigrationOptimizer: | |||||||
|         """Inner optimization loop.""" |         """Inner optimization loop.""" | ||||||
|         new_operations = [] |         new_operations = [] | ||||||
|         for i, operation in enumerate(operations): |         for i, operation in enumerate(operations): | ||||||
|  |             right = True  # Should we reduce on the right or on the left. | ||||||
|             # Compare it to each operation after it |             # Compare it to each operation after it | ||||||
|             for j, other in enumerate(operations[i + 1:]): |             for j, other in enumerate(operations[i + 1:]): | ||||||
|                 in_between = operations[i + 1:i + j + 1] |                 in_between = operations[i + 1:i + j + 1] | ||||||
|                 result = operation.reduce(other, in_between, app_label) |                 result = operation.reduce(other, in_between, app_label) | ||||||
|                 if isinstance(result, list): |                 if isinstance(result, list): | ||||||
|                     # Add operations optimized through, optimized operations, |                     if right: | ||||||
|                     # and the remaining ones. |  | ||||||
|                         new_operations.extend(in_between) |                         new_operations.extend(in_between) | ||||||
|                         new_operations.extend(result) |                         new_operations.extend(result) | ||||||
|                     new_operations.extend(operations[i + j + 2:]) |                     elif all(op.reduce(other, [], app_label) is True for op in in_between): | ||||||
|                     return new_operations |                         # Perform a left reduction if all of the in-between | ||||||
|                 if not result: |                         # operations can optimize through other. | ||||||
|                     # We can't optimize across `other`. |                         new_operations.extend(result) | ||||||
|  |                         new_operations.extend(in_between) | ||||||
|  |                     else: | ||||||
|  |                         # Otherwise keep trying. | ||||||
|                         new_operations.append(operation) |                         new_operations.append(operation) | ||||||
|                         break |                         break | ||||||
|  |                     new_operations.extend(operations[i + j + 2:]) | ||||||
|  |                     return new_operations | ||||||
|  |                 elif not result: | ||||||
|  |                     # Can't perform a right reduction. | ||||||
|  |                     right = False | ||||||
|             else: |             else: | ||||||
|                 new_operations.append(operation) |                 new_operations.append(operation) | ||||||
|         return new_operations |         return new_operations | ||||||
|   | |||||||
| @@ -270,8 +270,6 @@ class OptimizerTests(SimpleTestCase): | |||||||
|             app_label="testapp", |             app_label="testapp", | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         # This could be optimized a bit more but it generates a valid set of |  | ||||||
|         # operations. |  | ||||||
|         self.assertOptimizesTo( |         self.assertOptimizesTo( | ||||||
|             [ |             [ | ||||||
|                 migrations.CreateModel('Book', [('name', models.CharField(max_length=255))]), |                 migrations.CreateModel('Book', [('name', models.CharField(max_length=255))]), | ||||||
| @@ -284,18 +282,12 @@ class OptimizerTests(SimpleTestCase): | |||||||
|                 migrations.DeleteModel('Person'), |                 migrations.DeleteModel('Person'), | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 migrations.CreateModel('Person', [('name', models.CharField(max_length=255))]), |                 migrations.CreateModel('Book', [('name', models.CharField(max_length=255))]), | ||||||
|                 migrations.CreateModel('Book', [ |  | ||||||
|                     ('name', models.CharField(max_length=255)), |  | ||||||
|                     ('author', models.ForeignKey('test_app.Person', models.CASCADE)), |  | ||||||
|                 ]), |  | ||||||
|                 migrations.CreateModel('Reviewer', [('name', models.CharField(max_length=255))]), |                 migrations.CreateModel('Reviewer', [('name', models.CharField(max_length=255))]), | ||||||
|                 migrations.CreateModel('Review', [ |                 migrations.CreateModel('Review', [ | ||||||
|                     ('book', models.ForeignKey('test_app.Book', models.CASCADE)), |                     ('book', models.ForeignKey('test_app.Book', models.CASCADE)), | ||||||
|                     ('reviewer', models.ForeignKey('test_app.Reviewer', models.CASCADE)), |                     ('reviewer', models.ForeignKey('test_app.Reviewer', models.CASCADE)), | ||||||
|                 ]), |                 ]), | ||||||
|                 migrations.RemoveField('book', 'author'), |  | ||||||
|                 migrations.DeleteModel('Person'), |  | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user