что происходит с существующими данными, когда мы меняем модель Charfield на IntegerField в django? - PullRequest
0 голосов
/ 27 сентября 2019

Когда применяется следующая миграция,

class Migration(migrations.Migration):
    dependencies = [
        ('outlets', '0009_auto_20190920_1155'),
    ]

    operations = [
        migrations.AlterField(
            model_name='outlet',
            name='country',
            field=models.IntegerField(choices=[(1, 'UAE'), (2, 'India')],
                                      verbose_name='Country'),
        ),
    ]

Произошла следующая ошибка,

return self.cursor.execute(sql, params)
django.db.utils.DataError: invalid input syntax for integer: "United States"

Это потому, что я меняю CharField на IntegerField и уже следую данным "United States"существует в БД, также я читал в аналогичном вопросе, миграция django не может обработать такие изменения.

Есть ли способ выполнить эту операцию, не удаляя существующие данные?

Ответы [ 2 ]

1 голос
/ 27 сентября 2019

Если вы хотите, чтобы дела шли гладко, вам нужно интегрировать мутацию данных в миграцию

Как пример:

from django.db import migrations, models
import django.db.models.deletion

def migrate_country(apps, schema_editor):
    Outlet = apps.get_model('your_app_name', 'Outlet')
    # You can either do
    Outlet.objects.all().update(
        temp_country= 'what_ever_update you want'
    )
    # Or
    for outlet in Outlet.objects.all():
        outlet.temp_country='your_transformation_result'
        outlet.save() # Please note django will use it's save method. 
                      # Your eventual override will just be ignore

def reverse_country(apps, schema_editor):
    """
    Do want you want when rolling back your migration
    """
    ....

class Migration(migrations.Migration):

    dependencies = [
        ...
    ]

    operations = [
        migrations.AddField(
            model_name='Outlet',
            name='temp_outlet',
            name='country',
            field=models.IntegerField(choices=[(1, 'UAE'), (2, 'India')],
                                      verbose_name='Country'),
        ),
        # Now, you have temp_outlet available on your schema!
        migrations.RunPython(migrate_country, reverse_country),
        # We dump "old" field
        migrations.RemoveField(
            model_name='article',
            name='outlet',
        ),
        # We rename the 'temp' field as excpected!
        migrations.RenameField(
        model_name='Outlet',
            old_name='oldname',
            new_name='newname',
    ),
    ]
0 голосов
/ 27 сентября 2019

Вам нужно будет сделать это в трех отдельных миграциях.

Во-первых, добавить IntegerField, либо null = True, либо значение по умолчанию 0.

Во-вторых, добавить миграцию данныхкоторый принимает значение в существующем CharField, ищет соответствующее целочисленное значение и устанавливает его в новое IntegerField.

В-третьих, удалите CharField, переименуйте IntegerField и удалите значение по умолчанию / null.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...