使用 Django Rest Framework 访问 'ManyRelatedManager' 上的不同字段?
Access different fields on 'ManyRelatedManager' using Django Rest Framework?
我正在尝试访问我的 ManyToMany link 的 through table 上的字段,以通过 Django Rest Frameworks 序列化为 JSON。
我的多对多涉及的模型有:
class Mage(models.Model):
arcana = models.ManyToManyField('ArcanumAbility', through='CharacterArcanumLink', related_name='mage_by_arcana')
class ArcanumAbility(models.Model):
class Arcana(AutoNumber):
FATE = ()
MIND = ()
SPIRIT = ()
DEATH = ()
FORCES = ()
TIME = ()
SPACE = ()
LIFE = ()
MATTER = ()
PRIME = ()
arcanum = EnumField(Arcana)
class Meta:
verbose_name_plural = "Arcana Abilities"
def __str__(self):
return self.arcanum.label
class CharacterArcanumLink(Trait):
PRIORITY_CHOICES = (
(0, 'Unassigned'), (1, 'Ruling'), (2, 'Common'), (3, 'Inferior')
)
priority = models.PositiveSmallIntegerField(
choices=PRIORITY_CHOICES, default=0)
mage = models.ForeignKey('Mage')
arcana = models.ForeignKey('ArcanumAbility')
class Meta:
unique_together = ('mage', 'arcana')
def __str__(self):
return self.arcana.arcanum.label
其中 Trait
mixin 提供了 current_value
为了将上述关系序列化为我的 JSON,我在我的序列化器上尝试了这两种模式:
class CharacterArcanumLinkSerializer(serializers.ModelSerializer):
class Meta:
model = CharacterArcanumLink
fields = ('current_value', 'arcana')
class MageSerializer(serializers.ModelSerializer):
arcana = CharacterArcanumLinkSerializer()
....
class Meta:
model = Mage
fields = (...., 'arcana', ....)
depth = 1
但这给了我这个错误:
AttributeError at /mages
'ManyRelatedManager' object has no attribute 'arcana'
来自(最终):
C:\Python34\lib\site-packages\rest_framework\fields.py in get_attribute
if instance is None:
# Break out early if we get `None` at any point in a nested lookup.
return None
try:
if isinstance(instance, collections.Mapping):
instance = instance[attr]
else:
instance = getattr(instance, attr) ...
except ObjectDoesNotExist:
return None
if is_simple_callable(instance):
instance = instance()
return instance
▼ Local vars
Variable: Value
instance: <django.db.models.fields.related.create_many_related_manager.<locals>.ManyRelatedManager object at 0x0000000004E4D4A8>
attr: 'arcana'
attrs: ['arcana']
(问题:我需要什么技巧才能从 ManyRelatedManager 转到它的字段?)
而且我也尝试过不指定特殊的序列化程序,只是在我的字段中包含 'arcana'
,然后从我的模型中提取它。这导致了这个错误:
TypeError at /mages
<Arcana.FATE: 1> is not JSON serializable
其中 1
来自 PK 的 ArcanumAbility,而不是通过 table 的值。这里的问题是 Mage class 有一个指向 'ArcanumAbility'
模型的 M2M 字段,所以 DRF 试图做的就是序列化它上面的 Enum。
那么,如果我想要一个包含从 Mage 到 ArcanumAbility 的所有关系的 JSON 字典以及从 table 到 table 的数据,我应该使用什么方法?
回复 ,我希望它看起来像这样:
....
"arcanum": {
"Fate": 2,
"Spirit": 0,
"Mind": 3,
....
}
希望这是一个足够清晰的示例。
如前所述,如果您在 CharacterArcanumLink class 中的法师字段中添加 related_name="linked_arcana",您应该能够执行如下操作:
class MageSerializer(serializers.ModelSerializer):
arcana = serializers.SerializerMethodField()
def get_arcana(self, obj):
if obj:
return {str(x): x.current_value for x in obj.linked_arcana.all()}
我是这样工作的:
arcana = serializers.SerializerMethodField()
def get_arcana(self, obj):
if obj:
return {str(link): link.current_value
for link in CharacterArcanumLink.objects.filter(mage=obj)}
深受 回答的启发。
我正在尝试访问我的 ManyToMany link 的 through table 上的字段,以通过 Django Rest Frameworks 序列化为 JSON。
我的多对多涉及的模型有:
class Mage(models.Model):
arcana = models.ManyToManyField('ArcanumAbility', through='CharacterArcanumLink', related_name='mage_by_arcana')
class ArcanumAbility(models.Model):
class Arcana(AutoNumber):
FATE = ()
MIND = ()
SPIRIT = ()
DEATH = ()
FORCES = ()
TIME = ()
SPACE = ()
LIFE = ()
MATTER = ()
PRIME = ()
arcanum = EnumField(Arcana)
class Meta:
verbose_name_plural = "Arcana Abilities"
def __str__(self):
return self.arcanum.label
class CharacterArcanumLink(Trait):
PRIORITY_CHOICES = (
(0, 'Unassigned'), (1, 'Ruling'), (2, 'Common'), (3, 'Inferior')
)
priority = models.PositiveSmallIntegerField(
choices=PRIORITY_CHOICES, default=0)
mage = models.ForeignKey('Mage')
arcana = models.ForeignKey('ArcanumAbility')
class Meta:
unique_together = ('mage', 'arcana')
def __str__(self):
return self.arcana.arcanum.label
其中 Trait
mixin 提供了 current_value
为了将上述关系序列化为我的 JSON,我在我的序列化器上尝试了这两种模式:
class CharacterArcanumLinkSerializer(serializers.ModelSerializer):
class Meta:
model = CharacterArcanumLink
fields = ('current_value', 'arcana')
class MageSerializer(serializers.ModelSerializer):
arcana = CharacterArcanumLinkSerializer()
....
class Meta:
model = Mage
fields = (...., 'arcana', ....)
depth = 1
但这给了我这个错误:
AttributeError at /mages
'ManyRelatedManager' object has no attribute 'arcana'
来自(最终):
C:\Python34\lib\site-packages\rest_framework\fields.py in get_attribute
if instance is None:
# Break out early if we get `None` at any point in a nested lookup.
return None
try:
if isinstance(instance, collections.Mapping):
instance = instance[attr]
else:
instance = getattr(instance, attr) ...
except ObjectDoesNotExist:
return None
if is_simple_callable(instance):
instance = instance()
return instance
▼ Local vars
Variable: Value
instance: <django.db.models.fields.related.create_many_related_manager.<locals>.ManyRelatedManager object at 0x0000000004E4D4A8>
attr: 'arcana'
attrs: ['arcana']
(问题:我需要什么技巧才能从 ManyRelatedManager 转到它的字段?)
而且我也尝试过不指定特殊的序列化程序,只是在我的字段中包含 'arcana'
,然后从我的模型中提取它。这导致了这个错误:
TypeError at /mages
<Arcana.FATE: 1> is not JSON serializable
其中 1
来自 PK 的 ArcanumAbility,而不是通过 table 的值。这里的问题是 Mage class 有一个指向 'ArcanumAbility'
模型的 M2M 字段,所以 DRF 试图做的就是序列化它上面的 Enum。
那么,如果我想要一个包含从 Mage 到 ArcanumAbility 的所有关系的 JSON 字典以及从 table 到 table 的数据,我应该使用什么方法?
回复
....
"arcanum": {
"Fate": 2,
"Spirit": 0,
"Mind": 3,
....
}
希望这是一个足够清晰的示例。
如前所述,如果您在 CharacterArcanumLink class 中的法师字段中添加 related_name="linked_arcana",您应该能够执行如下操作:
class MageSerializer(serializers.ModelSerializer):
arcana = serializers.SerializerMethodField()
def get_arcana(self, obj):
if obj:
return {str(x): x.current_value for x in obj.linked_arcana.all()}
我是这样工作的:
arcana = serializers.SerializerMethodField()
def get_arcana(self, obj):
if obj:
return {str(link): link.current_value
for link in CharacterArcanumLink.objects.filter(mage=obj)}
深受