routes.mapper
– Mapper and Sub-Mapper¶
Mapper and Sub-Mapper
Module Contents¶
- class routes.mapper.SubMapperParent¶
Base class for Mapper and SubMapper, both of which may be the parent of SubMapper objects
- 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)¶
Create a submapper that represents a collection.
This results in a
routes.mapper.SubMapper
object, with amember
property of the same type that represents the collection’s member resources.Its interface is the same as the
submapper
together withmember_prefix
,member_actions
andmember_options
which are passed to themember
submapper aspath_prefix
,actions
and keyword arguments respectively.Example:
>>> from routes.util import url_for >>> map = Mapper(controller_scan=None) >>> c = map.collection('entries', 'entry') >>> c.member.link('ping', method='POST') >>> url_for('entries') == '/entries' True >>> url_for('edit_entry', id=1) == '/entries/1/edit' True >>> url_for('ping_entry', id=1) == '/entries/1/ping' True
- submapper(**kargs)¶
Create a partial version of the Mapper with the designated options set
This results in a
routes.mapper.SubMapper
object.If keyword arguments provided to this method also exist in the keyword arguments provided to the submapper, their values will be merged with the saved options going first.
In addition to
routes.route.Route
arguments, submapper can also take apath_prefix
argument which will be prepended to the path of all routes that are connected.Example:
>>> map = Mapper(controller_scan=None) >>> map.connect('home', '/', controller='home', action='splash') >>> map.matchlist[0].name == 'home' True >>> m = map.submapper(controller='home') >>> m.connect('index', '/index', action='index') >>> map.matchlist[1].name == 'index' True >>> map.matchlist[1].defaults['controller'] == 'home' True
Optional
collection_name
andresource_name
arguments are used in the generation of route names by theaction
andlink
methods. These in turn are used by theindex
,new
,create
,show
,edit
,update
anddelete
methods which may be invoked indirectly by listing them in theactions
argument. If theformatted
argument is set toTrue
(the default), generated paths are given the suffix ‘{.format}’ which matches or generates an optional format extension.Example:
>>> from routes.util import url_for >>> map = Mapper(controller_scan=None) >>> m = map.submapper(path_prefix='/entries', collection_name='entries', resource_name='entry', actions=['index', 'new']) >>> url_for('entries') == '/entries' True >>> url_for('new_entry', format='xml') == '/entries/new.xml' True
- class routes.mapper.SubMapper(obj, resource_name=None, collection_name=None, actions=None, formatted=None, **kwargs)¶
Partial mapper for use with_options
- action(name=None, action=None, method='GET', formatted=None, **kwargs)¶
Generates a named route at the base path of a submapper.
Example:
>>> from routes import url_for >>> map = Mapper(controller_scan=None) >>> c = map.submapper(path_prefix='/entries', controller='entry') >>> c.action(action='index', name='entries', formatted=True) >>> c.action(action='create', method='POST') >>> url_for(controller='entry', action='index', method='GET') == '/entries' True >>> url_for(controller='entry', action='index', method='GET', format='xml') == '/entries.xml' True >>> url_for(controller='entry', action='create', method='POST') == '/entries' True
- add_actions(actions, **kwargs)¶
- connect(routename, path=None, **kwargs)¶
- create(**kwargs)¶
Generates the “create” action for a collection submapper.
- delete(**kwargs)¶
Generates the “delete” action for a collection member submapper.
- edit(**kwargs)¶
Generates the “edit” link for a collection member submapper.
- index(name=None, **kwargs)¶
Generates the “index” action for a collection submapper.
- link(rel=None, name=None, action=None, method='GET', formatted=None, **kwargs)¶
Generates a named route for a subresource.
Example:
>>> from routes.util import url_for >>> map = Mapper(controller_scan=None) >>> c = map.collection('entries', 'entry') >>> c.link('recent', name='recent_entries') >>> c.member.link('ping', method='POST', formatted=True) >>> url_for('entries') == '/entries' True >>> url_for('recent_entries') == '/entries/recent' True >>> url_for('ping_entry', id=1) == '/entries/1/ping' True >>> url_for('ping_entry', id=1, format='xml') == '/entries/1/ping.xml' True
- new(**kwargs)¶
Generates the “new” link for a collection submapper.
- show(name=None, **kwargs)¶
Generates the “show” action for a collection member submapper.
- update(**kwargs)¶
Generates the “update” action for a collection member submapper.
- class routes.mapper.Mapper(controller_scan=<function controller_scan>, directory=None, always_scan=False, register=True, explicit=True)¶
Mapper handles URL generation and URL recognition in a web application.
Mapper is built handling dictionary’s. It is assumed that the web application will handle the dictionary returned by URL recognition to dispatch appropriately.
URL generation is done by passing keyword parameters into the generate function, a URL is then returned.
Create a new Mapper instance
All keyword arguments are optional.
controller_scan
Function reference that will be used to return a list of valid controllers used during URL matching. If
directory
keyword arg is present, it will be passed into the function during its call. This option defaults to a function that will scan a directory for controllers.Alternatively, a list of controllers or None can be passed in which are assumed to be the definitive list of controller names valid when matching ‘controller’.
directory
Passed into controller_scan for the directory to scan. It should be an absolute path if using the default
controller_scan
function.always_scan
Whether or not the
controller_scan
function should be run during every URL match. This is typically a good idea during development so the server won’t need to be restarted anytime a controller is added.register
Boolean used to determine if the Mapper should use
request_config
to register itself as the mapper. Since it’s done on a thread-local basis, this is typically best used during testing though it won’t hurt in other cases.explicit
Boolean used to determine if routes should be connected with implicit defaults of:
{'controller':'content','action':'index','id':None}
When set to True, these defaults will not be added to route connections and
url_for
will not use Route memory.
Additional attributes that may be set after mapper initialization (ie, map.ATTRIBUTE = ‘something’):
encoding
Used to indicate alternative encoding/decoding systems to use with both incoming URL’s, and during Route generation when passed a Unicode string. Defaults to ‘utf-8’.
decode_errors
How to handle errors in the encoding, generally ignoring any chars that don’t convert should be sufficient. Defaults to ‘ignore’.
minimization
Boolean used to indicate whether or not Routes should minimize URL’s and the generated URL’s, or require every part where it appears in the path. Defaults to False.
hardcode_names
Whether or not Named Routes result in the default options for the route being used or if they actually force url generation to use the route. Defaults to False.
- connect(*args, **kargs)¶
Create and connect a new Route to the Mapper.
Usage:
m = Mapper() m.connect(':controller/:action/:id') m.connect('date/:year/:month/:day', controller="blog", action="view") m.connect('archives/:page', controller="blog", action="by_page", requirements = { 'page':'\d{1,2}' }) m.connect('category_list', 'archives/category/:section', controller='blog', action='category', section='home', type='list') m.connect('home', '', controller='blog', action='view', section='home')
- create_regs(*args, **kwargs)¶
Atomically creates regular expressions for all connected routes
- extend(routes, path_prefix='')¶
Extends the mapper routes with a list of Route objects
If a path_prefix is provided, all the routes will have their path prepended with the path_prefix.
Example:
>>> map = Mapper(controller_scan=None) >>> map.connect('home', '/', controller='home', action='splash') >>> map.matchlist[0].name == 'home' True >>> routes = [Route('index', '/index.htm', controller='home', ... action='index')] >>> map.extend(routes) >>> len(map.matchlist) == 2 True >>> map.extend(routes, path_prefix='/subapp') >>> len(map.matchlist) == 3 True >>> map.matchlist[2].routepath == '/subapp/index.htm' True
Note
This function does not merely extend the mapper with the given list of routes, it actually creates new routes with identical calling arguments.
- generate(*args, **kargs)¶
Generate a route from a set of keywords
Returns the url text, or None if no URL could be generated.
m.generate(controller='content',action='view',id=10)
- make_route(*args, **kargs)¶
Make a new Route object
A subclass can override this method to use a custom Route class.
- match(url=None, environ=None)¶
Match a URL against against one of the routes contained.
Will return None if no valid match is found.
resultdict = m.match('/joe/sixpack')
- redirect(match_path, destination_path, *args, **kwargs)¶
Add a redirect route to the mapper
Redirect routes bypass the wrapped WSGI application and instead result in a redirect being issued by the RoutesMiddleware. As such, this method is only meaningful when using RoutesMiddleware.
By default, a 302 Found status code is used, this can be changed by providing a
_redirect_code
keyword argument which will then be used instead. Note that the entire status code string needs to be present.When using keyword arguments, all arguments that apply to matching will be used for the match, while generation specific options will be used during generation. Thus all options normally available to connected Routes may be used with redirect routes as well.
Example:
map = Mapper() map.redirect('/legacyapp/archives/{url:.*}', '/archives/{url}') map.redirect('/home/index', '/', _redirect_code='301 Moved Permanently')
- resource(member_name, collection_name, **kwargs)¶
Generate routes for a controller resource
The member_name name should be the appropriate singular version of the resource given your locale and used with members of the collection. The collection_name name will be used to refer to the resource collection methods and should be a plural version of the member_name argument. By default, the member_name name will also be assumed to map to a controller you create.
The concept of a web resource maps somewhat directly to ‘CRUD’ operations. The overlying things to keep in mind is that mapping a resource is about handling creating, viewing, and editing that resource.
All keyword arguments are optional.
controller
If specified in the keyword args, the controller will be the actual controller used, but the rest of the naming conventions used for the route names and URL paths are unchanged.
collection
Additional action mappings used to manipulate/view the entire set of resources provided by the controller.
Example:
map.resource('message', 'messages', collection={'rss':'GET'}) # GET /message/rss (maps to the rss action) # also adds named route "rss_message"
member
Additional action mappings used to access an individual ‘member’ of this controllers resources.
Example:
map.resource('message', 'messages', member={'mark':'POST'}) # POST /message/1/mark (maps to the mark action) # also adds named route "mark_message"
new
Action mappings that involve dealing with a new member in the controller resources.
Example:
map.resource('message', 'messages', new={'preview':'POST'}) # POST /message/new/preview (maps to the preview action) # also adds a url named "preview_new_message"
path_prefix
Prepends the URL path for the Route with the path_prefix given. This is most useful for cases where you want to mix resources or relations between resources.
name_prefix
Perpends the route names that are generated with the name_prefix given. Combined with the path_prefix option, it’s easy to generate route names and paths that represent resources that are in relations.
Example:
map.resource('message', 'messages', controller='categories', path_prefix='/category/:category_id', name_prefix="category_") # GET /category/7/message/1 # has named route "category_message"
requirements
A dictionary that restricts the matching of a variable. Can be used when matching variables with path_prefix.
Example:
map.resource('message', 'messages', path_prefix='{project_id}/', requirements={"project_id": R"\d+"}) # POST /01234/message # success, project_id is set to "01234" # POST /foo/message # 404 not found, won't be matched by this route
parent_resource
A
dict
containing information about the parent resource, for creating a nested resource. It should contain themember_name
andcollection_name
of the parent resource. Thisdict
will be available via the associatedRoute
object which can be accessed during a request viarequest.environ['routes.route']
If
parent_resource
is supplied andpath_prefix
isn’t,path_prefix
will be generated fromparent_resource
as “<parent collection name>/:<parent member name>_id”.If
parent_resource
is supplied andname_prefix
isn’t,name_prefix
will be generated fromparent_resource
as “<parent member name>_”.Example:
>>> from routes.util import url_for >>> m = Mapper() >>> m.resource('location', 'locations', ... parent_resource=dict(member_name='region', ... collection_name='regions')) >>> # path_prefix is "regions/:region_id" >>> # name prefix is "region_" >>> url_for('region_locations', region_id=13) '/regions/13/locations' >>> url_for('region_new_location', region_id=13) '/regions/13/locations/new' >>> url_for('region_location', region_id=13, id=60) '/regions/13/locations/60' >>> url_for('region_edit_location', region_id=13, id=60) '/regions/13/locations/60/edit'
Overriding generated
path_prefix
:>>> m = Mapper() >>> m.resource('location', 'locations', ... parent_resource=dict(member_name='region', ... collection_name='regions'), ... path_prefix='areas/:area_id') >>> # name prefix is "region_" >>> url_for('region_locations', area_id=51) '/areas/51/locations'
Overriding generated
name_prefix
:>>> m = Mapper() >>> m.resource('location', 'locations', ... parent_resource=dict(member_name='region', ... collection_name='regions'), ... name_prefix='') >>> # path_prefix is "regions/:region_id" >>> url_for('locations', region_id=51) '/regions/51/locations'
- routematch(url=None, environ=None)¶
Match a URL against against one of the routes contained.
Will return None if no valid match is found, otherwise a result dict and a route object is returned.
resultdict, route_obj = m.match('/joe/sixpack')