mirror of
https://github.com/django/django.git
synced 2025-10-24 14:16:09 +00:00
Fixed #22667 -- Replaced leader/follower terminology with primary/replica
This commit is contained in:
committed by
Tim Graham
parent
ad994a3c5b
commit
beec05686c
@@ -649,10 +649,11 @@ Default: ``None``
|
||||
The alias of the database that this database should mirror during
|
||||
testing.
|
||||
|
||||
This setting exists to allow for testing of leader/follower
|
||||
This setting exists to allow for testing of primary/replica
|
||||
(referred to as master/slave by some databases)
|
||||
configurations of multiple databases. See the documentation on
|
||||
:ref:`testing leader/follower configurations
|
||||
<topics-testing-leaderfollower>` for details.
|
||||
:ref:`testing primary/replica configurations
|
||||
<topics-testing-primaryreplica>` for details.
|
||||
|
||||
.. setting:: TEST_NAME
|
||||
|
||||
|
@@ -222,29 +222,29 @@ won't appear in the models cache, but the model details can be used
|
||||
for routing purposes.
|
||||
|
||||
For example, the following router would direct all cache read
|
||||
operations to ``cache_follower``, and all write operations to
|
||||
``cache_leader``. The cache table will only be synchronized onto
|
||||
``cache_leader``::
|
||||
operations to ``cache_replica``, and all write operations to
|
||||
``cache_primary``. The cache table will only be synchronized onto
|
||||
``cache_primary``::
|
||||
|
||||
class CacheRouter(object):
|
||||
"""A router to control all database cache operations"""
|
||||
|
||||
def db_for_read(self, model, **hints):
|
||||
"All cache read operations go to the follower"
|
||||
"All cache read operations go to the replica"
|
||||
if model._meta.app_label in ('django_cache',):
|
||||
return 'cache_follower'
|
||||
return 'cache_replica'
|
||||
return None
|
||||
|
||||
def db_for_write(self, model, **hints):
|
||||
"All cache write operations go to leader"
|
||||
"All cache write operations go to primary"
|
||||
if model._meta.app_label in ('django_cache',):
|
||||
return 'cache_leader'
|
||||
return 'cache_primary'
|
||||
return None
|
||||
|
||||
def allow_migrate(self, db, model):
|
||||
"Only install the cache model on leader"
|
||||
"Only install the cache model on primary"
|
||||
if model._meta.app_label in ('django_cache',):
|
||||
return db == 'cache_leader'
|
||||
return db == 'cache_primary'
|
||||
return None
|
||||
|
||||
If you don't specify routing directions for the database cache model,
|
||||
|
@@ -197,17 +197,17 @@ Using routers
|
||||
|
||||
Database routers are installed using the :setting:`DATABASE_ROUTERS`
|
||||
setting. This setting defines a list of class names, each specifying a
|
||||
router that should be used by the leader router
|
||||
router that should be used by the master router
|
||||
(``django.db.router``).
|
||||
|
||||
The leader router is used by Django's database operations to allocate
|
||||
The master router is used by Django's database operations to allocate
|
||||
database usage. Whenever a query needs to know which database to use,
|
||||
it calls the leader router, providing a model and a hint (if
|
||||
it calls the master router, providing a model and a hint (if
|
||||
available). Django then tries each router in turn until a database
|
||||
suggestion can be found. If no suggestion can be found, it tries the
|
||||
current ``_state.db`` of the hint instance. If a hint instance wasn't
|
||||
provided, or the instance doesn't currently have database state, the
|
||||
leader router will allocate the ``default`` database.
|
||||
master router will allocate the ``default`` database.
|
||||
|
||||
An example
|
||||
----------
|
||||
@@ -225,16 +225,17 @@ An example
|
||||
introduce referential integrity problems that Django can't
|
||||
currently handle.
|
||||
|
||||
The leader/follower configuration described is also flawed -- it
|
||||
The primary/replica (referred to as master/slave by some databases)
|
||||
configuration described is also flawed -- it
|
||||
doesn't provide any solution for handling replication lag (i.e.,
|
||||
query inconsistencies introduced because of the time taken for a
|
||||
write to propagate to the followers). It also doesn't consider the
|
||||
write to propagate to the replicas). It also doesn't consider the
|
||||
interaction of transactions with the database utilization strategy.
|
||||
|
||||
So - what does this mean in practice? Let's consider another sample
|
||||
configuration. This one will have several databases: one for the
|
||||
``auth`` application, and all other apps using a leader/follower setup
|
||||
with two read followers. Here are the settings specifying these
|
||||
``auth`` application, and all other apps using a primary/replica setup
|
||||
with two read replicas. Here are the settings specifying these
|
||||
databases::
|
||||
|
||||
DATABASES = {
|
||||
@@ -244,20 +245,20 @@ databases::
|
||||
'USER': 'mysql_user',
|
||||
'PASSWORD': 'swordfish',
|
||||
},
|
||||
'leader': {
|
||||
'NAME': 'leader',
|
||||
'primary': {
|
||||
'NAME': 'primary',
|
||||
'ENGINE': 'django.db.backends.mysql',
|
||||
'USER': 'mysql_user',
|
||||
'PASSWORD': 'spam',
|
||||
},
|
||||
'follower1': {
|
||||
'NAME': 'follower1',
|
||||
'replica1': {
|
||||
'NAME': 'replica1',
|
||||
'ENGINE': 'django.db.backends.mysql',
|
||||
'USER': 'mysql_user',
|
||||
'PASSWORD': 'eggs',
|
||||
},
|
||||
'follower2': {
|
||||
'NAME': 'follower2',
|
||||
'replica2': {
|
||||
'NAME': 'replica2',
|
||||
'ENGINE': 'django.db.backends.mysql',
|
||||
'USER': 'mysql_user',
|
||||
'PASSWORD': 'bacon',
|
||||
@@ -309,30 +310,30 @@ send queries for the ``auth`` app to ``auth_db``::
|
||||
return None
|
||||
|
||||
And we also want a router that sends all other apps to the
|
||||
leader/follower configuration, and randomly chooses a follower to read
|
||||
primary/replica configuration, and randomly chooses a replica to read
|
||||
from::
|
||||
|
||||
import random
|
||||
|
||||
class LeaderFollowerRouter(object):
|
||||
class PrimaryReplicaRouter(object):
|
||||
def db_for_read(self, model, **hints):
|
||||
"""
|
||||
Reads go to a randomly-chosen follower.
|
||||
Reads go to a randomly-chosen replica.
|
||||
"""
|
||||
return random.choice(['follower1', 'follower2'])
|
||||
return random.choice(['replica1', 'replica2'])
|
||||
|
||||
def db_for_write(self, model, **hints):
|
||||
"""
|
||||
Writes always go to leader.
|
||||
Writes always go to primary.
|
||||
"""
|
||||
return 'leader'
|
||||
return 'primary'
|
||||
|
||||
def allow_relation(self, obj1, obj2, **hints):
|
||||
"""
|
||||
Relations between objects are allowed if both objects are
|
||||
in the leader/follower pool.
|
||||
in the primary/replica pool.
|
||||
"""
|
||||
db_list = ('leader', 'follower1', 'follower2')
|
||||
db_list = ('primary', 'replica1', 'replica2')
|
||||
if obj1._state.db in db_list and obj2._state.db in db_list:
|
||||
return True
|
||||
return None
|
||||
@@ -347,17 +348,17 @@ Finally, in the settings file, we add the following (substituting
|
||||
``path.to.`` with the actual python path to the module(s) where the
|
||||
routers are defined)::
|
||||
|
||||
DATABASE_ROUTERS = ['path.to.AuthRouter', 'path.to.LeaderFollowerRouter']
|
||||
DATABASE_ROUTERS = ['path.to.AuthRouter', 'path.to.PrimaryReplicaRouter']
|
||||
|
||||
The order in which routers are processed is significant. Routers will
|
||||
be queried in the order the are listed in the
|
||||
:setting:`DATABASE_ROUTERS` setting . In this example, the
|
||||
``AuthRouter`` is processed before the ``LeaderFollowerRouter``, and as a
|
||||
``AuthRouter`` is processed before the ``PrimaryReplicaRouter``, and as a
|
||||
result, decisions concerning the models in ``auth`` are processed
|
||||
before any other decision is made. If the :setting:`DATABASE_ROUTERS`
|
||||
setting listed the two routers in the other order,
|
||||
``LeaderFollowerRouter.allow_migrate()`` would be processed first. The
|
||||
catch-all nature of the LeaderFollowerRouter implementation would mean
|
||||
``PrimaryReplicaRouter.allow_migrate()`` would be processed first. The
|
||||
catch-all nature of the PrimaryReplicaRouter implementation would mean
|
||||
that all models would be available on all databases.
|
||||
|
||||
With this setup installed, lets run some Django code::
|
||||
@@ -369,7 +370,7 @@ With this setup installed, lets run some Django code::
|
||||
>>> # This save will also be directed to 'auth_db'
|
||||
>>> fred.save()
|
||||
|
||||
>>> # These retrieval will be randomly allocated to a follower database
|
||||
>>> # These retrieval will be randomly allocated to a replica database
|
||||
>>> dna = Person.objects.get(name='Douglas Adams')
|
||||
|
||||
>>> # A new object has no database allocation when created
|
||||
@@ -379,10 +380,10 @@ With this setup installed, lets run some Django code::
|
||||
>>> # the same database as the author object
|
||||
>>> mh.author = dna
|
||||
|
||||
>>> # This save will force the 'mh' instance onto the leader database...
|
||||
>>> # This save will force the 'mh' instance onto the primary database...
|
||||
>>> mh.save()
|
||||
|
||||
>>> # ... but if we re-retrieve the object, it will come back on a follower
|
||||
>>> # ... but if we re-retrieve the object, it will come back on a replica
|
||||
>>> mh = Book.objects.get(title='Mostly Harmless')
|
||||
|
||||
|
||||
@@ -690,7 +691,7 @@ In addition, some objects are automatically created just after
|
||||
database).
|
||||
|
||||
For common setups with multiple databases, it isn't useful to have these
|
||||
objects in more than one database. Common setups include leader / follower and
|
||||
objects in more than one database. Common setups include primary/replica and
|
||||
connecting to external databases. Therefore, it's recommended:
|
||||
|
||||
- either to run :djadmin:`migrate` only for the default database;
|
||||
|
@@ -64,16 +64,17 @@ The following is a simple unit test using the request factory::
|
||||
Tests and multiple databases
|
||||
============================
|
||||
|
||||
.. _topics-testing-leaderfollower:
|
||||
.. _topics-testing-primaryreplica:
|
||||
|
||||
Testing leader/follower configurations
|
||||
Testing primary/replica configurations
|
||||
-----------------------------------
|
||||
|
||||
If you're testing a multiple database configuration with leader/follower
|
||||
replication, this strategy of creating test databases poses a problem.
|
||||
If you're testing a multiple database configuration with primary/replica
|
||||
(referred to as master/slave by some databases) replication, this strategy of
|
||||
creating test databases poses a problem.
|
||||
When the test databases are created, there won't be any replication,
|
||||
and as a result, data created on the leader won't be seen on the
|
||||
follower.
|
||||
and as a result, data created on the primary won't be seen on the
|
||||
replica.
|
||||
|
||||
To compensate for this, Django allows you to define that a database is
|
||||
a *test mirror*. Consider the following (simplified) example database
|
||||
@@ -83,34 +84,34 @@ configuration::
|
||||
'default': {
|
||||
'ENGINE': 'django.db.backends.mysql',
|
||||
'NAME': 'myproject',
|
||||
'HOST': 'dbleader',
|
||||
'HOST': 'dbprimary',
|
||||
# ... plus some other settings
|
||||
},
|
||||
'follower': {
|
||||
'replica': {
|
||||
'ENGINE': 'django.db.backends.mysql',
|
||||
'NAME': 'myproject',
|
||||
'HOST': 'dbfollower',
|
||||
'HOST': 'dbreplica',
|
||||
'TEST_MIRROR': 'default'
|
||||
# ... plus some other settings
|
||||
}
|
||||
}
|
||||
|
||||
In this setup, we have two database servers: ``dbleader``, described
|
||||
by the database alias ``default``, and ``dbfollower`` described by the
|
||||
alias ``follower``. As you might expect, ``dbfollower`` has been configured
|
||||
by the database administrator as a read follower of ``dbleader``, so in
|
||||
normal activity, any write to ``default`` will appear on ``follower``.
|
||||
In this setup, we have two database servers: ``dbprimary``, described
|
||||
by the database alias ``default``, and ``dbreplica`` described by the
|
||||
alias ``replica``. As you might expect, ``dbreplica`` has been configured
|
||||
by the database administrator as a read replica of ``dbprimary``, so in
|
||||
normal activity, any write to ``default`` will appear on ``replica``.
|
||||
|
||||
If Django created two independent test databases, this would break any
|
||||
tests that expected replication to occur. However, the ``follower``
|
||||
tests that expected replication to occur. However, the ``replica``
|
||||
database has been configured as a test mirror (using the
|
||||
:setting:`TEST_MIRROR` setting), indicating that under testing,
|
||||
``follower`` should be treated as a mirror of ``default``.
|
||||
``replica`` should be treated as a mirror of ``default``.
|
||||
|
||||
When the test environment is configured, a test version of ``follower``
|
||||
will *not* be created. Instead the connection to ``follower``
|
||||
When the test environment is configured, a test version of ``replica``
|
||||
will *not* be created. Instead the connection to ``replica``
|
||||
will be redirected to point at ``default``. As a result, writes to
|
||||
``default`` will appear on ``follower`` -- but because they are actually
|
||||
``default`` will appear on ``replica`` -- but because they are actually
|
||||
the same database, not because there is data replication between the
|
||||
two databases.
|
||||
|
||||
|
@@ -980,7 +980,7 @@ class MultiDBOperationTests(MigrationTestBase):
|
||||
multi_db = True
|
||||
|
||||
def setUp(self):
|
||||
# Make the 'other' database appear to be a follower of the 'default'
|
||||
# Make the 'other' database appear to be a replica of the 'default'
|
||||
self.old_routers = router.routers
|
||||
router.routers = [MigrateNothingRouter()]
|
||||
|
||||
|
@@ -4,7 +4,7 @@ from django.db import DEFAULT_DB_ALIAS
|
||||
|
||||
|
||||
class TestRouter(object):
|
||||
# A test router. The behavior is vaguely leader/follower, but the
|
||||
# A test router. The behavior is vaguely primary/replica, but the
|
||||
# databases aren't assumed to propagate changes.
|
||||
def db_for_read(self, model, instance=None, **hints):
|
||||
if instance:
|
||||
|
@@ -854,7 +854,7 @@ class QueryTestCase(TestCase):
|
||||
self.assertEqual(book.editor._state.db, 'other')
|
||||
|
||||
def test_subquery(self):
|
||||
"""Make sure as_sql works with subqueries and leader/follower."""
|
||||
"""Make sure as_sql works with subqueries and primary/replica."""
|
||||
sub = Person.objects.using('other').filter(name='fff')
|
||||
qs = Book.objects.filter(editor__in=sub)
|
||||
|
||||
@@ -919,7 +919,7 @@ class RouterTestCase(TestCase):
|
||||
multi_db = True
|
||||
|
||||
def setUp(self):
|
||||
# Make the 'other' database appear to be a follower of the 'default'
|
||||
# Make the 'other' database appear to be a replica of the 'default'
|
||||
self.old_routers = router.routers
|
||||
router.routers = [TestRouter()]
|
||||
|
||||
@@ -1071,7 +1071,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
dive.editor = marty
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments of original objects haven't changed...
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1089,7 +1089,7 @@ class RouterTestCase(TestCase):
|
||||
except Book.DoesNotExist:
|
||||
self.fail('Source database should have a copy of saved object')
|
||||
|
||||
# This isn't a real leader-follower database, so restore the original from other
|
||||
# This isn't a real primary/replica database, so restore the original from other
|
||||
dive = Book.objects.using('other').get(title='Dive into Python')
|
||||
self.assertEqual(dive._state.db, 'other')
|
||||
|
||||
@@ -1097,7 +1097,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
marty.edited = [pro, dive]
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Assignment implies a save, so database assignments of original objects have changed...
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1111,7 +1111,7 @@ class RouterTestCase(TestCase):
|
||||
except Book.DoesNotExist:
|
||||
self.fail('Source database should have a copy of saved object')
|
||||
|
||||
# This isn't a real leader-follower database, so restore the original from other
|
||||
# This isn't a real primary/replica database, so restore the original from other
|
||||
dive = Book.objects.using('other').get(title='Dive into Python')
|
||||
self.assertEqual(dive._state.db, 'other')
|
||||
|
||||
@@ -1119,7 +1119,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
marty.edited.add(dive)
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Add implies a save, so database assignments of original objects have changed...
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1133,7 +1133,7 @@ class RouterTestCase(TestCase):
|
||||
except Book.DoesNotExist:
|
||||
self.fail('Source database should have a copy of saved object')
|
||||
|
||||
# This isn't a real leader-follower database, so restore the original from other
|
||||
# This isn't a real primary/replica database, so restore the original from other
|
||||
dive = Book.objects.using('other').get(title='Dive into Python')
|
||||
|
||||
# If you assign a FK object when the base object hasn't
|
||||
@@ -1196,7 +1196,7 @@ class RouterTestCase(TestCase):
|
||||
mark = Person.objects.using('default').create(pk=2, name="Mark Pilgrim")
|
||||
|
||||
# Now save back onto the usual database.
|
||||
# This simulates leader/follower - the objects exist on both database,
|
||||
# This simulates primary/replica - the objects exist on both database,
|
||||
# but the _state.db is as it is for all other tests.
|
||||
pro.save(using='default')
|
||||
marty.save(using='default')
|
||||
@@ -1213,7 +1213,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
marty.book_set = [pro, dive]
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments don't change
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1232,7 +1232,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
marty.book_set.add(dive)
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments don't change
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1251,7 +1251,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
dive.authors = [mark, marty]
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments don't change
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1273,7 +1273,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
dive.authors.add(marty)
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments don't change
|
||||
self.assertEqual(marty._state.db, 'default')
|
||||
@@ -1311,7 +1311,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
bob.userprofile = alice_profile
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments of original objects haven't changed...
|
||||
self.assertEqual(alice._state.db, 'default')
|
||||
@@ -1342,7 +1342,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
review1.content_object = dive
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments of original objects haven't changed...
|
||||
self.assertEqual(pro._state.db, 'default')
|
||||
@@ -1361,7 +1361,7 @@ class RouterTestCase(TestCase):
|
||||
except Book.DoesNotExist:
|
||||
self.fail('Source database should have a copy of saved object')
|
||||
|
||||
# This isn't a real leader-follower database, so restore the original from other
|
||||
# This isn't a real primary/replica database, so restore the original from other
|
||||
dive = Book.objects.using('other').get(title='Dive into Python')
|
||||
self.assertEqual(dive._state.db, 'other')
|
||||
|
||||
@@ -1369,7 +1369,7 @@ class RouterTestCase(TestCase):
|
||||
try:
|
||||
dive.reviews.add(review1)
|
||||
except ValueError:
|
||||
self.fail("Assignment across leader/follower databases with a common source should be ok")
|
||||
self.fail("Assignment across primary/replica databases with a common source should be ok")
|
||||
|
||||
# Database assignments of original objects haven't changed...
|
||||
self.assertEqual(pro._state.db, 'default')
|
||||
@@ -1444,7 +1444,7 @@ class RouterTestCase(TestCase):
|
||||
self.assertEqual(pro.reviews.db_manager('default').all().db, 'default')
|
||||
|
||||
def test_subquery(self):
|
||||
"""Make sure as_sql works with subqueries and leader/follower."""
|
||||
"""Make sure as_sql works with subqueries and primary/replica."""
|
||||
# Create a book and author on the other database
|
||||
|
||||
mark = Person.objects.using('other').create(name="Mark Pilgrim")
|
||||
@@ -1482,7 +1482,7 @@ class AuthTestCase(TestCase):
|
||||
multi_db = True
|
||||
|
||||
def setUp(self):
|
||||
# Make the 'other' database appear to be a follower of the 'default'
|
||||
# Make the 'other' database appear to be a replica of the 'default'
|
||||
self.old_routers = router.routers
|
||||
router.routers = [AuthRouter()]
|
||||
|
||||
|
Reference in New Issue
Block a user