2010-02-08 7 views
30

У меня есть приложение Django, в котором я хочу изменить поле из ForeignKey на ManyToManyField. Я хочу сохранить свои старые данные. Каков самый простой/лучший процесс для этого? Если это имеет значение, я использую sqlite3 в качестве базы данных моей базы данных.Перенос данных Django при изменении поля на ManyToMany

Если мое резюме проблемы неясно, вот пример. Скажем, у меня две модели:

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) 

Я не хочу, чтобы «потерять» все мои предыдущие данные.

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

Кен

+1

Если это уже не очевидно, убедитесь, что вы создали резервные копии своих данных, прежде чем пытаться выполнить какие-либо миграции. к счастью, копирование sqlite так же просто, как команда 'cp' –

+0

Отъезд [юг] (http://south.aeracode.org/). – Zach

+0

Более конкретно ознакомьтесь с разделом «миграции данных» в учебном пособии: http://south.aeracode.org/wiki/Tutorial3 Это хорошая привычка использовать Юг для всех ваших миграций. –

ответ

40

Я понимаю, что этот вопрос старый и в то время лучший вариант для миграции данных использует Юг. Теперь 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

Очень хорошо !!! Это интересно. ☺ –

+2

Отличный ответ, спасибо! – Ward

+0

Вам не нужно «book.save()' в 'make_many_authors'? –

2

Вероятно, самый лучший и самый простой, что вы должны сделать, было бы:

Создайте Многие ко многим поле с другим именем сказать

authors = models.ManyToManyField(Author) 

написать небольшую функцию для преобразования ForeignKey значения величин M2M:

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

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

+0

in django 1.10 i get a AttributeError: объект ManyRelatedManager 'не имеет атрибута' append'' i удален .id и имел 'book.authors = li', и это сработало. Я считаю, что вы также можете сделать book.authors.add (li) ', и это должно решить проблему. – hachacha

+0

Вы можете добавлять и сохранять в моделях в любом случае. Вопрос и ответ, в частности, заботятся о миграции. – sprksh

Смежные вопросы