1. from django.apps import apps
    
  2. from django.apps.registry import Apps
    
  3. from django.conf import settings
    
  4. from django.contrib.sites import models
    
  5. from django.contrib.sites.checks import check_site_id
    
  6. from django.contrib.sites.management import create_default_site
    
  7. from django.contrib.sites.middleware import CurrentSiteMiddleware
    
  8. from django.contrib.sites.models import Site, clear_site_cache
    
  9. from django.contrib.sites.requests import RequestSite
    
  10. from django.contrib.sites.shortcuts import get_current_site
    
  11. from django.core import checks
    
  12. from django.core.exceptions import ObjectDoesNotExist, ValidationError
    
  13. from django.db.models.signals import post_migrate
    
  14. from django.http import HttpRequest, HttpResponse
    
  15. from django.test import SimpleTestCase, TestCase, modify_settings, override_settings
    
  16. from django.test.utils import captured_stdout
    
  17. 
    
  18. 
    
  19. @modify_settings(INSTALLED_APPS={"append": "django.contrib.sites"})
    
  20. class SitesFrameworkTests(TestCase):
    
  21.     databases = {"default", "other"}
    
  22. 
    
  23.     @classmethod
    
  24.     def setUpTestData(cls):
    
  25.         cls.site = Site(id=settings.SITE_ID, domain="example.com", name="example.com")
    
  26.         cls.site.save()
    
  27. 
    
  28.     def setUp(self):
    
  29.         Site.objects.clear_cache()
    
  30. 
    
  31.     def tearDown(self):
    
  32.         Site.objects.clear_cache()
    
  33. 
    
  34.     def test_site_manager(self):
    
  35.         # Make sure that get_current() does not return a deleted Site object.
    
  36.         s = Site.objects.get_current()
    
  37.         self.assertIsInstance(s, Site)
    
  38.         s.delete()
    
  39.         with self.assertRaises(ObjectDoesNotExist):
    
  40.             Site.objects.get_current()
    
  41. 
    
  42.     def test_site_cache(self):
    
  43.         # After updating a Site object (e.g. via the admin), we shouldn't return a
    
  44.         # bogus value from the SITE_CACHE.
    
  45.         site = Site.objects.get_current()
    
  46.         self.assertEqual("example.com", site.name)
    
  47.         s2 = Site.objects.get(id=settings.SITE_ID)
    
  48.         s2.name = "Example site"
    
  49.         s2.save()
    
  50.         site = Site.objects.get_current()
    
  51.         self.assertEqual("Example site", site.name)
    
  52. 
    
  53.     def test_delete_all_sites_clears_cache(self):
    
  54.         # When all site objects are deleted the cache should also
    
  55.         # be cleared and get_current() should raise a DoesNotExist.
    
  56.         self.assertIsInstance(Site.objects.get_current(), Site)
    
  57.         Site.objects.all().delete()
    
  58.         with self.assertRaises(Site.DoesNotExist):
    
  59.             Site.objects.get_current()
    
  60. 
    
  61.     @override_settings(ALLOWED_HOSTS=["example.com"])
    
  62.     def test_get_current_site(self):
    
  63.         # The correct Site object is returned
    
  64.         request = HttpRequest()
    
  65.         request.META = {
    
  66.             "SERVER_NAME": "example.com",
    
  67.             "SERVER_PORT": "80",
    
  68.         }
    
  69.         site = get_current_site(request)
    
  70.         self.assertIsInstance(site, Site)
    
  71.         self.assertEqual(site.id, settings.SITE_ID)
    
  72. 
    
  73.         # An exception is raised if the sites framework is installed
    
  74.         # but there is no matching Site
    
  75.         site.delete()
    
  76.         with self.assertRaises(ObjectDoesNotExist):
    
  77.             get_current_site(request)
    
  78. 
    
  79.         # A RequestSite is returned if the sites framework is not installed
    
  80.         with self.modify_settings(INSTALLED_APPS={"remove": "django.contrib.sites"}):
    
  81.             site = get_current_site(request)
    
  82.             self.assertIsInstance(site, RequestSite)
    
  83.             self.assertEqual(site.name, "example.com")
    
  84. 
    
  85.     @override_settings(SITE_ID=None, ALLOWED_HOSTS=["example.com"])
    
  86.     def test_get_current_site_no_site_id(self):
    
  87.         request = HttpRequest()
    
  88.         request.META = {
    
  89.             "SERVER_NAME": "example.com",
    
  90.             "SERVER_PORT": "80",
    
  91.         }
    
  92.         del settings.SITE_ID
    
  93.         site = get_current_site(request)
    
  94.         self.assertEqual(site.name, "example.com")
    
  95. 
    
  96.     @override_settings(SITE_ID=None, ALLOWED_HOSTS=["example.com"])
    
  97.     def test_get_current_site_host_with_trailing_dot(self):
    
  98.         """
    
  99.         The site is matched if the name in the request has a trailing dot.
    
  100.         """
    
  101.         request = HttpRequest()
    
  102.         request.META = {
    
  103.             "SERVER_NAME": "example.com.",
    
  104.             "SERVER_PORT": "80",
    
  105.         }
    
  106.         site = get_current_site(request)
    
  107.         self.assertEqual(site.name, "example.com")
    
  108. 
    
  109.     @override_settings(SITE_ID=None, ALLOWED_HOSTS=["example.com", "example.net"])
    
  110.     def test_get_current_site_no_site_id_and_handle_port_fallback(self):
    
  111.         request = HttpRequest()
    
  112.         s1 = self.site
    
  113.         s2 = Site.objects.create(domain="example.com:80", name="example.com:80")
    
  114. 
    
  115.         # Host header without port
    
  116.         request.META = {"HTTP_HOST": "example.com"}
    
  117.         site = get_current_site(request)
    
  118.         self.assertEqual(site, s1)
    
  119. 
    
  120.         # Host header with port - match, no fallback without port
    
  121.         request.META = {"HTTP_HOST": "example.com:80"}
    
  122.         site = get_current_site(request)
    
  123.         self.assertEqual(site, s2)
    
  124. 
    
  125.         # Host header with port - no match, fallback without port
    
  126.         request.META = {"HTTP_HOST": "example.com:81"}
    
  127.         site = get_current_site(request)
    
  128.         self.assertEqual(site, s1)
    
  129. 
    
  130.         # Host header with non-matching domain
    
  131.         request.META = {"HTTP_HOST": "example.net"}
    
  132.         with self.assertRaises(ObjectDoesNotExist):
    
  133.             get_current_site(request)
    
  134. 
    
  135.         # Ensure domain for RequestSite always matches host header
    
  136.         with self.modify_settings(INSTALLED_APPS={"remove": "django.contrib.sites"}):
    
  137.             request.META = {"HTTP_HOST": "example.com"}
    
  138.             site = get_current_site(request)
    
  139.             self.assertEqual(site.name, "example.com")
    
  140. 
    
  141.             request.META = {"HTTP_HOST": "example.com:80"}
    
  142.             site = get_current_site(request)
    
  143.             self.assertEqual(site.name, "example.com:80")
    
  144. 
    
  145.     def test_domain_name_with_whitespaces(self):
    
  146.         # Regression for #17320
    
  147.         # Domain names are not allowed contain whitespace characters
    
  148.         site = Site(name="test name", domain="test test")
    
  149.         with self.assertRaises(ValidationError):
    
  150.             site.full_clean()
    
  151.         site.domain = "test\ttest"
    
  152.         with self.assertRaises(ValidationError):
    
  153.             site.full_clean()
    
  154.         site.domain = "test\ntest"
    
  155.         with self.assertRaises(ValidationError):
    
  156.             site.full_clean()
    
  157. 
    
  158.     @override_settings(ALLOWED_HOSTS=["example.com"])
    
  159.     def test_clear_site_cache(self):
    
  160.         request = HttpRequest()
    
  161.         request.META = {
    
  162.             "SERVER_NAME": "example.com",
    
  163.             "SERVER_PORT": "80",
    
  164.         }
    
  165.         self.assertEqual(models.SITE_CACHE, {})
    
  166.         get_current_site(request)
    
  167.         expected_cache = {self.site.id: self.site}
    
  168.         self.assertEqual(models.SITE_CACHE, expected_cache)
    
  169. 
    
  170.         with self.settings(SITE_ID=None):
    
  171.             get_current_site(request)
    
  172. 
    
  173.         expected_cache.update({self.site.domain: self.site})
    
  174.         self.assertEqual(models.SITE_CACHE, expected_cache)
    
  175. 
    
  176.         clear_site_cache(Site, instance=self.site, using="default")
    
  177.         self.assertEqual(models.SITE_CACHE, {})
    
  178. 
    
  179.     @override_settings(SITE_ID=None, ALLOWED_HOSTS=["example2.com"])
    
  180.     def test_clear_site_cache_domain(self):
    
  181.         site = Site.objects.create(name="example2.com", domain="example2.com")
    
  182.         request = HttpRequest()
    
  183.         request.META = {
    
  184.             "SERVER_NAME": "example2.com",
    
  185.             "SERVER_PORT": "80",
    
  186.         }
    
  187.         get_current_site(request)  # prime the models.SITE_CACHE
    
  188.         expected_cache = {site.domain: site}
    
  189.         self.assertEqual(models.SITE_CACHE, expected_cache)
    
  190. 
    
  191.         # Site exists in 'default' database so using='other' shouldn't clear.
    
  192.         clear_site_cache(Site, instance=site, using="other")
    
  193.         self.assertEqual(models.SITE_CACHE, expected_cache)
    
  194.         # using='default' should clear.
    
  195.         clear_site_cache(Site, instance=site, using="default")
    
  196.         self.assertEqual(models.SITE_CACHE, {})
    
  197. 
    
  198.     def test_unique_domain(self):
    
  199.         site = Site(domain=self.site.domain)
    
  200.         msg = "Site with this Domain name already exists."
    
  201.         with self.assertRaisesMessage(ValidationError, msg):
    
  202.             site.validate_unique()
    
  203. 
    
  204.     def test_site_natural_key(self):
    
  205.         self.assertEqual(Site.objects.get_by_natural_key(self.site.domain), self.site)
    
  206.         self.assertEqual(self.site.natural_key(), (self.site.domain,))
    
  207. 
    
  208.     @override_settings(SITE_ID="1")
    
  209.     def test_check_site_id(self):
    
  210.         self.assertEqual(
    
  211.             check_site_id(None),
    
  212.             [
    
  213.                 checks.Error(
    
  214.                     msg="The SITE_ID setting must be an integer",
    
  215.                     id="sites.E101",
    
  216.                 ),
    
  217.             ],
    
  218.         )
    
  219. 
    
  220.     def test_valid_site_id(self):
    
  221.         for site_id in [1, None]:
    
  222.             with self.subTest(site_id=site_id), self.settings(SITE_ID=site_id):
    
  223.                 self.assertEqual(check_site_id(None), [])
    
  224. 
    
  225. 
    
  226. @override_settings(ALLOWED_HOSTS=["example.com"])
    
  227. class RequestSiteTests(SimpleTestCase):
    
  228.     def setUp(self):
    
  229.         request = HttpRequest()
    
  230.         request.META = {"HTTP_HOST": "example.com"}
    
  231.         self.site = RequestSite(request)
    
  232. 
    
  233.     def test_init_attributes(self):
    
  234.         self.assertEqual(self.site.domain, "example.com")
    
  235.         self.assertEqual(self.site.name, "example.com")
    
  236. 
    
  237.     def test_str(self):
    
  238.         self.assertEqual(str(self.site), "example.com")
    
  239. 
    
  240.     def test_save(self):
    
  241.         msg = "RequestSite cannot be saved."
    
  242.         with self.assertRaisesMessage(NotImplementedError, msg):
    
  243.             self.site.save()
    
  244. 
    
  245.     def test_delete(self):
    
  246.         msg = "RequestSite cannot be deleted."
    
  247.         with self.assertRaisesMessage(NotImplementedError, msg):
    
  248.             self.site.delete()
    
  249. 
    
  250. 
    
  251. class JustOtherRouter:
    
  252.     def allow_migrate(self, db, app_label, **hints):
    
  253.         return db == "other"
    
  254. 
    
  255. 
    
  256. @modify_settings(INSTALLED_APPS={"append": "django.contrib.sites"})
    
  257. class CreateDefaultSiteTests(TestCase):
    
  258.     databases = {"default", "other"}
    
  259. 
    
  260.     @classmethod
    
  261.     def setUpTestData(cls):
    
  262.         # Delete the site created as part of the default migration process.
    
  263.         Site.objects.all().delete()
    
  264. 
    
  265.     def setUp(self):
    
  266.         self.app_config = apps.get_app_config("sites")
    
  267. 
    
  268.     def test_basic(self):
    
  269.         """
    
  270.         #15346, #15573 - create_default_site() creates an example site only if
    
  271.         none exist.
    
  272.         """
    
  273.         with captured_stdout() as stdout:
    
  274.             create_default_site(self.app_config)
    
  275.         self.assertEqual(Site.objects.count(), 1)
    
  276.         self.assertIn("Creating example.com", stdout.getvalue())
    
  277. 
    
  278.         with captured_stdout() as stdout:
    
  279.             create_default_site(self.app_config)
    
  280.         self.assertEqual(Site.objects.count(), 1)
    
  281.         self.assertEqual("", stdout.getvalue())
    
  282. 
    
  283.     @override_settings(DATABASE_ROUTERS=[JustOtherRouter()])
    
  284.     def test_multi_db_with_router(self):
    
  285.         """
    
  286.         #16353, #16828 - The default site creation should respect db routing.
    
  287.         """
    
  288.         create_default_site(self.app_config, using="default", verbosity=0)
    
  289.         create_default_site(self.app_config, using="other", verbosity=0)
    
  290.         self.assertFalse(Site.objects.using("default").exists())
    
  291.         self.assertTrue(Site.objects.using("other").exists())
    
  292. 
    
  293.     def test_multi_db(self):
    
  294.         create_default_site(self.app_config, using="default", verbosity=0)
    
  295.         create_default_site(self.app_config, using="other", verbosity=0)
    
  296.         self.assertTrue(Site.objects.using("default").exists())
    
  297.         self.assertTrue(Site.objects.using("other").exists())
    
  298. 
    
  299.     def test_save_another(self):
    
  300.         """
    
  301.         #17415 - Another site can be created right after the default one.
    
  302. 
    
  303.         On some backends the sequence needs to be reset after saving with an
    
  304.         explicit ID. There shouldn't be a sequence collisions by saving another
    
  305.         site. This test is only meaningful with databases that use sequences
    
  306.         for automatic primary keys such as PostgreSQL and Oracle.
    
  307.         """
    
  308.         create_default_site(self.app_config, verbosity=0)
    
  309.         Site(domain="example2.com", name="example2.com").save()
    
  310. 
    
  311.     def test_signal(self):
    
  312.         """
    
  313.         #23641 - Sending the ``post_migrate`` signal triggers creation of the
    
  314.         default site.
    
  315.         """
    
  316.         post_migrate.send(
    
  317.             sender=self.app_config, app_config=self.app_config, verbosity=0
    
  318.         )
    
  319.         self.assertTrue(Site.objects.exists())
    
  320. 
    
  321.     @override_settings(SITE_ID=35696)
    
  322.     def test_custom_site_id(self):
    
  323.         """
    
  324.         #23945 - The configured ``SITE_ID`` should be respected.
    
  325.         """
    
  326.         create_default_site(self.app_config, verbosity=0)
    
  327.         self.assertEqual(Site.objects.get().pk, 35696)
    
  328. 
    
  329.     @override_settings()  # Restore original ``SITE_ID`` afterward.
    
  330.     def test_no_site_id(self):
    
  331.         """
    
  332.         #24488 - The pk should default to 1 if no ``SITE_ID`` is configured.
    
  333.         """
    
  334.         del settings.SITE_ID
    
  335.         create_default_site(self.app_config, verbosity=0)
    
  336.         self.assertEqual(Site.objects.get().pk, 1)
    
  337. 
    
  338.     def test_unavailable_site_model(self):
    
  339.         """
    
  340.         #24075 - A Site shouldn't be created if the model isn't available.
    
  341.         """
    
  342.         apps = Apps()
    
  343.         create_default_site(self.app_config, verbosity=0, apps=apps)
    
  344.         self.assertFalse(Site.objects.exists())
    
  345. 
    
  346. 
    
  347. class MiddlewareTest(TestCase):
    
  348.     def test_request(self):
    
  349.         def get_response(request):
    
  350.             return HttpResponse(str(request.site.id))
    
  351. 
    
  352.         response = CurrentSiteMiddleware(get_response)(HttpRequest())
    
  353.         self.assertContains(response, settings.SITE_ID)