记Django数据库迁移过程中遇到的一些问题

时间:2021-04-01 14:22:00

首先描述一下问题,Django 数据库使用的mysql, 然后开始没注意,没建一个default库,就把第一个数据库当成默认的了,结果Django的admin相关的那些表,都自动生成到这个库里了,现在想迁移出来,放到一个新的数据库里,遇到的问题就是,migration是成功的,但是迁移的时候就是不成功

最终问题是解决了,可能原因有以下几个:

1. 我发现我的urls.py 文件中,有两个不同路径对应了一个相同的试图函数,其实就是写了两个url来对应admin, 然后我把这两个注释了一个,最后竟然就好了,反正我是不会相信是这个问题导致的

2. 最开始的时候是使用sqlite的,所以服务器上还有sqlite数据库文件,我把它删掉了
3. 当我在数据库手动清空django_migrations表的时候,使用的delete命令,删掉数据后,id还是从之前的顺序排,后来换成了直接用truncate命令清空的数据表

4. 我单独建了一个default数据库,为了把Django自带的哪些admin,session, auth 这些表单独放到一个数据库,我现在看来,问题出在这的可能性最大, 应该是还有个地方存migrations文件的,所以每次迁移,都没有删掉这个文件里的内容,导致出错



首先, 你得保证,不是migrations文件夹中的文件导致的冲突,如果当你迁移数据库的时候,报错说是xx表不存在,或者xx表已存在,首先应该想到的就是解决一下这个问题,当然最暴力的方法就是
1. 删除这个文件夹下的除了__init__.py文件之外的所有文件;

2. 从数据库中删掉django_migrations表中的数据;

当然如果你能找到冲突的那条记录,只删那一条记录是最好的

删除成功后,在python manage.py makemigrations 
  python manage.py migrate

然后记录一下Django使用mysql, 使用多数据库时应该注意的一些地方:
首先要在settings中配置好多个数据库,

DATABASES = {
    # 'default': {
    #     'ENGINE': 'django.db.backends.sqlite3',
    #     'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
    # },
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'dja......t',
        'USER': 'root',
        'PASSWORD': 't.....',
        'HOST': '34.....',
        'PORT': '3306',
        "OPTIONS": {
            "init_command": "SET sql_mode='STRICT_TRANS_TABLES'",

        }

    },
    'slots_config': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'slo....',
        'USER': 'root',
        'PASSWORD': 't......',
        'HOST': '34.2.......',
        'PORT': '3306',
        "OPTIONS": {
            "init_command": "SET sql_mode='STRICT_TRANS_TABLES'",
            # 'charset': 'utf8mb4'
        }

    },
    'permission': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'sl.....',
        'USER': 'root',
        'PASSWORD': 't....',
        'HOST': '34.237....',
        'PORT': '3306',
        "OPTIONS": {
            "init_command": "SET sql_mode='STRICT_TRANS_TABLES'",
            # 'charset': 'utf8mb4'
        }
    },
}

我的做法是一个app配一个数据库,所以要把app和数据库对应起来,写一个map映射关系

DATABASE_APPS_MAPPING = {
    # "default": "default",
    "permission_control": "permission",
    "slo...": "s.....",
    "auth": "default",
    "session": "default",
    "contenttypes": "default",
    "admin": "default",
}

同时,还需要一个路由,来指定model 和 数据库之间的路由关系,在settings同级目录下写一个文件,从settings中引用:

DATABASE_ROUTERS = ['s......nd.database_router.DatabaseAppsRouter']  # 数据库路由规则

文件内容:

from django.conf import settings

DATABASE_MAPPING = settings.DATABASE_APPS_MAPPING


class DatabaseAppsRouter(object):
    """
    A router to control all database operations on models for different
    databases.

    In case an app is not set in settings.DATABASE_APPS_MAPPING, the router
    will fallback to the `default` database.

    Settings example:

    DATABASE_APPS_MAPPING = {'app1': 'db1', 'app2': 'db2'}
    """

    def db_for_read(self, model, **hints):
        """"Point all read operations to the specific database."""
        if model._meta.app_label in DATABASE_MAPPING:
            return DATABASE_MAPPING[model._meta.app_label]
        return None

    def db_for_write(self, model, **hints):
        """Point all write operations to the specific database."""
        if model._meta.app_label in DATABASE_MAPPING:
            return DATABASE_MAPPING[model._meta.app_label]
        return None

    def allow_relation(self, obj1, obj2, **hints):
        """Allow any relation between apps that use the same database."""
        db_obj1 = DATABASE_MAPPING.get(obj1._meta.app_label)
        db_obj2 = DATABASE_MAPPING.get(obj2._meta.app_label)
        if db_obj1 and db_obj2:
            if db_obj1 == db_obj2:
                return True
            else:
                return False
        return None

    def allow_syncdb(self, db, model):
        """Make sure that apps only appear in the related database."""

        if db in DATABASE_MAPPING.values():
            return DATABASE_MAPPING.get(model._meta.app_label) == db
        elif model._meta.app_label in DATABASE_MAPPING:
            return False
        return None

    def allow_migrate(self, db, app_label, model=None, **hints):
        """
        Make sure the auth app only appears in the 'auth_db'
        database.
        """
        if db in DATABASE_MAPPING.values():
            return DATABASE_MAPPING.get(app_label) == db
        elif app_label in DATABASE_MAPPING:
            return False
        return None

ok, 这样就按app来区分了不同的数据库
这样在views中,操作数据的时候,也不用use 这种语法了,能自动根据app去连接对应的数据库
哦,对了,在model中,每个表都要明确指定app名称,类似这样

class App(models.Model):
    """
    app表
    """

    caption = models.CharField(verbose_name="应用名", null=True, blank=True, max_length=64)
    app_id = models.IntegerField(verbose_name="app_id")
    product = models.ForeignKey(Product, verbose_name="所属产品", related_name="apps")
    role = models.ManyToManyField(to="Role", verbose_name="拥有角色", blank=True)

    class Meta:
        app_label = "permission_control"    # 这里,指明app名称,用来对应app 和 数据库的map表
        db_table = "app"