2009-10-20 00:32:06 -04:00
|
|
|
require 'rack/mount'
|
|
|
|
require 'forwardable'
|
2010-04-16 08:44:08 -04:00
|
|
|
require 'active_support/core_ext/object/to_query'
|
2010-02-28 17:39:01 -05:00
|
|
|
require 'action_dispatch/routing/deprecated_mapper'
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2009-10-20 11:14:46 -04:00
|
|
|
module ActionDispatch
|
2008-01-17 17:45:54 -05:00
|
|
|
module Routing
|
2008-07-28 14:38:20 -04:00
|
|
|
class RouteSet #:nodoc:
|
2009-10-20 00:32:06 -04:00
|
|
|
PARAMETERS_KEY = 'action_dispatch.request.path_parameters'
|
|
|
|
|
2010-03-31 09:54:07 -04:00
|
|
|
class Dispatcher #:nodoc:
|
2010-02-16 18:14:49 -05:00
|
|
|
def initialize(options={})
|
|
|
|
@defaults = options[:defaults]
|
2009-10-20 00:32:06 -04:00
|
|
|
@glob_param = options.delete(:glob)
|
2010-06-04 20:28:04 -04:00
|
|
|
@controllers = {}
|
2009-10-20 00:32:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def call(env)
|
|
|
|
params = env[PARAMETERS_KEY]
|
2009-12-11 01:11:16 -05:00
|
|
|
prepare_params!(params)
|
2010-02-16 18:14:49 -05:00
|
|
|
|
|
|
|
# Just raise undefined constant errors if a controller was specified as default.
|
|
|
|
unless controller = controller(params, @defaults.key?(:controller))
|
|
|
|
return [404, {'X-Cascade' => 'pass'}, []]
|
|
|
|
end
|
|
|
|
|
|
|
|
controller.action(params[:action]).call(env)
|
2009-12-11 01:11:16 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def prepare_params!(params)
|
2009-10-20 00:32:06 -04:00
|
|
|
merge_default_action!(params)
|
|
|
|
split_glob_param!(params) if @glob_param
|
2009-12-11 01:11:16 -05:00
|
|
|
end
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-02-16 18:14:49 -05:00
|
|
|
def controller(params, raise_error=true)
|
2010-06-04 20:28:04 -04:00
|
|
|
if params && params.key?(:controller)
|
|
|
|
controller_param = params[:controller]
|
|
|
|
unless controller = @controllers[controller_param]
|
|
|
|
controller_name = "#{controller_param.camelize}Controller"
|
|
|
|
controller = @controllers[controller_param] =
|
|
|
|
ActiveSupport::Dependencies.ref(controller_name)
|
|
|
|
end
|
|
|
|
|
|
|
|
controller.get
|
2009-10-20 00:32:06 -04:00
|
|
|
end
|
2010-02-06 05:39:51 -05:00
|
|
|
rescue NameError => e
|
2010-02-16 18:14:49 -05:00
|
|
|
raise ActionController::RoutingError, e.message, e.backtrace if raise_error
|
2009-10-20 00:32:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
def merge_default_action!(params)
|
|
|
|
params[:action] ||= 'index'
|
|
|
|
end
|
|
|
|
|
|
|
|
def split_glob_param!(params)
|
|
|
|
params[@glob_param] = params[@glob_param].split('/').map { |v| URI.unescape(v) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2008-01-17 17:45:54 -05: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.
|
|
|
|
class NamedRouteCollection #:nodoc:
|
|
|
|
include Enumerable
|
2010-03-17 17:04:41 -04:00
|
|
|
attr_reader :routes, :helpers, :module
|
2008-01-17 17:45:54 -05:00
|
|
|
|
|
|
|
def initialize
|
|
|
|
clear!
|
|
|
|
end
|
|
|
|
|
2010-06-07 15:29:34 -04:00
|
|
|
def helper_names
|
|
|
|
self.module.instance_methods.map(&:to_s)
|
|
|
|
end
|
|
|
|
|
2008-01-17 17:45:54 -05:00
|
|
|
def clear!
|
|
|
|
@routes = {}
|
|
|
|
@helpers = []
|
|
|
|
|
2010-01-07 09:26:31 -05:00
|
|
|
@module ||= Module.new do
|
|
|
|
instance_methods.each { |selector| remove_method(selector) }
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def add(name, route)
|
|
|
|
routes[name.to_sym] = route
|
|
|
|
define_named_route_methods(name, route)
|
|
|
|
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
|
|
|
|
|
|
|
|
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
|
|
|
|
Array(destinations).each do |dest|
|
2008-08-31 16:15:26 -04:00
|
|
|
dest.__send__(:include, @module)
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
def url_helper_name(name, kind = :url)
|
|
|
|
:"#{name}_#{kind}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def hash_access_name(name, kind = :url)
|
|
|
|
:"hash_for_#{name}_#{kind}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def define_named_route_methods(name, route)
|
|
|
|
{:url => {:only_path => false}, :path => {:only_path => true}}.each do |kind, opts|
|
|
|
|
hash = route.defaults.merge(:use_route => name).merge(opts)
|
|
|
|
define_hash_access route, name, kind, hash
|
|
|
|
define_url_helper route, name, kind, hash
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def define_hash_access(route, name, kind, options)
|
|
|
|
selector = hash_access_name(name, kind)
|
2010-01-06 03:32:29 -05:00
|
|
|
|
|
|
|
# We use module_eval to avoid leaks
|
|
|
|
@module.module_eval <<-END_EVAL, __FILE__, __LINE__ + 1
|
2008-12-28 14:48:05 -05:00
|
|
|
def #{selector}(options = nil) # def hash_for_users_url(options = nil)
|
|
|
|
options ? #{options.inspect}.merge(options) : #{options.inspect} # options ? {:only_path=>false}.merge(options) : {:only_path=>false}
|
|
|
|
end # end
|
|
|
|
protected :#{selector} # protected :hash_for_users_url
|
2010-01-06 03:32:29 -05:00
|
|
|
END_EVAL
|
2008-01-17 17:45:54 -05:00
|
|
|
helpers << selector
|
|
|
|
end
|
|
|
|
|
2010-01-06 03:32:29 -05:00
|
|
|
# Create a url helper allowing 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')
|
|
|
|
#
|
2008-01-17 17:45:54 -05:00
|
|
|
def define_url_helper(route, name, kind, options)
|
|
|
|
selector = url_helper_name(name, kind)
|
|
|
|
hash_access_method = hash_access_name(name, kind)
|
|
|
|
|
2010-01-06 03:32:29 -05:00
|
|
|
@module.module_eval <<-END_EVAL, __FILE__, __LINE__ + 1
|
2010-01-07 09:26:31 -05:00
|
|
|
def #{selector}(*args)
|
2010-02-25 18:05:10 -05:00
|
|
|
options = #{hash_access_method}(args.extract_options!)
|
|
|
|
|
|
|
|
if args.any?
|
|
|
|
options[:_positional_args] = args
|
|
|
|
options[:_positional_keys] = #{route.segment_keys.inspect}
|
2010-01-07 09:26:31 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
url_for(options)
|
|
|
|
end
|
2010-01-06 03:32:29 -05:00
|
|
|
END_EVAL
|
2008-01-17 17:45:54 -05:00
|
|
|
helpers << selector
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-06-27 03:03:39 -04:00
|
|
|
attr_accessor :set, :routes, :named_routes
|
2010-01-24 05:06:06 -05:00
|
|
|
attr_accessor :disable_clear_and_finalize, :resources_path_names
|
2010-04-03 23:23:23 -04:00
|
|
|
attr_accessor :default_url_options, :request_class
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-01-13 21:21:04 -05:00
|
|
|
def self.default_resources_path_names
|
|
|
|
{ :new => 'new', :edit => 'edit' }
|
|
|
|
end
|
|
|
|
|
2010-04-03 23:23:23 -04:00
|
|
|
def initialize(request_class = ActionDispatch::Request)
|
2008-01-17 17:45:54 -05:00
|
|
|
self.routes = []
|
|
|
|
self.named_routes = NamedRouteCollection.new
|
2010-01-24 05:06:06 -05:00
|
|
|
self.resources_path_names = self.class.default_resources_path_names.dup
|
|
|
|
self.controller_namespaces = Set.new
|
2010-03-09 22:20:13 -05:00
|
|
|
self.default_url_options = {}
|
2010-04-03 23:23:23 -04:00
|
|
|
self.request_class = request_class
|
2008-07-28 14:41:42 -04:00
|
|
|
|
2009-12-14 18:54:41 -05:00
|
|
|
@disable_clear_and_finalize = false
|
2010-03-04 22:01:48 -05:00
|
|
|
clear!
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2010-03-05 00:59:42 -05:00
|
|
|
def draw(&block)
|
2009-12-14 18:54:41 -05:00
|
|
|
clear! unless @disable_clear_and_finalize
|
2009-12-09 21:46:32 -05:00
|
|
|
|
|
|
|
mapper = Mapper.new(self)
|
|
|
|
if block.arity == 1
|
|
|
|
mapper.instance_exec(DeprecatedMapper.new(self), &block)
|
|
|
|
else
|
|
|
|
mapper.instance_exec(&block)
|
|
|
|
end
|
|
|
|
|
2009-12-14 18:54:41 -05:00
|
|
|
finalize! unless @disable_clear_and_finalize
|
2009-12-14 17:51:13 -05:00
|
|
|
|
|
|
|
nil
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2009-12-14 18:54:41 -05:00
|
|
|
def finalize!
|
2010-03-16 15:38:37 -04:00
|
|
|
return if @finalized
|
|
|
|
@finalized = true
|
2009-12-14 18:54:41 -05:00
|
|
|
@set.freeze
|
|
|
|
end
|
|
|
|
|
2008-01-17 17:45:54 -05:00
|
|
|
def clear!
|
2009-12-14 18:54:41 -05:00
|
|
|
# Clear the controller cache so we may discover new ones
|
|
|
|
@controller_constraints = nil
|
2010-03-16 15:38:37 -04:00
|
|
|
@finalized = false
|
2008-01-17 17:45:54 -05:00
|
|
|
routes.clear
|
|
|
|
named_routes.clear
|
2010-04-03 23:23:23 -04:00
|
|
|
@set = ::Rack::Mount::RouteSet.new(
|
|
|
|
:parameters_key => PARAMETERS_KEY,
|
|
|
|
:request_class => request_class
|
|
|
|
)
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def install_helpers(destinations = [ActionController::Base, ActionView::Base], regenerate_code = false)
|
|
|
|
Array(destinations).each { |d| d.module_eval { include Helpers } }
|
|
|
|
named_routes.install(destinations, regenerate_code)
|
|
|
|
end
|
|
|
|
|
2010-02-26 18:00:33 -05:00
|
|
|
def url_helpers
|
|
|
|
@url_helpers ||= begin
|
2010-03-17 17:04:41 -04:00
|
|
|
routes = self
|
2010-02-24 19:01:03 -05:00
|
|
|
|
2010-03-17 17:04:41 -04:00
|
|
|
helpers = Module.new do
|
2010-02-24 19:01:03 -05:00
|
|
|
extend ActiveSupport::Concern
|
2010-03-09 21:57:43 -05:00
|
|
|
include UrlFor
|
2010-02-24 19:01:03 -05:00
|
|
|
|
2010-03-17 17:04:41 -04:00
|
|
|
@routes = routes
|
|
|
|
class << self
|
|
|
|
delegate :url_for, :to => '@routes'
|
|
|
|
end
|
|
|
|
extend routes.named_routes.module
|
|
|
|
|
2010-02-24 19:01:03 -05:00
|
|
|
# ROUTES TODO: install_helpers isn't great... can we make a module with the stuff that
|
|
|
|
# we can include?
|
2010-03-09 21:57:43 -05:00
|
|
|
# Yes plz - JP
|
2010-02-24 19:01:03 -05:00
|
|
|
included do
|
2010-03-17 17:04:41 -04:00
|
|
|
routes.install_helpers(self)
|
2010-03-19 01:21:25 -04:00
|
|
|
singleton_class.send(:define_method, :_router) { routes }
|
2010-02-24 19:01:03 -05:00
|
|
|
end
|
2010-03-09 21:57:43 -05:00
|
|
|
|
2010-03-17 17:04:41 -04:00
|
|
|
define_method(:_router) { routes }
|
2010-02-24 19:01:03 -05:00
|
|
|
end
|
2010-03-17 17:04:41 -04:00
|
|
|
|
|
|
|
helpers
|
2010-02-24 19:01:03 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2008-01-17 17:45:54 -05:00
|
|
|
def empty?
|
|
|
|
routes.empty?
|
|
|
|
end
|
|
|
|
|
2010-03-08 15:24:49 -05:00
|
|
|
def add_route(app, conditions = {}, requirements = {}, defaults = {}, name = nil, anchor = true)
|
|
|
|
route = Route.new(app, conditions, requirements, defaults, name, anchor)
|
2009-10-24 19:08:54 -04:00
|
|
|
@set.add_route(*route)
|
2009-10-20 11:46:27 -04:00
|
|
|
named_routes[name] = route if name
|
2008-01-17 17:45:54 -05:00
|
|
|
routes << route
|
|
|
|
route
|
|
|
|
end
|
|
|
|
|
2010-03-31 09:54:07 -04:00
|
|
|
class Generator #:nodoc:
|
2010-03-09 13:20:48 -05:00
|
|
|
attr_reader :options, :recall, :set, :script_name, :named_route
|
|
|
|
|
|
|
|
def initialize(options, recall, set, extras = false)
|
|
|
|
@script_name = options.delete(:script_name)
|
|
|
|
@named_route = options.delete(:use_route)
|
|
|
|
@options = options.dup
|
|
|
|
@recall = recall.dup
|
|
|
|
@set = set
|
|
|
|
@extras = extras
|
|
|
|
|
|
|
|
normalize_options!
|
2010-06-27 03:03:39 -04:00
|
|
|
normalize_recall!
|
2010-03-09 13:20:48 -05:00
|
|
|
normalize_controller_action_id!
|
|
|
|
use_relative_controller!
|
|
|
|
controller.sub!(%r{^/}, '') if controller
|
|
|
|
handle_nil_action!
|
|
|
|
end
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def controller
|
|
|
|
@controller ||= @options[:controller]
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def current_controller
|
|
|
|
@recall[:controller]
|
|
|
|
end
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def use_recall_for(key)
|
|
|
|
if @recall[key] && (!@options.key?(key) || @options[key] == @recall[key])
|
|
|
|
@options[key] = @recall.delete(key)
|
|
|
|
end
|
|
|
|
end
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def normalize_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".
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
if options[:controller]
|
|
|
|
options[:action] ||= 'index'
|
|
|
|
options[:controller] = options[:controller].to_s
|
|
|
|
end
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
if options[:action]
|
|
|
|
options[:action] = options[:action].to_s
|
|
|
|
end
|
|
|
|
end
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-06-27 03:03:39 -04:00
|
|
|
def normalize_recall!
|
|
|
|
# If the target route is not a standard route then remove controller and action
|
|
|
|
# from the options otherwise they will appear in the url parameters
|
|
|
|
if block_or_proc_route_target?
|
|
|
|
recall.delete(:controller) unless segment_keys.include?(:controller)
|
|
|
|
recall.delete(:action) unless segment_keys.include?(:action)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
# This pulls :controller, :action, and :id out of the recall.
|
|
|
|
# The recall key is only used if there is no key in the options
|
|
|
|
# or if the key in the options is identical. If any of
|
|
|
|
# :controller, :action or :id is not found, don't pull any
|
|
|
|
# more keys from the recall.
|
|
|
|
def normalize_controller_action_id!
|
|
|
|
@recall[:action] ||= 'index' if current_controller
|
|
|
|
|
|
|
|
use_recall_for(:controller) or return
|
|
|
|
use_recall_for(:action) or return
|
|
|
|
use_recall_for(:id)
|
|
|
|
end
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
# if the current controller is "foo/bar/baz" and :controller => "baz/bat"
|
|
|
|
# is specified, the controller becomes "foo/baz/bat"
|
|
|
|
def use_relative_controller!
|
|
|
|
if !named_route && different_controller?
|
|
|
|
old_parts = current_controller.split('/')
|
|
|
|
size = controller.count("/") + 1
|
|
|
|
parts = old_parts[0...-size] << controller
|
|
|
|
@controller = @options[:controller] = parts.join("/")
|
|
|
|
end
|
|
|
|
end
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
# This handles the case of :action => nil being explicitly passed.
|
|
|
|
# It is identical to :action => "index"
|
|
|
|
def handle_nil_action!
|
|
|
|
if options.has_key?(:action) && options[:action].nil?
|
|
|
|
options[:action] = 'index'
|
2009-10-20 00:32:06 -04:00
|
|
|
end
|
2010-03-09 13:20:48 -05:00
|
|
|
recall[:action] = options.delete(:action) if options[:action] == 'index'
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def generate
|
|
|
|
error = ActionController::RoutingError.new("No route matches #{options.inspect}")
|
2010-06-27 03:03:39 -04:00
|
|
|
path, params = @set.set.generate(:path_info, named_route, options, recall, opts)
|
2009-10-20 00:32:06 -04:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
raise error unless path
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
params.reject! {|k,v| !v }
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
return [path, params.keys] if @extras
|
2009-11-14 23:00:38 -05:00
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
path << "?#{params.to_query}" if params.any?
|
|
|
|
"#{script_name}#{path}"
|
|
|
|
rescue Rack::Mount::RoutingError
|
|
|
|
raise error
|
2009-12-02 13:33:33 -05:00
|
|
|
end
|
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def opts
|
|
|
|
parameterize = lambda do |name, value|
|
|
|
|
if name == :controller
|
|
|
|
value
|
|
|
|
elsif value.is_a?(Array)
|
|
|
|
value.map { |v| Rack::Mount::Utils.escape_uri(v.to_param) }.join('/')
|
|
|
|
else
|
|
|
|
Rack::Mount::Utils.escape_uri(value.to_param)
|
|
|
|
end
|
2009-12-02 13:33:33 -05:00
|
|
|
end
|
2010-03-09 13:20:48 -05:00
|
|
|
{:parameterize => parameterize}
|
2009-12-02 13:33:33 -05:00
|
|
|
end
|
|
|
|
|
2010-03-09 13:20:48 -05:00
|
|
|
def different_controller?
|
|
|
|
return false unless current_controller
|
|
|
|
controller.to_param != current_controller.to_param
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
2010-06-27 03:03:39 -04:00
|
|
|
|
|
|
|
private
|
|
|
|
def named_route_exists?
|
|
|
|
named_route && set.named_routes[named_route]
|
|
|
|
end
|
|
|
|
|
|
|
|
def block_or_proc_route_target?
|
|
|
|
named_route_exists? && !set.named_routes[named_route].app.is_a?(Dispatcher)
|
|
|
|
end
|
|
|
|
|
|
|
|
def segment_keys
|
|
|
|
named_route_exists? ? set.named_routes[named_route].segment_keys : []
|
|
|
|
end
|
2010-03-09 13:20:48 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
# 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, true)
|
|
|
|
end
|
|
|
|
|
|
|
|
def generate(options, recall = {}, extras = false)
|
2010-06-27 03:03:39 -04:00
|
|
|
Generator.new(options, recall, self, extras).generate
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2010-03-09 22:25:09 -05:00
|
|
|
RESERVED_OPTIONS = [:anchor, :params, :only_path, :host, :protocol, :port, :trailing_slash]
|
2010-03-09 21:50:35 -05:00
|
|
|
|
2010-03-09 22:00:24 -05:00
|
|
|
def url_for(options)
|
2010-03-16 15:38:37 -04:00
|
|
|
finalize!
|
2010-03-09 22:20:13 -05:00
|
|
|
options = default_url_options.merge(options || {})
|
|
|
|
|
2010-03-09 21:50:35 -05:00
|
|
|
handle_positional_args(options)
|
|
|
|
|
|
|
|
rewritten_url = ""
|
|
|
|
|
|
|
|
path_segments = options.delete(:_path_segments)
|
|
|
|
|
|
|
|
unless options[:only_path]
|
|
|
|
rewritten_url << (options[:protocol] || "http")
|
|
|
|
rewritten_url << "://" unless rewritten_url.match("://")
|
|
|
|
rewritten_url << rewrite_authentication(options)
|
|
|
|
|
|
|
|
raise "Missing host to link to! Please provide :host parameter or set default_url_options[:host]" unless options[:host]
|
|
|
|
|
|
|
|
rewritten_url << options[:host]
|
|
|
|
rewritten_url << ":#{options.delete(:port)}" if options.key?(:port)
|
|
|
|
end
|
|
|
|
|
|
|
|
path_options = options.except(*RESERVED_OPTIONS)
|
|
|
|
path_options = yield(path_options) if block_given?
|
|
|
|
path = generate(path_options, path_segments || {})
|
|
|
|
|
|
|
|
# ROUTES TODO: This can be called directly, so script_name should probably be set in the router
|
|
|
|
rewritten_url << (options[:trailing_slash] ? path.sub(/\?|\z/) { "/" + $& } : path)
|
2010-06-25 20:54:13 -04:00
|
|
|
rewritten_url << "##{Rack::Mount::Utils.escape_uri(options[:anchor].to_param.to_s)}" if options[:anchor]
|
2010-03-09 21:50:35 -05:00
|
|
|
|
|
|
|
rewritten_url
|
|
|
|
end
|
|
|
|
|
2009-10-20 00:32:06 -04:00
|
|
|
def call(env)
|
2010-03-16 15:38:37 -04:00
|
|
|
finalize!
|
2009-10-20 00:32:06 -04:00
|
|
|
@set.call(env)
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2009-12-11 01:01:22 -05:00
|
|
|
def recognize_path(path, environment = {})
|
2009-10-20 00:32:06 -04:00
|
|
|
method = (environment[:method] || "GET").to_s.upcase
|
2009-12-11 01:11:16 -05:00
|
|
|
path = Rack::Mount::Utils.normalize_path(path)
|
2008-01-17 17:45:54 -05:00
|
|
|
|
2009-10-20 00:32:06 -04:00
|
|
|
begin
|
|
|
|
env = Rack::MockRequest.env_for(path, {:method => method})
|
|
|
|
rescue URI::InvalidURIError => e
|
2009-10-20 11:14:46 -04:00
|
|
|
raise ActionController::RoutingError, e.message
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
|
2009-12-11 01:11:16 -05:00
|
|
|
req = Rack::Request.new(env)
|
2010-02-15 12:16:56 -05:00
|
|
|
@set.recognize(req) do |route, matches, params|
|
2010-06-04 14:49:55 -04:00
|
|
|
params.each do |key, value|
|
|
|
|
if value.is_a?(String)
|
|
|
|
value = value.dup.force_encoding(Encoding::BINARY) if value.encoding_aware?
|
|
|
|
params[key] = URI.unescape(value)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2009-12-11 01:11:16 -05:00
|
|
|
dispatcher = route.app
|
2010-02-16 18:14:49 -05:00
|
|
|
if dispatcher.is_a?(Dispatcher) && dispatcher.controller(params, false)
|
2009-12-11 01:11:16 -05:00
|
|
|
dispatcher.prepare_params!(params)
|
|
|
|
return params
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2009-12-21 18:34:53 -05:00
|
|
|
raise ActionController::RoutingError, "No route matches #{path.inspect}"
|
2009-08-09 23:38:50 -04:00
|
|
|
end
|
2010-03-09 21:50:35 -05:00
|
|
|
|
|
|
|
private
|
|
|
|
def handle_positional_args(options)
|
|
|
|
return unless args = options.delete(:_positional_args)
|
|
|
|
|
|
|
|
keys = options.delete(:_positional_keys)
|
|
|
|
keys -= options.keys if args.size < keys.size - 1 # take format into account
|
|
|
|
|
|
|
|
args = args.zip(keys).inject({}) do |h, (v, k)|
|
|
|
|
h[k] = v
|
|
|
|
h
|
|
|
|
end
|
|
|
|
|
|
|
|
# Tell url_for to skip default_url_options
|
|
|
|
options.merge!(args)
|
|
|
|
end
|
|
|
|
|
|
|
|
def rewrite_authentication(options)
|
|
|
|
if options[:user] && options[:password]
|
|
|
|
"#{Rack::Utils.escape(options.delete(:user))}:#{Rack::Utils.escape(options.delete(:password))}@"
|
|
|
|
else
|
|
|
|
""
|
|
|
|
end
|
|
|
|
end
|
2008-01-17 17:45:54 -05:00
|
|
|
end
|
|
|
|
end
|
2008-07-28 14:38:20 -04:00
|
|
|
end
|