Миграция данных Django при изменении поля на ManyToMany - PullRequest
37 голосов
/ 08 февраля 2010

У меня есть приложение 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)

Скажем, у меня много данных в моей базе данных. Теперь я хочу изменить модель Книги следующим образом:

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

Я не хочу "потерять" все свои предыдущие данные.

Каков наилучший / самый простой способ сделать это?

Ken

Ответы [ 2 ]

58 голосов
/ 22 декабря 2015

Я понимаю, что этот вопрос старый и в то время лучшим вариантом для переноса данных было использование Юга. Теперь у Django есть собственная команда migrate, и процесс немного отличается.

Я добавил эти модели в приложение под названием books - отрегулируйте соответственно, если это не ваш случай.

Сначала добавьте поле к Book и к related_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().

1 голос
/ 06 сентября 2016

Вероятно, лучшая и самая легкая вещь, которую вы должны сделать, была бы:

Создайте поле «Многие ко многим» с другим именем, скажем

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()

После запуска вы можете удалить поле автора из таблицы и снова запустить миграцию.

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