Module: ActionController::Routing
- Defined in:
- lib/action_controller/routing.rb
Overview
Routing
The routing module provides URL rewriting in native Ruby. It’s a way to redirect incoming requests to controllers and actions. This replaces mod_rewrite rules. Best of all Rails’ Routing works with any web server. Routes are defined in routes.rb in your RAILS_ROOT/config directory.
Consider the following route, installed by Rails when you generate your application:
map.connect ':controller/:action/:id'
This route states that it expects requests to consist of a :controller followed by an :action that in turns is fed by some :id
Suppose you get an incoming request for /blog/edit/22
, you’ll end up with:
params = { :controller => 'blog',
:action => 'edit'
:id => '22'
}
Think of creating routes as drawing a map for your requests. The map tells them where to go based on some predefined pattern:
ActionController::Routing::Routes.draw do |map|
Pattern 1 tells some request to go to one place
Pattern 2 tell them to go to another
...
end
The following symbols are special:
:controller maps to your controller name
:action maps to an action with your controllers
Other names simply map to a parameter as in the case of :id
.
Route priority
Not all routes are created equally. Routes have priority defined by the order of appearance of the routes in the routes.rb file. The priority goes from top to bottom. The last route in that file is at the lowest priority will be applied last. If no route matches, 404 is returned.
Within blocks, the empty pattern goes first i.e. is at the highest priority. In practice this works out nicely:
ActionController::Routing::Routes.draw do |map|
map.with_options :controller => 'blog' do |blog|
blog.show '', :action => 'list'
end
map.connect ':controller/:action/:view
end
In this case, invoking blog controller (with an URL like ‘/blog/’) without parameters will activate the ‘list’ action by default.
Defaults routes and default parameters
Setting a default route is straightforward in Rails because by appending a Hash to the end of your mapping you can set default parameters.
Example:
ActionController::Routing:Routes.draw do |map|
map.connect ':controller/:action/:id', :controller => 'blog'
end
This sets up blog
as the default controller if no other is specified. This means visiting ‘/’ would invoke the blog controller.
More formally, you can define defaults in a route with the :defaults
key.
map.connect ':controller/:id/:action', :action => 'show', :defaults => { :page => 'Dashboard' }
Named routes
Routes can be named with the syntax map.name_of_route options
, allowing for easy reference within your source as name_of_route_url
.
Example:
# In routes.rb
map.login 'login', :controller => 'accounts', :action => 'login'
# With render, redirect_to, tests, etc.
redirect_to login_url
Arguments can be passed as well.
redirect_to show_item_url(:id => 25)
When using with_options
, the name goes after the item passed to the block.
ActionController::Routing::Routes.draw do |map|
map.with_options :controller => 'blog' do |blog|
blog.show '', :action => 'list'
blog.delete 'delete/:id', :action => 'delete',
blog.edit 'edit/:id', :action => 'edit'
end
map.connect ':controller/:action/:view
end
You would then use the named routes in your views:
link_to @article.title, show_url(:id => @article.id)
Pretty URL’s
Routes can generate pretty URLs. For example:
map.connect 'articles/:year/:month/:day',
:controller => 'articles',
:action => 'find_by_date',
:year => /\d{4}/,
:month => /\d{1,2}/,
:day => /\d{1,2}/
# Using the route above, the url below maps to:
# params = {:year => '2005', :month => '11', :day => '06'}
# http://localhost:3000/articles/2005/11/06
Regular Expressions and parameters
You can specify a reqular expression to define a format for a parameter.
map.geocode 'geocode/:postalcode', :controller => 'geocode',
:action => 'show', :postalcode => /\d{5}(-\d{4})?/
or more formally:
map.geocode 'geocode/:postalcode', :controller => 'geocode',
:action => 'show',
:requirements { :postalcode => /\d{5}(-\d{4})?/ }
Route globbing
Specifying *[string]
as part of a rule like :
map.connect '*path' , :controller => 'blog' , :action => 'unrecognized?'
will glob all remaining parts of the route that were not recognized earlier. This idiom must appear at the end of the path. The globbed values are in params[:path]
in this case.
Reloading routes
You can reload routes if you feel you must:
Action::Controller::Routes.reload
This will clear all named routes and reload routes.rb
Testing Routes
The two main methods for testing your routes:
assert_routing
def test_movie_route_properly_splits
opts = {:controller => "plugin", :action => "checkout", :id => "2"}
assert_routing "plugin/checkout/2", opts
end
assert_routing
lets you test whether or not the route properly resolves into options.
assert_recognizes
def
opts = {:controller => "plugin", :action => "show", :id => "12"}
assert_recognizes opts, "/plugins/show/12"
end
Note the subtle difference between the two: assert_routing
tests that an URL fits options while assert_recognizes
tests that an URL breaks into parameters properly.
In tests you can simply pass the URL or named route to get
or post
.
def send_to_jail
get '/jail'
assert_response :success
assert_template "jail/front"
end
def goes_to_login
get login_url
#...
end
Defined Under Namespace
Classes: ControllerSegment, DividerSegment, DynamicSegment, PathSegment, Route, RouteBuilder, RouteSet, Segment, StaticSegment
Constant Summary collapse
Class Method Summary collapse
- .controller_relative_to(controller, previous) ⇒ Object
- .normalize_paths(paths) ⇒ Object
- .possible_controllers ⇒ Object
- .use_controllers!(controller_names) ⇒ Object
- .with_controllers(names) ⇒ Object
Class Method Details
.controller_relative_to(controller, previous) ⇒ Object
299 300 301 302 303 304 305 |
# File 'lib/action_controller/routing.rb', line 299 def controller_relative_to(controller, previous) if controller.nil? then previous elsif controller[0] == ?/ then controller[1..-1] elsif %r{^(.*)/} =~ previous then "#{$1}/#{controller}" else controller end end |
.normalize_paths(paths) ⇒ Object
253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 |
# File 'lib/action_controller/routing.rb', line 253 def normalize_paths(paths) # do the hokey-pokey of path normalization... paths = paths.collect do |path| path = path. gsub("//", "/"). # replace double / chars with a single gsub("\\\\", "\\"). # replace double \ chars with a single gsub(%r{(.)[\\/]$}, '\1') # drop final / or \ if path ends with it # eliminate .. paths where possible re = %r{\w+[/\\]\.\.[/\\]} path.gsub!(%r{\w+[/\\]\.\.[/\\]}, "") while path.match(re) path end # start with longest path, first paths = paths.uniq.sort_by { |path| - path.length } end |
.possible_controllers ⇒ Object
271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 |
# File 'lib/action_controller/routing.rb', line 271 def possible_controllers unless @possible_controllers @possible_controllers = [] paths = controller_paths.select { |path| File.directory?(path) && path != "." } seen_paths = Hash.new {|h, k| h[k] = true; false} normalize_paths(paths).each do |load_path| Dir["#{load_path}/**/*_controller.rb"].collect do |path| next if seen_paths[path.gsub(%r{^\.[/\\]}, "")] controller_name = path[(load_path.length + 1)..-1] controller_name.gsub!(/_controller\.rb\Z/, '') @possible_controllers << controller_name end end # remove duplicates @possible_controllers.uniq! end @possible_controllers end |
.use_controllers!(controller_names) ⇒ Object
295 296 297 |
# File 'lib/action_controller/routing.rb', line 295 def use_controllers!(controller_names) @possible_controllers = controller_names end |
.with_controllers(names) ⇒ Object
245 246 247 248 249 250 251 |
# File 'lib/action_controller/routing.rb', line 245 def with_controllers(names) prior_controllers = @possible_controllers use_controllers! names yield ensure use_controllers! prior_controllers end |