При использовании django restframework Multiple database, serializer.is_valid() всегда идет в базу данных по умолчанию для проверки

Я использую django restframework и хочу работать с несколькими базами данных. Я использую функцию django с использованием (псевдоним) и switch_db(псевдоним) для ручного переключения между базами данных, когда я хочу получить, опубликовать или обновить данные.

Я сталкиваюсь с проблемой при публикации и обновлении data.ie всякий раз, когда будет вызван serializer.is_valid().

serializer.is_valid() пойдет и сначала проверит db_alias в файле model.py. Если я не указал db_alias в meta, он выберет базу данных по умолчанию для проверки. Если я укажу db_alias в модели, он выберет эту базу данных для проверки.

Но я не хочу указывать db_alias в модели, поскольку мой сценарий использования заключается в хранении данных в другой базе данных на основе некоторой логики в моем файле представления. Так что динамически я хочу выбрать базу данных из представления и сохранить данные в ней.

Почти я реализовал, но я столкнулся с проблемой, когда моя модель имеет поле справки. В этом случае serializer.is_valid собирается в базу данных по умолчанию для проверки этого ссылочного поля.

Необходимые детали: я использую mongoengine(0.9.0), документ, сериализатор документов.

Мои файлы как ниже:

model.py:

class ngroup(Document):

    groupname = StringField(max_length=100, required=True)
    description = StringField(max_length=100, required=False)
    parent = ReferenceField('ngroup',null=True)
    created_ts = DateTimeField(default=datetime.now())
    modified_ts = DateTimeField(default=datetime.now())
    is_deleted = BooleanField(default=False)

serializer.py:

from device_management.models import ngroup
from rest_framework_mongoengine.serializers import DocumentSerializer
from mongoengine import EmbeddedDocumentField, ReferenceField, StringField, ObjectIdField, IntField, BooleanField, FloatField, DateTimeField,ListField


class ngroupSerializer(DocumentSerializer):

    class Meta:
        model = ngroup

    def setOrgId(self, orgid):
        self.orgid = orgid

    def create(self, validated_data):
        ngroup_data = ngroup(**validated_data).switch_db(self.orgid)
        ngroup_data.save()
        return ngroup_data

    def update(self, instance, validated_data):
        ngroup_data = ngroup.objects.using(self.orgid).get(id = instance.id)
        ngroup_data = ngroup_data.switch_db(self.orgid)
        ngroup_data = ngroup_data.update(**validated_data)
        return validated_data

    def to_internal_value(self, data):
        print "data:" , data
        return super(DocumentSerializer, self).to_internal_value(data)  

view.py:

def create(self, request, format=None):
    orgid = str(request.user.orgid.id)
    data=request.data

    serializer = ngroupSerializer(data=data)
    if serializer.is_valid():
        try:
            serializer.save()
        except Exception as e:
            log.error("create" , extra={'extra':{'error': str(e),'message' :strings.DATA_VALIDATION_ERROR }})
            return response.errorResponse(message=strings.SERIALIZATION_ERROR_MSG,error=str(e),rstatus=status.HTTP_400_BAD_REQUEST)
        return response.successResponse(res_data=serializer.data, message=strings.POST_SUCCESS_MSG, rstatus=status.HTTP_201_CREATED)
    log.error("create" , extra={'extra':{'error': serializer.errors,'message' :strings.DATA_VALIDATION_ERROR }})
    return response.errorResponse(message=strings.DATA_VALIDATION_ERROR,error=serializer.errors,rstatus=status.HTTP_400_BAD_REQUEST)

settings.py:

DATABASES = {
     'default': {
        'ENGINE': 'django_mongodb_engine',
        'NAME': 'mydb',
        'USER': 'admin',
        'PASSWORD':'admin123',
        'HOST': '127.0.0.1',
        'PORT': 27017,
        'DBTYPE' : "mongo",
    },
    '586e47c784413825f2b5bc49': {
        'ENGINE': 'django_mongodb_engine',
        'NAME': 'mydb1',
        'USER': 'admin',
        'PASSWORD':'admin123',
        'HOST': '127.0.0.1',
        'PORT': 27017,
        'DBTYPE' : "mongo",
    },
    # Enter super_user organisation here. This DB will be same as default db only always
    '58996fb28441384430dc8ae6': {
        'ENGINE': 'django_mongodb_engine',
        'NAME': 'mydb',
        'USER': 'admin',
        'PASSWORD':'admin123',
        'HOST': '127.0.0.1',
        'PORT': 27017,
        'DBTYPE' : "mongo",
    },
}

pip freeze (Установочные версии):

Django==1.5.11
django-browserid==2.0.2
django-classy-tags==0.8.0
django-missing==0.1.18
django-mongo-auth==0.1.3
django-mongodb-engine==0.6.0
django-mongoengine==0.2.1
django-redis-sessions==0.5.6
django-rest-framework-mongoengine==3.3.0
django-sekizai==0.10.0
django-websocket-redis==0.4.7
djangorestframework==3.1.2
djangorestframework-jwt==1.9.0
djangotoolbox==1.8.0
gevent==1.1.2
greenlet==0.4.10
httplib2==0.9.2
mongoengine==0.9.0
oauthlib==2.0.1
pika==0.10.0
Pygments==2.1.3
PyJWT==1.4.2
pymongo==2.8
python-dateutil==2.6.0
python-openid==2.2.5
pytz==2016.10
redis==2.10.5
requests==2.12.3
requests-oauthlib==0.7.0
rest-condition==1.0.3
six==1.10.0
tweepy==3.5.0
twilio==5.7.0

У меня есть overide создать в сериализаторе, чтобы заботиться о базе данных при вызове serializer.save(), но как обрабатывать serializer.is_valid().

Мой проект застрял на этом этапе. Любая помощь будет оценена...

2 ответа

Это не точное решение вышеуказанной проблемы, но у нас есть 2 варианта.

1) Не переходите на serializer.is_valid() или serializer.save(). Непосредственно создайте ngroup:

def my_create(self, validated_data):
    gateway = Gateway(**validated_data).switch_db(self.orgid)
    gateway.save()
    return gateway

2) Другим решением является использование моделей и сериализаторов django-mogodb-engine и django вместо сериализаторов документов и документов.

Я попытался следовать этому с Django-mongodb-engine и работает хорошо:

-> JWT authentication
-> custom user
-> foreign key
-> embedded model
-> list of embedded model
-> dict field
-> **Routers for switching between databases.(Manual switching DB is not required)**

Я также могу использовать middleware_classes для указания времени выполнения в каждом запросе, какую базу данных использовать. Ссылка: Django аутентифицирует несколько баз данных

Хусайн,

К сожалению, вы смешиваете несовместимые проекты. Mongoengine, django-mongoengine а также Django-REST-Framework-Mongoengine проекты являются альтернативой django-mongodb-engineони не предназначены для совместного использования.

Насколько я знаю, django-mongodb-engine Проект был мертв в течение 2 лет или даже дольше, если честно. В то же время, Mongoengine Стек работает в производстве, хотя разработка не слишком активна. Я действительно хочу создать надлежащий сервер базы данных django из Mongoengine, чтобы сделать его первоклассным гражданином в мире Django, и похоже, что парни из Django тоже смотрят в этом направлении.

Возможно, вы также захотите заглянуть в этот пост.

Это моя вторая попытка. Я пытался переключить соединение с базой данных в поле зрения create(), Не работает для меня:

settings.py

# We define 2 Mongo databases - default (project) and project2
MONGODB_DATABASES = {
    "project": {
        "name": "project",
        "host": "localhost",
        "port": 27017,
        "tz_aware": True,  # if you use timezones in django (USE_TZ = True)
    },

    "project2": {
        "name": "project2",
        "host": "localhost",
        "port": 27017,
        "tz_aware": True,  # if you use timezones in django (USE_TZ = True)
    }
}


mongoengine.register_connection(alias='default', name=MONGODB_DATABASES["project"]["name"], host="local")
mongoengine.register_connection(alias='project2', name=MONGODB_DATABASES["project2"]["name"], host="local")

connection = mongoengine.connect(db="project", alias='default')

views.py

class AuthorViewSet(MongoModelViewSet):
    lookup_field = 'id'
    serializer_class = AuthorSerializer

    def create(self, request, format=None):
        global Author
        mongoengine.connection.disconnect(alias='project')
        mongoengine.connect('project2', alias='project2')
        return super(AuthorViewSet, self).create(request, format)

    def get_queryset(self):
        return Author.objects.all()
Другие вопросы по тегам