Routes 起步

  1. from routes import Mapper  
  2. map = Mapper()  
  3. map.connect('spch''/blog', controller='main', action='index')  
  4.   
  5. result = map.match('/blog')  
  6. print result  
  7. {'action': u'index''controller': u'main'}  

1.2 行创建一个mapper
3.   行注册一条路由, 路由名称为'spch', 路径为'/blog', controller为main,
      action为index
      可以这样认为,匹配到此条路由的请求交由controller处理,请求预调用的
      函数为index

5. 创建好路由条目后,即可以进行匹配,调用match方法,匹配路径'blog'
6. 输出匹配结果

[python]  view plain copy print ?
  1. map.connect(None"/error/{action}/{id}", controller="error")  
  2. result = map.match('/error/index/2')  
  3. print result  
  4. {'action': u'index''controller': u'error''id': u'2'}  

1.注册了一条无名路由,并且action从匹配路由中获得
  同样,我们可以省掉None
  map.connect("/error/{action}/{id}", controller="error")
  上述语句同样注册了一条无名路由。


Conditions

Conditions用于限制进行路由匹配,比如method

[python]  view plain copy print ?
  1. m.connect("/user/list", controller="user", action="list", conditions=dict(method=["GET""HEAD"]))  
只匹配GET,HEAD请求。


Requirements

有时只想匹配数字,或者匹配可选的几个条目

[python]  view plain copy print ?
  1. map.connect(R"/blog/{id:\d+}")  
  2. map.connect(R"/download/{platform:windows|mac}/{filename}")  

\d表示匹配1位数字,\d+表示匹配多位
windows|mac 表示只匹配windows或者mac
可以将上述写成
[python]  view plain copy print ?
  1. map.connect("/blog/{id}", requirements={"id": R"\d+"}  
  2. map.connect("/download/{platform}/{filename}",  
  3.     requirements={"platform": R"windows|mac"})  

Format extensions

通过{.format}来指定匹配格式

[python]  view plain copy print ?
  1. map.connect('/entries/{id}{.format}')  
  2. print map.match('/entries/2')  
  3. {'id': u'2''format'None}  
  4. print map.match('/entries/2.mp3')  
  5. {'id': u'2''format': u'mp3'}  

[python]  view plain copy print ?
  1. map.connect('/entries/{id:\d+}{.format:mp3}')  
  2. print map.match('/entries/2.mp3')  
  3. {'id': u'2''format': u'mp3'}  
  4. print map.match('/entries/2')  
  5. {'id': u'2''format'None}  
  6. print map.match('/entries/2.mp4')  
  7. None  

注意:{id:\d+}, 如果没有\d+, print map.match('/entries/2.mp4')将输出 {'id': u'2.mp4', 'format': None}是可以成功的。
有了\d+后,由于没有匹配format,同时\d+要求只匹配数字,所有2.mp4匹配失败


当路由条目过多时,需要一条一条注册,过于麻烦,此时可以通过resource route简化

[python]  view plain copy print ?
  1. map.connect("messages""/messages",  
  2.     controller="messages", action="create",  
  3.     conditions=dict(method=["POST"]))  
  4. map.connect("messages""/messages",  
  5.     controller="messages", action="index",  
  6.     conditions=dict(method=["GET"]))  
  7. map.connect("formatted_messages""/messages.{format}",  
  8.     controller="messages", action="index",  
  9.     conditions=dict(method=["GET"]))  
  10. map.connect("new_message""/messages/new",  
  11.     controller="messages", action="new",  
  12.     conditions=dict(method=["GET"]))  
  13. map.connect("formatted_new_message""/messages/new.{format}",  
  14.     controller="messages", action="new",  
  15.     conditions=dict(method=["GET"]))  
  16. map.connect("/messages/{id}",  
  17.     controller="messages", action="update",  
  18.     conditions=dict(method=["PUT"]))  
  19. map.connect("/messages/{id}",  
  20.     controller="messages", action="delete",  
  21.     conditions=dict(method=["DELETE"]))  
  22. map.connect("edit_message""/messages/{id}/edit",  
  23.     controller="messages", action="edit",  
  24.     conditions=dict(method=["GET"]))  
  25. map.connect("formatted_edit_message""/messages/{id}.{format}/edit",  
  26.     controller="messages", action="edit",  
  27.     conditions=dict(method=["GET"]))  
  28. map.connect("message""/messages/{id}",  
  29.     controller="messages", action="show",  
  30.     conditions=dict(method=["GET"]))  
  31. map.connect("formatted_message""/messages/{id}.{format}",  
  32.     controller="messages", action="show",  
  33.     conditions=dict(method=["GET"]))  

上述路由条目可以使用这一条语句代替。

[python]  view plain copy print ?
  1. map.resource("message""messages")   
两个参数,一个指定单数,为member路由名字;一个指定复数,为collection路由名字。

函数原型:resource(member_name, collection_name, **kwargs)


[plain]  view plain copy print ?
  1. GET    /messages        => messages.index()    => url("messages")  
  2. POST   /messages        => messages.create()   => url("messages")  
  3. GET    /messages/new    => messages.new()      => url("new_message")  
  4. PUT    /messages/1      => messages.update(id) => url("message", id=1)  
  5. DELETE /messages/1      => messages.delete(id) => url("message", id=1)  
  6. GET    /messages/1      => messages.show(id)   => url("message", id=1)  
  7. GET    /messages/1/edit => messages.edit(id)   => url("edit_message", id=1)  


这里有必要说一下member 路由与 collection路由。

上述的路由模型

[plain]  view plain copy print ?
  1. GET    /messages        => messages.index()     
  2. POST   /messages        => messages.create()    
  3. GET    /messages/new    => messages.new()       
  4. PUT    /messages/1      => messages.update(id)  
  5. DELETE /messages/1      => messages.delete(id)  
  6. GET    /messages/1      => messages.show(id)    
  7. GET    /messages/1/edit => messages.edit(id)   

1. 有的路由有id, 指向一个具体的对象
2. 有的路由没有id, 指向全体对象
3. 有的路由(index/create, show/update/delete)有相同的URL,但是HTTP method不同
4. 有的路由(show/edit)HTTP method和前缀相同,仅后缀不同


一个member路由指定具体实例,也就是说它们有id。而一个collection路由,
没有指定的实例,即没有给定id
综上:member路由操作一个单独的实例,而collection操作全体实例。


另一个函数collection也可以完成上述功能。

函数原型:collection(collection_name, resource_name, path_prefix=None, member_prefix='/{id}', controller=None, collection_actions=['index', 'create', 'new'],member_actions=['show', 'update', 'delete', 'edit'], member_options=None, **kwargs)


用法:

map.collection('entries', 'entry')

RoutesMiddleware将请求应声到相应WSGI程序,它将路由匹配结果存到environ环境变量中去。

[python]  view plain copy print ?
  1. from routes.middleware import RoutesMiddleware  
  2. app = RoutesMiddleware(wsgi_app, map)     # ``map`` is a routes.Mapper.  

map调用match匹配URL,并设置WSGI环境变量

[python]  view plain copy print ?
  1. environ['wsgiorg.routing_args'] = ((url, match))  
  2. environ['routes.route'] = route  
  3. environ['routes.url'] = url  

route为匹配到的路由,url为一个URLGenerator对象,match为匹配所得条目。


app为一个RoutesMiddleware对象,内部重载__call__(def __call__(self, environ, start_response))仍为一个wsgi应用。

wsgi_app为一个wsgi程序,RoutesMiddleware将环境变量(environ)设置好后,调用wsgi_app进行后续处理。


下面是一个实际的输出:

[plain]  view plain copy print ?
  1. wsgiorg.routing_args = (<routes.util.URLGenerator object at 0x0287AFB0>,   
  2.                         {'action': u'index', 'controller': <__main__.Resourse instance at 0x02876E40>})  
  3. routes.route = <routes.route.Route object at 0x02871F10>  
  4. routes.url = <routes.util.URLGenerator object at 0x0287AFB0>  

本文地址:http://blog.csdn.net/spch2008/article/details/9004926


1.下载库文件 

   webob库:http://download.csdn.net/detail/spch2008/5497755

   routes库:http://download.csdn.net/detail/spch2008/5497757

   repoze库:http://download.csdn.net/detail/spch2008/5499231

2. 组织代码

    

3. 代码

   

[python]  view plain copy print ?
  1. ''''' 
  2. Created on 2013-6-1 
  3.  
  4. @author: spch2008 
  5. '''  
  6.   
  7. from wsgiref.simple_server import make_server  
  8.   
  9. import routes.middleware  
  10. import webob.dec  
  11. import webob.exc  
  12.   
  13. class Controller:  
  14.     @webob.dec.wsgify  
  15.     def __call__(self, req):  
  16.         return webob.Response("Hello World!")  
  17.   
  18.   
  19.    
  20. class Router(object):  
  21.     def __init__(self):  
  22.         self._mapper = routes.Mapper()  
  23.         self._mapper.connect('/spch',    
  24.                         controller=Controller(),    
  25.                         action='index',    
  26.                         conditions={'method': ['GET']})    
  27.           
  28.         self._router = routes.middleware.RoutesMiddleware(self._dispatch, self._mapper)  
  29.  
  30.     @webob.dec.wsgify  
  31.     def __call__(self, req):  
  32.           
  33.         return self._router  
  34.  
  35.     @staticmethod  
  36.     @webob.dec.wsgify  
  37.     def _dispatch(req):  
  38.         match = req.environ['wsgiorg.routing_args'][1]  
  39.                   
  40.         if not match:  
  41.             return webob.exc.HTTPNotFound()  
  42.           
  43.         app = match['controller']    
  44.         return app  
  45.           
  46.         
  47.   
  48. app = Router()  
  49. httpd = make_server('localhost'8282, app)    
  50. httpd.serve_forever()   


  22行:创建一个mapper

  23行:#注册一个路由

  28行:创建一个RoutesMiddleware对象,匹配路由,修改环境变量后,调用self._dispatch


4. 运行结果

    


  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值