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>wanghaikuo@gmail.com</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>2222b@baidu.com</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发布,确实很简单,不是吗?

 


posted @ 2012-02-09 15:30  心内求法  阅读(3228)  评论(0编辑  收藏  举报