我正试图解决这个问题:
用户可以将技能添加到他的配置文件中,而其他用户可以在他的配置文件上添加技能。
我已经在系统中实现了添加技能。接下来,我尝试构建一个技能(管理员已经在系统中添加了这个技能),以添加到用户配置文件中。
但是在我的POST API中,我总是收到以下错误
{
"user": {
"user": [
"This field is required."
]
}
}正文输入:
{
"user":{
"username": "USERN",
"email": "diahu@gail.com",
"first_name": "",
"last_name": ""
},
"new_user_skill":
{
"id": 1,
"skill_name": "C"
}
}我的观点:
elif request.method == 'POST':
data = { 'user':request.data.get('user'),'skill_item':request.data.get('new_user_skill')}
serializer = UserSkillSerializer(data=data)
print("-------------------> serializer ")
print(serializer)
if serializer.is_valid():
serializer.save()
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) 模型:
class UserModelSerializer(serializers.ModelSerializer):
class Meta:
model = User
fields = ('username', 'first_name', 'last_name', 'email')
extra_kwargs = {
'username': {
'validators': [],
}
}
class UserProfileSerializer(serializers.ModelSerializer):
user=UserModelSerializer()
class Meta:
model = UserProfile
fields = '__all__'
def create(self, validated_data):
user_serializer = UserModelSerializer.create(UserModelSerializer(),validated_data = validated_data)
user,created=UserProfile.objects.update_or_create(user=user_serializer)
return user
class UserSkillSerializer(serializers.ModelSerializer):
user = UserProfileSerializer(required=True)
skill_item = SkillSerializer(required=True)
class Meta:
model = UserSkill
fields= '__all__'
def create (self,validated_data):
user_data = validated_data.pop('user')
user = UserProfileSerializer.create(UserProfileSerializer(),validated_data= user_data)
skill_data = validated_data.pop('skill_item')
skill_item = SkillSerializer.create(SkillSerializer(),validated_data=skill_data)
user_skill, created = UserSkill.objects.update_or_create(user=user,skill_item=skill_item)
return user_skill序列化器:
class SkillSerializer(serializers.ModelSerializer):
class Meta:
model = Skill
fields = '__all__'
extra_kwargs = {
'skill_name': {
'validators': [],
}
}
class UserModelSerializer(serializers.ModelSerializer):
class Meta:
model = User
fields = ('username', 'first_name', 'last_name', 'email')
extra_kwargs = {
'username': {
'validators': [],
}
}
class UserProfileSerializer(serializers.ModelSerializer):
user=UserModelSerializer()
class Meta:
model = UserProfile
fields = '__all__'
def create(self, validated_data):
user_serializer = UserModelSerializer.create(UserModelSerializer(),validated_data = validated_data)
user,created=UserProfile.objects.update_or_create(user=user_serializer)
return user
class UserSkillSerializer(serializers.ModelSerializer):
user = UserProfileSerializer(required=True)
skill_item = SkillSerializer(required=True)
class Meta:
model = UserSkill
fields= '__all__'
def create (self,validated_data):
user_data = validated_data.pop('user')
user = UserProfileSerializer.create(UserProfileSerializer(),validated_data= user_data)
skill_data = validated_data.pop('skill_item')
skill_item = SkillSerializer.create(SkillSerializer(),validated_data=skill_data)
user_skill, created = UserSkill.objects.update_or_create(user=user,skill_item=skill_item)
return user_skill我已经删除了验证器(不确定这是否正确)
编辑1
我按照答案编辑了我的大部分代码(我已经研究并实现了反向关系),即使我有null=True,也会面临一个非空约束。
我的更新代码:
序列化器:
class UserSerializer(serializers.ModelSerializer):
class Meta:
model = User
fields = ('username', 'first_name', 'last_name', 'email','user_profile')
class UserSkillSerializer(serializers.ModelSerializer):
user = serializers.PrimaryKeyRelatedField(read_only=True)
skill_item = serializers.PrimaryKeyRelatedField(read_only=True)
class Meta:
model = UserSkill
fields= ('user', 'skill_item')意见:
elif request.method == 'POST':
current_user = User.objects.get(pk=request.data.get('user')) # get the user key
user_profile_id = current_user.user_profile.pk # get the user profile from reverse relation in model
data = { 'user':user_profile_id,'skill_item':request.data.get('skill_id')}
serializer = UserSkillSerializer(data=data)
if serializer.is_valid():
serializer.save()
return Response(serializer.data, status=status.HTTP_201_CREATED)
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) 模型:
# This class will more or less map to a table in the database and defines the many to many relationship between user-skill, this is our intermediate model
class UserSkill(models.Model):
""" A Model for representing skill in user profile """
unique_together = (('user', 'skill_item'),)
user = models.ForeignKey('UserProfile',on_delete=models.CASCADE,related_name='current_user_skills')
skill_item = models.ForeignKey(Skill,on_delete=models.CASCADE,null=True)
def __str__(self):
"""Return a human readable representation of the model instance."""
return "{}".format(self.skill_item.skill_name)
# this class adds a Many to Many field in existing django-rest auth UserProfile class for user and his/her skills
class UserProfile(models.Model):
user = models.OneToOneField('auth.User',unique=True,on_delete=models.CASCADE,related_name='user_profile')
user_skills = models.ManyToManyField(
Skill,
through='UserSkill',
through_fields=('user','skill_item'),null=True
)
#create user profile signal handler
def create_user_profile(sender, instance, created, **kwargs):
if created:
UserProfile.objects.create(user=instance)
post_save.connect(create_user_profile, sender=User)错误:
返回func(*args,**kwargs)文件"C:\code\django\wantedly\src\wantedly_webapp\views\AllViews.py",行 61,在user_skill_collection serializer.save()中 /api/v1/user/ IntegrityError / NULL约束失败: wantedly_webapp_userskill.skill_item_id
发布于 2018-01-21 10:36:20
生成此错误是因为UserProfileSerializer有一个需要完整用户定义的user字段。通常,我相信您在序列化程序中混合了“用户配置文件”/“用户”的概念,这会造成混淆。
一旦创建了User及其关联的UserProfile和Skill,将Skill添加到UserProfile的一种方法是通过主键选择它们:
class UserSkillSerializer(serializers.ModelSerializer):
user = serializers.PrimaryKeyRelatedField()
skill_item = serializers.PrimaryKeyRelatedField()
class Meta:
model = UserSkill
fields= ('user', 'skill_item')然后,将Skill分配给UserProfile的POST数据如下:
{
"user": 3, # PK of the UserProfile, not the User!
"skill_item": 42 # PK of the SkillItem you want to assign
}https://stackoverflow.com/questions/48365600
复制相似问题