2006-06-01 11:42:08 -04:00
require 'cgi'
2006-12-28 16:04:44 -05:00
require 'uri'
2007-05-12 17:12:31 -04:00
require 'action_controller/polymorphic_routes'
2007-09-08 20:18:55 -04:00
require 'action_controller/routing_optimisation'
2006-06-01 11:42:08 -04:00
class Object
def to_param
to_s
end
end
class TrueClass
def to_param
self
end
end
class FalseClass
def to_param
self
end
end
class NilClass
def to_param
self
end
end
2007-01-26 16:37:38 -05:00
class Regexp #:nodoc:
2006-06-01 11:42:08 -04:00
def number_of_captures
Regexp . new ( " | #{ source } " ) . match ( '' ) . captures . length
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
class << self
def optionalize ( pattern )
case unoptionalize ( pattern )
when / \ A(.| \ (.* \ )) \ Z / then " #{ pattern } ? "
else " (?: #{ pattern } )? "
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def unoptionalize ( pattern )
[ / \ A \ ( \ ?:(.*) \ ) \ ? \ Z / , / \ A(.| \ (.* \ )) \ ? \ Z / ] . each do | regexp |
return $1 if regexp =~ pattern
end
return pattern
end
end
end
2005-02-14 20:45:35 -05:00
module ActionController
2007-11-06 03:09:39 -05:00
# == Routing
2006-09-04 16:34:19 -04:00
#
# The routing module provides URL rewriting in native Ruby. It's a way to
# redirect incoming requests to controllers and actions. This replaces
2007-11-06 03:09:39 -05:00
# mod_rewrite rules. Best of all, Rails' Routing works with any web server.
2006-09-04 16:34:19 -04:00
# Routes are defined in routes.rb in your RAILS_ROOT/config directory.
#
2007-11-06 03:09:39 -05:00
# Consider the following route, installed by Rails when you generate your
2006-09-04 16:34:19 -04:00
# application:
#
# map.connect ':controller/:action/:id'
#
2007-11-06 03:09:39 -05:00
# This route states that it expects requests to consist of a
# :controller followed by an :action that in turn is fed some :id.
2006-09-04 16:34:19 -04:00
#
2007-11-06 03:09:39 -05:00
# Suppose you get an incoming request for <tt>/blog/edit/22</tt>, you'll end up
2006-09-04 16:34:19 -04:00
# with:
#
# params = { :controller => 'blog',
2007-11-06 03:09:39 -05:00
# :action => 'edit',
2006-09-04 16:34:19 -04:00
# :id => '22'
# }
#
2007-11-06 03:09:39 -05:00
# Think of creating routes as drawing a map for your requests. The map tells
2006-09-04 16:34:19 -04:00
# 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
2007-11-06 03:09:39 -05:00
#
2006-09-04 16:34:19 -04:00
# Other names simply map to a parameter as in the case of +:id+.
2007-11-06 03:09:39 -05:00
#
2006-09-04 16:34:19 -04:00
# == Route priority
#
2007-11-06 03:09:39 -05:00
# Not all routes are created equally. Routes have priority defined by the
2006-09-04 16:34:19 -04:00
# 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
2007-11-06 03:09:39 -05:00
# and will be applied last. If no route matches, 404 is returned.
2006-09-04 16:34:19 -04:00
#
2007-11-06 03:09:39 -05:00
# Within blocks, the empty pattern is at the highest priority.
2006-09-04 16:34:19 -04:00
# In practice this works out nicely:
#
2007-11-06 03:09:39 -05:00
# ActionController::Routing::Routes.draw do |map|
2006-09-04 16:34:19 -04:00
# map.with_options :controller => 'blog' do |blog|
2007-11-06 03:09:39 -05:00
# blog.show '', :action => 'list'
2006-09-04 16:34:19 -04:00
# end
2007-11-06 03:09:39 -05:00
# map.connect ':controller/:action/:view'
2006-09-04 16:34:19 -04:00
# end
#
2007-11-06 03:09:39 -05:00
# In this case, invoking blog controller (with an URL like '/blog/')
2006-09-04 16:34:19 -04:00
# without parameters will activate the 'list' action by default.
#
# == Defaults routes and default parameters
#
2007-11-06 03:09:39 -05:00
# Setting a default route is straightforward in Rails - you simply append a
# Hash at the end of your mapping to set any default parameters.
2006-09-04 16:34:19 -04:00
#
# Example:
# ActionController::Routing:Routes.draw do |map|
# map.connect ':controller/:action/:id', :controller => 'blog'
# end
#
2007-11-06 03:09:39 -05:00
# This sets up +blog+ as the default controller if no other is specified.
2006-09-04 16:34:19 -04:00
# This means visiting '/' would invoke the blog controller.
#
# More formally, you can define defaults in a route with the +:defaults+ key.
2007-11-06 03:09:39 -05:00
#
# map.connect ':controller/:action/:id', :action => 'show', :defaults => { :page => 'Dashboard' }
2006-09-04 16:34:19 -04:00
#
# == Named routes
#
# Routes can be named with the syntax <tt>map.name_of_route options</tt>,
2007-01-28 09:52:25 -05:00
# allowing for easy reference within your source as +name_of_route_url+
# for the full URL and +name_of_route_path+ for the URI path.
2006-09-04 16:34:19 -04:00
#
# 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.
#
2007-01-28 09:52:25 -05:00
# redirect_to show_item_path(:id => 25)
2006-09-04 16:34:19 -04:00
#
2007-11-06 03:09:39 -05:00
# Use <tt>map.root</tt> as a shorthand to name a route for the root path "".
2006-09-04 16:34:19 -04:00
#
2007-01-28 09:52:25 -05:00
# # In routes.rb
# map.root :controller => 'blogs'
#
# # would recognize http://www.example.com/ as
# params = { :controller => 'blogs', :action => 'index' }
#
# # and provide these named routes
# root_url # => 'http://www.example.com/'
# root_path # => ''
2006-09-04 16:34:19 -04:00
#
2007-01-28 09:52:25 -05:00
# Note: when using +with_options+, the route is simply named after the
# method you call on the block parameter rather than map.
#
# # In routes.rb
# map.with_options :controller => 'blog' do |blog|
# blog.show '', :action => 'list'
# blog.delete 'delete/:id', :action => 'delete',
# blog.edit 'edit/:id', :action => 'edit'
# end
2006-09-04 16:34:19 -04:00
#
2007-01-28 09:52:25 -05:00
# # provides named routes for show, delete, and edit
2007-11-06 03:09:39 -05:00
# link_to @article.title, show_path(:id => @article.id)
2006-09-04 16:34:19 -04:00
#
2007-01-28 09:52:25 -05:00
# == Pretty URLs
2006-09-04 16:34:19 -04:00
#
# Routes can generate pretty URLs. For example:
#
# map.connect 'articles/:year/:month/:day',
2007-11-06 03:09:39 -05:00
# :controller => 'articles',
2006-09-04 16:34:19 -04:00
# :action => 'find_by_date',
# :year => /\d{4}/,
2007-11-06 03:09:39 -05:00
# :month => /\d{1,2}/,
# :day => /\d{1,2}/
#
2006-09-04 16:34:19 -04:00
# # 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
2007-11-06 03:09:39 -05:00
# You can specify a regular expression to define a format for a parameter.
2006-09-04 16:34:19 -04:00
#
# map.geocode 'geocode/:postalcode', :controller => 'geocode',
# :action => 'show', :postalcode => /\d{5}(-\d{4})?/
#
2007-05-26 16:29:05 -04:00
# or, more formally:
2006-09-04 16:34:19 -04:00
#
2007-11-06 03:09:39 -05:00
# map.geocode 'geocode/:postalcode', :controller => 'geocode',
2007-05-26 16:29:05 -04:00
# :action => 'show', :requirements => { :postalcode => /\d{5}(-\d{4})?/ }
2006-09-04 16:34:19 -04:00
#
# == Route globbing
#
2007-10-12 23:28:35 -04:00
# Specifying <tt>*[string]</tt> as part of a rule like:
2006-09-04 16:34:19 -04:00
#
# map.connect '*path' , :controller => 'blog' , :action => 'unrecognized?'
#
2007-10-12 23:28:35 -04:00
# 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 <tt>params[:path]</tt> in
# this case.
2006-09-04 16:34:19 -04:00
#
2007-11-06 13:54:33 -05:00
# == Route conditions
#
# With conditions you can define restrictions on routes. Currently the only valid condition is <tt>:method</tt>.
#
# * <tt>:method</tt> - Allows you to specify which method can access the route. Possible values are <tt>:post</tt>,
# <tt>:get</tt>, <tt>:put</tt>, <tt>:delete</tt> and <tt>:any</tt>. The default value is <tt>:any</tt>,
# <tt>:any</tt> means that any method can access the route.
#
# Example:
#
# map.connect 'post/:id', :controller => 'posts', :action => 'show',
# :conditions => { :method => :get }
# map.connect 'post/:id', :controller => 'posts', :action => 'create_comment',
# :conditions => { :method => :post }
#
# Now, if you POST to <tt>/posts/:id</tt>, it will route to the <tt>create_comment</tt> action. A GET on the same
# URL will route to the <tt>show</tt> action.
#
2006-09-04 16:34:19 -04:00
# == Reloading routes
#
# You can reload routes if you feel you must:
#
2007-05-06 00:24:27 -04:00
# ActionController::Routing::Routes.reload
2006-09-04 16:34:19 -04:00
#
2007-10-12 23:28:35 -04:00
# This will clear all named routes and reload routes.rb if the file has been modified from
# last load. To absolutely force reloading, use +reload!+.
2006-09-04 16:34:19 -04:00
#
# == Testing Routes
#
# The two main methods for testing your routes:
#
# === +assert_routing+
2007-11-06 03:09:39 -05:00
#
2006-09-04 16:34:19 -04:00
# def test_movie_route_properly_splits
# opts = {:controller => "plugin", :action => "checkout", :id => "2"}
# assert_routing "plugin/checkout/2", opts
# end
2007-11-06 03:09:39 -05:00
#
2006-09-04 16:34:19 -04:00
# +assert_routing+ lets you test whether or not the route properly resolves into options.
#
# === +assert_recognizes+
#
# def test_route_has_options
# opts = {:controller => "plugin", :action => "show", :id => "12"}
2007-11-06 03:09:39 -05:00
# assert_recognizes opts, "/plugins/show/12"
2006-09-04 16:34:19 -04:00
# end
2007-11-06 03:09:39 -05:00
#
2006-09-04 16:34:19 -04:00
# Note the subtle difference between the two: +assert_routing+ tests that
2007-11-06 03:09:39 -05:00
# a URL fits options while +assert_recognizes+ tests that a URL
2006-09-04 16:34:19 -04:00
# 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
#
2007-12-13 13:21:19 -05:00
# == View a list of all your routes
#
# Run <tt>rake routes</tt>.
#
2006-06-01 11:42:08 -04:00
module Routing
2007-09-23 17:58:02 -04:00
SEPARATORS = %w( / . ? )
2006-06-01 11:42:08 -04:00
2007-05-26 16:07:34 -04:00
HTTP_METHODS = [ :get , :head , :post , :put , :delete ]
2007-11-20 16:25:25 -05:00
ALLOWED_REQUIREMENTS_FOR_OPTIMISATION = [ :controller , :action ] . to_set
2006-08-14 21:28:06 -04:00
# The root paths which may contain controller files
mattr_accessor :controller_paths
self . controller_paths = [ ]
2007-11-06 03:09:39 -05:00
2007-05-12 00:18:46 -04:00
# A helper module to hold URL related helpers.
module Helpers
2007-05-12 17:12:31 -04:00
include PolymorphicRoutes
2007-05-12 00:18:46 -04:00
end
2007-11-06 03:09:39 -05:00
2005-06-24 12:40:01 -04:00
class << self
2006-06-01 11:42:08 -04:00
def with_controllers ( names )
2006-09-23 13:25:06 -04:00
prior_controllers = @possible_controllers
2006-06-01 11:42:08 -04:00
use_controllers! names
yield
ensure
2006-09-23 13:25:06 -04:00
use_controllers! prior_controllers
2006-06-01 11:42:08 -04:00
end
2006-06-07 12:16:37 -04:00
2006-08-14 21:28:06 -04:00
def normalize_paths ( paths )
2006-06-07 12:16:37 -04:00
# 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
2006-06-01 11:42:08 -04:00
def possible_controllers
unless @possible_controllers
@possible_controllers = [ ]
2007-11-06 03:09:39 -05:00
2006-08-14 21:28:06 -04:00
paths = controller_paths . select { | path | File . directory? ( path ) && path != " . " }
2006-06-07 12:16:37 -04:00
2006-06-01 11:42:08 -04:00
seen_paths = Hash . new { | h , k | h [ k ] = true ; false }
2006-06-07 12:16:37 -04:00
normalize_paths ( paths ) . each do | load_path |
2006-06-01 11:42:08 -04:00
Dir [ " #{ load_path } /**/*_controller.rb " ] . collect do | path |
2006-06-07 12:16:37 -04:00
next if seen_paths [ path . gsub ( %r{ ^ \ .[/ \\ ] } , " " ) ]
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
controller_name = path [ ( load_path . length + 1 ) .. - 1 ]
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
controller_name . gsub! ( / _controller \ .rb \ Z / , '' )
@possible_controllers << controller_name
end
end
2006-06-07 12:16:37 -04:00
# remove duplicates
@possible_controllers . uniq!
2006-06-01 11:42:08 -04:00
end
@possible_controllers
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def use_controllers! ( controller_names )
@possible_controllers = controller_names
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
2005-06-24 12:40:01 -04:00
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
2007-11-06 03:09:39 -05:00
end
end
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2007-01-26 16:37:38 -05:00
class Route #:nodoc:
2007-09-08 20:18:55 -04:00
attr_accessor :segments , :requirements , :conditions , :optimise
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def initialize
@segments = [ ]
@requirements = { }
@conditions = { }
2007-11-20 16:25:25 -05:00
@optimise = true
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2007-09-08 20:18:55 -04:00
# Indicates whether the routes should be optimised with the string interpolation
# version of the named routes methods.
def optimise?
2007-10-03 01:47:41 -04:00
@optimise && ActionController :: Base :: optimise_named_routes
2007-09-08 20:18:55 -04:00
end
2007-11-06 03:09:39 -05:00
2007-09-08 20:18:55 -04:00
def segment_keys
segments . collect do | segment |
segment . key if segment . respond_to? :key
end . compact
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Write and compile a +generate+ method for this Route.
def write_generation
# Build the main body of the generation
2006-10-14 23:11:08 -04:00
body = " expired = false \n #{ generation_extraction } \n #{ generation_structure } "
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# If we have conditions that must be tested first, nest the body inside an if
body = " if #{ generation_requirements } \n #{ body } \n end " if generation_requirements
args = " options, hash, expire_on = {} "
# Nest the body inside of a def block, and then compile it.
2006-06-29 22:36:17 -04:00
raw_method = method_decl = " def generate_raw( #{ args } ) \n path = begin \n #{ body } \n end \n [path, hash] \n end "
2006-06-01 11:42:08 -04:00
instance_eval method_decl , " generated code ( #{ __FILE__ } : #{ __LINE__ } ) "
2006-06-06 13:59:54 -04:00
# expire_on.keys == recall.keys; in other words, the keys in the expire_on hash
# are the same as the keys that were recalled from the previous request. Thus,
# we can use the expire_on.keys to determine which keys ought to be used to build
# the query string. (Never use keys from the recalled request when building the
# query string.)
2006-12-08 23:42:32 -05:00
method_decl = " def generate( #{ args } ) \n path, hash = generate_raw(options, hash, expire_on) \n append_query_string(path, hash, extra_keys(options)) \n end "
2006-06-01 11:42:08 -04:00
instance_eval method_decl , " generated code ( #{ __FILE__ } : #{ __LINE__ } ) "
2005-06-24 12:40:01 -04:00
2006-12-08 23:42:32 -05:00
method_decl = " def generate_extras( #{ args } ) \n path, hash = generate_raw(options, hash, expire_on) \n [path, extra_keys(options)] \n end "
2006-06-01 11:42:08 -04:00
instance_eval method_decl , " generated code ( #{ __FILE__ } : #{ __LINE__ } ) "
2006-06-29 22:36:17 -04:00
raw_method
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Build several lines of code that extract values from the options hash. If any
# of the values are missing or rejected then a return will be executed.
def generation_extraction
segments . collect do | segment |
segment . extraction_code
end . compact * " \n "
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Produce a condition expression that will check the requirements of this route
# upon generation.
def generation_requirements
requirement_conditions = requirements . collect do | key , req |
if req . is_a? Regexp
value_regexp = Regexp . new " \\ A #{ req . source } \\ Z "
" hash[: #{ key } ] && #{ value_regexp . inspect } =~ options[: #{ key } ] "
2005-08-26 06:57:43 -04:00
else
2006-06-01 11:42:08 -04:00
" hash[: #{ key } ] == #{ req . inspect } "
2005-08-26 06:57:43 -04:00
end
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
requirement_conditions * ' && ' unless requirement_conditions . empty?
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def generation_structure
segments . last . string_structure segments [ 0 .. - 2 ]
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Write and compile a +recognize+ method for this Route.
def write_recognition
# Create an if structure to extract the params from a match if it occurs.
body = " params = parameter_shell.dup \n #{ recognition_extraction * " \n " } \n params "
body = " if #{ recognition_conditions . join ( " && " ) } \n #{ body } \n end "
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Build the method declaration and compile it
method_decl = " def recognize(path, env={}) \n #{ body } \n end "
instance_eval method_decl , " generated code ( #{ __FILE__ } : #{ __LINE__ } ) "
method_decl
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
# Plugins may override this method to add other conditions, like checks on
# host, subdomain, and so forth. Note that changes here only affect route
# recognition, not generation.
def recognition_conditions
result = [ " (match = #{ Regexp . new ( recognition_pattern ) . inspect } .match(path)) " ]
result << " conditions[:method] === env[:method] " if conditions [ :method ]
result
2006-04-21 11:17:02 -04:00
end
2006-06-01 11:42:08 -04:00
# Build the regular expression pattern that will match this route.
def recognition_pattern ( wrap = true )
pattern = ''
segments . reverse_each do | segment |
pattern = segment . build_pattern pattern
2006-04-21 13:21:26 -04:00
end
2006-06-01 11:42:08 -04:00
wrap ? ( " \\ A " + pattern + " \\ Z " ) : pattern
2006-04-21 11:17:02 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Write the code to extract the parameters from a matched route.
def recognition_extraction
next_capture = 1
extraction = segments . collect do | segment |
2007-02-21 05:05:07 -05:00
x = segment . match_extraction ( next_capture )
2006-06-01 11:42:08 -04:00
next_capture += Regexp . new ( segment . regexp_chunk ) . number_of_captures
x
end
extraction . compact
2006-04-21 11:17:02 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Write the real generation implementation and then resend the message.
def generate ( options , hash , expire_on = { } )
write_generation
generate options , hash , expire_on
2006-04-21 11:17:02 -04:00
end
2006-06-01 11:42:08 -04:00
def generate_extras ( options , hash , expire_on = { } )
write_generation
generate_extras options , hash , expire_on
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
# Generate the query string with any extra keys in the hash and append
# it to the given path, returning the new path.
2006-06-06 13:59:54 -04:00
def append_query_string ( path , hash , query_keys = nil )
2006-06-01 11:42:08 -04:00
return nil unless path
2006-06-06 13:59:54 -04:00
query_keys || = extra_keys ( hash )
" #{ path } #{ build_query_string ( hash , query_keys ) } "
end
# Determine which keys in the given hash are "extra". Extra keys are
# those that were not used to generate a particular route. The extra
# keys also do not include those recalled from the prior request, nor
# do they include any keys that were implied in the route (like a
# :controller that is required, but not explicitly used in the text of
# the route.)
def extra_keys ( hash , recall = { } )
( hash || { } ) . keys . map { | k | k . to_sym } - ( recall || { } ) . keys - significant_keys
2006-06-01 11:42:08 -04:00
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
# Build a query string from the keys of the given hash. If +only_keys+
# is given (as an array), only the keys indicated will be used to build
# the query string. The query string will correctly build array parameter
# values.
2007-03-06 02:47:23 -05:00
def build_query_string ( hash , only_keys = nil )
2006-06-01 11:42:08 -04:00
elements = [ ]
2006-04-21 11:17:02 -04:00
2007-03-06 02:47:23 -05:00
( only_keys || hash . keys ) . each do | key |
if value = hash [ key ]
elements << value . to_query ( key )
end
end
elements . empty? ? '' : " ? #{ elements . sort * '&' } "
2006-06-01 11:42:08 -04:00
end
2007-03-06 02:47:23 -05:00
2006-06-01 11:42:08 -04:00
# Write the real recognition implementation and then resend the message.
def recognize ( path , environment = { } )
write_recognition
recognize path , environment
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# A route's parameter shell contains parameter values that are not in the
# route's path, but should be placed in the recognized hash.
2007-11-06 03:09:39 -05:00
#
2006-06-01 11:42:08 -04:00
# For example, +{:controller => 'pages', :action => 'show'} is the shell for the route:
2007-11-06 03:09:39 -05:00
#
2006-06-01 11:42:08 -04:00
# map.connect '/page/:id', :controller => 'pages', :action => 'show', :id => /\d+/
2007-11-06 03:09:39 -05:00
#
2006-06-01 11:42:08 -04:00
def parameter_shell
@parameter_shell || = returning ( { } ) do | shell |
requirements . each do | key , requirement |
shell [ key ] = requirement unless requirement . is_a? Regexp
end
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Return an array containing all the keys that are used in this route. This
# includes keys that appear inside the path, and keys that have requirements
# placed upon them.
def significant_keys
@significant_keys || = returning [ ] do | sk |
segments . each { | segment | sk << segment . key if segment . respond_to? :key }
sk . concat requirements . keys
sk . uniq!
end
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
# Return a hash of key/value pairs representing the keys in the route that
# have defaults, or which are specified by non-regexp requirements.
def defaults
@defaults || = returning ( { } ) do | hash |
segments . each do | segment |
next unless segment . respond_to? :default
hash [ segment . key ] = segment . default unless segment . default . nil?
end
requirements . each do | key , req |
next if Regexp === req || req . nil?
hash [ key ] = req
end
2006-04-21 11:17:02 -04:00
end
2006-06-01 11:42:08 -04:00
end
2007-01-28 02:16:55 -05:00
2006-06-01 11:42:08 -04:00
def matches_controller_and_action? ( controller , action )
2007-01-28 02:16:55 -05:00
unless defined? @matching_prepared
2006-06-01 11:42:08 -04:00
@controller_requirement = requirement_for ( :controller )
@action_requirement = requirement_for ( :action )
@matching_prepared = true
end
( @controller_requirement . nil? || @controller_requirement === controller ) &&
( @action_requirement . nil? || @action_requirement === action )
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
def to_s
2006-06-06 13:59:54 -04:00
@to_s || = begin
2006-06-29 22:36:17 -04:00
segs = segments . inject ( " " ) { | str , s | str << s . to_s }
" %-6s %-40s %s " % [ ( conditions [ :method ] || :any ) . to_s . upcase , segs , requirements . inspect ]
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
protected
def requirement_for ( key )
return requirements [ key ] if requirements . key? key
segments . each do | segment |
return segment . regexp if segment . respond_to? ( :key ) && segment . key == key
2006-04-21 11:17:02 -04:00
end
2006-06-01 11:42:08 -04:00
nil
end
2007-03-06 02:47:23 -05:00
2006-04-21 11:17:02 -04:00
end
2007-01-26 16:37:38 -05:00
class Segment #:nodoc:
2007-09-23 17:58:02 -04:00
RESERVED_PCHAR = ':@&=+$,;'
2007-05-14 07:14:30 -04:00
UNSAFE_PCHAR = Regexp . new ( " [^ #{ URI :: REGEXP :: PATTERN :: UNRESERVED } #{ RESERVED_PCHAR } ] " , false , 'N' ) . freeze
2006-06-01 11:42:08 -04:00
attr_accessor :is_optional
alias_method :optional? , :is_optional
def initialize
self . is_optional = false
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def extraction_code
nil
end
2007-09-06 10:28:32 -04:00
2006-06-01 11:42:08 -04:00
# Continue generating string for the prior segments.
def continue_string_structure ( prior_segments )
if prior_segments . empty?
interpolation_statement ( prior_segments )
else
new_priors = prior_segments [ 0 .. - 2 ]
prior_segments . last . string_structure ( new_priors )
2005-02-14 20:45:35 -05:00
end
end
2007-05-14 07:14:30 -04:00
def interpolation_chunk
URI . escape ( value , UNSAFE_PCHAR )
end
2006-06-01 11:42:08 -04:00
# Return a string interpolation statement for this segment and those before it.
def interpolation_statement ( prior_segments )
chunks = prior_segments . collect { | s | s . interpolation_chunk }
chunks << interpolation_chunk
" \" #{ chunks * '' } \" #{ all_optionals_available_condition ( prior_segments ) } "
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def string_structure ( prior_segments )
optional? ? continue_string_structure ( prior_segments ) : interpolation_statement ( prior_segments )
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Return an if condition that is true if all the prior segments can be generated.
# If there are no optional segments before this one, then nil is returned.
def all_optionals_available_condition ( prior_segments )
optional_locals = prior_segments . collect { | s | s . local_name if s . optional? && s . respond_to? ( :local_name ) } . compact
optional_locals . empty? ? nil : " if #{ optional_locals * ' && ' } "
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Recognition
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def match_extraction ( next_capture )
nil
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Warning
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Returns true if this segment is optional? because of a default. If so, then
# no warning will be emitted regarding this segment.
def optionality_implied?
false
2005-06-24 12:40:01 -04:00
end
end
2005-03-20 09:04:33 -05:00
2007-01-26 16:37:38 -05:00
class StaticSegment < Segment #:nodoc:
2006-06-01 11:42:08 -04:00
attr_accessor :value , :raw
alias_method :raw? , :raw
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def initialize ( value = nil )
super ( )
self . value = value
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def interpolation_chunk
2007-05-14 07:14:30 -04:00
raw? ? value : super
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def regexp_chunk
2007-05-14 07:14:30 -04:00
chunk = Regexp . escape ( value )
2006-06-01 11:42:08 -04:00
optional? ? Regexp . optionalize ( chunk ) : chunk
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def build_pattern ( pattern )
escaped = Regexp . escape ( value )
if optional? && ! pattern . empty?
" (?: #{ Regexp . optionalize escaped } \\ Z| #{ escaped } #{ Regexp . unoptionalize pattern } ) "
elsif optional?
Regexp . optionalize escaped
2005-06-24 12:40:01 -04:00
else
2006-06-01 11:42:08 -04:00
escaped + pattern
2005-06-24 12:40:01 -04:00
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def to_s
value
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
end
2005-03-20 09:04:33 -05:00
2007-01-26 16:37:38 -05:00
class DividerSegment < StaticSegment #:nodoc:
2006-06-01 11:42:08 -04:00
def initialize ( value = nil )
super ( value )
self . raw = true
self . is_optional = true
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def optionality_implied?
true
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
end
2005-06-24 12:40:01 -04:00
2007-01-26 16:37:38 -05:00
class DynamicSegment < Segment #:nodoc:
2006-06-01 11:42:08 -04:00
attr_accessor :key , :default , :regexp
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def initialize ( key = nil , options = { } )
super ( )
self . key = key
self . default = options [ :default ] if options . key? :default
self . is_optional = true if options [ :optional ] || options . key? ( :default )
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def to_s
" : #{ key } "
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# The local variable name that the value of this segment will be extracted to.
def local_name
" #{ key } _value "
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def extract_value
2007-03-06 02:47:23 -05:00
" #{ local_name } = hash[: #{ key } ] && hash[: #{ key } ].to_param #{ " || #{ default . inspect } " if default } "
2006-06-01 11:42:08 -04:00
end
def value_check
if default # Then we know it won't be nil
" #{ value_regexp . inspect } =~ #{ local_name } " if regexp
elsif optional?
# If we have a regexp check that the value is not given, or that it matches.
# If we have no regexp, return nil since we do not require a condition.
" #{ local_name } .nil? || #{ value_regexp . inspect } =~ #{ local_name } " if regexp
else # Then it must be present, and if we have a regexp, it must match too.
" #{ local_name } #{ " && #{ value_regexp . inspect } =~ #{ local_name } " if regexp } "
2005-02-14 20:45:35 -05:00
end
2006-06-01 11:42:08 -04:00
end
def expiry_statement
2006-10-14 23:11:08 -04:00
" expired, hash = true, options if !expired && expire_on[: #{ key } ] "
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def extraction_code
s = extract_value
vc = value_check
s << " \n return [nil,nil] unless #{ vc } " if vc
s << " \n #{ expiry_statement } "
end
2007-11-06 03:09:39 -05:00
2007-10-03 01:47:41 -04:00
def interpolation_chunk ( value_code = " #{ local_name } " )
" \# {URI.escape( #{ value_code } .to_s, ActionController::Routing::Segment::UNSAFE_PCHAR)} "
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def string_structure ( prior_segments )
if optional? # We have a conditional to do...
# If we should not appear in the url, just write the code for the prior
# segments. This occurs if our value is the default value, or, if we are
# optional, if we have nil as our value.
2007-11-06 03:09:39 -05:00
" if #{ local_name } == #{ default . inspect } \n " +
continue_string_structure ( prior_segments ) +
2006-06-01 11:42:08 -04:00
" \n else \n " + # Otherwise, write the code up to here
" #{ interpolation_statement ( prior_segments ) } \n end "
else
interpolation_statement ( prior_segments )
2005-02-14 20:45:35 -05:00
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def value_regexp
Regexp . new " \\ A #{ regexp . source } \\ Z " if regexp
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def regexp_chunk
2006-06-05 10:31:38 -04:00
regexp ? " ( #{ regexp . source } ) " : " ([^ #{ Routing :: SEPARATORS . join } ]+) "
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def build_pattern ( pattern )
chunk = regexp_chunk
chunk = " ( #{ chunk } ) " if Regexp . new ( chunk ) . number_of_captures == 0
pattern = " #{ chunk } #{ pattern } "
optional? ? Regexp . optionalize ( pattern ) : pattern
end
def match_extraction ( next_capture )
2007-05-14 07:14:30 -04:00
# All non code-related keys (such as :id, :slug) are URI-unescaped as
# path parameters.
2007-02-21 05:05:07 -05:00
default_value = default ? default . inspect : nil
%[
value = if ( m = match [ #{next_capture}])
2007-05-14 07:14:30 -04:00
URI . unescape ( m )
2007-02-21 05:05:07 -05:00
else
#{default_value}
end
params [ : #{key}] = value if value
]
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def optionality_implied?
[ :action , :id ] . include? key
end
2007-11-06 03:09:39 -05:00
2005-06-24 12:40:01 -04:00
end
2007-01-26 16:37:38 -05:00
class ControllerSegment < DynamicSegment #:nodoc:
2006-06-01 11:42:08 -04:00
def regexp_chunk
possible_names = Routing . possible_controllers . collect { | name | Regexp . escape name }
" (?i-:( #{ ( regexp || Regexp . union ( * possible_names ) ) . source } )) "
2005-06-24 12:40:01 -04:00
end
2007-05-14 07:14:30 -04:00
# Don't URI.escape the controller name since it may contain slashes.
2007-10-03 01:47:41 -04:00
def interpolation_chunk ( value_code = " #{ local_name } " )
" \# { #{ value_code } .to_s} "
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
# Make sure controller names like Admin/Content are correctly normalized to
# admin/content
def extract_value
" #{ local_name } = (hash[: #{ key } ] #{ " || #{ default . inspect } " if default } ).downcase "
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def match_extraction ( next_capture )
2006-09-04 16:09:15 -04:00
if default
" params[: #{ key } ] = match[ #{ next_capture } ] ? match[ #{ next_capture } ].downcase : ' #{ default } ' "
else
" params[: #{ key } ] = match[ #{ next_capture } ].downcase if match[ #{ next_capture } ] "
end
2005-06-24 12:40:01 -04:00
end
end
2007-01-26 16:37:38 -05:00
class PathSegment < DynamicSegment #:nodoc:
2007-05-14 07:14:30 -04:00
RESERVED_PCHAR = " #{ Segment :: RESERVED_PCHAR } / "
UNSAFE_PCHAR = Regexp . new ( " [^ #{ URI :: REGEXP :: PATTERN :: UNRESERVED } #{ RESERVED_PCHAR } ] " , false , 'N' ) . freeze
2007-10-03 01:47:41 -04:00
def interpolation_chunk ( value_code = " #{ local_name } " )
" \# {URI.escape( #{ value_code } .to_s, ActionController::Routing::PathSegment::UNSAFE_PCHAR)} "
2006-06-01 11:42:08 -04:00
end
2005-07-08 04:56:24 -04:00
2006-06-01 11:42:08 -04:00
def default
''
2005-07-08 04:56:24 -04:00
end
2006-06-01 11:42:08 -04:00
def default = ( path )
raise RoutingError , " paths cannot have non-empty default values " unless path . blank?
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def match_extraction ( next_capture )
" params[: #{ key } ] = PathSegment::Result.new_escaped((match[ #{ next_capture } ] #{ " || " + default . inspect if default } ).split('/')) #{ " if match[ " + next_capture + " ] " if ! default } "
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
def regexp_chunk
regexp || " (.*) "
end
2007-09-21 00:52:18 -04:00
def optionality_implied?
true
end
2005-09-17 02:37:50 -04:00
class Result < :: Array #:nodoc:
2007-11-06 03:09:39 -05:00
def to_s ( ) join '/' end
2005-07-07 15:43:03 -04:00
def self . new_escaped ( strings )
2006-12-28 16:04:44 -05:00
new strings . collect { | str | URI . unescape str }
2007-11-06 03:09:39 -05:00
end
2007-09-21 00:52:18 -04:00
end
2005-06-24 12:40:01 -04:00
end
2005-02-14 20:45:35 -05:00
2007-01-26 16:37:38 -05:00
class RouteBuilder #:nodoc:
2006-06-01 11:42:08 -04:00
attr_accessor :separators , :optional_separators
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def initialize
self . separators = Routing :: SEPARATORS
self . optional_separators = %w( / )
2005-02-14 20:45:35 -05:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def separator_pattern ( inverted = false )
" [ #{ '^' if inverted } #{ Regexp . escape ( separators . join ) } ] "
2005-02-14 20:45:35 -05:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def interval_regexp
Regexp . new " (.*?)( #{ separators . source } |$) "
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Accepts a "route path" (a string defining a route), and returns the array
# of segments that corresponds to it. Note that the segment array is only
# partially initialized--the defaults and requirements, for instance, need
# to be set separately, via the #assign_route_options method, and the
# #optional? method for each segment will not be reliable until after
# #assign_route_options is called, as well.
def segments_for_route_path ( path )
rest , segments = path , [ ]
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
until rest . empty?
segment , rest = segment_for rest
segments << segment
end
segments
end
2005-06-24 12:40:01 -04:00
2006-06-01 11:42:08 -04:00
# A factory method that returns a new segment instance appropriate for the
# format of the given string.
def segment_for ( string )
segment = case string
when / \ A:( \ w+) /
key = $1 . to_sym
case key
when :controller then ControllerSegment . new ( key )
else DynamicSegment . new key
end
when / \ A \ *( \ w+) / then PathSegment . new ( $1 . to_sym , :optional = > true )
when / \ A \ ?(.*?) \ ? /
returning segment = StaticSegment . new ( $1 ) do
segment . is_optional = true
end
when / \ A( #{ separator_pattern ( :inverted ) } +) / then StaticSegment . new ( $1 )
when Regexp . new ( separator_pattern ) then
returning segment = DividerSegment . new ( $& ) do
segment . is_optional = ( optional_separators . include? $& )
2005-07-14 05:09:39 -04:00
end
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
[ segment , $~ . post_match ]
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Split the given hash of options into requirement and default hashes. The
# segments are passed alongside in order to distinguish between default values
# and requirements.
def divide_route_options ( segments , options )
2006-06-07 12:27:14 -04:00
options = options . dup
2007-11-06 03:09:39 -05:00
2007-05-19 12:26:44 -04:00
if options [ :namespace ]
options [ :controller ] = " #{ options [ :path_prefix ] } / #{ options [ :controller ] } "
options . delete ( :path_prefix )
options . delete ( :name_prefix )
options . delete ( :namespace )
2007-11-06 03:09:39 -05:00
end
2006-06-07 12:27:14 -04:00
requirements = ( options . delete ( :requirements ) || { } ) . dup
defaults = ( options . delete ( :defaults ) || { } ) . dup
conditions = ( options . delete ( :conditions ) || { } ) . dup
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
path_keys = segments . collect { | segment | segment . key if segment . respond_to? ( :key ) } . compact
options . each do | key , value |
hash = ( path_keys . include? ( key ) && ! value . is_a? ( Regexp ) ) ? defaults : requirements
hash [ key ] = value
2006-04-13 01:44:23 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
[ defaults , requirements , conditions ]
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Takes a hash of defaults and a hash of requirements, and assigns them to
# the segments. Any unused requirements (which do not correspond to a segment)
# are returned as a hash.
def assign_route_options ( segments , defaults , requirements )
route_requirements = { } # Requirements that do not belong to a segment
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
segment_named = Proc . new do | key |
segments . detect { | segment | segment . key == key if segment . respond_to? ( :key ) }
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
requirements . each do | key , requirement |
segment = segment_named [ key ]
if segment
raise TypeError , " #{ key } : requirements on a path segment must be regular expressions " unless requirement . is_a? ( Regexp )
2006-08-14 22:04:11 -04:00
if requirement . source =~ %r{ \ A( \\ A| \ ^)|( \\ Z| \\ z| \ $) \ Z }
2006-08-05 18:12:50 -04:00
raise ArgumentError , " Regexp anchor characters are not allowed in routing requirements: #{ requirement . inspect } "
end
2006-06-01 11:42:08 -04:00
segment . regexp = requirement
else
route_requirements [ key ] = requirement
2005-02-14 20:45:35 -05:00
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
defaults . each do | key , default |
segment = segment_named [ key ]
raise ArgumentError , " #{ key } : No matching segment exists; cannot assign default " unless segment
segment . is_optional = true
segment . default = default . to_param if default
2005-07-13 20:13:06 -04:00
end
2007-11-06 03:09:39 -05:00
2006-09-20 13:45:03 -04:00
assign_default_route_options ( segments )
2006-06-01 11:42:08 -04:00
ensure_required_segments ( segments )
route_requirements
2005-06-24 13:48:14 -04:00
end
2007-11-06 03:09:39 -05:00
2006-09-20 13:45:03 -04:00
# Assign default options, such as 'index' as a default for :action. This
# method must be run *after* user supplied requirements and defaults have
# been applied to the segments.
def assign_default_route_options ( segments )
segments . each do | segment |
next unless segment . is_a? DynamicSegment
case segment . key
when :action
if segment . regexp . nil? || segment . regexp . match ( 'index' ) . to_s == 'index'
segment . default || = 'index'
segment . is_optional = true
end
when :id
if segment . default . nil? && segment . regexp . nil? || segment . regexp =~ ''
segment . is_optional = true
end
end
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Makes sure that there are no optional segments that precede a required
# segment. If any are found that precede a required segment, they are
# made required.
def ensure_required_segments ( segments )
allow_optional = true
segments . reverse_each do | segment |
allow_optional && = segment . optional?
if ! allow_optional && segment . optional?
unless segment . optionality_implied?
warn " Route segment \" #{ segment . to_s } \" cannot be optional because it precedes a required segment. This segment will be required. "
end
segment . is_optional = false
2007-09-22 15:02:51 -04:00
elsif allow_optional && segment . respond_to? ( :default ) && segment . default
2006-06-01 11:42:08 -04:00
# if a segment has a default, then it is optional
segment . is_optional = true
end
2005-02-23 20:38:10 -05:00
end
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Construct and return a route with the given path and options.
def build ( path , options )
# Wrap the path with slashes
path = " / #{ path } " unless path [ 0 ] == ?/
2007-11-06 03:09:39 -05:00
path = " #{ path } / " unless path [ - 1 ] == ?/
2007-11-26 17:41:28 -05:00
path = " / #{ options [ :path_prefix ] . to_s . gsub ( / ^ \/ / , '' ) } #{ path } " if options [ :path_prefix ]
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
segments = segments_for_route_path ( path )
defaults , requirements , conditions = divide_route_options ( segments , options )
requirements = assign_route_options ( segments , defaults , requirements )
route = Route . new
2007-09-08 20:18:55 -04:00
2006-06-01 11:42:08 -04:00
route . segments = segments
route . requirements = requirements
route . conditions = conditions
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
if ! route . significant_keys . include? ( :action ) && ! route . requirements [ :action ]
route . requirements [ :action ] = " index "
route . significant_keys << :action
end
2007-11-20 16:25:25 -05:00
# Routes cannot use the current string interpolation method
# if there are user-supplied :requirements as the interpolation
# code won't raise RoutingErrors when generating
if options . key? ( :requirements ) || route . requirements . keys . to_set != Routing :: ALLOWED_REQUIREMENTS_FOR_OPTIMISATION
route . optimise = false
end
2006-11-22 11:31:00 -05:00
if ! route . significant_keys . include? ( :controller )
raise ArgumentError , " Illegal route: the :controller must be specified! "
end
2006-06-01 11:42:08 -04:00
route
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
end
2007-12-14 13:12:08 -05:00
class RouteSet #:nodoc:
2006-06-01 11:42:08 -04:00
# Mapper instances are used to build routes. The object passed to the draw
# block in config/routes.rb is a Mapper instance.
2007-11-06 03:09:39 -05:00
#
2006-06-01 11:42:08 -04:00
# Mapper instances have relatively few instance methods, in order to avoid
# clashes with named routes.
2007-12-14 13:12:08 -05:00
class Mapper #:doc:
def initialize ( set ) #:nodoc:
2006-06-01 11:42:08 -04:00
@set = set
end
2007-11-06 03:09:39 -05:00
# Create an unnamed route with the provided +path+ and +options+. See
2007-12-14 13:12:08 -05:00
# ActionController::Routing for an introduction to routes.
2006-06-01 11:42:08 -04:00
def connect ( path , options = { } )
@set . add_route ( path , options )
end
2006-06-01 12:00:15 -04:00
2006-12-03 19:12:00 -05:00
# Creates a named route called "root" for matching the root level request.
def root ( options = { } )
named_route ( " root " , '' , options )
end
2007-12-14 13:12:08 -05:00
def named_route ( name , path , options = { } ) #:nodoc:
2006-06-01 12:00:15 -04:00
@set . add_named_route ( name , path , options )
end
2007-11-06 03:09:39 -05:00
2007-05-19 12:26:44 -04:00
# Enables the use of resources in a module by setting the name_prefix, path_prefix, and namespace for the model.
# Example:
#
# map.namespace(:admin) do |admin|
# admin.resources :products,
# :has_many => [ :tags, :images, :variants ]
# end
#
2007-12-14 13:12:08 -05:00
# This will create +admin_products_url+ pointing to "admin/products", which will look for an Admin::ProductsController.
# It'll also create +admin_product_tags_url+ pointing to "admin/products/#{product_id}/tags", which will look for
2007-05-19 12:26:44 -04:00
# Admin::TagsController.
def namespace ( name , options = { } , & block )
if options [ :namespace ]
with_options ( { :path_prefix = > " #{ options . delete ( :path_prefix ) } / #{ name } " , :name_prefix = > " #{ options . delete ( :name_prefix ) } #{ name } _ " , :namespace = > " #{ options . delete ( :namespace ) } #{ name } / " } . merge ( options ) , & block )
else
with_options ( { :path_prefix = > name , :name_prefix = > " #{ name } _ " , :namespace = > " #{ name } / " } . merge ( options ) , & block )
end
end
2007-11-06 03:09:39 -05:00
2007-12-14 13:12:08 -05:00
def method_missing ( route_name , * args , & proc ) #:nodoc:
2006-06-01 11:42:08 -04:00
super unless args . length > = 1 && proc . nil?
@set . add_named_route ( route_name , * args )
end
2005-06-25 15:05:44 -04:00
end
2006-06-01 11:42:08 -04:00
# A NamedRouteCollection instance is a collection of named routes, and also
# maintains an anonymous module that can be used to install helpers for the
# named routes.
2007-01-26 16:37:38 -05:00
class NamedRouteCollection #:nodoc:
2006-06-01 11:42:08 -04:00
include Enumerable
2007-09-08 20:18:55 -04:00
include ActionController :: Routing :: Optimisation
2007-09-06 10:28:32 -04:00
attr_reader :routes , :helpers
2006-06-01 11:42:08 -04:00
def initialize
clear!
end
def clear!
@routes = { }
@helpers = [ ]
2007-11-06 03:09:39 -05:00
2006-11-17 17:29:02 -05:00
@module || = Module . new
@module . instance_methods . each do | selector |
2007-10-02 01:32:14 -04:00
@module . class_eval { remove_method selector }
2006-11-17 17:29:02 -05:00
end
2006-06-01 11:42:08 -04:00
end
def add ( name , route )
routes [ name . to_sym ] = route
2006-06-29 22:36:17 -04:00
define_named_route_methods ( name , route )
2006-06-01 11:42:08 -04:00
end
def get ( name )
routes [ name . to_sym ]
end
alias [ ] = add
alias [ ] get
alias clear clear!
def each
routes . each { | name , route | yield name , route }
self
end
def names
routes . keys
end
def length
routes . length
end
2007-09-17 05:30:18 -04:00
def reset!
old_routes = routes . dup
clear!
old_routes . each do | name , route |
add ( name , route )
end
end
def install ( destinations = [ ActionController :: Base , ActionView :: Base ] , regenerate = false )
reset! if regenerate
2007-10-02 01:32:14 -04:00
Array ( destinations ) . each do | dest |
dest . send! :include , @module
end
2005-02-14 20:45:35 -05:00
end
2006-04-21 11:17:02 -04:00
2006-06-01 11:42:08 -04:00
private
2006-06-29 22:36:17 -04:00
def url_helper_name ( name , kind = :url )
:" #{ name } _ #{ kind } "
2006-06-01 11:42:08 -04:00
end
2006-06-29 22:36:17 -04:00
def hash_access_name ( name , kind = :url )
:" hash_for_ #{ name } _ #{ kind } "
2006-06-01 11:42:08 -04:00
end
2006-06-29 22:36:17 -04:00
def define_named_route_methods ( name , route )
2006-10-20 14:00:20 -04:00
{ :url = > { :only_path = > false } , :path = > { :only_path = > true } } . each do | kind , opts |
2006-06-29 22:36:17 -04:00
hash = route . defaults . merge ( :use_route = > name ) . merge ( opts )
define_hash_access route , name , kind , hash
define_url_helper route , name , kind , hash
2006-06-01 11:42:08 -04:00
end
end
2007-11-06 03:09:39 -05:00
2006-06-29 22:36:17 -04:00
def define_hash_access ( route , name , kind , options )
selector = hash_access_name ( name , kind )
2007-10-02 01:32:14 -04:00
@module . module_eval <<-end_eval # We use module_eval to avoid leaks
2006-07-17 13:03:47 -04:00
def #{selector}(options = nil)
options ? #{options.inspect}.merge(options) : #{options.inspect}
end
2007-10-02 01:32:14 -04:00
protected : #{selector}
2006-07-17 13:03:47 -04:00
end_eval
2006-06-29 22:36:17 -04:00
helpers << selector
end
2007-09-15 18:10:20 -04:00
2006-06-29 22:36:17 -04:00
def define_url_helper ( route , name , kind , options )
selector = url_helper_name ( name , kind )
# The segment keys used for positional paramters
2007-09-15 18:10:20 -04:00
2006-06-29 22:36:17 -04:00
hash_access_method = hash_access_name ( name , kind )
2007-09-15 18:10:20 -04:00
# allow ordered parameters to be associated with corresponding
# dynamic segments, so you can do
#
# foo_url(bar, baz, bang)
#
# instead of
#
# foo_url(:bar => bar, :baz => baz, :bang => bang)
#
# Also allow options hash, so you can do
#
# foo_url(bar, baz, bang, :sort_by => 'baz')
#
2007-10-02 01:32:14 -04:00
@module . module_eval <<-end_eval # We use module_eval to avoid leaks
2006-07-17 13:03:47 -04:00
def #{selector}(*args)
2007-09-08 20:18:55 -04:00
#{generate_optimisation_block(route, kind)}
2007-09-15 18:10:20 -04:00
2006-07-17 13:03:47 -04:00
opts = if args . empty? || Hash === args . first
args . first || { }
else
2007-12-27 22:58:01 -05:00
options = args . extract_options!
2007-09-08 20:18:55 -04:00
args = args . zip ( #{route.segment_keys.inspect}).inject({}) do |h, (v, k)|
2006-07-17 13:03:47 -04:00
h [ k ] = v
h
end
2007-09-05 19:37:17 -04:00
options . merge ( args )
2005-06-24 12:40:01 -04:00
end
2007-09-15 18:10:20 -04:00
2006-07-17 13:03:47 -04:00
url_for ( #{hash_access_method}(opts))
2005-06-24 12:40:01 -04:00
end
2007-10-02 01:32:14 -04:00
protected : #{selector}
2006-07-17 13:03:47 -04:00
end_eval
2006-06-29 22:36:17 -04:00
helpers << selector
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
end
2007-09-15 18:10:20 -04:00
2006-06-01 11:42:08 -04:00
attr_accessor :routes , :named_routes
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def initialize
self . routes = [ ]
self . named_routes = NamedRouteCollection . new
2005-02-14 20:45:35 -05:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
# Subclasses and plugins may override this method to specify a different
# RouteBuilder instance, so that other route DSL's can be created.
def builder
@builder || = RouteBuilder . new
end
2005-06-24 13:48:14 -04:00
2006-06-01 11:42:08 -04:00
def draw
clear!
yield Mapper . new ( self )
2007-05-12 00:18:46 -04:00
install_helpers
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def clear!
routes . clear
named_routes . clear
@combined_regexp = nil
@routes_by_controller = nil
2005-06-24 12:40:01 -04:00
end
2007-11-06 03:09:39 -05:00
2007-09-17 05:30:18 -04:00
def install_helpers ( destinations = [ ActionController :: Base , ActionView :: Base ] , regenerate_code = false )
2007-10-02 01:32:14 -04:00
Array ( destinations ) . each { | d | d . module_eval { include Helpers } }
2007-09-17 05:30:18 -04:00
named_routes . install ( destinations , regenerate_code )
2007-05-12 00:18:46 -04:00
end
2005-06-24 12:40:01 -04:00
2006-06-01 11:42:08 -04:00
def empty?
routes . empty?
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def load!
2006-09-23 13:25:06 -04:00
Routing . use_controllers! nil # Clear the controller cache so we may discover new ones
2006-06-01 11:42:08 -04:00
clear!
load_routes!
2007-05-12 00:18:46 -04:00
install_helpers
2005-06-24 12:40:01 -04:00
end
2007-08-28 23:01:11 -04:00
# reload! will always force a reload whereas load checks the timestamp first
alias reload! load !
2007-11-06 03:09:39 -05:00
2007-08-28 23:01:11 -04:00
def reload
2007-10-12 23:28:35 -04:00
if @routes_last_modified && defined? ( RAILS_ROOT )
mtime = File . stat ( " #{ RAILS_ROOT } /config/routes.rb " ) . mtime
2007-08-28 23:01:11 -04:00
# if it hasn't been changed, then just return
return if mtime == @routes_last_modified
# if it has changed then record the new time and fall to the load! below
2007-10-12 23:28:35 -04:00
@routes_last_modified = mtime
2007-08-28 23:01:11 -04:00
end
load !
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def load_routes!
if defined? ( RAILS_ROOT ) && defined? ( :: ActionController :: Routing :: Routes ) && self == :: ActionController :: Routing :: Routes
load File . join ( " #{ RAILS_ROOT } /config/routes.rb " )
2007-10-12 23:28:35 -04:00
@routes_last_modified = File . stat ( " #{ RAILS_ROOT } /config/routes.rb " ) . mtime
2006-06-01 11:42:08 -04:00
else
add_route " :controller/:action/:id "
2005-02-14 20:45:35 -05:00
end
2007-08-28 23:14:15 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def add_route ( path , options = { } )
route = builder . build ( path , options )
routes << route
2005-06-24 12:40:01 -04:00
route
2005-02-14 20:45:35 -05:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def add_named_route ( name , path , options = { } )
2007-06-27 04:38:55 -04:00
# TODO - is options EVER used?
2007-05-19 12:26:44 -04:00
name = options [ :name_prefix ] + name . to_s if options [ :name_prefix ]
named_routes [ name . to_sym ] = add_route ( path , options )
2005-07-14 06:32:37 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def options_as_params ( options )
# If an explicit :controller was given, always make :action explicit
# too, so that action expiry works as expected for things like
#
# generate({:controller => 'content'}, {:controller => 'content', :action => 'show'})
#
# (the above is from the unit tests). In the above case, because the
# controller was explicitly given, but no action, the action is implied to
# be "index", not the recalled action of "show".
#
# great fun, eh?
2005-06-29 06:40:26 -04:00
2007-03-06 02:47:23 -05:00
options_as_params = options . clone
options_as_params [ :action ] || = 'index' if options [ :controller ]
options_as_params [ :action ] = options_as_params [ :action ] . to_s if options_as_params [ :action ]
2006-06-01 11:42:08 -04:00
options_as_params
2005-06-29 06:40:26 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def build_expiry ( options , recall )
recall . inject ( { } ) do | expiry , ( key , recalled_value ) |
2007-03-26 15:02:19 -04:00
expiry [ key ] = ( options . key? ( key ) && options [ key ] . to_param != recalled_value . to_param )
2006-06-01 11:42:08 -04:00
expiry
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
end
2005-06-24 12:40:01 -04:00
2006-06-01 11:42:08 -04:00
# Generate the path indicated by the arguments, and return an array of
# the keys that were not used to generate it.
def extra_keys ( options , recall = { } )
generate_extras ( options , recall ) . last
end
def generate_extras ( options , recall = { } )
generate ( options , recall , :generate_extras )
end
def generate ( options , recall = { } , method = :generate )
2006-08-08 20:02:08 -04:00
named_route_name = options . delete ( :use_route )
2007-01-28 12:29:51 -05:00
generate_all = options . delete ( :generate_all )
2006-08-08 20:02:08 -04:00
if named_route_name
named_route = named_routes [ named_route_name ]
2006-06-01 11:42:08 -04:00
options = named_route . parameter_shell . merge ( options )
2005-06-24 12:40:01 -04:00
end
2006-06-01 11:42:08 -04:00
options = options_as_params ( options )
expire_on = build_expiry ( options , recall )
2007-10-07 15:12:02 -04:00
if options [ :controller ]
options [ :controller ] = options [ :controller ] . to_s
end
2006-06-01 11:42:08 -04:00
# if the controller has changed, make sure it changes relative to the
# current controller module, if any. In other words, if we're currently
# on admin/get, and the new controller is 'set', the new controller
# should really be admin/set.
2006-06-06 15:09:56 -04:00
if ! named_route && expire_on [ :controller ] && options [ :controller ] && options [ :controller ] [ 0 ] != ?/
2006-06-05 11:48:29 -04:00
old_parts = recall [ :controller ] . split ( '/' )
new_parts = options [ :controller ] . split ( '/' )
parts = old_parts [ 0 .. - ( new_parts . length + 1 ) ] + new_parts
2006-06-01 11:42:08 -04:00
options [ :controller ] = parts . join ( '/' )
2005-07-13 19:13:15 -04:00
end
2006-06-01 11:42:08 -04:00
# drop the leading '/' on the controller name
options [ :controller ] = options [ :controller ] [ 1 .. - 1 ] if options [ :controller ] && options [ :controller ] [ 0 ] == ?/
merged = recall . merge ( options )
2006-12-08 23:42:32 -05:00
2006-06-01 11:42:08 -04:00
if named_route
2006-08-13 14:00:08 -04:00
path = named_route . generate ( options , merged , expire_on )
2007-11-06 03:09:39 -05:00
if path . nil?
2007-02-04 14:07:08 -05:00
raise_named_route_error ( options , named_route , named_route_name )
else
return path
end
2006-06-01 11:42:08 -04:00
else
merged [ :action ] || = 'index'
options [ :action ] || = 'index'
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
controller = merged [ :controller ]
action = merged [ :action ]
2006-08-08 20:02:08 -04:00
raise RoutingError , " Need controller and action! " unless controller && action
2007-11-06 03:09:39 -05:00
2007-01-28 12:29:51 -05:00
if generate_all
# Used by caching to expire all paths for a resource
return routes . collect do | route |
2007-10-02 01:32:14 -04:00
route . send! ( method , options , merged , expire_on )
2007-01-28 12:29:51 -05:00
end . compact
end
2007-11-06 03:09:39 -05:00
2006-06-06 13:59:54 -04:00
# don't use the recalled keys when determining which routes to check
routes = routes_by_controller [ controller ] [ action ] [ options . keys . sort_by { | x | x . object_id } ]
2006-06-01 11:42:08 -04:00
routes . each do | route |
2007-10-02 01:32:14 -04:00
results = route . send! ( method , options , merged , expire_on )
2006-10-16 15:52:21 -04:00
return results if results && ( ! results . is_a? ( Array ) || results . first )
2005-09-11 09:45:55 -04:00
end
2005-07-13 19:13:15 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
raise RoutingError , " No route matches #{ options . inspect } "
end
2007-11-06 03:09:39 -05:00
2007-02-04 14:07:08 -05:00
# try to give a helpful error message when named route generation fails
def raise_named_route_error ( options , named_route , named_route_name )
diff = named_route . requirements . diff ( options )
unless diff . empty?
raise RoutingError , " #{ named_route_name } _url failed to generate from #{ options . inspect } , expected: #{ named_route . requirements . inspect } , diff: #{ named_route . requirements . diff ( options ) . inspect } "
else
required_segments = named_route . segments . select { | seg | ( ! seg . optional? ) && ( ! seg . is_a? ( DividerSegment ) ) }
required_keys_or_values = required_segments . map { | seg | seg . key rescue seg . value } # we want either the key or the value from the segment
2007-05-06 00:26:49 -04:00
raise RoutingError , " #{ named_route_name } _url failed to generate from #{ options . inspect } - you may have ambiguous routes, or you may need to supply additional parameters for this route. content_url has the following required parameters: #{ required_keys_or_values . inspect } - are they all satisfied? "
2007-02-04 14:07:08 -05:00
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def recognize ( request )
params = recognize_path ( request . path , extract_request_environment ( request ) )
request . path_parameters = params . with_indifferent_access
" #{ params [ :controller ] . camelize } Controller " . constantize
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def recognize_path ( path , environment = { } )
routes . each do | route |
result = route . recognize ( path , environment ) and return result
end
2007-05-26 16:07:34 -04:00
allows = HTTP_METHODS . select { | verb | routes . find { | r | r . recognize ( path , :method = > verb ) } }
if environment [ :method ] && ! HTTP_METHODS . include? ( environment [ :method ] )
raise NotImplemented . new ( * allows )
elsif ! allows . empty?
raise MethodNotAllowed . new ( * allows )
else
raise RoutingError , " No route matches #{ path . inspect } with #{ environment . inspect } "
end
2006-06-01 11:42:08 -04:00
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def routes_by_controller
@routes_by_controller || = Hash . new do | controller_hash , controller |
controller_hash [ controller ] = Hash . new do | action_hash , action |
action_hash [ action ] = Hash . new do | key_hash , keys |
key_hash [ keys ] = routes_for_controller_and_action_and_keys ( controller , action , keys )
2005-06-24 12:40:01 -04:00
end
end
end
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def routes_for ( options , merged , expire_on )
raise " Need controller and action! " unless controller && action
controller = merged [ :controller ]
merged = options if expire_on [ :controller ]
action = merged [ :action ] || 'index'
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
routes_by_controller [ controller ] [ action ] [ merged . keys ]
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def routes_for_controller_and_action ( controller , action )
selected = routes . select do | route |
route . matches_controller_and_action? controller , action
end
( selected . length == routes . length ) ? routes : selected
end
2007-11-06 03:09:39 -05:00
2006-06-01 11:42:08 -04:00
def routes_for_controller_and_action_and_keys ( controller , action , keys )
selected = routes . select do | route |
route . matches_controller_and_action? controller , action
end
selected . sort_by do | route |
( keys - route . significant_keys ) . length
end
end
# Subclasses and plugins may override this method to extract further attributes
# from the request, for use by route conditions and such.
def extract_request_environment ( request )
{ :method = > request . method }
end
2005-03-20 10:42:40 -05:00
end
2005-02-14 20:45:35 -05:00
Routes = RouteSet . new
2007-10-12 23:28:35 -04:00
:: Inflector . module_eval do
def inflections_with_route_reloading ( & block )
returning ( inflections_without_route_reloading ( & block ) ) {
ActionController :: Routing :: Routes . reload! if block_given?
}
end
alias_method_chain :inflections , :route_reloading
end
2005-02-14 20:45:35 -05:00
end
2007-09-15 18:10:20 -04:00
end