Проблема с обновлением профиля django у объекта 'User' нет атрибута 'profile'
я новичок в django и в настоящее время учусь, и я столкнулся с проблемой, которую, похоже, просто не могу решить.
from django.shortcuts import render, redirect
from django.contrib import messages
from .forms import UserRegisterForm, UserUpdateForm, ProfileUpdateForm
from django.contrib.auth.decorators import login_required
def register(request):
if request.method == 'POST':
form = UserRegisterForm(request.POST)
if form.is_valid():
form.save()
username = form.cleaned_data.get('username')
messages.success(request, f'Account Successfully created for {username}!')
return redirect('login')
else:
form = UserRegisterForm()
return render(request, 'users/register.html', {'form': form})
@login_required()
def profile(request):
if request.method == 'POST':
u_form = UserUpdateForm(request.POST, instance=request.user)
p_form = ProfileUpdateForm(request.POST, request.FILES, instance=request.user.profile)
if u_form.is_valid() and p_form.is_valid():
u_form.save()
p_form.save()
messages.success(request, f'Your account has been updated!')
return redirect('profile')
else:
u_form = UserUpdateForm(instance=request.user)
p_form = ProfileUpdateForm()
context = {
'u_form': u_form,
'p_form': p_form
}
return render(request, 'users/profile.html',context)
в этом коде, в
else:
u_form = UserUpdateForm(instance=request.user)
p_form = ProfileUpdateForm()
часть, всякий раз, когда я добавляю экземпляр в p_form как p_form = ProfileUpdateForm( instance=request.user.profile)
это выдает такую ошибку в браузере:
AttributeError at /profile/
'User' object has no attribute 'profile'
Request Method: GET
Request URL: http://127.0.0.1:8000/profile/
Django Version: 5.2.1
Exception Type: AttributeError
Exception Value:
'User' object has no attribute 'profile'
Exception Location: D:\Documents\pycharm\venv\Lib\site-packages\django\utils\functional.py, line 253, in inner
Raised during: users.views.profile
и да, именно из-за изменения в p_form возникает эта ошибка при входе в /profile. ПОЖАЛУЙСТА, ПОМОГИТЕ
вот мой models.py:
from django.contrib.auth.models import User
from django.db import models
class UserProfile(models.Model):
user = models.OneToOneField(User, on_delete=models.CASCADE)
pf_image = models.ImageField(default='default.jpg',upload_to='profile_pics/', null=True, blank=True)
def __str__(self):
return f'{self.user.username}Profile'
а вот и мой signals.py:
from django.db.models.signals import post_save
from django.contrib.auth.models import User
from django.dispatch import receiver
from .models import UserProfile
@receiver(post_save, sender=User)
def create_profile(sender, instance, created, **kwargs):
if created:
UserProfile.objects.create(user=instance)
@receiver(post_save, sender=User)
def save_profile(sender, instance, **kwargs):
instance.UserProfile.save()
Вы назвали модель UserProfile, так что это .userprofile, вместо :.profile
@login_required
def profile(request):
if request.method == 'POST':
u_form = UserUpdateForm(
request.POST, request.FILES, instance=request.user, prefix='user'
)
p_form = ProfileUpdateForm(
request.POST,
request.FILES,
instance=request.user.userprofile,
prefix='profile',
)
if u_form.is_valid() and p_form.is_valid():
u_form.save()
p_form.save()
messages.success(request, f'Your account has been updated!')
return redirect('profile')
else:
u_form = UserUpdateForm(instance=request.user, prefix='user')
p_form = ProfileUpdateForm(
instance=request.user.userprofile, prefix='profile'
)
context = {'u_form': u_form, 'p_form': p_form}
return render(request, 'users/profile.html', context)
Я бы также посоветовал использовать prefix=… [Django-doc] для форм, чтобы никакие два поля формы не могли в конечном итоге столкнуться.
Примечание: Сигналы часто не являются надежным механизмом. Я написал статью [Django-антипаттерны], в которой обсуждаются некоторые проблемы при использовании сигналов. Поэтому их следует использовать только в крайнем случае.