这个问题其实要分两步,第一步是清理数据库中django_migrations
这张表,第二步是重新生成迁移文件并且在不影响现有数据库结构的情况下实施到数据库中。
清理数据库中迁移记录
这里要分两种情况讨论,当你的迁移文件不小心丢失了一部分或者全部没了的时候,你需要手动去数据库里清空django_migrations
这张表。
而如果你本地拥有完整的迁移文件,就可以使用Django自带的命令manage.py migrate --fake APP_NAME zero
进行清理。参考 How to Reset Migrations 可以得知,使用manage.py migrate --fake APP_NAME zero
可以清理对应的迁移文件依赖,并且将它们从数据库里的django_migrations
表里删除。
migrate --fake
根据 Django Document 的描述,当执行manage.py migrate --fake
时,它并不会真的去运行SQL,只是把对应的迁移文件给填充进数据库中django_migrations
这张表里。
当然,在运行此命令之前,你必须保证迁移文件与数据库结构吻合。
封装成命令
对于某些爱好清理迁移文件以保持整个项目的迁移文件看起来很干净的人来说,一键清理很必要。
在任意一个(只需要一个就够了)APP里,新建立如上图的结构,并且在clearmigrations.py
文件里写入以下内容
import os
import sys
import shutil
from django.core.management.base import BaseCommand
from django.conf import settings
from django.apps import apps
class Command(BaseCommand):
help = "Clear all migration files in project."
def get_apps(self):
for app in apps.get_app_configs():
path = os.path.join(
settings.BASE_DIR, app.name.replace(".", "/"), "migrations"
)
if os.path.exists(path):
yield app, path
def execute_command(self, command: str):
self.stdout.write(f"Run command: {command}")
os.system(f"{sys.executable} {command}")
def handle(self, *args, **options):
self.execute_command(f"manage.py makemigrations")
self.execute_command(f"manage.py migrate")
for app, path in self.get_apps():
self.execute_command(
f"manage.py migrate --fake {app.name.split('.')[-1]} zero"
)
for app, path in self.get_apps():
shutil.rmtree(path)
os.makedirs(path)
with open(os.path.join(path, "__init__.py"), "w+") as file:
file.write("")
self.stdout.write(self.style.SUCCESS(f"Clear {path}"))
self.execute_command(f"manage.py makemigrations")
self.execute_command(f"manage.py migrate --fake")
self.stdout.write(self.style.SUCCESS("Successfully cleared!"))
然后当你执行python manage.py clearmigrations
的时候,就会自动的从数据库和本地清除迁移文件了。