Django实战(19):自定义many-to-many关系,实现Atom订阅

记得有人跟我说过,rails的has_many :through是一个”亮点“,在Django看来,该功能简直不值一提。rails中的many-to-many关联中,还需要你手工创建关联表(写migration的方式),而has_many :through的”语法“只不过是为了自定义关联关系:通过一个中间的、到两端都是many-to-one的模型类实现多对多关联。
在Django中,many-to-many的中间关系表是自动创建的,如果你要指定一个自己的Model类作为关系对象,只需要在需要获取对端的Model类中增加一个ManyToManyField属性,并指定though参数。比如现在我们已经有了这样两个many-to-one关系,LineItem --->Product,  LineItem-->Order,  如果需要从Product直接获取关联的Order,只需要增加一行,最终的Product如下:

class Product(models.Model):

    title = models.CharField(max_length=100,unique=True)

    description    = models.TextField()

    image_url = models.URLField(max_length=200)

    price = models.DecimalField(max_digits=8,decimal_places=2)

    date_available = models.DateField()

    orders = models.ManyToManyField(Order,through='LineItem')

 

之后就可以通过product对象直接找到包含该产品的订单:

$ python manage.py shell

>>> from depot.depotapp.models import Product

>>> p = Product(id=1)

>>> p.orders



>>> p.orders.all()

[, ]

 

实现这个关系的目的是我们要针对每个产品生成一个”订阅“,用于查看谁买了该产品。我们采用Atom作为格式的标准。生成的Atom发布格式如下:

<?xml version="1.0" encoding="UTF-8"?>

<feed xml:lang="en-US" xmlns="http://www.w3.org/2005/Atom">

    <id>tag:localhost,2005:/products/3/who_bought</id>

    <link type="text/html" href="http://localhost:3000/depotapp" rel="alternate"/>

    <link type="application/atom+xml" href="http://localhost:8000/depotapp/product/3/who_bought" rel="self"/>

        <title>谁购买了《黄瓜的黄 西瓜的西》</title>

    <updated>2012-01-02 12:02:02</updated> 

    <entry>

        <id>tag:localhost,2005:Order/1</id>

        <published>2012-01-02 12:02:02</published>

        <updated>2012-01-02 12:02:02</updated>

        <link rel="alternate" type="text/html" href="http://localhost:8000/orders/1"/>

        <title>订单1</title>

        <summary type="xhtml">

            <div xmlns="http://www.w3.org/1999/xhtml">

                <p>我住在北京</p>                

            </div>

        </summary>

        <author>

            <name>我是买家</name>

            <email>[email protected]</email>

        </author>

    </entry>

    <entry>

        <id>tag:localhost,2005:Order/3</id>

        <published>2012-01-02 12:02:02</published>

        <updated>2012-01-02 12:02:02</updated>

        <link rel="alternate" type="text/html" href="http://localhost:8000/orders/3"/>

        <title>订单3</title>

        <summary type="xhtml">

            <div xmlns="http://www.w3.org/1999/xhtml">

                <p>我住在哪里?</p>                

            </div>

        </summary>

        <author>

            <name>我是买家2</name>

            <email>[email protected]</email>

        </author>

    </entry>

</feed>

 

你可能想到,Atom是以xml为格式的,我们可以借助Django REST framework来实现,但是这不是一个好主意,因为REST framework生成的xml有其自身的格式,与Atom的格式完全不同。如果使用REST framework就需要对其进行定制,甚至要实现一个自己的renderer(比如,AtomRenderer),而这需要深入了解该框架的大量细节。为了简单起见,我们考虑用Django的模板来实现。

首先我们设计url为:http://localhost:8000/depotapp/product/[id]/who_bought,先在depot/depotapp/urls.py中增加urlpatterns:

(r'product/(?P[^/]+)/who_bought$', atom_of_order)

然后在depot/depotapp/views.py中实现视图函数:

def atom_of_order(request,id):

    product = Product.objects.get(id = id)

    t = get_template('depotapp/atom_order.xml')

    c=RequestContext(request,locals())    

    return HttpResponse(t.render(c), mimetype='application/atom+xml')

注意其中我们指定了mimetype,使浏览器知道返回的是xml而不是html。最后的工作就是编写模板了。depot/templates/depotapp/atom_order.xml如下:

<?xml version="1.0" encoding="UTF-8"?>

<feed xml:lang="en-US" xmlns="http://www.w3.org/2005/Atom">

    <id>tag:localhost,2005:/product/{{product.id}/who_bought</id>

    <link type="text/html" href="{% url depotapp.views.store_view %}" rel="alternate"/>

    <link type="application/atom+xml" href="{% url depotapp.views.atom_of_order product.id %}" rel="self"/>

        <title>谁购买了《{{product.title}}》</title>

    <updated>2012-01-02 12:02:02</updated> 

{% for order in product.orders.all %}

    <entry>

        <id>tag:localhost,2005:order/{{order.id}}</id>

        <published>2012-01-02 12:02:02</published>

        <updated>2012-01-02 12:02:02</updated>

        <link rel="alternate" type="text/html" href="{% url depotapp.views.atom_of_order order.id %}"/>

        <title>订单{{order.id}}</title>

        <summary type="xhtml">

            <div xmlns="http://www.w3.org/1999/xhtml">

                <p>{{order.address}}</p>                

            </div>

        </summary>

        <author>

            <name>{{order.name}}</name>

            <email>{{order.email}}</email>

        </author>

    </entry>

{% endfor %}

</feed>

 

用模板实现Atom发布,确实很简单,不是吗?

 


你可能感兴趣的:(django)