Django Admin не сохраняет данные в базу данных

Я пытаюсь сохранить данные из моего администратора Django в свою базу данных, но почему-то этого не происходит. Я создал форму в одном из своих приложений, которую использует мой администратор. Я новичок в Django, и любая помощь будет очень признательна. Ниже приведен соответствующий код:

модели.py

from __future__ import unicode_literals
from django.contrib.auth.models import User
from django_countries.fields import CountryField
from django.db import models

class UserProfile(models.Model):
    user = models.OneToOneField(User)
    name = models.CharField(max_length=100)
    bio = models.TextField(max_length=1000, blank=True, null=True)
    image = models.FileField()
    country = CountryField()
    city = models.CharField(max_length=100)
    twitter = models.CharField(max_length=100, null=True, blank=True)
    linkedin = models.CharField(max_length=100, null=True, blank=True)
    location = models.TextField()

    def __unicode__(self):
        return self.name

    def __str__(self):
        return self.name


class Mentor(models.Model):
    mentor = models.CharField(max_length=100)
    mentee = models.CharField(max_length=100)

    def __unicode__(self):
        return self.mentee

    def __str__(self):
        return self.mentee

формы.py

from django import forms
from models import UserProfile, Mentor
from django_countries.fields import CountryField
from django.contrib.auth.models import User
from reports.models import Reports

class UserProfileForm(forms.ModelForm):

    name = forms.CharField(max_length=100)
    bio = forms.Textarea()
    image = forms.FileField(label='Profile Photo')
    country = CountryField(blank_label='(Select Country)')
    city = forms.CharField(max_length=100)
    twitter = forms.CharField(max_length=100, required=False)
    linkedin = forms.CharField(max_length=100, required=False)

    class Meta:
        model = UserProfile
        exclude = ('user',)

class MentorForm(forms.ModelForm):
    mentor_choices = tuple(UserProfile.objects.filter(user__is_staff=1).order_by('name').values_list('name', 'name'))
    mentee_choices = tuple(UserProfile.objects.exclude(user__is_staff=1).order_by('name').values_list('name', 'name'))
    mentor_name = forms.ChoiceField(choices=mentor_choices)
    mentee_name = forms.ChoiceField(choices=mentee_choices)

    def save(self, commit=True):
        mentor_name = self.cleaned_data.get('mentor_name', None)
        mentor_name = self.cleaned_data.get('mentee_name', None)
        return super(MentorForm, self).save(commit=commit)

    class Meta:
        model = Mentor
        fields= ('mentor_name', 'mentee_name')

admin.py

from django.contrib import admin
from .models import UserProfile, Mentor
from.forms import MentorForm

class UserProfileAdmin(admin.ModelAdmin):
    list_display = ('user',)
    search_fields = ['user']

    def save_model(self, request, obj, form, change):
        obj.created_by = request.user
        obj.save()
admin.site.register(UserProfile, UserProfileAdmin )

class MentorAdmin(admin.ModelAdmin):
    list_display = ('__unicode__','mentee')
    search_fields = ['mentee', 'mentor']
    form = MentorForm

    fieldsets = (
        (None,{
            'fields': ('mentor_name', 'mentee_name'),
        }),
    )

    def save_model(self, request, obj, form, change):
        super(MentorAdmin, self).save_model(request, obj, form, change)

admin.site.register(Mentor, MentorAdmin )

UserProfile работает отлично, но форма Mentor в админке ничего не сохраняет в базе данных. Он создает пустые записи в базе данных, поэтому я знаю, что интерфейс и сервер общаются, но данные не передаются. Любая помощь будет очень полезна


person karan nayak    schedule 19.07.2016    source источник
comment
Вы уверены, что MentorAdmin.save_model() вообще звонят? Попробуйте добавить сообщение журнала вверху метода.   -  person John Gordon    schedule 19.07.2016
comment
@JohnGordon да, метод вызывается. Я запустил его в режиме отладки, чтобы проверить, вызывался ли метод и вызывается ли метод. Он показывает все параметры, как и ожидалось, кроме «изменения», которое получает значение False.   -  person karan nayak    schedule 19.07.2016


Ответы (1)


def save(self, commit=True):
    # here you define `mentor_name`. OK.
    mentor_name = self.cleaned_data.get('mentor_name', None)
    # here you redefine `mentor_name`. I guess it is a typo and should be `mentee_name`.
    mentor_name = self.cleaned_data.get('mentee_name', None)
    # And... you don't do anything with these variables.
    return super(MentorForm, self).save(commit=commit)

Этот метод эквивалентен:

def save(self, commit=True):
    return super(MentorForm, self).save(commit=commit)

Что эквивалентно полному отказу от переопределения метода сохранения.

А что насчет этого?

def save_model(self, request, obj, form, change):
    super(MentorAdmin, self).save_model(request, obj, form, change)

Какова цель переопределения метода и вызова родительского метода только с теми же аргументами?

Но на самом деле проблема здесь:

mentor_choices = tuple(UserProfile.objects.filter(user__is_staff=1).order_by('name').values_list('name', 'name'))
mentee_choices = tuple(UserProfile.objects.exclude(user__is_staff=1).order_by('name').values_list('name', 'name'))
mentor_name = forms.ChoiceField(choices=mentor_choices)
mentee_name = forms.ChoiceField(choices=mentee_choices)

class Meta:
    model = Mentor
    fields = ('mentor_name', 'mentee_name')

Вы используете ModelForm, но ни одно из полей Mentor не находится в fields. Что вы ожидаете от этого, кроме сохранения строки с Mentor.mentor = None и Mentor.mentee = None. Вы даже не упомянули эти поля.

И почему вы используете CharField для Mentor.mentor и Mentor.mentee, хотя вам, вероятно, нужен внешний ключ.

class Mentor(models.Model):
    mentor = models.ForeignKey(UserProfile, models.PROTECT,
                               related_name='mentees')
    mentee = models.ForeignKey(UserProfile, models.PROTECT,
                               related_name='mentors')

class MentorForm(forms.ModelForm):

    class Meta:
        model = Mentor
        fields = ('mentor', 'mentee')

    mentor = forms.ModelChoiceField(queryset=UserProfile.objects.filter(
        user__is_staff=True).order_by('name'))
    mentee = forms.ModelChoiceField(queryset=UserProfile.objects.exclude(
        user__is_staff=True).order_by('name'))

Или еще лучше:

class Mentor(models.Model):
    mentor = models.ForeignKey(
        UserProfile, models.PROTECT, related_name='mentees',
        limit_choices_to={'user__is_staff': True},
    )
    mentee = models.ForeignKey(
        UserProfile, models.PROTECT, related_name='mentors',
        limit_choices_to={'user__is_staff': False},
    )

Что позволяет избежать создания формы.

person Antoine Pinsard    schedule 19.07.2016
comment
Спасибо тебе большое за это. Это мой первый проект Django, отсюда и глупые ошибки. Спасибо за помощь - person karan nayak; 20.07.2016