2017-01-19 12:16:04 -05:00
|
|
|
from unittest import mock
|
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
from django.contrib import admin
|
|
|
|
from django.contrib.auth.models import User
|
2023-11-15 14:32:03 -05:00
|
|
|
from django.contrib.contenttypes.models import ContentType
|
|
|
|
from django.http import HttpResponse
|
2017-01-19 12:16:04 -05:00
|
|
|
from django.test import TestCase, override_settings
|
2018-12-07 17:52:28 -05:00
|
|
|
from django.urls import path, reverse
|
2016-09-14 21:06:39 +01:00
|
|
|
|
|
|
|
from .models import Book
|
|
|
|
|
|
|
|
|
2017-01-19 02:39:46 -05:00
|
|
|
class Router:
|
2016-09-14 21:06:39 +01:00
|
|
|
target_db = None
|
|
|
|
|
|
|
|
def db_for_read(self, model, **hints):
|
|
|
|
return self.target_db
|
|
|
|
|
|
|
|
db_for_write = db_for_read
|
|
|
|
|
2022-04-14 12:54:27 +02:00
|
|
|
def allow_relation(self, obj1, obj2, **hints):
|
|
|
|
return True
|
|
|
|
|
2016-11-12 20:41:23 +03:30
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
site = admin.AdminSite(name="test_adminsite")
|
|
|
|
site.register(Book)
|
|
|
|
|
2023-11-15 14:32:03 -05:00
|
|
|
|
|
|
|
def book(request, book_id):
|
|
|
|
b = Book.objects.get(id=book_id)
|
|
|
|
return HttpResponse(b.title)
|
|
|
|
|
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
urlpatterns = [
|
2018-12-07 17:52:28 -05:00
|
|
|
path("admin/", site.urls),
|
2023-11-15 14:32:03 -05:00
|
|
|
path("books/<book_id>/", book),
|
2016-09-14 21:06:39 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
@override_settings(ROOT_URLCONF=__name__, DATABASE_ROUTERS=["%s.Router" % __name__])
|
|
|
|
class MultiDatabaseTests(TestCase):
|
2018-07-12 00:12:20 -04:00
|
|
|
databases = {"default", "other"}
|
2024-06-11 19:27:49 +01:00
|
|
|
READ_ONLY_METHODS = {"get", "options", "head", "trace"}
|
2016-09-14 21:06:39 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
cls.superusers = {}
|
|
|
|
cls.test_book_ids = {}
|
2019-12-20 20:49:56 +01:00
|
|
|
for db in cls.databases:
|
2016-09-14 21:06:39 +01:00
|
|
|
Router.target_db = db
|
|
|
|
cls.superusers[db] = User.objects.create_superuser(
|
|
|
|
username="admin",
|
|
|
|
password="something",
|
|
|
|
email="test@test.org",
|
|
|
|
)
|
|
|
|
b = Book(name="Test Book")
|
|
|
|
b.save(using=db)
|
|
|
|
cls.test_book_ids[db] = b.id
|
|
|
|
|
2024-06-11 19:27:49 +01:00
|
|
|
def tearDown(self):
|
|
|
|
# Reset the routers' state between each test.
|
|
|
|
Router.target_db = None
|
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_add_view(self, mock):
|
2019-12-20 20:49:56 +01:00
|
|
|
for db in self.databases:
|
2017-03-07 21:00:43 +00:00
|
|
|
with self.subTest(db=db):
|
2024-06-11 19:27:49 +01:00
|
|
|
mock.mock_reset()
|
2017-03-07 21:00:43 +00:00
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
2024-06-11 19:27:49 +01:00
|
|
|
response = self.client.post(
|
2017-03-07 21:00:43 +00:00
|
|
|
reverse("test_adminsite:admin_views_book_add"),
|
|
|
|
{"name": "Foobar: 5th edition"},
|
|
|
|
)
|
2024-06-11 19:27:49 +01:00
|
|
|
self.assertEqual(response.status_code, 302)
|
|
|
|
self.assertEqual(
|
|
|
|
response.url, reverse("test_adminsite:admin_views_book_changelist")
|
|
|
|
)
|
2017-03-07 21:00:43 +00:00
|
|
|
mock.atomic.assert_called_with(using=db)
|
2016-09-14 21:06:39 +01:00
|
|
|
|
2024-06-11 19:27:49 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_read_only_methods_add_view(self, mock):
|
|
|
|
for db in self.databases:
|
|
|
|
for method in self.READ_ONLY_METHODS:
|
|
|
|
with self.subTest(db=db, method=method):
|
|
|
|
mock.mock_reset()
|
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
|
|
|
response = getattr(self.client, method)(
|
|
|
|
reverse("test_adminsite:admin_views_book_add"),
|
|
|
|
)
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
mock.atomic.assert_not_called()
|
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_change_view(self, mock):
|
2019-12-20 20:49:56 +01:00
|
|
|
for db in self.databases:
|
2017-03-07 21:00:43 +00:00
|
|
|
with self.subTest(db=db):
|
2024-06-11 19:27:49 +01:00
|
|
|
mock.mock_reset()
|
2017-03-07 21:00:43 +00:00
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
2024-06-11 19:27:49 +01:00
|
|
|
response = self.client.post(
|
2017-03-07 21:00:43 +00:00
|
|
|
reverse(
|
|
|
|
"test_adminsite:admin_views_book_change",
|
|
|
|
args=[self.test_book_ids[db]],
|
|
|
|
),
|
|
|
|
{"name": "Test Book 2: Test more"},
|
|
|
|
)
|
2024-06-11 19:27:49 +01:00
|
|
|
self.assertEqual(response.status_code, 302)
|
|
|
|
self.assertEqual(
|
|
|
|
response.url, reverse("test_adminsite:admin_views_book_changelist")
|
|
|
|
)
|
2017-03-07 21:00:43 +00:00
|
|
|
mock.atomic.assert_called_with(using=db)
|
2016-09-14 21:06:39 +01:00
|
|
|
|
2024-06-11 19:27:49 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_read_only_methods_change_view(self, mock):
|
|
|
|
for db in self.databases:
|
|
|
|
for method in self.READ_ONLY_METHODS:
|
|
|
|
with self.subTest(db=db, method=method):
|
|
|
|
mock.mock_reset()
|
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
|
|
|
response = getattr(self.client, method)(
|
|
|
|
reverse(
|
|
|
|
"test_adminsite:admin_views_book_change",
|
|
|
|
args=[self.test_book_ids[db]],
|
|
|
|
),
|
|
|
|
data={"name": "Test Book 2: Test more"},
|
|
|
|
)
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
mock.atomic.assert_not_called()
|
|
|
|
|
2016-09-14 21:06:39 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_delete_view(self, mock):
|
2019-12-20 20:49:56 +01:00
|
|
|
for db in self.databases:
|
2017-03-07 21:00:43 +00:00
|
|
|
with self.subTest(db=db):
|
2024-06-11 19:27:49 +01:00
|
|
|
mock.mock_reset()
|
2017-03-07 21:00:43 +00:00
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
2024-06-11 19:27:49 +01:00
|
|
|
response = self.client.post(
|
2017-03-07 21:00:43 +00:00
|
|
|
reverse(
|
|
|
|
"test_adminsite:admin_views_book_delete",
|
|
|
|
args=[self.test_book_ids[db]],
|
|
|
|
),
|
|
|
|
{"post": "yes"},
|
|
|
|
)
|
2024-06-11 19:27:49 +01:00
|
|
|
self.assertEqual(response.status_code, 302)
|
|
|
|
self.assertEqual(
|
|
|
|
response.url, reverse("test_adminsite:admin_views_book_changelist")
|
|
|
|
)
|
2017-03-07 21:00:43 +00:00
|
|
|
mock.atomic.assert_called_with(using=db)
|
2023-11-15 14:32:03 -05:00
|
|
|
|
2024-06-11 19:27:49 +01:00
|
|
|
@mock.patch("django.contrib.admin.options.transaction")
|
|
|
|
def test_read_only_methods_delete_view(self, mock):
|
|
|
|
for db in self.databases:
|
|
|
|
for method in self.READ_ONLY_METHODS:
|
|
|
|
with self.subTest(db=db, method=method):
|
|
|
|
mock.mock_reset()
|
|
|
|
Router.target_db = db
|
|
|
|
self.client.force_login(self.superusers[db])
|
|
|
|
response = getattr(self.client, method)(
|
|
|
|
reverse(
|
|
|
|
"test_adminsite:admin_views_book_delete",
|
|
|
|
args=[self.test_book_ids[db]],
|
|
|
|
)
|
|
|
|
)
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
mock.atomic.assert_not_called()
|
|
|
|
|
2023-11-15 14:32:03 -05:00
|
|
|
|
|
|
|
class ViewOnSiteRouter:
|
|
|
|
def db_for_read(self, model, instance=None, **hints):
|
|
|
|
if model._meta.app_label in {"auth", "sessions", "contenttypes"}:
|
|
|
|
return "default"
|
|
|
|
return "other"
|
|
|
|
|
|
|
|
def db_for_write(self, model, **hints):
|
|
|
|
if model._meta.app_label in {"auth", "sessions", "contenttypes"}:
|
|
|
|
return "default"
|
|
|
|
return "other"
|
|
|
|
|
|
|
|
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, app_label, **hints):
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
@override_settings(ROOT_URLCONF=__name__, DATABASE_ROUTERS=[ViewOnSiteRouter()])
|
|
|
|
class ViewOnSiteTests(TestCase):
|
|
|
|
databases = {"default", "other"}
|
|
|
|
|
|
|
|
def test_contenttype_in_separate_db(self):
|
|
|
|
ContentType.objects.using("other").all().delete()
|
|
|
|
book = Book.objects.using("other").create(name="other book")
|
|
|
|
user = User.objects.create_superuser(
|
|
|
|
username="super", password="secret", email="super@example.com"
|
|
|
|
)
|
|
|
|
|
|
|
|
book_type = ContentType.objects.get(app_label="admin_views", model="book")
|
|
|
|
|
|
|
|
self.client.force_login(user)
|
|
|
|
|
|
|
|
shortcut_url = reverse("admin:view_on_site", args=(book_type.pk, book.id))
|
|
|
|
response = self.client.get(shortcut_url, follow=False)
|
|
|
|
self.assertEqual(response.status_code, 302)
|
|
|
|
self.assertRegex(
|
|
|
|
response.url, f"http://(testserver|example.com)/books/{book.id}/"
|
|
|
|
)
|