1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00
rails--rails/actionpack/lib/action_dispatch/routing/mapper.rb

340 lines
9.9 KiB
Ruby
Raw Normal View History

2009-10-20 11:14:46 -04:00
module ActionDispatch
module Routing
2009-10-20 13:31:23 -04:00
class Mapper
module Resources
def resource(*resources, &block)
options = resources.last.is_a?(Hash) ? resources.pop : {}
if resources.length > 1
raise ArgumentError if block_given?
resources.each { |r| resource(r, options) }
return self
end
2009-10-20 13:31:23 -04:00
resource = resources.pop
2009-10-20 13:31:23 -04:00
if @scope[:scope_level] == :resources
member do
resource(resource, options, &block)
end
2009-10-20 13:31:23 -04:00
return self
end
singular = resource.to_s
plural = singular.pluralize
controller(plural) do
2009-10-20 13:31:23 -04:00
namespace(resource) do
with_scope_level(:resource) do
yield if block_given?
get "", :to => :show, :as => "#{singular}"
2009-10-20 13:31:23 -04:00
post "", :to => :create
put "", :to => :update
delete "", :to => :destroy
get "new", :to => :new, :as => "new_#{singular}"
get "edit", :to => :edit, :as => "edit_#{singular}"
2009-10-20 13:31:23 -04:00
end
end
end
2009-10-20 13:31:23 -04:00
self
end
2009-10-20 13:31:23 -04:00
def resources(*resources, &block)
options = resources.last.is_a?(Hash) ? resources.pop : {}
2009-10-20 13:31:23 -04:00
if resources.length > 1
raise ArgumentError if block_given?
resources.each { |r| resources(r, options) }
return self
end
2009-10-20 13:31:23 -04:00
resource = resources.pop
plural = resource.to_s
singular = plural.singularize
2009-10-20 13:31:23 -04:00
if @scope[:scope_level] == :resources
parent_resource = @scope[:scope_level_options][:name]
with_scope_level(:member) do
scope(":#{parent_resource}_id", :name_prefix => parent_resource) do
resources(resource, options, &block)
end
end
2009-10-20 13:31:23 -04:00
return self
end
if @scope[:options] && (prefix = @scope[:options][:name_prefix])
plural = "#{prefix}_#{plural}"
singular = "#{prefix}_#{singular}"
end
2009-10-20 13:31:23 -04:00
controller(resource) do
namespace(resource) do
with_scope_level(:resources, :name => singular) do
2009-10-20 13:31:23 -04:00
yield if block_given?
member do
get "", :to => :show, :as => singular
2009-10-20 13:31:23 -04:00
put "", :to => :update
delete "", :to => :destroy
get "edit", :to => :edit, :as => "edit_#{singular}"
2009-10-20 13:31:23 -04:00
end
collection do
get "", :to => :index, :as => plural
2009-10-20 13:31:23 -04:00
post "", :to => :create
get "new", :to => :new, :as => "new_#{singular}"
2009-10-20 13:31:23 -04:00
end
end
end
end
2009-10-20 13:31:23 -04:00
self
end
2009-10-20 13:31:23 -04:00
def collection
unless @scope[:scope_level] == :resources
raise ArgumentError, "can't use collection outside resources scope"
end
2009-10-20 13:31:23 -04:00
with_scope_level(:collection) do
yield
end
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
def member
unless @scope[:scope_level] == :resources
raise ArgumentError, "can't use member outside resources scope"
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
with_scope_level(:member) do
scope(":id") do
yield
end
end
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def match(*args)
options = args.last.is_a?(Hash) ? args.pop : {}
args.push(options)
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
case options.delete(:on)
when :collection
return collection { match(*args) }
when :member
return member { match(*args) }
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
if @scope[:scope_level] == :resources
raise ArgumentError, "can't define route directly in resources scope"
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
super
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
private
def with_scope_level(kind, options = {})
2009-10-20 13:31:23 -04:00
old, @scope[:scope_level] = @scope[:scope_level], kind
old_options, @scope[:scope_level_options] = @scope[:scope_level_options], options
2009-10-20 13:31:23 -04:00
yield
ensure
@scope[:scope_level] = old
@scope[:scope_level_options] = old_options
2009-10-20 13:31:23 -04:00
end
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
module Scoping
def scope(*args)
options = args.last.is_a?(Hash) ? args.pop : {}
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
constraints = options.delete(:constraints) || {}
unless constraints.is_a?(Hash)
block, constraints = constraints, {}
end
constraints, @scope[:constraints] = @scope[:constraints], (@scope[:constraints] || {}).merge(constraints)
blocks, @scope[:blocks] = @scope[:blocks], (@scope[:blocks] || []) + [block]
options, @scope[:options] = @scope[:options], (@scope[:options] || {}).merge(options)
path_set = controller_set = false
case args.first
when String
path_set = true
path = args.first
path, @scope[:path] = @scope[:path], "#{@scope[:path]}#{Rack::Mount::Utils.normalize_path(path)}"
when Symbol
controller_set = true
controller = args.first
controller, @scope[:controller] = @scope[:controller], controller
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
yield
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
self
ensure
@scope[:path] = path if path_set
@scope[:controller] = controller if controller_set
@scope[:options] = options
@scope[:blocks] = blocks
@scope[:constraints] = constraints
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def controller(controller)
scope(controller.to_sym) { yield }
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def namespace(path)
scope(path.to_s) { yield }
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def constraints(constraints = {})
scope(:constraints => constraints) { yield }
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
class Constraints
def initialize(app, constraints = [])
@app, @constraints = app, constraints
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def call(env)
req = Rack::Request.new(env)
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
@constraints.each { |constraint|
if constraint.respond_to?(:matches?) && !constraint.matches?(req)
2009-11-10 20:12:54 -05:00
return [417, {}, []]
2009-10-20 13:31:23 -04:00
elsif constraint.respond_to?(:call) && !constraint.call(req)
2009-11-10 20:12:54 -05:00
return [417, {}, []]
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
}
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
@app.call(env)
end
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def initialize(set)
@set = set
@scope = {}
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
extend Scoping
extend Resources
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def get(*args, &block)
map_method(:get, *args, &block)
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def post(*args, &block)
map_method(:post, *args, &block)
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
def put(*args, &block)
map_method(:put, *args, &block)
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
def delete(*args, &block)
map_method(:delete, *args, &block)
end
2009-10-20 11:14:46 -04:00
def root(options = {})
match '/', options.merge(:as => :root)
end
2009-10-20 13:31:23 -04:00
def match(*args)
options = args.last.is_a?(Hash) ? args.pop : {}
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
if args.length > 1
args.each { |path| match(path, options) }
return self
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
if args.first.is_a?(Symbol)
return match(args.first.to_s, options.merge(:to => args.first.to_sym))
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
path = args.first
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
options = (@scope[:options] || {}).merge(options)
conditions, defaults = {}, {}
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
path = nil if path == ""
path = Rack::Mount::Utils.normalize_path(path) if path
path = "#{@scope[:path]}#{path}" if @scope[:path]
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
raise ArgumentError, "path is required" unless path
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
constraints = options[:constraints] || {}
unless constraints.is_a?(Hash)
block, constraints = constraints, {}
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
blocks = ((@scope[:blocks] || []) + [block]).compact
constraints = (@scope[:constraints] || {}).merge(constraints)
options.each { |k, v| constraints[k] = v if v.is_a?(Regexp) }
2009-10-20 11:14:46 -04:00
conditions[:path_info] = path
requirements = constraints.dup
2009-10-20 11:14:46 -04:00
path_regexp = Rack::Mount::Strexp.compile(path, constraints, SEPARATORS)
segment_keys = Rack::Mount::RegexpWithNamedGroups.new(path_regexp).names
2009-10-20 13:31:23 -04:00
constraints.reject! { |k, v| segment_keys.include?(k.to_s) }
conditions.merge!(constraints)
2009-10-20 11:14:46 -04:00
2009-11-23 20:44:43 -05:00
requirements[:controller] ||= Routing.controller_constraints
2009-10-20 13:31:23 -04:00
if via = options[:via]
via = Array(via).map { |m| m.to_s.upcase }
conditions[:request_method] = Regexp.union(*via)
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
defaults[:controller] = @scope[:controller].to_s if @scope[:controller]
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
if options[:to].respond_to?(:call)
app = options[:to]
defaults.delete(:controller)
defaults.delete(:action)
elsif options[:to].is_a?(String)
defaults[:controller], defaults[:action] = options[:to].split('#')
elsif options[:to].is_a?(Symbol)
defaults[:action] = options[:to].to_s
2009-10-20 11:14:46 -04:00
end
2009-10-20 13:31:23 -04:00
app ||= Routing::RouteSet::Dispatcher.new(:defaults => defaults)
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
if app.is_a?(Routing::RouteSet::Dispatcher)
unless defaults.include?(:controller) || segment_keys.include?("controller")
raise ArgumentError, "missing :controller"
end
unless defaults.include?(:action) || segment_keys.include?("action")
raise ArgumentError, "missing :action"
2009-10-20 11:14:46 -04:00
end
end
2009-10-20 13:31:23 -04:00
app = Constraints.new(app, blocks) if blocks.any?
@set.add_route(app, conditions, requirements, defaults, options[:as])
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
self
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
def redirect(path, options = {})
status = options[:status] || 301
lambda { |env|
req = Rack::Request.new(env)
url = req.scheme + '://' + req.host + path
[status, {'Location' => url, 'Content-Type' => 'text/html'}, ['Moved Permanently']]
}
end
2009-10-20 11:14:46 -04:00
2009-10-20 13:31:23 -04:00
private
def map_method(method, *args, &block)
options = args.last.is_a?(Hash) ? args.pop : {}
options[:via] = method
args.push(options)
match(*args, &block)
self
2009-10-20 11:14:46 -04:00
end
end
end
end