{
  "id": "django__django-14999",
  "question": "RenameModel with db_table should be a noop.\nDescription\n\t\nA RenameModel operation that already has db_table defined must be a noop.\nIn Postgres, it drops and recreates foreign key constraints. In sqlite it recreates the table (as expected for a table renaming).\n",
  "answer": "{\"task_id\": \"6e76ad06-bd87-405e-89a7-c7368725de60\", \"workflow_run_id\": \"1556f542-f581-4409-9537-158fc88b2542\", \"data\": {\"id\": \"1556f542-f581-4409-9537-158fc88b2542\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/db/migrations/operations/models.py b/django/db/migrations/operations/models.py\\\\nindex abcdef1..1234567 100644\\\\n--- a/django/db/migrations/operations/models.py\\\\n+++ b/django/db/migrations/operations/models.py\\\\n@@ -52,6 +52,11 @@ class RenameModel(Operation):\\\\n         old_model = apps.get_model(app_label, self.old_name)\\\\n         new_model = apps.get_model(app_label, self.new_name)\\\\n         old_db_table = old_model._meta.db_table\\\\n \\\\n+        # Skip renaming if db_table is explicitly set, should be a noop\\\\n+        if old_model._meta.db_table:\\\\n+            return\\\\n+\\\\n         # Check if the old_table and new_table are the same, so we can rename the table\\\\n         new_db_table = new_model._meta.db_table\\\\n         if old_db_table == new_db_table:\\\\n@@ -82,6 +87,11 @@ class RenameModel(Operation):\\\\n         old_model = apps.get_model(app_label, self.old_name)\\\\n         new_model = apps.get_model(app_label, self.new_name)\\\\n         new_db_table = new_model._meta.db_table\\\\n \\\\n+        # Skip reversing if db_table is explicitly set, should be a noop\\\\n+        if new_model._meta.db_table:\\\\n+            return\\\\n+\\\\n         # Reverse renaming\\\\n         connection.schema_editor().alter_db_table(\\\\n             new_model, new_db_table, old_model._meta.db_table\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly addresses the issue by checking if `db_table` is explicitly set and returning early, making the operation a no-op. It is straightforward and adheres to best practices, minimizing the risk of introducing new bugs.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/migrations/operations/models.py b/django/db/migrations/operations/models.py\\\\nindex abcdef1..89efd34 100644\\\\n--- a/django/db/migrations/operations/models.py\\\\n+++ b/django/db/migrations/operations/models.py\\\\n@@ -53,6 +53,9 @@ class RenameModel(Operation):\\\\n         old_model = apps.get_model(app_label, self.old_name)\\\\n         new_model = apps.get_model(app_label, self.new_name)\\\\n         old_db_table = old_model._meta.db_table\\\\n \\\\n+        # If db_table is set, assume no operation is needed\\\\n+        if old_model._meta.original_attrs.get('db_table', None):\\\\n+            return\\\\n         # Alter table\\\\n         new_db_table = new_model._meta.db_table\\\\n         connection.schema_editor().alter_db_table(old_model, old_db_table, new_db_table)\\\\n@@ -83,6 +86,9 @@ class RenameModel(Operation):\\\\n         old_model = apps.get_model(app_label, self.old_name)\\\\n         new_model = apps.get_model(app_label, self.new_name)\\\\n         new_db_table = new_model._meta.db_table\\\\n \\\\n+        # Skip if db_table is set, assume no operation is needed\\\\n+        if new_model._meta.original_attrs.get('db_table', None):\\\\n+            return\\\\n         # Reverse alter table\\\\n         connection.schema_editor().alter_db_table(new_model, new_db_table, old_model._meta.db_table)\\\\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.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue but uses `original_attrs.get('db_table', None)` which might not be as clear or standard as directly checking `_meta.db_table`. It is slightly more complex and could potentially miss some edge cases.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/tests/migrations/test_operations.py b/tests/migrations/test_operations.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/tests/migrations/test_operations.py\\\\n+++ b/tests/migrations/test_operations.py\\\\n@@ -200,6 +200,23 @@ class OperationTests(unittest.TestCase):\\\\n         ...\\\\n \\\\n+    def test_rename_model_db_table_noop(self):\\\\n+        \\\\\\\"\\\\\\\"\\\\\\\"Test RenameModel is a noop when db_table is set.\\\\\\\"\\\\\\\"\\\\\\\"\\\\n+        class OldModel(models.Model):\\\\n+            field = models.IntegerField()\\\\n+\\\\n+            class Meta:\\\\n+                db_table = 'custom_table_name'\\\\n+\\\\n+        class NewModel(models.Model):\\\\n+            field = models.IntegerField()\\\\n+\\\\n+\\\\n+        migration = RenameModel('app', 'OldModel', 'NewModel')\\\\n+\\\\n+        with mock.patch('django.db.migrations.operations.models.RenameModel.database_forwards') as forwards_patch:\\\\n+            migration.state_forwards('app', state)\\\\n+            forwards_patch.assert_not_called()\\\\n+        with mock.patch('django.db.migrations.operations.models.RenameModel.database_backwards') as backwards_patch:\\\\n+            migration.state_backwards('app', state)\\\\n+            backwards_patch.assert_not_called()\\\\n+\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.7,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch focuses on adding a test rather than modifying the core logic. While it is useful for validation, it does not directly address the issue in the codebase. It is more of a supplementary patch.\\\",\\n      \\\"recommendation\\\": \\\"Useful for testing, but not a standalone fix\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective solution, providing a clear and maintainable fix with minimal risk. Patch 2 offers a similar solution but with a less conventional approach. Patch 3 adds valuable test coverage but does not directly resolve the issue.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_with_explicit_db_table\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_with_explicit_db_table():\\\\n    # Ensure that RenameModel operation is a noop when db_table is explicitly set\\\\n    class OldModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            db_table = 'custom_table_name'\\\\n\\\\n    class NewModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    migration = RenameModel('app', 'OldModel', 'NewModel')\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as alter_db_table_patch:\\\\n        migration.state_forwards('app', state)\\\\n        alter_db_table_patch.assert_not_called()\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as reverse_alter_db_table_patch:\\\\n        migration.state_backwards('app', state)\\\\n        reverse_alter_db_table_patch.assert_not_called()\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by using a RenameModel operation with db_table explicitly set\\\",\\n      \\\"expected_behavior\\\": \\\"The database operations should not be triggered as the RenameModel operation should be a no-op\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_without_db_table\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_without_db_table():\\\\n    # Ensure that RenameModel operation works as expected without db_table set\\\\n    class OldModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    class NewModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    migration = RenameModel('app', 'OldModel', 'NewModel')\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as alter_db_table_patch:\\\\n        migration.state_forwards('app', state)\\\\n        alter_db_table_patch.assert_called_once()\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as reverse_alter_db_table_patch:\\\\n        migration.state_backwards('app', state)\\\\n        reverse_alter_db_table_patch.assert_called_once()\\\",\\n      \\\"description\\\": \\\"This test ensures that RenameModel operation works as expected when db_table is not explicitly set\\\",\\n      \\\"expected_behavior\\\": \\\"The database operations should be triggered as the RenameModel operation is necessary\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_1():\\\\n    # Test the first patch implementation to validate the check for db_table\\\\n    class OldModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            db_table = 'custom_table_name'\\\\n\\\\n    class NewModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    migration = RenameModel('app', 'OldModel', 'NewModel')\\\\n    with mock.patch('django.db.migrations.operations.models.RenameModel.database_forwards') as forwards_patch:\\\\n        migration.state_forwards('app', state)\\\\n        forwards_patch.assert_not_called()\\\\n    with mock.patch('django.db.migrations.operations.models.RenameModel.database_backwards') as backwards_patch:\\\\n        migration.state_backwards('app', state)\\\\n        backwards_patch.assert_not_called()\\\",\\n      \\\"description\\\": \\\"Test the first patch implementation to ensure the RenameModel operation is a noop with db_table set\\\",\\n      \\\"expected_behavior\\\": \\\"The database operations should not be triggered\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_2():\\\\n    # Test the second patch implementation to validate the check for db_table\\\\n    class OldModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    class NewModel(models.Model):\\\\n        field = models.IntegerField()\\\\n\\\\n    migration = RenameModel('app', 'OldModel', 'NewModel')\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as alter_db_table_patch:\\\\n        migration.state_forwards('app', state)\\\\n        alter_db_table_patch.assert_called_once()\\\\n    with mock.patch('django.db.migrations.operations.models.connection.schema_editor().alter_db_table') as reverse_alter_db_table_patch:\\\\n        migration.state_backwards('app', state)\\\\n        reverse_alter_db_table_patch.assert_called_once()\\\",\\n      \\\"description\\\": \\\"Test the second patch implementation to ensure the RenameModel operation works without db_table set\\\",\\n      \\\"expected_behavior\\\": \\\"The database operations should be triggered as necessary\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases have been generated to reproduce the original issue using RenameModel operation with and without db_table explicitly set. Additionally, validation tests have been created to ensure the patches effectively make RenameModel a no-op when db_table is defined.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 465.815698, \"total_tokens\": 22501, \"total_steps\": 9, \"created_at\": 1753310577, \"finished_at\": 1753311043}}"
}