mirror of
				https://github.com/django/django.git
				synced 2025-10-26 15:16:09 +00:00 
			
		
		
		
	The new signature enables better support for routing RunPython and
RunSQL operations, especially w.r.t. reusable and third-party apps.
This commit also takes advantage of the deprecation cycle for the old
signature to remove the backward incompatibility introduced in #22583;
RunPython and RunSQL won't call allow_migrate() when when the router
has the old signature.
Thanks Aymeric Augustin and Tim Graham for helping shape up the patch.
Refs 22583.
Conflicts:
	django/db/utils.py
Backport of bed504d70b from master
			
			
This commit is contained in:
		| @@ -66,7 +66,7 @@ def create_permissions(app_config, verbosity=2, interactive=True, using=DEFAULT_ | |||||||
|     except LookupError: |     except LookupError: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     if not router.allow_migrate(using, Permission): |     if not router.allow_migrate_model(using, Permission): | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     from django.contrib.contenttypes.models import ContentType |     from django.contrib.contenttypes.models import ContentType | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ def update_contenttypes(app_config, verbosity=2, interactive=True, using=DEFAULT | |||||||
|     except LookupError: |     except LookupError: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     if not router.allow_migrate(using, ContentType): |     if not router.allow_migrate_model(using, ContentType): | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     ContentType.objects.clear_cache() |     ContentType.objects.clear_cache() | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ class Migration(migrations.Migration): | |||||||
|         migrations.RunPython( |         migrations.RunPython( | ||||||
|             migrations.RunPython.noop, |             migrations.RunPython.noop, | ||||||
|             add_legacy_name, |             add_legacy_name, | ||||||
|  |             hints={'model_name': 'contenttype'}, | ||||||
|         ), |         ), | ||||||
|         migrations.RemoveField( |         migrations.RemoveField( | ||||||
|             model_name='contenttype', |             model_name='contenttype', | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ def create_default_site(app_config, verbosity=2, interactive=True, using=DEFAULT | |||||||
|     except LookupError: |     except LookupError: | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     if not router.allow_migrate(using, Site): |     if not router.allow_migrate_model(using, Site): | ||||||
|         return |         return | ||||||
|  |  | ||||||
|     if not Site.objects.using(using).exists(): |     if not Site.objects.using(using).exists(): | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								django/core/cache/backends/db.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								django/core/cache/backends/db.py
									
									
									
									
										vendored
									
									
								
							| @@ -30,6 +30,7 @@ class Options(object): | |||||||
|         self.abstract = False |         self.abstract = False | ||||||
|         self.managed = True |         self.managed = True | ||||||
|         self.proxy = False |         self.proxy = False | ||||||
|  |         self.swapped = False | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseDatabaseCache(BaseCache): | class BaseDatabaseCache(BaseCache): | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ class Command(BaseCommand): | |||||||
|  |  | ||||||
|     def create_table(self, database, tablename): |     def create_table(self, database, tablename): | ||||||
|         cache = BaseDatabaseCache(tablename, {}) |         cache = BaseDatabaseCache(tablename, {}) | ||||||
|         if not router.allow_migrate(database, cache.cache_model_class): |         if not router.allow_migrate_model(database, cache.cache_model_class): | ||||||
|             return |             return | ||||||
|         connection = connections[database] |         connection = connections[database] | ||||||
|  |  | ||||||
|   | |||||||
| @@ -140,7 +140,7 @@ class Command(BaseCommand): | |||||||
|             for model in serializers.sort_dependencies(app_list.items()): |             for model in serializers.sort_dependencies(app_list.items()): | ||||||
|                 if model in excluded_models: |                 if model in excluded_models: | ||||||
|                     continue |                     continue | ||||||
|                 if not model._meta.proxy and router.allow_migrate(using, model): |                 if not model._meta.proxy and router.allow_migrate_model(using, model): | ||||||
|                     if use_base_manager: |                     if use_base_manager: | ||||||
|                         objects = model._base_manager |                         objects = model._base_manager | ||||||
|                     else: |                     else: | ||||||
|   | |||||||
| @@ -140,7 +140,7 @@ class Command(BaseCommand): | |||||||
|  |  | ||||||
|                 for obj in objects: |                 for obj in objects: | ||||||
|                     objects_in_fixture += 1 |                     objects_in_fixture += 1 | ||||||
|                     if router.allow_migrate(self.using, obj.object.__class__): |                     if router.allow_migrate_model(self.using, obj.object.__class__): | ||||||
|                         loaded_objects_in_fixture += 1 |                         loaded_objects_in_fixture += 1 | ||||||
|                         self.models.add(obj.object.__class__) |                         self.models.add(obj.object.__class__) | ||||||
|                         try: |                         try: | ||||||
|   | |||||||
| @@ -404,7 +404,7 @@ class BaseDatabaseCreation(object): | |||||||
|         def get_objects(): |         def get_objects(): | ||||||
|             for model in serializers.sort_dependencies(app_list): |             for model in serializers.sort_dependencies(app_list): | ||||||
|                 if (not model._meta.proxy and model._meta.managed and |                 if (not model._meta.proxy and model._meta.managed and | ||||||
|                         router.allow_migrate(self.connection.alias, model)): |                         router.allow_migrate_model(self.connection.alias, model)): | ||||||
|                     queryset = model._default_manager.using(self.connection.alias).order_by(model._meta.pk.name) |                     queryset = model._default_manager.using(self.connection.alias).order_by(model._meta.pk.name) | ||||||
|                     for obj in queryset.iterator(): |                     for obj in queryset.iterator(): | ||||||
|                         yield obj |                         yield obj | ||||||
|   | |||||||
| @@ -99,15 +99,17 @@ class Operation(object): | |||||||
|         """ |         """ | ||||||
|         return self.references_model(model_name, app_label) |         return self.references_model(model_name, app_label) | ||||||
|  |  | ||||||
|     def allowed_to_migrate(self, connection_alias, model, hints=None): |     def allow_migrate_model(self, connection_alias, model): | ||||||
|         """ |         """ | ||||||
|         Returns if we're allowed to migrate the model. |         Returns if we're allowed to migrate the model. | ||||||
|  |  | ||||||
|  |         This is a thin wrapper around router.allow_migrate_model() that | ||||||
|  |         preemptively rejects any proxy, swapped out, or unmanaged model. | ||||||
|         """ |         """ | ||||||
|         # Always skip if proxy, swapped out, or unmanaged. |         if model._meta.proxy or model._meta.swapped or not model._meta.managed: | ||||||
|         if model and (model._meta.proxy or model._meta.swapped or not model._meta.managed): |  | ||||||
|             return False |             return False | ||||||
|  |  | ||||||
|         return router.allow_migrate(connection_alias, model, **(hints or {})) |         return router.allow_migrate_model(connection_alias, model) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return "<%s %s%s>" % ( |         return "<%s %s%s>" % ( | ||||||
|   | |||||||
| @@ -52,7 +52,7 @@ class AddField(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.model_name) |             from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|             field = to_model._meta.get_field(self.name) |             field = to_model._meta.get_field(self.name) | ||||||
|             if not self.preserve_default: |             if not self.preserve_default: | ||||||
| @@ -66,7 +66,7 @@ class AddField(Operation): | |||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         from_model = from_state.apps.get_model(app_label, self.model_name) |         from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, from_model): |         if self.allow_migrate_model(schema_editor.connection.alias, from_model): | ||||||
|             schema_editor.remove_field(from_model, from_model._meta.get_field(self.name)) |             schema_editor.remove_field(from_model, from_model._meta.get_field(self.name)) | ||||||
|  |  | ||||||
|     def describe(self): |     def describe(self): | ||||||
| @@ -117,12 +117,12 @@ class RemoveField(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         from_model = from_state.apps.get_model(app_label, self.model_name) |         from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, from_model): |         if self.allow_migrate_model(schema_editor.connection.alias, from_model): | ||||||
|             schema_editor.remove_field(from_model, from_model._meta.get_field(self.name)) |             schema_editor.remove_field(from_model, from_model._meta.get_field(self.name)) | ||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.model_name) |             from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|             schema_editor.add_field(from_model, to_model._meta.get_field(self.name)) |             schema_editor.add_field(from_model, to_model._meta.get_field(self.name)) | ||||||
|  |  | ||||||
| @@ -184,7 +184,7 @@ class AlterField(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.model_name) |             from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|             from_field = from_model._meta.get_field(self.name) |             from_field = from_model._meta.get_field(self.name) | ||||||
|             to_field = to_model._meta.get_field(self.name) |             to_field = to_model._meta.get_field(self.name) | ||||||
| @@ -267,7 +267,7 @@ class RenameField(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.model_name) |             from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|             schema_editor.alter_field( |             schema_editor.alter_field( | ||||||
|                 from_model, |                 from_model, | ||||||
| @@ -277,7 +277,7 @@ class RenameField(Operation): | |||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.model_name) |         to_model = to_state.apps.get_model(app_label, self.model_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.model_name) |             from_model = from_state.apps.get_model(app_label, self.model_name) | ||||||
|             schema_editor.alter_field( |             schema_editor.alter_field( | ||||||
|                 from_model, |                 from_model, | ||||||
|   | |||||||
| @@ -55,12 +55,12 @@ class CreateModel(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = to_state.apps.get_model(app_label, self.name) |         model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, model): |         if self.allow_migrate_model(schema_editor.connection.alias, model): | ||||||
|             schema_editor.create_model(model) |             schema_editor.create_model(model) | ||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = from_state.apps.get_model(app_label, self.name) |         model = from_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, model): |         if self.allow_migrate_model(schema_editor.connection.alias, model): | ||||||
|             schema_editor.delete_model(model) |             schema_editor.delete_model(model) | ||||||
|  |  | ||||||
|     def describe(self): |     def describe(self): | ||||||
| @@ -111,12 +111,12 @@ class DeleteModel(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = from_state.apps.get_model(app_label, self.name) |         model = from_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, model): |         if self.allow_migrate_model(schema_editor.connection.alias, model): | ||||||
|             schema_editor.delete_model(model) |             schema_editor.delete_model(model) | ||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         model = to_state.apps.get_model(app_label, self.name) |         model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, model): |         if self.allow_migrate_model(schema_editor.connection.alias, model): | ||||||
|             schema_editor.create_model(model) |             schema_editor.create_model(model) | ||||||
|  |  | ||||||
|     def references_model(self, name, app_label=None): |     def references_model(self, name, app_label=None): | ||||||
| @@ -189,7 +189,7 @@ class RenameModel(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.new_name) |         new_model = to_state.apps.get_model(app_label, self.new_name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, new_model): |         if self.allow_migrate_model(schema_editor.connection.alias, new_model): | ||||||
|             old_model = from_state.apps.get_model(app_label, self.old_name) |             old_model = from_state.apps.get_model(app_label, self.old_name) | ||||||
|             # Move the main table |             # Move the main table | ||||||
|             schema_editor.alter_db_table( |             schema_editor.alter_db_table( | ||||||
| @@ -287,7 +287,7 @@ class AlterModelTable(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, new_model): |         if self.allow_migrate_model(schema_editor.connection.alias, new_model): | ||||||
|             old_model = from_state.apps.get_model(app_label, self.name) |             old_model = from_state.apps.get_model(app_label, self.name) | ||||||
|             schema_editor.alter_db_table( |             schema_editor.alter_db_table( | ||||||
|                 new_model, |                 new_model, | ||||||
| @@ -347,7 +347,7 @@ class AlterUniqueTogether(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, new_model): |         if self.allow_migrate_model(schema_editor.connection.alias, new_model): | ||||||
|             old_model = from_state.apps.get_model(app_label, self.name) |             old_model = from_state.apps.get_model(app_label, self.name) | ||||||
|             schema_editor.alter_unique_together( |             schema_editor.alter_unique_together( | ||||||
|                 new_model, |                 new_model, | ||||||
| @@ -399,7 +399,7 @@ class AlterIndexTogether(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         new_model = to_state.apps.get_model(app_label, self.name) |         new_model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, new_model): |         if self.allow_migrate_model(schema_editor.connection.alias, new_model): | ||||||
|             old_model = from_state.apps.get_model(app_label, self.name) |             old_model = from_state.apps.get_model(app_label, self.name) | ||||||
|             schema_editor.alter_index_together( |             schema_editor.alter_index_together( | ||||||
|                 new_model, |                 new_model, | ||||||
| @@ -448,7 +448,7 @@ class AlterOrderWithRespectTo(Operation): | |||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         to_model = to_state.apps.get_model(app_label, self.name) |         to_model = to_state.apps.get_model(app_label, self.name) | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, to_model): |         if self.allow_migrate_model(schema_editor.connection.alias, to_model): | ||||||
|             from_model = from_state.apps.get_model(app_label, self.name) |             from_model = from_state.apps.get_model(app_label, self.name) | ||||||
|             # Remove a field if we need to |             # Remove a field if we need to | ||||||
|             if from_model._meta.order_with_respect_to and not to_model._meta.order_with_respect_to: |             if from_model._meta.order_with_respect_to and not to_model._meta.order_with_respect_to: | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | from django.db import router | ||||||
|  |  | ||||||
| from .base import Operation | from .base import Operation | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -94,13 +96,13 @@ class RunSQL(Operation): | |||||||
|             state_operation.state_forwards(app_label, state) |             state_operation.state_forwards(app_label, state) | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, None, hints=self.hints): |         if router.allow_migrate(schema_editor.connection.alias, app_label, **self.hints): | ||||||
|             self._run_sql(schema_editor, self.sql) |             self._run_sql(schema_editor, self.sql) | ||||||
|  |  | ||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         if self.reverse_sql is None: |         if self.reverse_sql is None: | ||||||
|             raise NotImplementedError("You cannot reverse this operation") |             raise NotImplementedError("You cannot reverse this operation") | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, None, hints=self.hints): |         if router.allow_migrate(schema_editor.connection.alias, app_label, **self.hints): | ||||||
|             self._run_sql(schema_editor, self.reverse_sql) |             self._run_sql(schema_editor, self.reverse_sql) | ||||||
|  |  | ||||||
|     def describe(self): |     def describe(self): | ||||||
| @@ -171,7 +173,7 @@ class RunPython(Operation): | |||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def database_forwards(self, app_label, schema_editor, from_state, to_state): |     def database_forwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, None, hints=self.hints): |         if router.allow_migrate(schema_editor.connection.alias, app_label, **self.hints): | ||||||
|             # We now execute the Python code in a context that contains a 'models' |             # We now execute the Python code in a context that contains a 'models' | ||||||
|             # object, representing the versioned models as an app registry. |             # object, representing the versioned models as an app registry. | ||||||
|             # We could try to override the global cache, but then people will still |             # We could try to override the global cache, but then people will still | ||||||
| @@ -181,7 +183,7 @@ class RunPython(Operation): | |||||||
|     def database_backwards(self, app_label, schema_editor, from_state, to_state): |     def database_backwards(self, app_label, schema_editor, from_state, to_state): | ||||||
|         if self.reverse_code is None: |         if self.reverse_code is None: | ||||||
|             raise NotImplementedError("You cannot reverse this operation") |             raise NotImplementedError("You cannot reverse this operation") | ||||||
|         if self.allowed_to_migrate(schema_editor.connection.alias, None, hints=self.hints): |         if router.allow_migrate(schema_editor.connection.alias, app_label, **self.hints): | ||||||
|             self.reverse_code(from_state.apps, schema_editor) |             self.reverse_code(from_state.apps, schema_editor) | ||||||
|  |  | ||||||
|     def describe(self): |     def describe(self): | ||||||
|   | |||||||
| @@ -1582,7 +1582,7 @@ class Model(six.with_metaclass(ModelBase)): | |||||||
|         # Find the minimum max allowed length among all specified db_aliases. |         # Find the minimum max allowed length among all specified db_aliases. | ||||||
|         for db in settings.DATABASES.keys(): |         for db in settings.DATABASES.keys(): | ||||||
|             # skip databases where the model won't be created |             # skip databases where the model won't be created | ||||||
|             if not router.allow_migrate(db, cls): |             if not router.allow_migrate_model(db, cls): | ||||||
|                 continue |                 continue | ||||||
|             connection = connections[db] |             connection = connections[db] | ||||||
|             max_name_length = connection.ops.max_name_length() |             max_name_length = connection.ops.max_name_length() | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import inspect | ||||||
| import os | import os | ||||||
| import pkgutil | import pkgutil | ||||||
| import warnings | import warnings | ||||||
| @@ -8,7 +9,9 @@ from django.conf import settings | |||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils._os import upath | from django.utils._os import upath | ||||||
| from django.utils.deprecation import RemovedInDjango19Warning | from django.utils.deprecation import ( | ||||||
|  |     RemovedInDjango19Warning, RemovedInDjango20Warning, | ||||||
|  | ) | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| from django.utils.module_loading import import_string | from django.utils.module_loading import import_string | ||||||
|  |  | ||||||
| @@ -315,7 +318,7 @@ class ConnectionRouter(object): | |||||||
|                     return allow |                     return allow | ||||||
|         return obj1._state.db == obj2._state.db |         return obj1._state.db == obj2._state.db | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model, **hints): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         for router in self.routers: |         for router in self.routers: | ||||||
|             try: |             try: | ||||||
|                 try: |                 try: | ||||||
| @@ -328,16 +331,36 @@ class ConnectionRouter(object): | |||||||
|                         RemovedInDjango19Warning, stacklevel=2) |                         RemovedInDjango19Warning, stacklevel=2) | ||||||
|             except AttributeError: |             except AttributeError: | ||||||
|                 # If the router doesn't have a method, skip to the next one. |                 # If the router doesn't have a method, skip to the next one. | ||||||
|                 pass |                 continue | ||||||
|  |  | ||||||
|  |             argspec = inspect.getargspec(router.allow_migrate) | ||||||
|  |             if len(argspec.args) == 3 and not argspec.keywords: | ||||||
|  |                 warnings.warn( | ||||||
|  |                     "The signature of allow_migrate has changed from " | ||||||
|  |                     "allow_migrate(self, db, model) to " | ||||||
|  |                     "allow_migrate(self, db, app_label, model_name=None, **hints). " | ||||||
|  |                     "Support for the old signature will be removed in Django 2.0.", | ||||||
|  |                     RemovedInDjango20Warning) | ||||||
|  |                 model = hints.get('model') | ||||||
|  |                 allow = None if model is None else method(db, model) | ||||||
|             else: |             else: | ||||||
|                 allow = method(db, model, **hints) |                 allow = method(db, app_label, **hints) | ||||||
|                 if allow is not None: |  | ||||||
|                     return allow |             if allow is not None: | ||||||
|  |                 return allow | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |     def allow_migrate_model(self, db, model): | ||||||
|  |         return self.allow_migrate( | ||||||
|  |             db, | ||||||
|  |             model._meta.app_label, | ||||||
|  |             model_name=model._meta.model_name, | ||||||
|  |             model=model, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def get_migratable_models(self, app_config, db, include_auto_created=False): |     def get_migratable_models(self, app_config, db, include_auto_created=False): | ||||||
|         """ |         """ | ||||||
|         Return app models allowed to be synchronized on provided db. |         Return app models allowed to be synchronized on provided db. | ||||||
|         """ |         """ | ||||||
|         models = app_config.get_models(include_auto_created=include_auto_created) |         models = app_config.get_models(include_auto_created=include_auto_created) | ||||||
|         return [model for model in models if self.allow_migrate(db, model)] |         return [model for model in models if self.allow_migrate_model(db, model)] | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ method of database routers as ``**hints``: | |||||||
|  |  | ||||||
|     class MyRouter(object): |     class MyRouter(object): | ||||||
|  |  | ||||||
|         def allow_migrate(self, db, model, **hints): |         def allow_migrate(self, db, app_label, model_name=None, **hints): | ||||||
|             if 'target_db' in hints: |             if 'target_db' in hints: | ||||||
|                 return db == hints['target_db'] |                 return db == hints['target_db'] | ||||||
|             return True |             return True | ||||||
| @@ -68,6 +68,10 @@ Then, to leverage this in your migrations, do the following:: | |||||||
|             migrations.RunPython(forwards, hints={'target_db': 'default'}), |             migrations.RunPython(forwards, hints={'target_db': 'default'}), | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
|  | If your ``RunPython`` or ``RunSQL`` operation only affects one model, it's good | ||||||
|  | practice to pass ``model_name`` as a hint to make it as transparent as possible | ||||||
|  | to the router. This is especially important for reusable and third-party apps. | ||||||
|  |  | ||||||
| Migrations that add unique fields | Migrations that add unique fields | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -168,6 +168,10 @@ details on these changes. | |||||||
| * Ability to specify ``ContentType.name`` when creating a content type instance | * Ability to specify ``ContentType.name`` when creating a content type instance | ||||||
|   will be removed. |   will be removed. | ||||||
|  |  | ||||||
|  | * Support for the old signature of ``allow_migrate`` will be removed. It changed | ||||||
|  |   from ``allow_migrate(self, db, model)`` to | ||||||
|  |   ``allow_migrate(self, db, app_label, model_name=None, **hints)``. | ||||||
|  |  | ||||||
| .. _deprecation-removed-in-1.9: | .. _deprecation-removed-in-1.9: | ||||||
|  |  | ||||||
| 1.9 | 1.9 | ||||||
|   | |||||||
| @@ -479,11 +479,13 @@ Migrations | |||||||
|   method/attribute were added to ease in making ``RunPython`` and ``RunSQL`` |   method/attribute were added to ease in making ``RunPython`` and ``RunSQL`` | ||||||
|   operations reversible. |   operations reversible. | ||||||
|  |  | ||||||
| * The :class:`~django.db.migrations.operations.RunPython` and | * The migration operations :class:`~django.db.migrations.operations.RunPython` | ||||||
|   :class:`~django.db.migrations.operations.RunSQL` operations now accept a |   and :class:`~django.db.migrations.operations.RunSQL` now call the | ||||||
|   ``hints`` parameter that will be passed to :meth:`allow_migrate`. To take |   :meth:`allow_migrate` method of database routers. The router can use the | ||||||
|   advantage of this feature you must ensure that the ``allow_migrate()`` method |   newly introduced ``app_label`` and ``hints`` arguments to make a routing | ||||||
|   of all your routers accept ``**hints``. |   decision. To take advantage of this feature you need to update the router to | ||||||
|  |   the new ``allow_migrate`` signature, see the :ref:`deprecation section | ||||||
|  |   <deprecated-signature-of-allow-migrate>` for more details. | ||||||
|  |  | ||||||
| Models | Models | ||||||
| ^^^^^^ | ^^^^^^ | ||||||
| @@ -1145,14 +1147,6 @@ Miscellaneous | |||||||
| * :func:`django.utils.translation.get_language()` now returns ``None`` instead | * :func:`django.utils.translation.get_language()` now returns ``None`` instead | ||||||
|   of :setting:`LANGUAGE_CODE` when translations are temporarily deactivated. |   of :setting:`LANGUAGE_CODE` when translations are temporarily deactivated. | ||||||
|  |  | ||||||
| * The migration operations :class:`~django.db.migrations.operations.RunPython` |  | ||||||
|   and :class:`~django.db.migrations.operations.RunSQL` now call the |  | ||||||
|   :meth:`allow_migrate` method of database routers. In these cases the |  | ||||||
|   ``model`` argument of ``allow_migrate()`` is set to ``None``, so the router |  | ||||||
|   must properly handle this value. This is most useful when used together with |  | ||||||
|   the newly introduced ``hints`` parameter for these operations, but it can |  | ||||||
|   also be used to disable migrations from running on a particular database. |  | ||||||
|  |  | ||||||
| * The ``name`` field of :class:`django.contrib.contenttypes.models.ContentType` | * The ``name`` field of :class:`django.contrib.contenttypes.models.ContentType` | ||||||
|   has been removed by a migration and replaced by a property. That means it's |   has been removed by a migration and replaced by a property. That means it's | ||||||
|   not possible to query or filter a ``ContentType`` by this field any longer. |   not possible to query or filter a ``ContentType`` by this field any longer. | ||||||
| @@ -1650,6 +1644,23 @@ aggregate methods are deprecated and should be replaced by their function-based | |||||||
| aggregate equivalents (``Collect``, ``Extent``, ``Extent3D``, ``MakeLine``, and | aggregate equivalents (``Collect``, ``Extent``, ``Extent3D``, ``MakeLine``, and | ||||||
| ``Union``). | ``Union``). | ||||||
|  |  | ||||||
|  | .. _deprecated-signature-of-allow-migrate: | ||||||
|  |  | ||||||
|  | Signature of the ``allow_migrate`` router method | ||||||
|  | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  | The signature of the :meth:`allow_migrate` method of database routers has | ||||||
|  | changed from ``allow_migrate(db, model)`` to | ||||||
|  | ``allow_migrate(db, app_label, model_name=None, **hints)``. | ||||||
|  |  | ||||||
|  | When ``model_name`` is set, the value that was previously given through the | ||||||
|  | ``model`` positional argument may now be found inside the ``hints`` dictionary | ||||||
|  | under the key ``'model'``. | ||||||
|  |  | ||||||
|  | After switching to the new signature the router will also be called by the | ||||||
|  | :class:`~django.db.migrations.operations.RunPython` and | ||||||
|  | :class:`~django.db.migrations.operations.RunSQL` operations. | ||||||
|  |  | ||||||
| .. removed-features-1.8: | .. removed-features-1.8: | ||||||
|  |  | ||||||
| Features removed in 1.8 | Features removed in 1.8 | ||||||
|   | |||||||
| @@ -233,19 +233,19 @@ operations to ``cache_replica``, and all write operations to | |||||||
|  |  | ||||||
|         def db_for_read(self, model, **hints): |         def db_for_read(self, model, **hints): | ||||||
|             "All cache read operations go to the replica" |             "All cache read operations go to the replica" | ||||||
|             if model._meta.app_label in ('django_cache',): |             if model._meta.app_label == 'django_cache': | ||||||
|                 return 'cache_replica' |                 return 'cache_replica' | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         def db_for_write(self, model, **hints): |         def db_for_write(self, model, **hints): | ||||||
|             "All cache write operations go to primary" |             "All cache write operations go to primary" | ||||||
|             if model._meta.app_label in ('django_cache',): |             if model._meta.app_label == 'django_cache': | ||||||
|                 return 'cache_primary' |                 return 'cache_primary' | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         def allow_migrate(self, db, model): |         def allow_migrate(self, db, app_label, model_name=None, **hints): | ||||||
|             "Only install the cache model on primary" |             "Only install the cache model on primary" | ||||||
|             if model._meta.app_label in ('django_cache',): |             if app_label == 'django_cache': | ||||||
|                 return db == 'cache_primary' |                 return db == 'cache_primary' | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|   | |||||||
| @@ -128,7 +128,7 @@ A database Router is a class that provides up to four methods: | |||||||
|     provided in the ``hints`` dictionary. Details on valid hints are |     provided in the ``hints`` dictionary. Details on valid hints are | ||||||
|     provided :ref:`below <topics-db-multi-db-hints>`. |     provided :ref:`below <topics-db-multi-db-hints>`. | ||||||
|  |  | ||||||
|     Returns None if there is no suggestion. |     Returns ``None`` if there is no suggestion. | ||||||
|  |  | ||||||
| .. method:: db_for_write(model, **hints) | .. method:: db_for_write(model, **hints) | ||||||
|  |  | ||||||
| @@ -140,32 +140,53 @@ A database Router is a class that provides up to four methods: | |||||||
|     provided in the ``hints`` dictionary. Details on valid hints are |     provided in the ``hints`` dictionary. Details on valid hints are | ||||||
|     provided :ref:`below <topics-db-multi-db-hints>`. |     provided :ref:`below <topics-db-multi-db-hints>`. | ||||||
|  |  | ||||||
|     Returns None if there is no suggestion. |     Returns ``None`` if there is no suggestion. | ||||||
|  |  | ||||||
| .. method:: allow_relation(obj1, obj2, **hints) | .. method:: allow_relation(obj1, obj2, **hints) | ||||||
|  |  | ||||||
|     Return True if a relation between obj1 and obj2 should be |     Return ``True`` if a relation between ``obj1`` and ``obj2`` should be | ||||||
|     allowed, False if the relation should be prevented, or None if |     allowed, ``False`` if the relation should be prevented, or ``None`` if | ||||||
|     the router has no opinion. This is purely a validation operation, |     the router has no opinion. This is purely a validation operation, | ||||||
|     used by foreign key and many to many operations to determine if a |     used by foreign key and many to many operations to determine if a | ||||||
|     relation should be allowed between two objects. |     relation should be allowed between two objects. | ||||||
|  |  | ||||||
| .. method:: allow_migrate(db, model, **hints) | .. method:: allow_migrate(db, app_label, model_name=None, **hints) | ||||||
|  |  | ||||||
|     Determine if the ``model`` should have tables/indexes created in the |     Determine if the migration operation is allowed to run on the database with | ||||||
|     database with alias ``db``. Return True if the model should be |     alias ``db``. Return ``True`` if the operation should run, ``False`` if it | ||||||
|     migrated, False if it should not be migrated, or None if |     shouldn't run, or ``None`` if the router has no opinion. | ||||||
|     the router has no opinion. This method can be used to determine |  | ||||||
|     the availability of a model on a given database. |  | ||||||
|  |  | ||||||
|     Note that migrations will just silently not perform any operations |     The ``app_label`` positional argument is the label of the application | ||||||
|     on a model for which this returns ``False``. This may result in broken |     being migrated. | ||||||
|     ForeignKeys, extra tables or missing tables if you change it once you |  | ||||||
|     have applied some migrations. |  | ||||||
|  |  | ||||||
|     The value passed for ``model`` may be a |     ``model_name`` is set by most migration operations to the value of | ||||||
|     :ref:`historical model <historical-models>`, and thus not have any |     ``model._meta.model_name`` (the lowercased version of the model | ||||||
|     custom attributes, methods or managers. You should only rely on ``_meta``. |     ``__name__``) of the model being migrated. Its value is ``None`` for the | ||||||
|  |     :class:`~django.db.migrations.operations.RunPython` and | ||||||
|  |     :class:`~django.db.migrations.operations.RunSQL` operations unless they | ||||||
|  |     provide it using hints. | ||||||
|  |  | ||||||
|  |     ``hints`` are used by certain operations to communicate additional | ||||||
|  |     information to the router. | ||||||
|  |  | ||||||
|  |     When ``model_name`` is set, ``hints`` normally contains the model class | ||||||
|  |     under the key ``'model'``. Note that it may be a :ref:`historical model | ||||||
|  |     <historical-models>`, and thus not have any custom attributes, methods, or | ||||||
|  |     managers. You should only rely on ``_meta``. | ||||||
|  |  | ||||||
|  |     This method can also be used to determine the availability of a model on a | ||||||
|  |     given database. | ||||||
|  |  | ||||||
|  |     Note that migrations will just silently not perform any operations on a | ||||||
|  |     model for which this returns ``False``. This may result in broken foreign | ||||||
|  |     keys, extra tables, or missing tables if you change it once you have | ||||||
|  |     applied some migrations. | ||||||
|  |  | ||||||
|  |     .. versionchanged:: 1.8 | ||||||
|  |  | ||||||
|  |         The signature of ``allow_migrate`` has changed significantly from previous | ||||||
|  |         versions. See the :ref:`deprecation notes | ||||||
|  |         <deprecated-signature-of-allow-migrate>` for more details. | ||||||
|  |  | ||||||
| A router doesn't have to provide *all* these methods -- it may omit one | A router doesn't have to provide *all* these methods -- it may omit one | ||||||
| or more of them. If one of the methods is omitted, Django will skip | or more of them. If one of the methods is omitted, Django will skip | ||||||
| @@ -293,15 +314,13 @@ send queries for the ``auth`` app to ``auth_db``:: | |||||||
|                return True |                return True | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         def allow_migrate(self, db, model, **hints): |         def allow_migrate(self, db, app_label, model, **hints): | ||||||
|             """ |             """ | ||||||
|             Make sure the auth app only appears in the 'auth_db' |             Make sure the auth app only appears in the 'auth_db' | ||||||
|             database. |             database. | ||||||
|             """ |             """ | ||||||
|             if db == 'auth_db': |             if app_label == 'auth': | ||||||
|                 return model._meta.app_label == 'auth' |                 return db == 'auth_db' | ||||||
|             elif model._meta.app_label == 'auth': |  | ||||||
|                 return False |  | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
| And we also want a router that sends all other apps to the | And we also want a router that sends all other apps to the | ||||||
| @@ -333,7 +352,7 @@ from:: | |||||||
|                 return True |                 return True | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         def allow_migrate(self, db, model, **hints): |         def allow_migrate(self, db, app_label, model, **hints): | ||||||
|             """ |             """ | ||||||
|             All non-auth models end up in this pool. |             All non-auth models end up in this pool. | ||||||
|             """ |             """ | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								tests/cache/tests.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								tests/cache/tests.py
									
									
									
									
										vendored
									
									
								
							| @@ -959,14 +959,17 @@ class DBCacheRouter(object): | |||||||
|     def db_for_read(self, model, **hints): |     def db_for_read(self, model, **hints): | ||||||
|         if model._meta.app_label == 'django_cache': |         if model._meta.app_label == 'django_cache': | ||||||
|             return 'other' |             return 'other' | ||||||
|  |         return None | ||||||
|  |  | ||||||
|     def db_for_write(self, model, **hints): |     def db_for_write(self, model, **hints): | ||||||
|         if model._meta.app_label == 'django_cache': |         if model._meta.app_label == 'django_cache': | ||||||
|             return 'other' |             return 'other' | ||||||
|  |         return None | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         if model._meta.app_label == 'django_cache': |         if app_label == 'django_cache': | ||||||
|             return db == 'other' |             return db == 'other' | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |  | ||||||
| @override_settings( | @override_settings( | ||||||
|   | |||||||
| @@ -322,7 +322,7 @@ class OtherRouter(object): | |||||||
|     def allow_relation(self, obj1, obj2, **hints): |     def allow_relation(self, obj1, obj2, **hints): | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ class AgnosticRouter(object): | |||||||
|     """ |     """ | ||||||
|     A router that doesn't have an opinion regarding migrating. |     A router that doesn't have an opinion regarding migrating. | ||||||
|     """ |     """ | ||||||
|     def allow_migrate(self, db, model, **hints): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -24,7 +24,7 @@ class MigrateNothingRouter(object): | |||||||
|     """ |     """ | ||||||
|     A router that doesn't allow migrating. |     A router that doesn't allow migrating. | ||||||
|     """ |     """ | ||||||
|     def allow_migrate(self, db, model, **hints): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return False |         return False | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -32,7 +32,7 @@ class MigrateEverythingRouter(object): | |||||||
|     """ |     """ | ||||||
|     A router that always allows migrating. |     A router that always allows migrating. | ||||||
|     """ |     """ | ||||||
|     def allow_migrate(self, db, model, **hints): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -40,7 +40,7 @@ class MigrateWhenFooRouter(object): | |||||||
|     """ |     """ | ||||||
|     A router that allows migrating depending on a hint. |     A router that allows migrating depending on a hint. | ||||||
|     """ |     """ | ||||||
|     def allow_migrate(self, db, model, **hints): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return hints.get('foo', False) |         return hints.get('foo', False) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,8 +4,11 @@ from django.db import DEFAULT_DB_ALIAS | |||||||
|  |  | ||||||
|  |  | ||||||
| class TestRouter(object): | class TestRouter(object): | ||||||
|     # A test router. The behavior is vaguely primary/replica, but the |     """ | ||||||
|     # databases aren't assumed to propagate changes. |     Vaguely behave like primary/replica, but the databases aren't assumed to | ||||||
|  |     propagate changes. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|     def db_for_read(self, model, instance=None, **hints): |     def db_for_read(self, model, instance=None, **hints): | ||||||
|         if instance: |         if instance: | ||||||
|             return instance._state.db or 'other' |             return instance._state.db or 'other' | ||||||
| @@ -17,13 +20,14 @@ class TestRouter(object): | |||||||
|     def allow_relation(self, obj1, obj2, **hints): |     def allow_relation(self, obj1, obj2, **hints): | ||||||
|         return obj1._state.db in ('default', 'other') and obj2._state.db in ('default', 'other') |         return obj1._state.db in ('default', 'other') and obj2._state.db in ('default', 'other') | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthRouter(object): | class AuthRouter(object): | ||||||
|     """A router to control all database operations on models in |     """ | ||||||
|     the contrib.auth application""" |     Control all database operations on models in the contrib.auth application. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|     def db_for_read(self, model, **hints): |     def db_for_read(self, model, **hints): | ||||||
|         "Point all read operations on auth models to 'default'" |         "Point all read operations on auth models to 'default'" | ||||||
| @@ -45,12 +49,10 @@ class AuthRouter(object): | |||||||
|             return True |             return True | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         "Make sure the auth app only appears on the 'other' db" |         "Make sure the auth app only appears on the 'other' db" | ||||||
|         if db == 'other': |         if app_label == 'auth': | ||||||
|             return model._meta.app_label == 'auth' |             return db == 'other' | ||||||
|         elif model._meta.app_label == 'auth': |  | ||||||
|             return False |  | ||||||
|         return None |         return None | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ from __future__ import unicode_literals | |||||||
|  |  | ||||||
| import datetime | import datetime | ||||||
| import pickle | import pickle | ||||||
|  | import warnings | ||||||
| from operator import attrgetter | from operator import attrgetter | ||||||
|  |  | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| @@ -11,6 +12,7 @@ from django.db import DEFAULT_DB_ALIAS, connections, router, transaction | |||||||
| from django.db.models import signals | from django.db.models import signals | ||||||
| from django.db.utils import ConnectionRouter | from django.db.utils import ConnectionRouter | ||||||
| from django.test import TestCase, override_settings | from django.test import TestCase, override_settings | ||||||
|  | from django.utils.encoding import force_text | ||||||
| from django.utils.six import StringIO | from django.utils.six import StringIO | ||||||
|  |  | ||||||
| from .models import Book, Person, Pet, Review, UserProfile | from .models import Book, Person, Pet, Review, UserProfile | ||||||
| @@ -901,28 +903,58 @@ class RouterTestCase(TestCase): | |||||||
|     def test_migrate_selection(self): |     def test_migrate_selection(self): | ||||||
|         "Synchronization behavior is predictable" |         "Synchronization behavior is predictable" | ||||||
|  |  | ||||||
|         self.assertTrue(router.allow_migrate('default', User)) |         self.assertTrue(router.allow_migrate_model('default', User)) | ||||||
|         self.assertTrue(router.allow_migrate('default', Book)) |         self.assertTrue(router.allow_migrate_model('default', Book)) | ||||||
|  |  | ||||||
|         self.assertTrue(router.allow_migrate('other', User)) |         self.assertTrue(router.allow_migrate_model('other', User)) | ||||||
|         self.assertTrue(router.allow_migrate('other', Book)) |         self.assertTrue(router.allow_migrate_model('other', Book)) | ||||||
|  |  | ||||||
|         with override_settings(DATABASE_ROUTERS=[TestRouter(), AuthRouter()]): |         with override_settings(DATABASE_ROUTERS=[TestRouter(), AuthRouter()]): | ||||||
|             # Add the auth router to the chain. TestRouter is a universal |             # Add the auth router to the chain. TestRouter is a universal | ||||||
|             # synchronizer, so it should have no effect. |             # synchronizer, so it should have no effect. | ||||||
|             self.assertTrue(router.allow_migrate('default', User)) |             self.assertTrue(router.allow_migrate_model('default', User)) | ||||||
|             self.assertTrue(router.allow_migrate('default', Book)) |             self.assertTrue(router.allow_migrate_model('default', Book)) | ||||||
|  |  | ||||||
|             self.assertTrue(router.allow_migrate('other', User)) |             self.assertTrue(router.allow_migrate_model('other', User)) | ||||||
|             self.assertTrue(router.allow_migrate('other', Book)) |             self.assertTrue(router.allow_migrate_model('other', Book)) | ||||||
|  |  | ||||||
|         with override_settings(DATABASE_ROUTERS=[AuthRouter(), TestRouter()]): |         with override_settings(DATABASE_ROUTERS=[AuthRouter(), TestRouter()]): | ||||||
|             # Now check what happens if the router order is reversed. |             # Now check what happens if the router order is reversed. | ||||||
|             self.assertFalse(router.allow_migrate('default', User)) |             self.assertFalse(router.allow_migrate_model('default', User)) | ||||||
|             self.assertTrue(router.allow_migrate('default', Book)) |             self.assertTrue(router.allow_migrate_model('default', Book)) | ||||||
|  |  | ||||||
|             self.assertTrue(router.allow_migrate('other', User)) |             self.assertTrue(router.allow_migrate_model('other', User)) | ||||||
|             self.assertFalse(router.allow_migrate('other', Book)) |             self.assertTrue(router.allow_migrate_model('other', Book)) | ||||||
|  |  | ||||||
|  |     def test_migrate_legacy_router(self): | ||||||
|  |         class LegacyRouter(object): | ||||||
|  |             def allow_migrate(self, db, model): | ||||||
|  |                 """ | ||||||
|  |                 Deprecated allow_migrate signature should trigger | ||||||
|  |                 RemovedInDjango20Warning. | ||||||
|  |                 """ | ||||||
|  |                 assert db == 'default' | ||||||
|  |                 assert model is User | ||||||
|  |                 return True | ||||||
|  |  | ||||||
|  |         with override_settings(DATABASE_ROUTERS=[LegacyRouter()]): | ||||||
|  |             with warnings.catch_warnings(record=True) as recorded: | ||||||
|  |                 warnings.filterwarnings('always') | ||||||
|  |  | ||||||
|  |                 msg = ( | ||||||
|  |                     "The signature of allow_migrate has changed from " | ||||||
|  |                     "allow_migrate(self, db, model) to " | ||||||
|  |                     "allow_migrate(self, db, app_label, model_name=None, **hints). " | ||||||
|  |                     "Support for the old signature will be removed in Django 2.0." | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |                 self.assertTrue(router.allow_migrate_model('default', User)) | ||||||
|  |                 self.assertEqual(force_text(recorded.pop().message), msg) | ||||||
|  |  | ||||||
|  |                 self.assertEqual(recorded, []) | ||||||
|  |  | ||||||
|  |                 self.assertTrue(router.allow_migrate('default', 'app_label')) | ||||||
|  |                 self.assertEqual(force_text(recorded.pop().message), msg) | ||||||
|  |  | ||||||
|     def test_partial_router(self): |     def test_partial_router(self): | ||||||
|         "A router can choose to implement a subset of methods" |         "A router can choose to implement a subset of methods" | ||||||
| @@ -939,8 +971,8 @@ class RouterTestCase(TestCase): | |||||||
|  |  | ||||||
|         self.assertTrue(router.allow_relation(dive, dive)) |         self.assertTrue(router.allow_relation(dive, dive)) | ||||||
|  |  | ||||||
|         self.assertTrue(router.allow_migrate('default', User)) |         self.assertTrue(router.allow_migrate_model('default', User)) | ||||||
|         self.assertTrue(router.allow_migrate('default', Book)) |         self.assertTrue(router.allow_migrate_model('default', Book)) | ||||||
|  |  | ||||||
|         with override_settings(DATABASE_ROUTERS=[WriteRouter(), AuthRouter(), TestRouter()]): |         with override_settings(DATABASE_ROUTERS=[WriteRouter(), AuthRouter(), TestRouter()]): | ||||||
|             self.assertEqual(router.db_for_read(User), 'default') |             self.assertEqual(router.db_for_read(User), 'default') | ||||||
| @@ -951,8 +983,8 @@ class RouterTestCase(TestCase): | |||||||
|  |  | ||||||
|             self.assertTrue(router.allow_relation(dive, dive)) |             self.assertTrue(router.allow_relation(dive, dive)) | ||||||
|  |  | ||||||
|             self.assertFalse(router.allow_migrate('default', User)) |             self.assertFalse(router.allow_migrate_model('default', User)) | ||||||
|             self.assertTrue(router.allow_migrate('default', Book)) |             self.assertTrue(router.allow_migrate_model('default', Book)) | ||||||
|  |  | ||||||
|     def test_database_routing(self): |     def test_database_routing(self): | ||||||
|         marty = Person.objects.using('default').create(name="Marty Alchin") |         marty = Person.objects.using('default').create(name="Marty Alchin") | ||||||
| @@ -1492,11 +1524,11 @@ class AntiPetRouter(object): | |||||||
|     # A router that only expresses an opinion on migrate, |     # A router that only expresses an opinion on migrate, | ||||||
|     # passing pets to the 'other' database |     # passing pets to the 'other' database | ||||||
|  |  | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, model_name, **hints): | ||||||
|         if db == 'other': |         if db == 'other': | ||||||
|             return model._meta.object_name == 'Pet' |             return model_name == 'pet' | ||||||
|         else: |         else: | ||||||
|             return model._meta.object_name != 'Pet' |             return model_name != 'pet' | ||||||
|  |  | ||||||
|  |  | ||||||
| class FixtureTestCase(TestCase): | class FixtureTestCase(TestCase): | ||||||
| @@ -1757,7 +1789,7 @@ class RouterModelArgumentTestCase(TestCase): | |||||||
|  |  | ||||||
|  |  | ||||||
| class SyncOnlyDefaultDatabaseRouter(object): | class SyncOnlyDefaultDatabaseRouter(object): | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return db == DEFAULT_DB_ALIAS |         return db == DEFAULT_DB_ALIAS | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -137,7 +137,7 @@ class SitesFrameworkTests(TestCase): | |||||||
|  |  | ||||||
|  |  | ||||||
| class JustOtherRouter(object): | class JustOtherRouter(object): | ||||||
|     def allow_migrate(self, db, model): |     def allow_migrate(self, db, app_label, **hints): | ||||||
|         return db == 'other' |         return db == 'other' | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user