Mysql
 sql >> база данни >  >> RDS >> Mysql

Външен ключ на Django в друга схема

Бях се опитал да използвам две бази данни, за да симулирам вашия случай и да намеря решението по-долу:

1. Сценарий:

  1. база данни schema1 , който се управлява от django (четене и писане)
  2. база данни schema2 , което е НЕ управлявано от django

2. Стъпки:

  1. създайте миграции python manage.py makemigrations за вашите модели
  2. Генерирайте SQL за вашата миграция:python manage.py sqlmigrate app 0001 .(да предположим, че генерираното име на файл за миграция е 0001_initial.py от стъпка 1 )

Sql за тази миграция трябва да изглежда така:

CREATE TABLE `user_info` (`id_id` integer NOT NULL PRIMARY KEY, `name` varchar(20) NOT NULL);
ALTER TABLE `user_info` ADD CONSTRAINT `user_info_id_id_e8dc4652_fk_schema2.user_extra_info_id` FOREIGN KEY (`id_id`) REFERENCES `user_extra_info` (`id`);
COMMIT;

Ако стартирате директно горния sql, ще получите грешка като тази:

django.db.utils.OperationalError: (1824, "Failed to open the referenced table 'user_extra_info'")

Това е, защото django приема, че всичките ви стъпки за мигриране се изпълняват в една и съща база данни . Така че не може да открие user_extra_info в schema1 база данни.

3. Следните стъпки:

  1. Изрично посочете базата данни schema2 за таблица user_extra_info :

    ALTER TABLE `user_info` ADD CONSTRAINT `user_info_id_id_e8dc4652_fk_schema2.user_extra_info_id` FOREIGN KEY (`id_id`) REFERENCES schema2.user_extra_info (`id`);
    
  2. Изпълнете ръчно ревизирания sql в schema1 база данни.

  3. Кажете на django, че сам стартирах миграцията:python manage.py migrate --fake

  4. Готово!!

Изходен код За справка:

models.py

from django.db import models


class UserExtraInfo(models.Model):
    # table in schema2, not managed by django
    name = models.CharField('name', max_length=20)

    class Meta:
        managed = False
        db_table = 'user_extra_info'


class UserInfo(models.Model):
    # table in schema1, managed by django
    id = models.OneToOneField(
        UserExtraInfo,
        on_delete=models.CASCADE,
        primary_key=True
    )
    name = models.CharField('user name', max_length=20)

    class Meta:
        db_table = 'user_info'

settings.py

# Database
# https://docs.djangoproject.com/en/2.1/ref/settings/#databases

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'schema1',
        'USER': 'USER',
        'PASSWORD': 'PASSWORD',
        'HOST': 'localhost',
        'PORT': 3306,
    },
    'extra': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'schema2',
        'USER': 'USER',
        'PASSWORD': 'PASSWORD',
        'HOST': 'localhost',
        'PORT': 3306,
    }
}

DATABASE_ROUTERS = ['two_schemas.router.DBRouter']

router.py

class DBRouter(object):
    """
    A router to control all database operations on models in the
    auth application.
    """
    def db_for_read(self, model, **hints):
        """
        Attempts to read auth models go to auth_db.
        """
        if model._meta.db_table == 'user_extra_info':
            # specify the db for `user_extra_info` table
            return 'extra'
        if model._meta.app_label == 'app':
            return 'default'
        return None

    def db_for_write(self, model, **hints):
        """
        Attempts to write auth models go to auth_db.
        """
        if model._meta.db_table == 'user_extra_info':
            # specify the db for `user_extra_info` table
            return 'extra'
        if model._meta.app_label == 'app':
            return 'default'
        return None

    def allow_relation(self, obj1, obj2, **hints):
        """
        Relations between objects are allowed if both objects are
        in the primary/replica pool.
        """
        db_list = ('default', 'extra')
        if obj1._state.db in db_list and obj2._state.db in db_list:
            return True
        return None

    def allow_migrate(self, db, app_label, model_name=None, **hints):
        """
        Make sure the auth app only appears in the 'auth_db'
        database.
        """
        if app_label == 'app':
            return db == 'default'
        return None


  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Кой е най-добрият начин за внедряване на таблица на опашката за съобщения в mysql

  2. вмъкване на заявка с ajax без презареждане на цялата страница

  3. Някой намери ли ефикасен начин за съхраняване на стойности на BSON ObjectId в SQL база данни?

  4. Съхраняване на номера на кредитни карти в SESSION – начини за заобикаляне?

  5. Множество докер контейнери за mysql или един екземпляр с множество бази данни