mirror of https://github.com/django/django.git
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:
parent
564487601e
commit
3a6c37fce4
|
@ -66,7 +66,7 @@ def create_permissions(app_config, verbosity=2, interactive=True, using=DEFAULT_
|
|||
except LookupError:
|
||||
return
|
||||
|
||||
if not router.allow_migrate(using, Permission):
|
||||
if not router.allow_migrate_model(using, Permission):
|
||||
return
|
||||
|
||||
from django.contrib.contenttypes.models import ContentType
|
||||
|
|
|
@ -17,7 +17,7 @@ def update_contenttypes(app_config, verbosity=2, interactive=True, using=DEFAULT
|
|||
except LookupError:
|
||||
return
|
||||
|
||||
if not router.allow_migrate(using, ContentType):
|
||||
if not router.allow_migrate_model(using, ContentType):
|
||||
return
|
||||
|
||||
ContentType.objects.clear_cache()
|
||||
|
|
|
@ -33,6 +33,7 @@ class Migration(migrations.Migration):
|
|||
migrations.RunPython(
|
||||
migrations.RunPython.noop,
|
||||
add_legacy_name,
|
||||
hints={'model_name': 'contenttype'},
|
||||
),
|
||||
migrations.RemoveField(
|
||||
model_name='contenttype',
|
||||
|
|
|
@ -14,7 +14,7 @@ def create_default_site(app_config, verbosity=2, interactive=True, using=DEFAULT
|
|||
except LookupError:
|
||||
return
|
||||
|
||||
if not router.allow_migrate(using, Site):
|
||||
if not router.allow_migrate_model(using, Site):
|
||||
return
|
||||
|
||||
if not Site.objects.using(using).exists():
|
||||
|
|
|
@ -30,6 +30,7 @@ class Options(object):
|
|||
self.abstract = False
|
||||
self.managed = True
|
||||
self.proxy = False
|
||||
self.swapped = False
|
||||
|
||||
|
||||
class BaseDatabaseCache(BaseCache):
|
||||
|
|
|
@ -38,7 +38,7 @@ class Command(BaseCommand):
|
|||
|
||||
def create_table(self, database, 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
|
||||
connection = connections[database]
|
||||
|
||||
|
|
|
@ -140,7 +140,7 @@ class Command(BaseCommand):
|
|||
for model in serializers.sort_dependencies(app_list.items()):
|
||||
if model in excluded_models:
|
||||
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:
|
||||
objects = model._base_manager
|
||||
else:
|
||||
|
|
|
@ -140,7 +140,7 @@ class Command(BaseCommand):
|
|||
|
||||
for obj in objects:
|
||||
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
|
||||
self.models.add(obj.object.__class__)
|
||||
try:
|
||||
|
|
|
@ -404,7 +404,7 @@ class BaseDatabaseCreation(object):
|
|||
def get_objects():
|
||||
for model in serializers.sort_dependencies(app_list):
|
||||
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)
|
||||
for obj in queryset.iterator():
|
||||
yield obj
|
||||
|
|
|
@ -99,15 +99,17 @@ class Operation(object):
|
|||
"""
|
||||
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.
|
||||
|
||||
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 and (model._meta.proxy or model._meta.swapped or not model._meta.managed):
|
||||
if model._meta.proxy or model._meta.swapped or not model._meta.managed:
|
||||
return False
|
||||
|
||||
return router.allow_migrate(connection_alias, model, **(hints or {}))
|
||||
return router.allow_migrate_model(connection_alias, model)
|
||||
|
||||
def __repr__(self):
|
||||
return "<%s %s%s>" % (
|
||||
|
|
|
@ -52,7 +52,7 @@ class AddField(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
field = to_model._meta.get_field(self.name)
|
||||
if not self.preserve_default:
|
||||
|
@ -66,7 +66,7 @@ class AddField(Operation):
|
|||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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))
|
||||
|
||||
def describe(self):
|
||||
|
@ -117,12 +117,12 @@ class RemoveField(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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))
|
||||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
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):
|
||||
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_field = from_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):
|
||||
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)
|
||||
schema_editor.alter_field(
|
||||
from_model,
|
||||
|
@ -277,7 +277,7 @@ class RenameField(Operation):
|
|||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
schema_editor.alter_field(
|
||||
from_model,
|
||||
|
|
|
@ -55,12 +55,12 @@ class CreateModel(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
|
||||
def describe(self):
|
||||
|
@ -111,12 +111,12 @@ class DeleteModel(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
|
||||
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):
|
||||
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)
|
||||
# Move the main 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):
|
||||
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)
|
||||
schema_editor.alter_db_table(
|
||||
new_model,
|
||||
|
@ -347,7 +347,7 @@ class AlterUniqueTogether(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
schema_editor.alter_unique_together(
|
||||
new_model,
|
||||
|
@ -399,7 +399,7 @@ class AlterIndexTogether(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
schema_editor.alter_index_together(
|
||||
new_model,
|
||||
|
@ -448,7 +448,7 @@ class AlterOrderWithRespectTo(Operation):
|
|||
|
||||
def database_forwards(self, app_label, schema_editor, from_state, to_state):
|
||||
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)
|
||||
# Remove a field if we need 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 django.db import router
|
||||
|
||||
from .base import Operation
|
||||
|
||||
|
||||
|
@ -94,13 +96,13 @@ class RunSQL(Operation):
|
|||
state_operation.state_forwards(app_label, 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)
|
||||
|
||||
def database_backwards(self, app_label, schema_editor, from_state, to_state):
|
||||
if self.reverse_sql is None:
|
||||
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)
|
||||
|
||||
def describe(self):
|
||||
|
@ -171,7 +173,7 @@ class RunPython(Operation):
|
|||
pass
|
||||
|
||||
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'
|
||||
# object, representing the versioned models as an app registry.
|
||||
# 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):
|
||||
if self.reverse_code is None:
|
||||
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)
|
||||
|
||||
def describe(self):
|
||||
|
|
|
@ -1582,7 +1582,7 @@ class Model(six.with_metaclass(ModelBase)):
|
|||
# Find the minimum max allowed length among all specified db_aliases.
|
||||
for db in settings.DATABASES.keys():
|
||||
# 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
|
||||
connection = connections[db]
|
||||
max_name_length = connection.ops.max_name_length()
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
import inspect
|
||||
import os
|
||||
import pkgutil
|
||||
import warnings
|
||||
|
@ -8,7 +9,9 @@ from django.conf import settings
|
|||
from django.core.exceptions import ImproperlyConfigured
|
||||
from django.utils import six
|
||||
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.module_loading import import_string
|
||||
|
||||
|
@ -315,7 +318,7 @@ class ConnectionRouter(object):
|
|||
return allow
|
||||
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:
|
||||
try:
|
||||
try:
|
||||
|
@ -328,16 +331,36 @@ class ConnectionRouter(object):
|
|||
RemovedInDjango19Warning, stacklevel=2)
|
||||
except AttributeError:
|
||||
# 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:
|
||||
allow = method(db, model, **hints)
|
||||
if allow is not None:
|
||||
return allow
|
||||
allow = method(db, app_label, **hints)
|
||||
|
||||
if allow is not None:
|
||||
return allow
|
||||
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):
|
||||
"""
|
||||
Return app models allowed to be synchronized on provided db.
|
||||
"""
|
||||
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):
|
||||
|
||||
def allow_migrate(self, db, model, **hints):
|
||||
def allow_migrate(self, db, app_label, model_name=None, **hints):
|
||||
if 'target_db' in hints:
|
||||
return db == hints['target_db']
|
||||
return True
|
||||
|
@ -68,6 +68,10 @@ Then, to leverage this in your migrations, do the following::
|
|||
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
|
||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
|
|
|
@ -168,6 +168,10 @@ details on these changes.
|
|||
* Ability to specify ``ContentType.name`` when creating a content type instance
|
||||
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:
|
||||
|
||||
1.9
|
||||
|
|
|
@ -479,11 +479,13 @@ Migrations
|
|||
method/attribute were added to ease in making ``RunPython`` and ``RunSQL``
|
||||
operations reversible.
|
||||
|
||||
* The :class:`~django.db.migrations.operations.RunPython` and
|
||||
:class:`~django.db.migrations.operations.RunSQL` operations now accept a
|
||||
``hints`` parameter that will be passed to :meth:`allow_migrate`. To take
|
||||
advantage of this feature you must ensure that the ``allow_migrate()`` method
|
||||
of all your routers accept ``**hints``.
|
||||
* 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. The router can use the
|
||||
newly introduced ``app_label`` and ``hints`` arguments to make a routing
|
||||
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
|
||||
^^^^^^
|
||||
|
@ -1145,14 +1147,6 @@ Miscellaneous
|
|||
* :func:`django.utils.translation.get_language()` now returns ``None`` instead
|
||||
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`
|
||||
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.
|
||||
|
@ -1650,6 +1644,23 @@ aggregate methods are deprecated and should be replaced by their function-based
|
|||
aggregate equivalents (``Collect``, ``Extent``, ``Extent3D``, ``MakeLine``, and
|
||||
``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:
|
||||
|
||||
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):
|
||||
"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 None
|
||||
|
||||
def db_for_write(self, model, **hints):
|
||||
"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 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"
|
||||
if model._meta.app_label in ('django_cache',):
|
||||
if app_label == 'django_cache':
|
||||
return db == 'cache_primary'
|
||||
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 :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)
|
||||
|
||||
|
@ -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 :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)
|
||||
|
||||
Return True if a relation between obj1 and obj2 should be
|
||||
allowed, False if the relation should be prevented, or None if
|
||||
Return ``True`` if a relation between ``obj1`` and ``obj2`` should be
|
||||
allowed, ``False`` if the relation should be prevented, or ``None`` if
|
||||
the router has no opinion. This is purely a validation operation,
|
||||
used by foreign key and many to many operations to determine if a
|
||||
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
|
||||
database with alias ``db``. Return True if the model should be
|
||||
migrated, False if it should not be migrated, or None if
|
||||
the router has no opinion. This method can be used to determine
|
||||
the availability of a model on a given database.
|
||||
Determine if the migration operation is allowed to run on the database with
|
||||
alias ``db``. Return ``True`` if the operation should run, ``False`` if it
|
||||
shouldn't run, or ``None`` if the router has no opinion.
|
||||
|
||||
Note that migrations will just silently not perform any operations
|
||||
on a model for which this returns ``False``. This may result in broken
|
||||
ForeignKeys, extra tables or missing tables if you change it once you
|
||||
have applied some migrations.
|
||||
The ``app_label`` positional argument is the label of the application
|
||||
being migrated.
|
||||
|
||||
The value passed for ``model`` may be a
|
||||
:ref:`historical model <historical-models>`, and thus not have any
|
||||
custom attributes, methods or managers. You should only rely on ``_meta``.
|
||||
``model_name`` is set by most migration operations to the value of
|
||||
``model._meta.model_name`` (the lowercased version of the model
|
||||
``__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
|
||||
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 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'
|
||||
database.
|
||||
"""
|
||||
if db == 'auth_db':
|
||||
return model._meta.app_label == 'auth'
|
||||
elif model._meta.app_label == 'auth':
|
||||
return False
|
||||
if app_label == 'auth':
|
||||
return db == 'auth_db'
|
||||
return None
|
||||
|
||||
And we also want a router that sends all other apps to the
|
||||
|
@ -333,7 +352,7 @@ from::
|
|||
return True
|
||||
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.
|
||||
"""
|
||||
|
|
|
@ -959,14 +959,17 @@ class DBCacheRouter(object):
|
|||
def db_for_read(self, model, **hints):
|
||||
if model._meta.app_label == 'django_cache':
|
||||
return 'other'
|
||||
return None
|
||||
|
||||
def db_for_write(self, model, **hints):
|
||||
if model._meta.app_label == 'django_cache':
|
||||
return 'other'
|
||||
return None
|
||||
|
||||
def allow_migrate(self, db, model):
|
||||
if model._meta.app_label == 'django_cache':
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
if app_label == 'django_cache':
|
||||
return db == 'other'
|
||||
return None
|
||||
|
||||
|
||||
@override_settings(
|
||||
|
|
|
@ -322,7 +322,7 @@ class OtherRouter(object):
|
|||
def allow_relation(self, obj1, obj2, **hints):
|
||||
return None
|
||||
|
||||
def allow_migrate(self, db, model):
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
return True
|
||||
|
||||
|
||||
|
|
|
@ -16,7 +16,7 @@ class AgnosticRouter(object):
|
|||
"""
|
||||
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
|
||||
|
||||
|
||||
|
@ -24,7 +24,7 @@ class MigrateNothingRouter(object):
|
|||
"""
|
||||
A router that doesn't allow migrating.
|
||||
"""
|
||||
def allow_migrate(self, db, model, **hints):
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
return False
|
||||
|
||||
|
||||
|
@ -32,7 +32,7 @@ class MigrateEverythingRouter(object):
|
|||
"""
|
||||
A router that always allows migrating.
|
||||
"""
|
||||
def allow_migrate(self, db, model, **hints):
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
return True
|
||||
|
||||
|
||||
|
@ -40,7 +40,7 @@ class MigrateWhenFooRouter(object):
|
|||
"""
|
||||
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)
|
||||
|
||||
|
||||
|
|
|
@ -4,8 +4,11 @@ from django.db import DEFAULT_DB_ALIAS
|
|||
|
||||
|
||||
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):
|
||||
if instance:
|
||||
return instance._state.db or 'other'
|
||||
|
@ -17,13 +20,14 @@ class TestRouter(object):
|
|||
def allow_relation(self, obj1, obj2, **hints):
|
||||
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
|
||||
|
||||
|
||||
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):
|
||||
"Point all read operations on auth models to 'default'"
|
||||
|
@ -45,12 +49,10 @@ class AuthRouter(object):
|
|||
return True
|
||||
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"
|
||||
if db == 'other':
|
||||
return model._meta.app_label == 'auth'
|
||||
elif model._meta.app_label == 'auth':
|
||||
return False
|
||||
if app_label == 'auth':
|
||||
return db == 'other'
|
||||
return None
|
||||
|
||||
|
||||
|
|
|
@ -2,6 +2,7 @@ from __future__ import unicode_literals
|
|||
|
||||
import datetime
|
||||
import pickle
|
||||
import warnings
|
||||
from operator import attrgetter
|
||||
|
||||
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.utils import ConnectionRouter
|
||||
from django.test import TestCase, override_settings
|
||||
from django.utils.encoding import force_text
|
||||
from django.utils.six import StringIO
|
||||
|
||||
from .models import Book, Person, Pet, Review, UserProfile
|
||||
|
@ -901,28 +903,58 @@ class RouterTestCase(TestCase):
|
|||
def test_migrate_selection(self):
|
||||
"Synchronization behavior is predictable"
|
||||
|
||||
self.assertTrue(router.allow_migrate('default', User))
|
||||
self.assertTrue(router.allow_migrate('default', Book))
|
||||
self.assertTrue(router.allow_migrate_model('default', User))
|
||||
self.assertTrue(router.allow_migrate_model('default', Book))
|
||||
|
||||
self.assertTrue(router.allow_migrate('other', User))
|
||||
self.assertTrue(router.allow_migrate('other', Book))
|
||||
self.assertTrue(router.allow_migrate_model('other', User))
|
||||
self.assertTrue(router.allow_migrate_model('other', Book))
|
||||
|
||||
with override_settings(DATABASE_ROUTERS=[TestRouter(), AuthRouter()]):
|
||||
# Add the auth router to the chain. TestRouter is a universal
|
||||
# synchronizer, so it should have no effect.
|
||||
self.assertTrue(router.allow_migrate('default', User))
|
||||
self.assertTrue(router.allow_migrate('default', Book))
|
||||
self.assertTrue(router.allow_migrate_model('default', User))
|
||||
self.assertTrue(router.allow_migrate_model('default', Book))
|
||||
|
||||
self.assertTrue(router.allow_migrate('other', User))
|
||||
self.assertTrue(router.allow_migrate('other', Book))
|
||||
self.assertTrue(router.allow_migrate_model('other', User))
|
||||
self.assertTrue(router.allow_migrate_model('other', Book))
|
||||
|
||||
with override_settings(DATABASE_ROUTERS=[AuthRouter(), TestRouter()]):
|
||||
# Now check what happens if the router order is reversed.
|
||||
self.assertFalse(router.allow_migrate('default', User))
|
||||
self.assertTrue(router.allow_migrate('default', Book))
|
||||
self.assertFalse(router.allow_migrate_model('default', User))
|
||||
self.assertTrue(router.allow_migrate_model('default', Book))
|
||||
|
||||
self.assertTrue(router.allow_migrate('other', User))
|
||||
self.assertFalse(router.allow_migrate('other', Book))
|
||||
self.assertTrue(router.allow_migrate_model('other', User))
|
||||
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):
|
||||
"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_migrate('default', User))
|
||||
self.assertTrue(router.allow_migrate('default', Book))
|
||||
self.assertTrue(router.allow_migrate_model('default', User))
|
||||
self.assertTrue(router.allow_migrate_model('default', Book))
|
||||
|
||||
with override_settings(DATABASE_ROUTERS=[WriteRouter(), AuthRouter(), TestRouter()]):
|
||||
self.assertEqual(router.db_for_read(User), 'default')
|
||||
|
@ -951,8 +983,8 @@ class RouterTestCase(TestCase):
|
|||
|
||||
self.assertTrue(router.allow_relation(dive, dive))
|
||||
|
||||
self.assertFalse(router.allow_migrate('default', User))
|
||||
self.assertTrue(router.allow_migrate('default', Book))
|
||||
self.assertFalse(router.allow_migrate_model('default', User))
|
||||
self.assertTrue(router.allow_migrate_model('default', Book))
|
||||
|
||||
def test_database_routing(self):
|
||||
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,
|
||||
# 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':
|
||||
return model._meta.object_name == 'Pet'
|
||||
return model_name == 'pet'
|
||||
else:
|
||||
return model._meta.object_name != 'Pet'
|
||||
return model_name != 'pet'
|
||||
|
||||
|
||||
class FixtureTestCase(TestCase):
|
||||
|
@ -1757,7 +1789,7 @@ class RouterModelArgumentTestCase(TestCase):
|
|||
|
||||
|
||||
class SyncOnlyDefaultDatabaseRouter(object):
|
||||
def allow_migrate(self, db, model):
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
return db == DEFAULT_DB_ALIAS
|
||||
|
||||
|
||||
|
|
|
@ -137,7 +137,7 @@ class SitesFrameworkTests(TestCase):
|
|||
|
||||
|
||||
class JustOtherRouter(object):
|
||||
def allow_migrate(self, db, model):
|
||||
def allow_migrate(self, db, app_label, **hints):
|
||||
return db == 'other'
|
||||
|
||||
|
||||
|
|
Loading…
Reference in New Issue