Как зарегистрировать другого пользователя с профилем в django? - PullRequest
0 голосов
/ 25 сентября 2018

Я новичок в django и начал новый проект.Вскоре я хочу зарегистрировать различные типы пользователей (Teacher, Student, Stuff) с их фотографией в профиле.Я сделал одно-одно поле от пользователя к профилю, и ниже мой код, и я получаю следующую ошибку.если у вас есть какие-либо предложения для моей цели регистрации, я хочу зарегистрировать другого типа пользователя с изображением профиля.

models.py

from django.db import models
from django.contrib.auth.models import User,AbstractUser
from django.conf import settings
from django.dispatch import receiver
from django.db.models.signals import post_save

class User(AbstractUser):
    USER_TYPE_CHOICES = (
        (1,'Student'),
        (2,'Teacher'),
        (3,'Stuff')
    )
    profile = models.OneToOneField(on_delete=models.CASCADE, related_name='user')
    user_type = models.PositiveSmallIntegerField(choices=USER_TYPE_CHOICES)

class Profile(models.Model):
    user = models.OneToOneField(settings.AUTH_USER_MODEL,on_delete=models.CASCADE, related_name='profile')
    photo = models.ImageField(upload_to='users/%Y/%m/%d/')

    def __str__(self):
        return 'Profile {}'.format(self.user.username)

это форма

from django import forms
from django.contrib.auth.models import User
from .models import Profile

class UserCreationForm(forms.ModelForm):
    password = forms.CharField(widget=forms.PasswordInput, label='Password')
    password2 = forms.CharField(widget=forms.PasswordInput, label='Repeat Password')

    class Meta:
        model = User
        fields = ('username', 'first_name', 'last_name')

    def clean_password(self):
        cd = self.cleaned_data
        if cd['password']!=cd['password2']:
            raise forms.ValidationError('password no match')
        return cd['password2']


class ProfileCreationForm(forms.ModelForm):
    class Meta:
        model = Profile
        fields = ('photo',)

это мой модуль просмотра

from django.shortcuts import render
from .forms import ProfileCreationForm, UserCreationForm
from .models import Profile
from django.http import HttpResponse
def sign_up(request):
    if request.method == 'POST':
        user_form = UserCreationForm(instance=request.user,data=request.POST)
        profile_form = ProfileCreationForm(instance=request.user.profile,data=request.POST,
                                           files=request.FILES)
        if user_form.is_valid() and profile_form.is_valid():
            new_user = user_form.save(commit=False)
            new_user.set_password(user_form.cleaned_data['password'])
            new_user.save()
            Profile.objects.create(user=new_user)
            return HttpResponse('user created')

    else:
        user_form = UserCreationForm()
        profile_form = ProfileCreationForm()
    return render(request, '',{'user_form':user_form, 'profile_form':profile_form})

вот ошибка:

Traceback (most recent call last):
  File "manage.py", line 15, in <module>
    execute_from_command_line(sys.argv)
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\site-packages\django\core\management\__init__.py", line 381, in execute_from_command_line
    utility.execute()
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\site-packages\django\core\management\__init__.py", line 357, in execute
    django.setup()
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\site-packages\django\__init__.py", line 24, in setup
    apps.populate(settings.INSTALLED_APPS)
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\site-packages\django\apps\registry.py", line 112, in populate
    app_config.import_models()
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\site-packages\django\apps\config.py", line 198, in import_models
    self.models_module = import_module(models_module_name)
  File "C:\Users\Mahdi\AppData\Local\Programs\Python\Python37\lib\importlib\__init__.py", line 127, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
  File "<frozen importlib._bootstrap>", line 983, in _find_and_load
  File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
  File "<frozen importlib._bootstrap_external>", line 728, in exec_module
  File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
  File "C:\Users\Mahdi\PycharmProjects\MyProject\accounts\models.py", line 7, in <module>
    class User(AbstractUser):
  File "C:\Users\Mahdi\PycharmProjects\MyProject\accounts\models.py", line 13, in User
    profile = models.OneToOneField(on_delete=models.CASCADE, related_name='user')
TypeError: __init__() missing 1 required positional argument: 'to'

Ответы [ 2 ]

0 голосов
/ 27 сентября 2018

вот пример, вы должны только создать модель с OneToOne для пользователя

class Profile(models.Model):
    usuario = models.OneToOneField(User, on_delete=models.CASCADE, 
    related_name='profile')

    foto = models.ImageField(
        upload_to='usuarios/fotos/',
        null=True,
        blank=True,
    )
0 голосов
/ 25 сентября 2018

В вашей модели вы забыли указать, на какую модель должен указывать OneToOneField модели User (Profile), поскольку этот профиль определен позже в файле Python, вы не можете использовать идентификатор,однако вы можете использовать строковый литерал, который содержит имя модели:

class User(AbstractUser):
    USER_TYPE_CHOICES = (
        (1,'Student'),
        (2,'Teacher'),
        (3,'Stuff')
    )
    profile = models.OneToOneField(<b>'Profile'</b>, on_delete=models.CASCADE, related_name='user')
    user_type = models.PositiveSmallIntegerField(choices=USER_TYPE_CHOICES)

. Вам нужно будет выполнить миграцию и выполнить миграцию, чтобы добавить таблицу, соответствующую вашей модели.в базе данных.

Поскольку вы ссылаетесь на Profile, это означает, что вам необходимо сначала создать Profile перед сохранением User (поскольку в противном случае profile будет NULL, не здесь разрешено):

from django.shortcuts import render
from .forms import ProfileCreationForm, UserCreationForm
from .models import Profile
from django.http import HttpResponse

def sign_up(request):
    if request.method == 'POST':
        user_form = UserCreationForm(instance=request.user,data=request.POST)
        profile_form = ProfileCreationForm(instance=request.user.profile,data=request.POST,
                                           files=request.FILES)
        if user_form.is_valid() and profile_form.is_valid():
            profile = profile_form.save()
            new_user = user_form.save(commit=False)
            new_user.set_password(user_form.cleaned_data['password'])
            <b>new_user.profile = profile</b>
            new_user.save()
            return HttpResponse('user created')

    else:
        user_form = UserCreationForm()
        profile_form = ProfileCreationForm()
    return render(request, <b>'my_template.html'</b>,{'user_form':user_form, 'profile_form':profile_form})

Вы также забыли указать шаблон.Кроме того, рекомендуется возвращать redirect(..) в представление, так как в противном случае, если пользователь обновит браузер, будет сделан второй POST-запрос, что может вызвать создание другого пользователь.

...