{
  "id": "django__django-11141",
  "question": "Allow migrations directories without __init__.py files\nDescription\n\t \n\t\t(last modified by Tim Graham)\n\t \nBackground: In python 3 a package with no __init__.py is implicitly a namespace package, so it has no __file__ attribute. \nThe migrate command currently checks for existence of a __file__ attribute on the migrations package. This check was introduced in #21015, because the __file__ attribute was used in migration file discovery. \nHowever, in #23406 migration file discovery was changed to use pkgutil.iter_modules (), instead of direct filesystem access. pkgutil. iter_modules() uses the package's __path__ list, which exists on implicit namespace packages.\nAs a result, the __file__ check is no longer needed, and in fact prevents migrate from working on namespace packages (implicit or otherwise). \nRelated work: #29091\n",
  "answer": "{\"task_id\": \"f1f9b6a1-df30-4116-8272-0288471383bc\", \"workflow_run_id\": \"7ea8ec58-2098-4cd6-86be-9fbb21893dd8\", \"data\": {\"id\": \"7ea8ec58-2098-4cd6-86be-9fbb21893dd8\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,8 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n-            if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n+            migrations_module_name = '%s.migrations' % app_config.name\\n+            if module_has_submodule(app_config.module, 'migrations'):\\n+                migrations_module = import_module(migrations_module_name)\\n+                if not hasattr(migrations_module, '__path__'):\\n                     raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n+                        \\\"Django can't use migrations for app '%s' because migrations module has no __path__.\\\" % app_config.label\\n                     )\", \"--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,12 +90,8 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                # Import migrations module - pkgutil.iter_modules will handle discovery\\n+                import_module('%s.migrations' % app_config.name)\", \"--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,14 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                try:\\n+                    migrations_module = import_module('%s.migrations' % app_config.name)\\n+                    # Check for __path__ which exists on both regular and namespace packages\\n+                    if not hasattr(migrations_module, '__path__'):\\n+                        raise CommandError(\\n+                            \\\"Django can't use migrations for app '%s' because migrations module is not a package.\\\" % app_config.label\\n+                        )\\n+                except ImportError:\\n+                    # Module exists but can't be imported - let migration loader handle this\\n+                    pass\", \"--- a/django/db/migrations/loader.py\\n+++ b/django/db/migrations/loader.py\\n@@ -250,8 +250,10 @@ class MigrationLoader:\\n         \\\"\\\"\\\"\\n         Load the migrations from all INSTALLED_APPS from disk.\\n         \\\"\\\"\\\"\\n         self.disk_migrations = {}\\n         self.unmigrated_apps = set()\\n         self.migrated_apps = set()\\n         for app_config in apps.get_app_configs():\\n             # Get the migrations module directory\\n             module_name = self.migrations_module(app_config.label)\\n             if module_name is None:\\n@@ -260,7 +262,8 @@ class MigrationLoader:\\n             try:\\n                 module = import_module(module_name)\\n             except ImportError:\\n                 # I hate doing this, but I don't want to squash other import errors.\\n                 # Might be better to try a directory check directly.\\n                 was_loaded = module_name in sys.modules\\n                 try:\\n@@ -275,11 +278,11 @@ class MigrationLoader:\\n                 else:\\n                     self.unmigrated_apps.add(app_config.label)\\n                 continue\\n-            # Module is not a package (e.g. migrations.py).\\n-            if not hasattr(module, '__path__'):\\n+            # Check if module is a package (regular or namespace)\\n+            # Both should have __path__ attribute\\n+            if not hasattr(module, '__path__'):\\n                 self.unmigrated_apps.add(app_config.label)\\n                 continue\\n             # Force a reload if it's already loaded (tests need this)\\n             if was_loaded:\\n                 reload(module)\", \"--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,6 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                # Import migrations module - migration loader will validate package structure\\n+                import_module('%s.migrations' % app_config.name)\", \"diff\\n--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,14 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                try:\\n+                    migrations_module = import_module('%s.migrations' % app_config.name)\\n+                    # Check for __path__ which exists on both regular and namespace packages\\n+                    if not hasattr(migrations_module, '__path__'):\\n+                        raise CommandError(\\n+                            \\\"Django can't use migrations for app '%s' because migrations module is not a package.\\\" % app_config.label\\n+                        )\\n+                except ImportError:\\n+                    # Module exists but can't be imported - let migration loader handle this\\n+                    pass\\n\", \"diff\\n--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,14 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                try:\\n+                    migrations_module = import_module('%s.migrations' % app_config.name)\\n+                    # Check for __path__ which exists on both regular and namespace packages\\n+                    if not hasattr(migrations_module, '__path__'):\\n+                        raise CommandError(\\n+                            \\\"Django can't use migrations for app '%s' because migrations module is not a package.\\\" % app_config.label\\n+                        )\\n+                except ImportError:\\n+                    # Module exists but can't be imported - let migration loader handle this\\n+                    pass\\n\", \"diff\\n--- a/django/core/management/commands/migrate.py\\n+++ b/django/core/management/commands/migrate.py\\n@@ -90,10 +90,14 @@ class Command(BaseCommand):\\n         # Import the 'management' module within each installed app, to register\\n         # dispatcher events.\\n         for app_config in apps.get_app_configs():\\n             if module_has_submodule(app_config.module, 'migrations'):\\n-                import_module('%s.migrations' % app_config.name)\\n-                migrations_module = import_module('%s.migrations' % app_config.name)\\n-                if not hasattr(migrations_module, '__file__'):\\n-                    raise CommandError(\\n-                        \\\"Django can't use migrations for app '%s' because it's not a package.\\\" % app_config.label\\n-                    )\\n+                try:\\n+                    migrations_module = import_module('%s.migrations' % app_config.name)\\n+                    # Check for __path__ which exists on both regular and namespace packages\\n+                    if not hasattr(migrations_module, '__path__'):\\n+                        raise CommandError(\\n+                            \\\"Django can't use migrations for app '%s' because migrations module is not a package.\\\" % app_config.label\\n+                        )\\n+                except ImportError:\\n+                    # Module exists but can't be imported - let migration loader handle this\\n+                    pass\\n\"]}, \"error\": \"\", \"elapsed_time\": 93.903773, \"total_tokens\": 38455, \"total_steps\": 30, \"created_at\": 1754647884, \"finished_at\": 1754647978}}"
}