{
  "Selected_candidate": {
    "pr_number": 14935,
    "pr_title": "Fixed #23408 -- Added migrations questioner prompt for adding unique fields with a callable default.",
    "pr_body": "ticket-23408",
    "issue_id": 23408,
    "issue_title": "Add makemigrations warning for unique fields with callable defaults",
    "issue_body": "Callables on properties for ModelFields are used for various reasons. One use case is to autocreate random file names or user passwords if not present.\nThe migration seems to call them only once because after the migration every \"Buchung\" has the same wlan_password.\nMy Model:\ndef random_wlan_key():\n    return ''.join(random.SystemRandom().choice(\"1234567890abcdefghkmnpqrstuvwxyz\") for i in range(9))\n\nclass Buchung(models.Model):\n    [...]\n    wlan_passwort = models.CharField(max_length=10, default=random_wlan_key)\nThe generated migration:\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\nimport main.models\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        ('main', '0001_initial'),\n    ]\n\n    operations = [\n        migrations.AddField(\n            model_name='buchung',\n            name='wlan_passwort',\n            field=models.CharField(default=main.models.random_wlan_key, max_length=10),\n            preserve_default=True,\n        ),\n    ]",
    "issue_closed_at": "2021-10-06T01:58:38",
    "base_commit": "0f3e1a54bfa19f034f88bf3c25c67402d19e906c",
    "changes": [
      {
        "file": "django/db/migrations/autodetector.py",
        "type": "function",
        "name": "_generate_added_field",
        "class_name": "MigrationAutodetector",
        "code": "def _generate_added_field(self, app_label, model_name, field_name):\n        field = self.to_state.models[app_label, model_name].get_field(field_name)\n        # Fields that are foreignkeys/m2ms depend on stuff\n        dependencies = []\n        if field.remote_field and field.remote_field.model:\n            dependencies.extend(self._get_dependencies_for_foreign_key(\n                app_label, model_name, field, self.to_state,\n            ))\n        # You can't just add NOT NULL fields with no default or fields\n        # which don't allow empty strings as default.\n        time_fields = (models.DateField, models.DateTimeField, models.TimeField)\n        preserve_default = (\n            field.null or field.has_default() or field.many_to_many or\n            (field.blank and field.empty_strings_allowed) or\n            (isinstance(field, time_fields) and field.auto_now)\n        )\n        if not preserve_default:\n            field = field.clone()\n            if isinstance(field, time_fields) and field.auto_now_add:\n                field.default = self.questioner.ask_auto_now_add_addition(field_name, model_name)\n            else:\n                field.default = self.questioner.ask_not_null_addition(field_name, model_name)\n        self.add_operation(\n            app_label,\n            operations.AddField(\n                model_name=model_name,\n                name=field_name,\n                field=field,\n                preserve_default=preserve_default,\n            ),\n            dependencies=dependencies,\n        )"
      },
      {
        "file": "django/db/migrations/questioner.py",
        "type": "line",
        "name": "line 6",
        "code": "from django.apps import apps\nfrom django.db.models import NOT_PROVIDED\nfrom django.utils import timezone\n\nfrom .loader import MigrationLoader\n"
      },
      {
        "file": "django/db/migrations/questioner.py",
        "type": "function",
        "name": "ask_auto_now_add_addition",
        "class_name": "NonInteractiveMigrationQuestioner",
        "code": "def ask_auto_now_add_addition(self, field_name, model_name):\n        # We can't ask the user, so act like the user aborted.\n        sys.exit(3)"
      },
      {
        "file": "django/db/migrations/questioner.py",
        "type": "function",
        "name": "ask_auto_now_add_addition",
        "class_name": "NonInteractiveMigrationQuestioner",
        "code": "def ask_auto_now_add_addition(self, field_name, model_name):\n        # We can't ask the user, so act like the user aborted.\n        sys.exit(3)"
      }
    ]
  },
  "Justification": "Candidate A addresses a very similar issue regarding callable defaults for model fields, which is central to the CURRENT bug. Both reports involve the handling of default values in migrations and the effects of referencing class methods incorrectly. This similarity in structure and symptom guarantees that solutions or insights derived from Candidate A will directly inform the debugging process of the CURRENT bug, making it the most relevant and helpful report for this situation.",
  "instance_id": "django__django-17087",
  "repo": "django/django",
  "created_at": "2023-07-17T20:28:41Z",
  "problem_statement": "Class methods from nested classes cannot be used as Field.default.\nDescription\n\t \n\t\t(last modified by Mariusz Felisiak)\n\t \nGiven the following model:\n \nclass Profile(models.Model):\n\tclass Capability(models.TextChoices):\n\t\tBASIC = (\"BASIC\", \"Basic\")\n\t\tPROFESSIONAL = (\"PROFESSIONAL\", \"Professional\")\n\t\t\n\t\t@classmethod\n\t\tdef default(cls) -> list[str]:\n\t\t\treturn [cls.BASIC]\n\tcapabilities = ArrayField(\n\t\tmodels.CharField(choices=Capability.choices, max_length=30, blank=True),\n\t\tnull=True,\n\t\tdefault=Capability.default\n\t)\nThe resulting migration contained the following:\n # ...\n\t migrations.AddField(\n\t\t model_name='profile',\n\t\t name='capabilities',\n\t\t field=django.contrib.postgres.fields.ArrayField(base_field=models.CharField(blank=True, choices=[('BASIC', 'Basic'), ('PROFESSIONAL', 'Professional')], max_length=30), default=appname.models.Capability.default, null=True, size=None),\n\t ),\n # ...\nAs you can see, migrations.AddField is passed as argument \"default\" a wrong value \"appname.models.Capability.default\", which leads to an error when trying to migrate. The right value should be \"appname.models.Profile.Capability.default\".\n",
  "patch": "diff --git a/django/db/migrations/serializer.py b/django/db/migrations/serializer.py\n--- a/django/db/migrations/serializer.py\n+++ b/django/db/migrations/serializer.py\n@@ -168,7 +168,7 @@ def serialize(self):\n         ):\n             klass = self.value.__self__\n             module = klass.__module__\n-            return \"%s.%s.%s\" % (module, klass.__name__, self.value.__name__), {\n+            return \"%s.%s.%s\" % (module, klass.__qualname__, self.value.__name__), {\n                 \"import %s\" % module\n             }\n         # Further error checking\n"
}