From 2804b8d2153505ec49b191db2168302dfb92c3af Mon Sep 17 00:00:00 2001 From: Nasir Hussain Date: Wed, 23 Jan 2019 03:49:30 +0500 Subject: [PATCH] Fixed #30111 -- Fixed AppRegistryNotReady error with django.contrib.postgres in INSTALLED_APPS. Regression in e192223ed996ed30fe83787efdfa7f2be6b1a2ee. --- django/db/migrations/recorder.py | 32 ++++++++++++++------ tests/postgres_tests/integration_settings.py | 5 +++ tests/postgres_tests/test_integration.py | 17 +++++++++++ 3 files changed, 44 insertions(+), 10 deletions(-) create mode 100644 tests/postgres_tests/integration_settings.py create mode 100644 tests/postgres_tests/test_integration.py diff --git a/django/db/migrations/recorder.py b/django/db/migrations/recorder.py index 3a972fe4c63..ad5435d906d 100644 --- a/django/db/migrations/recorder.py +++ b/django/db/migrations/recorder.py @@ -1,6 +1,7 @@ from django.apps.registry import Apps from django.db import models from django.db.utils import DatabaseError +from django.utils.decorators import classproperty from django.utils.timezone import now from .exceptions import MigrationSchemaMissing @@ -18,19 +19,30 @@ class MigrationRecorder: If a migration is unapplied its row is removed from the table. Having a row in the table always means a migration is applied. """ + _migration_class = None - class Migration(models.Model): - app = models.CharField(max_length=255) - name = models.CharField(max_length=255) - applied = models.DateTimeField(default=now) + @classproperty + def Migration(cls): + """ + Lazy load to avoid AppRegistryNotReady if installed apps import + MigrationRecorder. + """ + if cls._migration_class is None: + class Migration(models.Model): + app = models.CharField(max_length=255) + name = models.CharField(max_length=255) + applied = models.DateTimeField(default=now) - class Meta: - apps = Apps() - app_label = "migrations" - db_table = "django_migrations" + class Meta: + apps = Apps() + app_label = 'migrations' + db_table = 'django_migrations' - def __str__(self): - return "Migration %s for %s" % (self.name, self.app) + def __str__(self): + return 'Migration %s for %s' % (self.name, self.app) + + cls._migration_class = Migration + return cls._migration_class def __init__(self, connection): self.connection = connection diff --git a/tests/postgres_tests/integration_settings.py b/tests/postgres_tests/integration_settings.py new file mode 100644 index 00000000000..c4ec0d1157a --- /dev/null +++ b/tests/postgres_tests/integration_settings.py @@ -0,0 +1,5 @@ +SECRET_KEY = 'abcdefg' + +INSTALLED_APPS = [ + 'django.contrib.postgres', +] diff --git a/tests/postgres_tests/test_integration.py b/tests/postgres_tests/test_integration.py new file mode 100644 index 00000000000..829b8620276 --- /dev/null +++ b/tests/postgres_tests/test_integration.py @@ -0,0 +1,17 @@ +import os +import subprocess +import sys + +from tests.postgres_tests import PostgreSQLSimpleTestCase + + +class PostgresIntegrationTests(PostgreSQLSimpleTestCase): + def test_check(self): + os.chdir(os.path.dirname(__file__)) + result = subprocess.run( + [sys.executable, '-m', 'django', 'check', '--settings', 'integration_settings'], + stdout=subprocess.DEVNULL, + stderr=subprocess.PIPE, + ) + stderr = '\n'.join([e.decode() for e in result.stderr.splitlines()]) + self.assertEqual(result.returncode, 0, msg=stderr)