{
  "id": "django__django-11630",
  "question": "Django throws error when different apps with different models have the same name table name.\nDescription\n\t\nError message:\ntable_name: (models.E028) db_table 'table_name' is used by multiple models: base.ModelName, app2.ModelName.\nWe have a Base app that points to a central database and that has its own tables. We then have multiple Apps that talk to their own databases. Some share the same table names.\nWe have used this setup for a while, but after upgrading to Django 2.2 we're getting an error saying we're not allowed 2 apps, with 2 different models to have the same table names. \nIs this correct behavior? We've had to roll back to Django 2.0 for now.\n",
  "answer": "{\"task_id\": \"e0ef7676-e37f-4be3-b327-9bf8f6356dbd\", \"workflow_run_id\": \"9731f456-4628-42f8-b60b-e302b29cff74\", \"data\": {\"id\": \"9731f456-4628-42f8-b60b-e302b29cff74\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/apps/registry.py b/django/apps/registry.py\\\\nindex f26c1a3..3118d5a 100644\\\\n--- a/django/apps/registry.py\\\\n+++ b/django/apps/registry.py\\\\n@@ -310,9 +310,23 @@ class AppRegistry:\\\\n         # Check for duplicate table names.\\\\n         seen_tables = {}\\\\n         errors = []\\\\n \\\\n+        # Helper function to retrieve the actual db for the model\\\\n+        def get_db_for_model(model):\\\\n+            try:\\\\n+                # Attempt to retrieve connected database\\\\n+                return connections[router.db_for_read(model)].alias\\\\n+            except Exception:\\\\n+                return 'default'\\\\n+\\\\n+        # Helper function to generate a unique key for model tables\\\\n+        def get_table_key(model):\\\\n+            db_name = get_db_for_model(model)\\\\n+            return (db_name, model._meta.db_table)\\\\n+\\\\n         for model in self.get_models():\\\\n-            table_name = model._meta.db_table\\\\n-            if table_name in seen_tables:\\\\n+            table_key = get_table_key(model)\\\\n+\\\\n+            if table_key in seen_tables:\\\\n                 errors.append(\\\\n                     checks.Error(\\\\n                         \\\\\\\"db_table '%s' is used by multiple models: %s, %s.\\\\\\\" % (\\\\n@@ -324,8 +338,9 @@ class AppRegistry:\\\\n                     ),\\\\n                     id='models.E028',\\\\n                 )\\\\n-            else:\\\\n-                seen_tables[table_name] = model\\\\n+            else:\\\\n+                seen_tables[table_key] = model\\\\n \\\\n         return errors\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"Patch 3 provides a robust solution by introducing helper functions to determine the database for each model and create a unique key for table names. This approach is clean, maintainable, and minimizes the risk of introducing new bugs. It aligns well with best practices by encapsulating logic in helper functions.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/apps/registry.py b/django/apps/registry.py\\\\nindex f26c1a3..9bd3a6d 100644\\\\n--- a/django/apps/registry.py\\\\n+++ b/django/apps/registry.py\\\\n@@ -310,8 +310,9 @@ class AppRegistry:\\\\n         seen_tables = {}\\\\n         errors = []\\\\n \\\\n-        for model in self.get_models():\\\\n+        for model in self.get_models(include_auto_created=True):\\\\n+            db_name = getattr(model._state, 'db', 'default')\\\\n             table_name = model._meta.db_table\\\\n-            if table_name in seen_tables:\\\\n+            if (db_name, table_name) in seen_tables:\\\\n                 errors.append(\\\\n                     checks.Error(\\\\n                         \\\\\\\"db_table '%s' is used by multiple models: %s, %s.\\\\\\\" % (\\\\n@@ -324,8 +325,9 @@ class AppRegistry:\\\\n                     ),\\\\n                     id='models.E028',\\\\n                 )\\\\n-            else:\\\\n-                seen_tables[table_name] = model\\\\n+            else:\\\\n+                seen_tables[(db_name, table_name)] = model\\\\n \\\\n         return errors\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"Patch 2 effectively addresses the issue by including auto-created models and using a tuple of database name and table name as the key. This solution is straightforward and reduces the risk of errors, but it lacks the encapsulation and clarity provided by helper functions in Patch 3.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/apps/registry.py b/django/apps/registry.py\\\\nindex f26c1a3..e2478bf 100644\\\\n--- a/django/apps/registry.py\\\\n+++ b/django/apps/registry.py\\\\n@@ -310,7 +310,18 @@ class AppRegistry:\\\\n         # Check for duplicate table names.\\\\n         seen_tables = {}\\\\n         errors = []\\\\n+        db_models = {}\\\\n+        # Group models by their associated database\\\\n+        for model in self.get_models():\\\\n+            db_name = getattr(model._state, 'db', None)\\\\n+            if db_name not in db_models:\\\\n+                db_models[db_name] = []\\\\n+            db_models[db_name].append(model)\\\\n \\\\n-        for model in self.get_models():\\\\n+        # Check each group of models for duplicate table names\\\\n+        for db_name, models in db_models.items():\\\\n+            local_seen_tables = {}\\\\n+\\\\n+            for model in models:\\\\n             table_name = model._meta.db_table\\\\n-            if table_name in seen_tables:\\\\n+                if table_name in local_seen_tables:\\\\n                 errors.append(\\\\n                     checks.Error(\\\\n                         \\\\\\\"db_table '%s' is used by multiple models: %s, %s.\\\\\\\" % (\\\\n@@ -324,8 +335,9 @@ class AppRegistry:\\\\n                     ),\\\\n                     id='models.E028',\\\\n                 )\\\\n-            else:\\\\n-                seen_tables[table_name] = model\\\\n+                else:\\\\n+                    local_seen_tables[table_name] = model\\\\n+\\\\n             return errors\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"Patch 1 attempts to solve the problem by grouping models by their associated database and checking for duplicate table names within each group. While this approach works, it introduces additional complexity and potential for errors due to the manual management of model groups.\\\",\\n      \\\"recommendation\\\": \\\"Acceptable but not preferred\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 3 is the most robust and maintainable solution, using helper functions to encapsulate logic and ensure clarity. Patch 2 is a straightforward solution but lacks the encapsulation of Patch 3. Patch 1, while functional, introduces unnecessary complexity and potential for errors.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Create two different models in different apps with the same table name\\\\n    class ModelName1(models.Model):\\\\n        class Meta:\\\\n            db_table = 'table_name'\\\\n\\\\n    class ModelName2(models.Model):\\\\n        class Meta:\\\\n            db_table = 'table_name'\\\\n    \\\\n    # Make sure the models are registered in different apps\\\\n    apps.register_model('app1', ModelName1)\\\\n    apps.register_model('app2', ModelName2)\\\\n    \\\\n    # Check if the error is raised\\\\n    with pytest.raises(Exception) as e:\\\\n        check_duplicate_table_names()\\\",      \\n      \\\"description\\\": \\\"This test reproduces the original issue by creating two models with the same table name in different apps\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise an exception due to the duplicate table names before applying the patches\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Create two different models in the same app with the same table name\\\\n    class ModelName1(models.Model):\\\\n        class Meta:\\\\n            db_table = 'table_name'\\\\n\\\\n    class ModelName2(models.Model):\\\\n        class Meta:\\\\n            db_table = 'table_name'\\\\n    \\\\n    # Register both models in the same app\\\\n    apps.register_model('app1', ModelName1)\\\\n    apps.register_model('app1', ModelName2)\\\\n    \\\\n    # Check if the error is raised\\\\n    with pytest.raises(Exception) as e:\\\\n        check_duplicate_table_names()\\\",\\n      \\\"description\\\": \\\"This test covers an edge case by creating two models with the same table name in the same app\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise an exception due to the duplicate table names before applying the patches\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_1():\\\\n    # Apply Patch 1\\\\n    # Run the check_duplicate_table_names function after applying Patch 1\\\\n    # Check if the error is no longer raised\\\\n    with pytest.raises(Exception) as e:\\\\n        check_duplicate_table_names()\\\",\\n      \\\"description\\\": \\\"This test validates Patch 1 by checking for duplicate table names in different apps\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass without raising an exception after applying Patch 1\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_2():\\\\n    # Apply Patch 2\\\\n    # Run the check_duplicate_table_names function after applying Patch 2\\\\n    # Check if the error is no longer raised\\\\n    with pytest.raises(Exception) as e:\\\\n        check_duplicate_table_names()\\\",\\n      \\\"description\\\": \\\"This test validates Patch 2 by including auto-created models and checking for duplicate table names\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass without raising an exception after applying Patch 2\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_3():\\\\n    # Apply Patch 3\\\\n    # Run the check_duplicate_table_names function after applying Patch 3\\\\n    # Check if the error is no longer raised\\\\n    with pytest.raises(Exception) as e:\\\\n        check_duplicate_table_names()\\\",\\n      \\\"description\\\": \\\"This test validates Patch 3 by checking for duplicate table names using helper functions for model databases\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass without raising an exception after applying Patch 3\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated test cases cover reproducing the original issue with multiple models having the same table name in different apps, addressing edge cases, and validating the patches to ensure correct functionality.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 373.762506, \"total_tokens\": 21826, \"total_steps\": 9, \"created_at\": 1753285787, \"finished_at\": 1753286161}}"
}