8

我正在为 REST API 计划一个包含 Django 和 Tastypie 的站点,并且我很难找出在返回的资源中包含子资源的“正确”方式。

作为一个沙盒,我做了一个带有 Ticket 模型和 TicketComment 模型的小应用程序,其中评论属于一张票。我查看了关于嵌套资源的 Tastypie Cookbook 食谱(http://django-tastypie.readthedocs.org/en/latest/cookbook.html#nested-resources),但我很难理解为什么我应该这样做. 下面的代码使用 django.forms.models.model_to_dict() 将评论放入票证中,但我认为这里一定有一个“陷阱”。

我有理由不做我现在正在做的事吗?此外,有没有比食谱中列出的更干净的感觉模式?

型号如下:

# tickets/models.py

from django.db import models

class Ticket(models.Model):
    title = models.CharField(max_length=200)
    create_ts = models.DateTimeField(auto_now_add=True)
    submitter_email = models.EmailField()
    PRIORITY_CHOICES = (
        ('H', 'High'),
        ('M', 'Medium'),
        ('L', 'Low'),)
    priority = models.CharField(max_length=1, choices=PRIORITY_CHOICES)
    description = models.TextField()
    STATUS_CHOICES = (
        ('NEW', 'New & Unclaimed'),
        ('WIP', 'Work In Progress'),
        ('RES', 'Resolved'),
        ('CLS', 'Closed'),)
    status = models.CharField(max_length=3, choices=STATUS_CHOICES)

    def __unicode__(self):
        return "<Ticket:%d:%s>" % (self.id, self.title,)

class TicketComment(models.Model):
    ticket = models.ForeignKey(Ticket)
    comment_ts = models.DateTimeField(auto_now_add=True)
    commenter_email = models.EmailField()
    comment = models.TextField()

    def __unicode__(self):
        return "<TicketComment:%d:%d>" % (self.ticket.id, self.id,)

资源如下:

# tickets/api.py

from tastypie import fields
from tastypie.resources import ModelResource
from tickets.models import Ticket, TicketComment
from django.forms.models import model_to_dict

class TicketResource(ModelResource):

    class Meta:
        queryset = Ticket.objects.all()
        resource_name = 'ticket'

    def dehydrate(self, bundle):
        comments = TicketComment.objects.filter(ticket=bundle.data['id'])
        bundle.data['comments'] = [model_to_dict(c) for c in comments]
        return bundle

class TicketCommentResource(ModelResource):
    ticket = fields.ForeignKey(TicketResource, 'ticket')

    class Meta:
        queryset = TicketComment.objects.all()
        resource_name = 'comment'

输出如下:

{
   comments: [
        {
            comment: "This is the first comment.",
            commenter_email: "me@example.com",
            id: 1,
            ticket: 1
        },
        {
            comment: "This is the second comment.",
            commenter_email: "me@example.com",
            id: 2,
            ticket: 1
        }
    ],
    create_ts: "2011-10-17T15:55:11.372000",
    description: "This is the first ticket.",
    id: "1",
    priority: "M",
    resource_uri: "/api/v1/ticket/1/",
    status: "NEW",
    submitter_email: "me@example.com",
    title: "First Ticket"
}
4

2 回答 2

14

您正在寻找相关领域:http ://django-tastypie.readthedocs.org/en/latest/fields.html#relationship-fields

于 2011-10-19T02:19:16.597 回答
1

你能发布你的解决方案吗?

我有相同的用例/模型(在指向父表的“子表”中建立的外键关系)但无法解决它。

我看到的解决方案是将此行添加到您的 TicketResource 一个类级别而不是元子类中:

comments = fields.ToManyField('TicketCommentResource', 'ticket', full=True)

然后可能也将其添加到 TicketCommentResource,再次在类级别:

event = fields.ForeignKey('TicketResource', 'event')

但是我总是收到错误消息,即我的父对象资源(在您的示例 TicketResource 中)没​​有属性“ticket”,这是发送到 ToManyField 的第二个术语。

我已经尝试过很多次,但似乎无法获得获胜的组合。您最初发布的解决方案有效,但正如您自己指出的那样,它并不理想。

谢谢!

于 2013-01-17T20:53:13.220 回答