gpt4 book ai didi

django - 添加新的唯一字段时如何正确进行迁移

转载 作者:行者123 更新时间:2023-12-04 04:29:45 24 4
gpt4 key购买 nike

我向我的一个模型添加了一个新字段:

class Agency(models.Model):
email = models.EmailField(unique=True, verbose_name=_("e-mail"))

由于此字段不能为空, django-admin makemigrations要求我提供一次性违约,我做到了。这是生成的迁移:
# Generated by Django 1.9.4 on 2016-03-20 10:38
from __future__ import unicode_literals

from django.db import migrations, models


class Migration(migrations.Migration):

dependencies = [
('accounts', '0008_auto_20160226_1226'),
]

operations = [
migrations.AddField(
model_name='agency',
name='email',
field=models.EmailField(default='example@example.fr', max_length=254, unique=True, verbose_name='e-mail'),
preserve_default=False,
),
]

正如预期的那样, django-admin migrate抛出一个错误:
psycopg2.IntegrityError: could not create unique index "accounts_agency_email_key"
DETAIL: Key (email)=(example@example.fr) is duplicate.

我想我可以在使字段唯一之前编辑迁移以设置唯一值。所以我试过:
# -*- coding: utf-8 -*-
# Generated by Django 1.9.4 on 2016-03-20 10:38
from __future__ import unicode_literals

from django.db import migrations, models
from django.utils.text import slugify


def set_email(apps, schema_editor):
Agency = apps.get_model('accounts', 'Agency')
for agency in Agency.objects.all():
agency.email = '{}@example.fr'.format(slugify(agency.name))
agency.save()


class Migration(migrations.Migration):

dependencies = [
('accounts', '0008_auto_20160226_1226'),
]

operations = [
migrations.AddField(
model_name='agency',
name='email',
field=models.EmailField(default='', max_length=254, blank=True, verbose_name='e-mail'),
preserve_default=False,
),
migrations.RunPython(set_email),
migrations.AlterField(
model_name='agency',
name='email',
field=models.EmailField(max_length=254, unique=True, verbose_name='e-mail'),
preserve_default=False,
),
]

不幸的是,我在运行 django-admin migrate 时遇到此错误:
django.db.utils.OperationalError: cannot ALTER TABLE "accounts_agency" because it has pending trigger events

我的猜测是 operations不是同步执行的。

我想我可以通过将迁移分成两次迁移来解决这个问题,但我想知道我是否只能在一次迁移中完成。在模型中添加新的唯一字段时,创建迁移的常用方法是什么?

PS:我也尝试使用 F 表达式作为默认值( default=models.F('name') + '@example.fr' ),但失败了:
django.db.utils.IntegrityError: could not create unique index "accounts_agency_email_key"
DETAIL: Key (email)=(F(name) + Vallu(@example.fr)) is duplicated.

最佳答案

也许为时已晚,但也许对其他人有用

您可以通过使用 在一次迁移中完成此操作。迁移.RunSQL 方法

对于将新字段添加到模型并运行 后的示例代码python manage.py makemigrations 命令(这里如果你的表中有现有的行命令想要选择默认值,你可以选择“现在提供一次性默认值”选项并给出一些字符串值这并不重要,因为实际上我们没有使用它)然后去迁移文件并使用此更改操作部分(注意我使用 postgresql 您可以更改数据库的 SQL)

operations = [
migrations.RunSQL(
'ALTER TABLE "agency" ADD COLUMN "email" varchar(254) NULL;ALTER TABLE "agency" ALTER COLUMN "email" DROP DEFAULT;COMMIT;',
),
migrations.RunSQL(
"UPDATE agency SET email= Concat(country_code, '@example.fr');COMMIT;",
),
migrations.RunSQL(
'ALTER TABLE "agency" ALTER COLUMN "email" SET NOT NULL;ALTER TABLE "agency" ADD CONSTRAINT "agency_email_b551ad2a_uniq" UNIQUE ("email");ALTER TABLE "agency" ALTER COLUMN "email" DROP DEFAULT;CREATE INDEX "agency_email_b551ad2a_like" ON "agency" ("email" varchar_pattern_ops);COMMIT;'
)
]

然后运行“ python manage.py migrate ”命令
这就对了。

关于django - 添加新的唯一字段时如何正确进行迁移,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36113432/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com