将字段更改为多对多时的 Django 数据迁移

Posted

技术标签:

【中文标题】将字段更改为多对多时的 Django 数据迁移【英文标题】:Django data migration when changing a field to ManyToMany 【发布时间】:2011-01-14 12:41:39 【问题描述】:

我有一个 Django 应用程序,我想在其中将字段从 ForeignKey 更改为 ManyToManyField。我想保留我的旧数据。为此,最简单/最好的流程是什么?如果重要,我使用 sqlite3 作为我的数据库后端。

如果我对问题的总结不清楚,这里有一个例子。假设我有两个模型:

class Author(models.Model):  
    author = models.CharField(max_length=100) 

class Book(models.Model):  
    author = models.ForeignKey(Author)  
    title = models.CharField(max_length=100)

假设我的数据库中有很多数据。现在,我想将 Book 模型更改如下:

class Book(models.Model):  
    author = models.ManyToManyField(Author)  
    title = models.CharField(max_length=100) 

我不想“丢失”我之前的所有数据。

完成此任务的最佳/最简单方法是什么?

【问题讨论】:

查看south。 更具体地查看教程的“数据迁移”部分:south.aeracode.org/wiki/Tutorial3 无论如何,使用 South 进行所有迁移是一个好习惯。 很棒的工具,感谢您的链接! 如果这还不明显,请确保在尝试任何迁移之前备份数据。幸运的是,复制 sqlite 就像 cp 命令一样简单 我想知道是否有关于此的错误报告。 【参考方案1】:

我意识到这个问题已经过时了,当时数据迁移的最佳选择是使用 South。现在django有了自己的migrate命令,过程略有不同。

我已将这些模型添加到名为 books 的应用程序中——如果不是您的情况,请进行相应调整。

首先,将字段添加到Bookrelated_name 到至少一个或两个(否则它们会发生冲突):

class Book(models.Model):  
    author = models.ForeignKey(Author, related_name='book')
    authors = models.ManyToManyField(Author, related_name='books')
    title = models.CharField(max_length=100) 

生成迁移:

$ ./manage.py makemigrations
Migrations for 'books':
  0002_auto_20151222_1457.py:
    - Add field authors to book
    - Alter field author on book

现在,创建一个空迁移来保存数据本身的迁移:

./manage.py makemigrations books --empty
    Migrations for 'books':
0003_auto_20151222_1459.py:

并在其中添加以下内容。要准确了解其工作原理,请查看Data Migrations 上的文档。注意不要覆盖迁移依赖。

# -*- coding: utf-8 -*-
from __future__ import unicode_literals

from django.db import models, migrations


def make_many_authors(apps, schema_editor):
    """
        Adds the Author object in Book.author to the
        many-to-many relationship in Book.authors
    """
    Book = apps.get_model('books', 'Book')

    for book in Book.objects.all():
        book.authors.add(book.author)


class Migration(migrations.Migration):

    dependencies = [
        ('books', '0002_auto_20151222_1457'),
    ]

    operations = [
        migrations.RunPython(make_many_authors),
    ]

现在从模型中删除 author 字段——它应该如下所示:

class Book(models.Model):
    authors = models.ManyToManyField(Author, related_name='books')
    title = models.CharField(max_length=100)

为此创建一个新的迁移,然后全部运行:

$ ./manage.py makemigrations
Migrations for 'books':
  0004_remove_book_author.py:
    - Remove field author from book

$ ./manage.py migrate
Operations to perform:
  Synchronize unmigrated apps: messages, staticfiles
  Apply all migrations: admin, auth, sessions, books, contenttypes
Synchronizing apps without migrations:
  Creating tables...
    Running deferred SQL...
  Installing custom SQL...
Running migrations:
  Rendering model states... DONE
  Applying books.0002_auto_20151222_1457... OK
  Applying books.0003_auto_20151222_1459... OK
  Applying books.0004_remove_book_author... OK

就是这样。以前在book.author 上的作者现在应该在您从book.authors.all() 获得的查询集中。

【讨论】:

嘿@JanSegre:不是真的,多对多关系的变化在你运行.add()的那一刻就完成了。 哦,你是对的,谢谢。这就是我掩饰文档而不是正确阅读它们的结果。 您可以在一次迁移中完成,在operations 列出此命令中的写入命令:add_m2m_field、your_function_to_migrate_data、delete_fk_field、rename_m2m_field 嗨@Salvioner,apps 不是导入,它是RunPython 传递给make_many_authors 的参数。这个答案现在有 5 年历史了,我近年来没有运行过这段代码,但无论如何解决这个问题的方法都不会是添加导入。 嗨@Django-Rocks,我真的很抱歉你的数据不见了,但我已经好几年没和 Django 一起工作了(这个答案是 5 岁)但我要告诉你的第一件事要查找的是本地环境和生产环境之间的差异。祝你好运!【参考方案2】:

您应该做的最好和最简单的事情可能是:

使用不同的名称创建多对多字段

authors = models.ManyToManyField(Author)

编写一个小函数将外键值转换为 M2M 值:

def convert():
    books = Book.objects.all()
    for book in books:
        if book.author:
            li = [book.author.id]
            book.authors.append(li)
            book.save()

运行后,您可以从表中删除作者字段并再次运行迁移。

【讨论】:

在 django 1.10 中,我得到了一个 AttributeError: 'ManyRelatedManager' object has no attribute 'append' 我删除了 .id 并拥有了 book.authors = li 并且有效。我相信您也可以使用book.authors.add(li),这应该可以解决问题。 您可以随意添加和保存模型。问题和答案更具体地处理迁移。

以上是关于将字段更改为多对多时的 Django 数据迁移的主要内容,如果未能解决你的问题,请参考以下文章

将 m2m 更改为 char 时,Django South 迁移失败

Django 1.8:删除/重命名数据迁移中的模型字段

Django - 将 ForeignKey 关系更改为 OneToOne

Django 最佳实践——迁移数据

sql 2005中的数据迁移

将 Django 数据库后端从 MySql 更改为 PostgreSQL