mapper.rb 54.4 KB
Newer Older
1
require 'active_support/core_ext/hash/except'
B
Bogdan Gusiev 已提交
2
require 'active_support/core_ext/hash/reverse_merge'
3
require 'active_support/core_ext/hash/slice'
4
require 'active_support/core_ext/object/blank'
S
Santiago Pastorino 已提交
5
require 'active_support/core_ext/enumerable'
6
require 'active_support/inflector'
7
require 'action_dispatch/routing/redirection'
8

J
Joshua Peek 已提交
9 10
module ActionDispatch
  module Routing
J
Joshua Peek 已提交
11
    class Mapper
12
      class Constraints #:nodoc:
13
        def self.new(app, constraints, request = Rack::Request)
14
          if constraints.any?
15
            super(app, constraints, request)
16 17 18 19 20
          else
            app
          end
        end

21
        attr_reader :app, :constraints
22

23 24
        def initialize(app, constraints, request)
          @app, @constraints, @request = app, constraints, request
25 26
        end

27
        def matches?(env)
28
          req = @request.new(env)
29 30 31

          @constraints.each { |constraint|
            if constraint.respond_to?(:matches?) && !constraint.matches?(req)
32
              return false
33
            elsif constraint.respond_to?(:call) && !constraint.call(*constraint_args(constraint, req))
34
              return false
35 36 37
            end
          }

38
          return true
39 40
        ensure
          req.reset_parameters
41 42 43 44
        end

        def call(env)
          matches?(env) ? @app.call(env) : [ 404, {'X-Cascade' => 'pass'}, [] ]
45
        end
46 47 48 49 50

        private
          def constraint_args(constraint, request)
            constraint.arity == 1 ? [request] : [request.symbolized_path_parameters, request]
          end
51 52
      end

53
      class Mapping #:nodoc:
54
        IGNORE_OPTIONS = [:to, :as, :via, :on, :constraints, :defaults, :only, :except, :anchor, :shallow, :shallow_path, :shallow_prefix]
55
        ANCHOR_CHARACTERS_REGEX = %r{\A(\\A|\^)|(\\Z|\\z|\$)\Z}
56
        SHORTHAND_REGEX = %r{/[\w/]+$}
57
        WILDCARD_PATH = %r{\*([^/\)]+)\)?$}
58

59
        def initialize(set, scope, path, options)
60
          @set, @scope = set, scope
61
          @segment_keys = nil
62
          @options = (@scope[:options] || {}).merge(options)
63
          @path = normalize_path(path)
64
          normalize_options!
65 66 67 68 69 70 71 72 73 74

          via_all = @options.delete(:via) if @options[:via] == :all

          if !via_all && request_method_condition.empty?
            msg = "You should not use the `match` method in your router without specifying an HTTP method.\n" \
                  "If you want to expose your action to GET, use `get` in the router:\n\n" \
                  "  Instead of: match \"controller#action\"\n" \
                  "  Do: get \"controller#action\""
            raise msg
          end
75
        end
J
Joshua Peek 已提交
76

77
        def to_route
78
          [ app, conditions, requirements, defaults, @options[:as], @options[:anchor] ]
79
        end
J
Joshua Peek 已提交
80

81
        private
82 83 84

          def normalize_options!
            path_without_format = @path.sub(/\(\.:format\)$/, '')
85

86 87
            if using_match_shorthand?(path_without_format, @options)
              to_shorthand    = @options[:to].blank?
88
              @options[:to] ||= path_without_format.gsub(/\(.*\)/, "")[1..-1].sub(%r{/([^/]*)$}, '#\1')
89 90
            end

91
            @options.merge!(default_controller_and_action(to_shorthand))
92 93 94 95 96

            requirements.each do |name, requirement|
              # segment_keys.include?(k.to_s) || k == :controller
              next unless Regexp === requirement && !constraints[name]

97
              if requirement.source =~ ANCHOR_CHARACTERS_REGEX
98 99 100 101 102 103
                raise ArgumentError, "Regexp anchor characters are not allowed in routing requirements: #{requirement.inspect}"
              end
              if requirement.multiline?
                raise ArgumentError, "Regexp multiline option not allowed in routing requirements: #{requirement.inspect}"
              end
            end
104 105 106 107

            if @options[:constraints].is_a?(Hash)
              (@options[:defaults] ||= {}).reverse_merge!(defaults_from_constraints(@options[:constraints]))
            end
108
          end
109

110
          # match "account/overview"
111
          def using_match_shorthand?(path, options)
112
            path && (options[:to] || options[:action]).nil? && path =~ SHORTHAND_REGEX
113
          end
114

115
          def normalize_path(path)
116 117
            raise ArgumentError, "path is required" if path.blank?
            path = Mapper.normalize_path(path)
118 119 120 121 122 123 124 125

            if path.match(':controller')
              raise ArgumentError, ":controller segment is not allowed within a namespace block" if @scope[:module]

              # Add a default constraint for :controller path segments that matches namespaced
              # controllers with default routes like :controller/:action/:id(.:format), e.g:
              # GET /admin/products/show/1
              # => { :controller => 'admin/products', :action => 'show', :id => '1' }
126
              @options[:controller] ||= /.+?/
127 128
            end

129 130
            # Add a constraint for wildcard route to make it non-greedy and match the
            # optional format part of the route by default
131
            if path.match(WILDCARD_PATH) && @options[:format] != false
132
              @options[$1.to_sym] ||= /.+?/
133 134
            end

135 136 137
            if @options[:format] == false
              @options.delete(:format)
              path
138
            elsif path.include?(":format") || path.end_with?('/')
139
              path
140 141
            elsif @options[:format] == true
              "#{path}.:format"
142 143 144
            else
              "#{path}(.:format)"
            end
145
          end
146

147 148
          def app
            Constraints.new(
149
              to.respond_to?(:call) ? to : Routing::RouteSet::Dispatcher.new(:defaults => defaults),
150 151
              blocks,
              @set.request_class
152
            )
153 154
          end

155 156 157
          def conditions
            { :path_info => @path }.merge(constraints).merge(request_method_condition)
          end
J
Joshua Peek 已提交
158

159
          def requirements
160
            @requirements ||= (@options[:constraints].is_a?(Hash) ? @options[:constraints] : {}).tap do |requirements|
161 162 163 164
              requirements.reverse_merge!(@scope[:constraints]) if @scope[:constraints]
              @options.each { |k, v| requirements[k] = v if v.is_a?(Regexp) }
            end
          end
165

166
          def defaults
167 168 169 170 171 172
            @defaults ||= (@options[:defaults] || {}).tap do |defaults|
              defaults.reverse_merge!(@scope[:defaults]) if @scope[:defaults]
              @options.each { |k, v| defaults[k] = v unless v.is_a?(Regexp) || IGNORE_OPTIONS.include?(k.to_sym) }
            end
          end

173
          def default_controller_and_action(to_shorthand=nil)
174
            if to.respond_to?(:call)
175 176
              { }
            else
177
              if to.is_a?(String)
178
                controller, action = to.split('#')
179 180
              elsif to.is_a?(Symbol)
                action = to.to_s
181
              end
J
Joshua Peek 已提交
182

183 184
              controller ||= default_controller
              action     ||= default_action
185

186 187 188
              unless controller.is_a?(Regexp) || to_shorthand
                controller = [@scope[:module], controller].compact.join("/").presence
              end
189

190 191 192 193
              if controller.is_a?(String) && controller =~ %r{\A/}
                raise ArgumentError, "controller name should not start with a slash"
              end

194 195
              controller = controller.to_s unless controller.is_a?(Regexp)
              action     = action.to_s     unless action.is_a?(Regexp)
196

197
              if controller.blank? && segment_keys.exclude?("controller")
198 199
                raise ArgumentError, "missing :controller"
              end
J
Joshua Peek 已提交
200

201
              if action.blank? && segment_keys.exclude?("action")
202 203
                raise ArgumentError, "missing :action"
              end
J
Joshua Peek 已提交
204

A
Aaron Patterson 已提交
205
              hash = {}
A
Aaron Patterson 已提交
206 207
              hash[:controller] = controller unless controller.blank?
              hash[:action]     = action unless action.blank?
A
Aaron Patterson 已提交
208
              hash
209 210
            end
          end
211

212
          def blocks
213 214 215 216 217
            constraints = @options[:constraints]
            if constraints.present? && !constraints.is_a?(Hash)
              [constraints]
            else
              @scope[:blocks] || []
218 219
            end
          end
J
Joshua Peek 已提交
220

221 222 223
          def constraints
            @constraints ||= requirements.reject { |k, v| segment_keys.include?(k.to_s) || k == :controller }
          end
224

225 226
          def request_method_condition
            if via = @options[:via]
227 228
              list = Array(via).map { |m| m.to_s.dasherize.upcase }
              { :request_method => list }
229 230
            else
              { }
231
            end
232
          end
J
Joshua Peek 已提交
233

234
          def segment_keys
235 236 237
            return @segment_keys if @segment_keys

            @segment_keys = Journey::Path::Pattern.new(
238
              Journey::Router::Strexp.compile(@path, requirements, SEPARATORS)
239
            ).names
240
          end
241

242 243 244
          def to
            @options[:to]
          end
J
Joshua Peek 已提交
245

246
          def default_controller
247
            @options[:controller] || @scope[:controller]
248
          end
249 250

          def default_action
251
            @options[:action] || @scope[:action]
252
          end
253 254 255 256 257

          def defaults_from_constraints(constraints)
            url_keys = [:protocol, :subdomain, :domain, :host, :port]
            constraints.slice(*url_keys).select{ |k, v| v.is_a?(String) || v.is_a?(Fixnum) }
          end
258
      end
259

260
      # Invokes Rack::Mount::Utils.normalize path and ensure that
261 262
      # (:locale) becomes (/:locale) instead of /(:locale). Except
      # for root cases, where the latter is the correct one.
263
      def self.normalize_path(path)
264
        path = Journey::Router::Utils.normalize_path(path)
265
        path.gsub!(%r{/(\(+)/?}, '\1/') unless path =~ %r{^/\(+[^)]+\)$}
266 267 268
        path
      end

269
      def self.normalize_name(name)
270
        normalize_path(name)[1..-1].tr("/", "_")
271 272
      end

273
      module Base
274 275 276 277
        # You can specify what Rails should route "/" to with the root method:
        #
        #   root :to => 'pages#main'
        #
278
        # For options, see +match+, as +root+ uses it internally.
279
        #
B
Brian Cardarella 已提交
280 281 282 283
        # You can also pass a string which will expand
        #
        #   root 'pages#main'
        #
284 285 286
        # You should put the root route at the top of <tt>config/routes.rb</tt>,
        # because this means it will be matched first. As this is the most popular route
        # of most Rails applications, this is beneficial.
287
        def root(options = {})
B
Brian Cardarella 已提交
288
          options = { :to => options } if options.is_a?(String)
289
          match '/', { :as => :root, :via => :get }.merge(options)
290
        end
291

292 293 294
        # Matches a url pattern to one or more routes. Any symbols in a pattern
        # are interpreted as url query parameters and thus available as +params+
        # in an action:
295
        #
296
        #   # sets :controller, :action and :id in params
297
        #   match ':controller/:action/:id'
298
        #
299 300 301 302 303 304 305 306 307 308 309 310
        # Two of these symbols are special, +:controller+ maps to the controller
        # and +:action+ to the controller's action. A pattern can also map
        # wildcard segments (globs) to params:
        #
        #   match 'songs/*category/:title' => 'songs#show'
        #
        #   # 'songs/rock/classic/stairway-to-heaven' sets
        #   #  params[:category] = 'rock/classic'
        #   #  params[:title] = 'stairway-to-heaven'
        #
        # When a pattern points to an internal route, the route's +:action+ and
        # +:controller+ should be set in options or hash shorthand. Examples:
311 312 313 314
        #
        #   match 'photos/:id' => 'photos#show'
        #   match 'photos/:id', :to => 'photos#show'
        #   match 'photos/:id', :controller => 'photos', :action => 'show'
315
        #
316 317 318
        # A pattern can also point to a +Rack+ endpoint i.e. anything that
        # responds to +call+:
        #
A
Alexey Vakhov 已提交
319
        #   match 'photos/:id' => lambda {|hash| [200, {}, "Coming soon"] }
320 321 322 323
        #   match 'photos/:id' => PhotoRackApp
        #   # Yes, controller actions are just rack endpoints
        #   match 'photos/:id' => PhotosController.action(:show)
        #
324
        # === Options
325
        #
326
        # Any options not seen here are passed on as params with the url.
327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
        #
        # [:controller]
        #   The route's controller.
        #
        # [:action]
        #   The route's action.
        #
        # [:path]
        #   The path prefix for the routes.
        #
        # [:module]
        #   The namespace for :controller.
        #
        #     match 'path' => 'c#a', :module => 'sekret', :controller => 'posts'
        #     #=> Sekret::PostsController
        #
        #   See <tt>Scoping#namespace</tt> for its scope equivalent.
        #
        # [:as]
        #   The name used to generate routing helpers.
        #
        # [:via]
        #   Allowed HTTP verb(s) for route.
        #
        #      match 'path' => 'c#a', :via => :get
        #      match 'path' => 'c#a', :via => [:get, :post]
        #
        # [:to]
355 356
        #   Points to a +Rack+ endpoint. Can be an object that responds to
        #   +call+ or a string representing a controller's action.
357
        #
358
        #      match 'path', :to => 'controller#action'
J
Justin Woodbridge 已提交
359
        #      match 'path', :to => lambda { |env| [200, {}, "Success!"] }
360
        #      match 'path', :to => RackApp
361 362 363
        #
        # [:on]
        #   Shorthand for wrapping routes in a specific RESTful context. Valid
364
        #   values are +:member+, +:collection+, and +:new+. Only use within
365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380
        #   <tt>resource(s)</tt> block. For example:
        #
        #      resource :bar do
        #        match 'foo' => 'c#a', :on => :member, :via => [:get, :post]
        #      end
        #
        #   Is equivalent to:
        #
        #      resource :bar do
        #        member do
        #          match 'foo' => 'c#a', :via => [:get, :post]
        #        end
        #      end
        #
        # [:constraints]
        #   Constrains parameters with a hash of regular expressions or an
381
        #   object that responds to <tt>matches?</tt>
382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
        #
        #     match 'path/:id', :constraints => { :id => /[A-Z]\d{5}/ }
        #
        #     class Blacklist
        #       def matches?(request) request.remote_ip == '1.2.3.4' end
        #     end
        #     match 'path' => 'c#a', :constraints => Blacklist.new
        #
        #   See <tt>Scoping#constraints</tt> for more examples with its scope
        #   equivalent.
        #
        # [:defaults]
        #   Sets defaults for parameters
        #
        #     # Sets params[:format] to 'jpg' by default
        #     match 'path' => 'c#a', :defaults => { :format => 'jpg' }
        #
        #   See <tt>Scoping#defaults</tt> for its scope equivalent.
400 401
        #
        # [:anchor]
402
        #   Boolean to anchor a <tt>match</tt> pattern. Default is true. When set to
403 404 405 406
        #   false, the pattern matches any request prefixed with the given path.
        #
        #     # Matches any request starting with 'path'
        #     match 'path' => 'c#a', :anchor => false
407 408
        #
        # [:format]
409 410
        #   Allows you to specify the default value for optional +format+
        #   segment or disable it if you supply +false+.
411
        def match(path, options=nil)
412
        end
413

414 415
        # Mount a Rack-based application to be used within the application.
        #
R
Ryan Bigg 已提交
416
        #   mount SomeRackApp, :at => "some_route"
417 418 419
        #
        # Alternatively:
        #
R
Ryan Bigg 已提交
420
        #   mount(SomeRackApp => "some_route")
421
        #
422 423
        # For options, see +match+, as +mount+ uses it internally.
        #
424 425 426 427 428
        # All mounted applications come with routing helpers to access them.
        # These are named after the class specified, so for the above example
        # the helper is either +some_rack_app_path+ or +some_rack_app_url+.
        # To customize this helper's name, use the +:as+ option:
        #
R
Ryan Bigg 已提交
429
        #   mount(SomeRackApp => "some_route", :as => "exciting")
430 431 432
        #
        # This will generate the +exciting_path+ and +exciting_url+ helpers
        # which can be used to navigate to this mounted app.
433 434 435 436
        def mount(app, options = nil)
          if options
            path = options.delete(:at)
          else
437 438 439 440
            unless Hash === app
              raise ArgumentError, "must be called with mount point"
            end

441 442 443 444 445 446 447
            options = app
            app, path = options.find { |k, v| k.respond_to?(:call) }
            options.delete(app) if app
          end

          raise "A rack application must be specified" unless path

448 449
          options[:as] ||= app_name(app)

450
          match(path, options.merge(:to => app, :anchor => false, :format => false, :via => :all))
451 452

          define_generate_prefix(app, options[:as])
453 454 455
          self
        end

456 457 458 459
        def default_url_options=(options)
          @set.default_url_options = options
        end
        alias_method :default_url_options, :default_url_options=
460

461 462 463 464 465 466
        def with_default_scope(scope, &block)
          scope(scope) do
            instance_exec(&block)
          end
        end

467 468 469
        private
          def app_name(app)
            return unless app.respond_to?(:routes)
470 471 472 473 474

            if app.respond_to?(:railtie_name)
              app.railtie_name
            else
              class_name = app.class.is_a?(Class) ? app.name : app.class.name
475
              ActiveSupport::Inflector.underscore(class_name).tr("/", "_")
476
            end
477 478 479
          end

          def define_generate_prefix(app, name)
480
            return unless app.respond_to?(:routes) && app.routes.respond_to?(:define_mounted_helper)
481 482

            _route = @set.named_routes.routes[name.to_sym]
P
Piotr Sarnacki 已提交
483 484
            _routes = @set
            app.routes.define_mounted_helper(name)
J
José Valim 已提交
485 486 487 488 489
            app.routes.singleton_class.class_eval do
              define_method :mounted? do
                true
              end

490
              define_method :_generate_prefix do |options|
P
Piotr Sarnacki 已提交
491
                prefix_options = options.slice(*_route.segment_keys)
492 493
                # we must actually delete prefix segment keys to avoid passing them to next url_for
                _route.segment_keys.each { |k| options.delete(k) }
R
rails-noob 已提交
494 495 496
                prefix = _routes.url_helpers.send("#{name}_path", prefix_options)
                prefix = '' if prefix == '/'
                prefix
497 498 499
              end
            end
          end
500 501 502
      end

      module HttpHelpers
503
        # Define a route that only recognizes HTTP GET.
504
        # For supported arguments, see <tt>Base#match</tt>.
505
        #
506
        #   get 'bacon', :to => 'food#bacon'
507
        def get(*args, &block)
508
          map_method(:get, args, &block)
509 510
        end

511
        # Define a route that only recognizes HTTP POST.
512
        # For supported arguments, see <tt>Base#match</tt>.
513
        #
514
        #   post 'bacon', :to => 'food#bacon'
515
        def post(*args, &block)
516
          map_method(:post, args, &block)
517 518
        end

519 520 521 522 523 524 525 526
        # Define a route that only recognizes HTTP PATCH.
        # For supported arguments, see <tt>Base#match</tt>.
        #
        #   patch 'bacon', :to => 'food#bacon'
        def patch(*args, &block)
          map_method(:patch, args, &block)
        end

527
        # Define a route that only recognizes HTTP PUT.
528
        # For supported arguments, see <tt>Base#match</tt>.
529
        #
530
        #   put 'bacon', :to => 'food#bacon'
531
        def put(*args, &block)
532
          map_method(:put, args, &block)
533 534
        end

535
        # Define a route that only recognizes HTTP DELETE.
536
        # For supported arguments, see <tt>Base#match</tt>.
537
        #
538
        #   delete 'broccoli', :to => 'food#broccoli'
539
        def delete(*args, &block)
540
          map_method(:delete, args, &block)
541 542 543
        end

        private
544
          def map_method(method, args, &block)
545
            options = args.extract_options!
546 547
            options[:via]    = method
            options[:path] ||= args.first if args.first.is_a?(String)
548
            match(*args, options, &block)
549 550 551 552
            self
          end
      end

553 554 555
      # You may wish to organize groups of controllers under a namespace.
      # Most commonly, you might group a number of administrative controllers
      # under an +admin+ namespace. You would place these controllers under
S
Sebastian Martinez 已提交
556 557
      # the <tt>app/controllers/admin</tt> directory, and you can group them
      # together in your router:
558 559 560 561
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
562
      #
563
      # This will create a number of routes for each of the posts and comments
S
Sebastian Martinez 已提交
564
      # controller. For <tt>Admin::PostsController</tt>, Rails will create:
565
      #
566 567 568 569 570
      #   GET       /admin/posts
      #   GET       /admin/posts/new
      #   POST      /admin/posts
      #   GET       /admin/posts/1
      #   GET       /admin/posts/1/edit
571
      #   PATCH/PUT /admin/posts/1
572
      #   DELETE    /admin/posts/1
573
      #
574
      # If you want to route /posts (without the prefix /admin) to
S
Sebastian Martinez 已提交
575
      # <tt>Admin::PostsController</tt>, you could use
576
      #
577
      #   scope :module => "admin" do
578
      #     resources :posts
579 580 581
      #   end
      #
      # or, for a single case
582
      #
583
      #   resources :posts, :module => "admin"
584
      #
S
Sebastian Martinez 已提交
585
      # If you want to route /admin/posts to +PostsController+
586
      # (without the Admin:: module prefix), you could use
587
      #
588
      #   scope "/admin" do
589
      #     resources :posts
590 591 592
      #   end
      #
      # or, for a single case
593
      #
594
      #   resources :posts, :path => "/admin/posts"
595 596 597
      #
      # In each of these cases, the named routes remain the same as if you did
      # not use scope. In the last case, the following paths map to
S
Sebastian Martinez 已提交
598
      # +PostsController+:
599
      #
600 601 602 603 604
      #   GET       /admin/posts
      #   GET       /admin/posts/new
      #   POST      /admin/posts
      #   GET       /admin/posts/1
      #   GET       /admin/posts/1/edit
605
      #   PATCH/PUT /admin/posts/1
606
      #   DELETE    /admin/posts/1
607
      module Scoping
608
        # Scopes a set of routes to the given default options.
609 610 611 612 613 614 615 616
        #
        # Take the following route definition as an example:
        #
        #   scope :path => ":account_id", :as => "account" do
        #     resources :projects
        #   end
        #
        # This generates helpers such as +account_projects_path+, just like +resources+ does.
617 618
        # The difference here being that the routes generated are like /:account_id/projects,
        # rather than /accounts/:account_id/projects.
619
        #
620
        # === Options
621
        #
622
        # Takes same options as <tt>Base#match</tt> and <tt>Resources#resources</tt>.
623
        #
624
        # === Examples
625
        #
S
Sebastian Martinez 已提交
626
        #   # route /posts (without the prefix /admin) to <tt>Admin::PostsController</tt>
627 628 629
        #   scope :module => "admin" do
        #     resources :posts
        #   end
630
        #
631 632 633 634
        #   # prefix the posts resource's requests with '/admin'
        #   scope :path => "/admin" do
        #     resources :posts
        #   end
635
        #
S
Sebastian Martinez 已提交
636
        #   # prefix the routing helper name: +sekret_posts_path+ instead of +posts_path+
637 638 639
        #   scope :as => "sekret" do
        #     resources :posts
        #   end
640 641
        def scope(*args)
          options = args.extract_options!
642
          options = options.dup
643

644
          options[:path] = args.first if args.first.is_a?(String)
645
          recover = {}
646

647 648
          options[:constraints] ||= {}
          unless options[:constraints].is_a?(Hash)
649
            block, options[:constraints] = options[:constraints], {}
650
          end
651

652 653 654 655
          if options[:constraints].is_a?(Hash)
            (options[:defaults] ||= {}).reverse_merge!(defaults_from_constraints(options[:constraints]))
          end

656 657 658 659 660
          scope_options.each do |option|
            if value = options.delete(option)
              recover[option] = @scope[option]
              @scope[option]  = send("merge_#{option}_scope", @scope[option], value)
            end
661 662
          end

663 664 665 666 667 668
          recover[:block] = @scope[:blocks]
          @scope[:blocks] = merge_blocks_scope(@scope[:blocks], block)

          recover[:options] = @scope[:options]
          @scope[:options]  = merge_options_scope(@scope[:options], options)

669 670 671
          yield
          self
        ensure
672 673 674
          scope_options.each do |option|
            @scope[option] = recover[option] if recover.has_key?(option)
          end
675 676 677

          @scope[:options] = recover[:options]
          @scope[:blocks]  = recover[:block]
678 679
        end

680 681 682 683 684
        # Scopes routes to a specific controller
        #
        #   controller "food" do
        #     match "bacon", :action => "bacon"
        #   end
685 686 687
        def controller(controller, options={})
          options[:controller] = controller
          scope(options) { yield }
688 689
        end

690 691 692 693 694 695 696 697
        # Scopes routes to a specific namespace. For example:
        #
        #   namespace :admin do
        #     resources :posts
        #   end
        #
        # This generates the following routes:
        #
698 699 700 701 702
        #       admin_posts GET       /admin/posts(.:format)          admin/posts#index
        #       admin_posts POST      /admin/posts(.:format)          admin/posts#create
        #    new_admin_post GET       /admin/posts/new(.:format)      admin/posts#new
        #   edit_admin_post GET       /admin/posts/:id/edit(.:format) admin/posts#edit
        #        admin_post GET       /admin/posts/:id(.:format)      admin/posts#show
703
        #        admin_post PATCH/PUT /admin/posts/:id(.:format)      admin/posts#update
704
        #        admin_post DELETE    /admin/posts/:id(.:format)      admin/posts#destroy
705
        #
706
        # === Options
707
        #
708 709
        # The +:path+, +:as+, +:module+, +:shallow_path+ and +:shallow_prefix+
        # options all default to the name of the namespace.
710
        #
711 712
        # For options, see <tt>Base#match</tt>. For +:shallow_path+ option, see
        # <tt>Resources#resources</tt>.
713
        #
714
        # === Examples
715
        #
716 717 718 719
        #   # accessible through /sekret/posts rather than /admin/posts
        #   namespace :admin, :path => "sekret" do
        #     resources :posts
        #   end
720
        #
S
Sebastian Martinez 已提交
721
        #   # maps to <tt>Sekret::PostsController</tt> rather than <tt>Admin::PostsController</tt>
722 723 724
        #   namespace :admin, :module => "sekret" do
        #     resources :posts
        #   end
725
        #
S
Sebastian Martinez 已提交
726
        #   # generates +sekret_posts_path+ rather than +admin_posts_path+
727 728 729
        #   namespace :admin, :as => "sekret" do
        #     resources :posts
        #   end
730
        def namespace(path, options = {})
731
          path = path.to_s
732 733 734
          options = { :path => path, :as => path, :module => path,
                      :shallow_path => path, :shallow_prefix => path }.merge!(options)
          scope(options) { yield }
735
        end
736

R
Ryan Bigg 已提交
737 738 739 740
        # === Parameter Restriction
        # Allows you to constrain the nested routes based on a set of rules.
        # For instance, in order to change the routes to allow for a dot character in the +id+ parameter:
        #
M
mjy 已提交
741
        #   constraints(:id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
742 743 744 745 746
        #     resources :posts
        #   end
        #
        # Now routes such as +/posts/1+ will no longer be valid, but +/posts/1.1+ will be.
        # The +id+ parameter must match the constraint passed in for this example.
747
        #
R
R.T. Lechow 已提交
748
        # You may use this to also restrict other parameters:
R
Ryan Bigg 已提交
749 750
        #
        #   resources :posts do
M
mjy 已提交
751
        #     constraints(:post_id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
752 753
        #       resources :comments
        #     end
J
James Miller 已提交
754
        #   end
R
Ryan Bigg 已提交
755 756 757 758 759 760 761 762 763 764 765 766 767 768
        #
        # === Restricting based on IP
        #
        # Routes can also be constrained to an IP or a certain range of IP addresses:
        #
        #   constraints(:ip => /192.168.\d+.\d+/) do
        #     resources :posts
        #   end
        #
        # Any user connecting from the 192.168.* range will be able to see this resource,
        # where as any user connecting outside of this range will be told there is no such route.
        #
        # === Dynamic request matching
        #
R
R.T. Lechow 已提交
769
        # Requests to routes can be constrained based on specific criteria:
R
Ryan Bigg 已提交
770 771 772 773 774 775 776 777 778 779
        #
        #    constraints(lambda { |req| req.env["HTTP_USER_AGENT"] =~ /iPhone/ }) do
        #      resources :iphones
        #    end
        #
        # You are able to move this logic out into a class if it is too complex for routes.
        # This class must have a +matches?+ method defined on it which either returns +true+
        # if the user should be given access to that route, or +false+ if the user should not.
        #
        #    class Iphone
780
        #      def self.matches?(request)
R
Ryan Bigg 已提交
781 782 783 784 785 786 787 788 789 790 791
        #        request.env["HTTP_USER_AGENT"] =~ /iPhone/
        #      end
        #    end
        #
        # An expected place for this code would be +lib/constraints+.
        #
        # This class is then used like this:
        #
        #    constraints(Iphone) do
        #      resources :iphones
        #    end
792 793 794 795
        def constraints(constraints = {})
          scope(:constraints => constraints) { yield }
        end

R
Ryan Bigg 已提交
796
        # Allows you to set default parameters for a route, such as this:
797 798 799
        #   defaults :id => 'home' do
        #     match 'scoped_pages/(:id)', :to => 'pages#show'
        #   end
R
Ryan Bigg 已提交
800
        # Using this, the +:id+ parameter here will default to 'home'.
801 802 803 804
        def defaults(defaults = {})
          scope(:defaults => defaults) { yield }
        end

805
        private
J
José Valim 已提交
806
          def scope_options #:nodoc:
807 808 809
            @scope_options ||= private_methods.grep(/^merge_(.+)_scope$/) { $1.to_sym }
          end

J
José Valim 已提交
810
          def merge_path_scope(parent, child) #:nodoc:
811
            Mapper.normalize_path("#{parent}/#{child}")
812 813
          end

J
José Valim 已提交
814
          def merge_shallow_path_scope(parent, child) #:nodoc:
815 816 817
            Mapper.normalize_path("#{parent}/#{child}")
          end

J
José Valim 已提交
818
          def merge_as_scope(parent, child) #:nodoc:
819
            parent ? "#{parent}_#{child}" : child
820 821
          end

J
José Valim 已提交
822
          def merge_shallow_prefix_scope(parent, child) #:nodoc:
823 824 825
            parent ? "#{parent}_#{child}" : child
          end

J
José Valim 已提交
826
          def merge_module_scope(parent, child) #:nodoc:
827 828 829
            parent ? "#{parent}/#{child}" : child
          end

J
José Valim 已提交
830
          def merge_controller_scope(parent, child) #:nodoc:
831
            child
832 833
          end

J
José Valim 已提交
834
          def merge_path_names_scope(parent, child) #:nodoc:
835 836 837
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
838
          def merge_constraints_scope(parent, child) #:nodoc:
839 840 841
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
842
          def merge_defaults_scope(parent, child) #:nodoc:
843 844 845
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
846
          def merge_blocks_scope(parent, child) #:nodoc:
847 848 849
            merged = parent ? parent.dup : []
            merged << child if child
            merged
850 851
          end

J
José Valim 已提交
852
          def merge_options_scope(parent, child) #:nodoc:
853
            (parent || {}).except(*override_keys(child)).merge(child)
854
          end
855

J
José Valim 已提交
856
          def merge_shallow_scope(parent, child) #:nodoc:
857 858
            child ? true : false
          end
859

J
José Valim 已提交
860
          def override_keys(child) #:nodoc:
861 862
            child.key?(:only) || child.key?(:except) ? [:only, :except] : []
          end
863 864 865 866 867

          def defaults_from_constraints(constraints)
            url_keys = [:protocol, :subdomain, :domain, :host, :port]
            constraints.slice(*url_keys).select{ |k, v| v.is_a?(String) || v.is_a?(Fixnum) }
          end
868 869
      end

870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893
      # Resource routing allows you to quickly declare all of the common routes
      # for a given resourceful controller. Instead of declaring separate routes
      # for your +index+, +show+, +new+, +edit+, +create+, +update+ and +destroy+
      # actions, a resourceful route declares them in a single line of code:
      #
      #  resources :photos
      #
      # Sometimes, you have a resource that clients always look up without
      # referencing an ID. A common example, /profile always shows the profile of
      # the currently logged in user. In this case, you can use a singular resource
      # to map /profile (rather than /profile/:id) to the show action.
      #
      #  resource :profile
      #
      # It's common to have resources that are logically children of other
      # resources:
      #
      #   resources :magazines do
      #     resources :ads
      #   end
      #
      # You may wish to organize groups of controllers under a namespace. Most
      # commonly, you might group a number of administrative controllers under
      # an +admin+ namespace. You would place these controllers under the
S
Sebastian Martinez 已提交
894 895
      # <tt>app/controllers/admin</tt> directory, and you can group them together
      # in your router:
896 897 898 899 900
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
      #
S
Sebastian Martinez 已提交
901 902
      # By default the +:id+ parameter doesn't accept dots. If you need to
      # use dots as part of the +:id+ parameter add a constraint which
903 904 905 906
      # overrides this restriction, e.g:
      #
      #   resources :articles, :id => /[^\/]+/
      #
S
Sebastian Martinez 已提交
907
      # This allows any character other than a slash as part of your +:id+.
908
      #
J
Joshua Peek 已提交
909
      module Resources
910 911
        # CANONICAL_ACTIONS holds all actions that does not need a prefix or
        # a path appended since they fit properly in their scope level.
912
        VALID_ON_OPTIONS  = [:new, :collection, :member]
913
        RESOURCE_OPTIONS  = [:as, :controller, :path, :only, :except, :param]
914
        CANONICAL_ACTIONS = %w(index create new show update destroy)
915

916
        class Resource #:nodoc:
917
          attr_reader :controller, :path, :options, :param
918 919

          def initialize(entities, options = {})
920
            @name       = entities.to_s
921 922 923
            @path       = (options[:path] || @name).to_s
            @controller = (options[:controller] || @name).to_s
            @as         = options[:as]
924
            @param      = (options[:param] || :id).to_sym
925
            @options    = options
926 927
          end

928
          def default_actions
929
            [:index, :create, :new, :show, :update, :destroy, :edit]
930 931
          end

932
          def actions
933
            if only = @options[:only]
934
              Array(only).map(&:to_sym)
935
            elsif except = @options[:except]
936 937 938 939 940 941
              default_actions - Array(except).map(&:to_sym)
            else
              default_actions
            end
          end

942
          def name
943
            @as || @name
944 945
          end

946
          def plural
947
            @plural ||= name.to_s
948 949 950
          end

          def singular
951
            @singular ||= name.to_s.singularize
952 953
          end

954
          alias :member_name :singular
955

956
          # Checks for uncountable plurals, and appends "_index" if the plural
957
          # and singular form are the same.
958
          def collection_name
959
            singular == plural ? "#{plural}_index" : plural
960 961
          end

962
          def resource_scope
963
            { :controller => controller }
964 965
          end

966
          alias :collection_scope :path
967 968

          def member_scope
969
            "#{path}/:#{param}"
970 971
          end

972 973
          alias :shallow_scope :member_scope

974
          def new_scope(new_path)
975
            "#{path}/#{new_path}"
976 977
          end

978 979 980 981
          def nested_param
            :"#{singular}_#{param}"
          end

982
          def nested_scope
983
            "#{path}/:#{nested_param}"
984
          end
985

986 987 988
        end

        class SingletonResource < Resource #:nodoc:
989
          def initialize(entities, options)
990
            super
991
            @as         = nil
992 993
            @controller = (options[:controller] || plural).to_s
            @as         = options[:as]
994 995
          end

996 997 998 999
          def default_actions
            [:show, :create, :update, :destroy, :new, :edit]
          end

1000 1001
          def plural
            @plural ||= name.to_s.pluralize
1002 1003
          end

1004 1005
          def singular
            @singular ||= name.to_s
1006
          end
1007 1008 1009 1010 1011 1012

          alias :member_name :singular
          alias :collection_name :singular

          alias :member_scope :path
          alias :nested_scope :path
1013 1014
        end

1015 1016 1017 1018
        def resources_path_names(options)
          @scope[:path_names].merge!(options)
        end

1019 1020 1021 1022 1023 1024 1025 1026 1027
        # Sometimes, you have a resource that clients always look up without
        # referencing an ID. A common example, /profile always shows the
        # profile of the currently logged in user. In this case, you can use
        # a singular resource to map /profile (rather than /profile/:id) to
        # the show action:
        #
        #   resource :geocoder
        #
        # creates six different routes in your application, all mapping to
S
Sebastian Martinez 已提交
1028
        # the +GeoCoders+ controller (note that the controller is named after
1029 1030
        # the plural):
        #
1031 1032 1033 1034
        #   GET       /geocoder/new
        #   POST      /geocoder
        #   GET       /geocoder
        #   GET       /geocoder/edit
1035
        #   PATCH/PUT /geocoder
1036
        #   DELETE    /geocoder
1037
        #
1038
        # === Options
1039
        # Takes same options as +resources+.
J
Joshua Peek 已提交
1040
        def resource(*resources, &block)
J
Joshua Peek 已提交
1041
          options = resources.extract_options!
J
Joshua Peek 已提交
1042

1043
          if apply_common_behavior_for(:resource, resources, options, &block)
1044 1045 1046
            return self
          end

1047
          resource_scope(:resource, SingletonResource.new(resources.pop, options)) do
1048
            yield if block_given?
1049

1050
            collection do
1051
              post :create
1052
            end if parent_resource.actions.include?(:create)
1053

1054
            new do
1055
              get :new
1056
            end if parent_resource.actions.include?(:new)
1057

1058
            member do
1059 1060
              get :edit if parent_resource.actions.include?(:edit)
              get :show if parent_resource.actions.include?(:show)
1061
              if parent_resource.actions.include?(:update)
1062 1063
                patch :update
                put   :update
1064
              end
1065
              delete :destroy if parent_resource.actions.include?(:destroy)
1066 1067 1068
            end
          end

J
Joshua Peek 已提交
1069
          self
1070 1071
        end

1072 1073 1074 1075 1076 1077 1078 1079
        # In Rails, a resourceful route provides a mapping between HTTP verbs
        # and URLs and controller actions. By convention, each action also maps
        # to particular CRUD operations in a database. A single entry in the
        # routing file, such as
        #
        #   resources :photos
        #
        # creates seven different routes in your application, all mapping to
S
Sebastian Martinez 已提交
1080
        # the +Photos+ controller:
1081
        #
1082 1083 1084 1085 1086
        #   GET       /photos
        #   GET       /photos/new
        #   POST      /photos
        #   GET       /photos/:id
        #   GET       /photos/:id/edit
1087
        #   PATCH/PUT /photos/:id
1088
        #   DELETE    /photos/:id
1089
        #
1090 1091 1092 1093 1094 1095 1096 1097
        # Resources can also be nested infinitely by using this block syntax:
        #
        #   resources :photos do
        #     resources :comments
        #   end
        #
        # This generates the following comments routes:
        #
1098 1099 1100 1101 1102
        #   GET       /photos/:photo_id/comments
        #   GET       /photos/:photo_id/comments/new
        #   POST      /photos/:photo_id/comments
        #   GET       /photos/:photo_id/comments/:id
        #   GET       /photos/:photo_id/comments/:id/edit
1103
        #   PATCH/PUT /photos/:photo_id/comments/:id
1104
        #   DELETE    /photos/:photo_id/comments/:id
1105
        #
1106
        # === Options
1107 1108
        # Takes same options as <tt>Base#match</tt> as well as:
        #
1109
        # [:path_names]
A
Aviv Ben-Yosef 已提交
1110 1111
        #   Allows you to change the segment component of the +edit+ and +new+ actions.
        #   Actions not specified are not changed.
1112 1113 1114 1115
        #
        #     resources :posts, :path_names => { :new => "brand_new" }
        #
        #   The above example will now change /posts/new to /posts/brand_new
1116
        #
1117 1118 1119 1120 1121 1122 1123
        # [:path]
        #   Allows you to change the path prefix for the resource.
        #
        #     resources :posts, :path => 'postings'
        #
        #   The resource and all segments will now route to /postings instead of /posts
        #
1124 1125
        # [:only]
        #   Only generate routes for the given actions.
1126
        #
1127 1128
        #     resources :cows, :only => :show
        #     resources :cows, :only => [:show, :index]
1129
        #
1130 1131
        # [:except]
        #   Generate all routes except for the given actions.
1132
        #
1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146
        #     resources :cows, :except => :show
        #     resources :cows, :except => [:show, :index]
        #
        # [:shallow]
        #   Generates shallow routes for nested resource(s). When placed on a parent resource,
        #   generates shallow routes for all nested resources.
        #
        #     resources :posts, :shallow => true do
        #       resources :comments
        #     end
        #
        #   Is the same as:
        #
        #     resources :posts do
1147
        #       resources :comments, :except => [:show, :edit, :update, :destroy]
1148
        #     end
1149 1150 1151 1152 1153
        #     resources :comments, :only => [:show, :edit, :update, :destroy]
        #
        #   This allows URLs for resources that otherwise would be deeply nested such
        #   as a comment on a blog post like <tt>/posts/a-long-permalink/comments/1234</tt>
        #   to be shortened to just <tt>/comments/1234</tt>.
1154 1155 1156 1157
        #
        # [:shallow_path]
        #   Prefixes nested shallow routes with the specified path.
        #
1158 1159 1160 1161
        #     scope :shallow_path => "sekret" do
        #       resources :posts do
        #         resources :comments, :shallow => true
        #       end
1162 1163 1164 1165
        #     end
        #
        #   The +comments+ resource here will have the following routes generated for it:
        #
1166 1167 1168 1169 1170
        #     post_comments    GET       /posts/:post_id/comments(.:format)
        #     post_comments    POST      /posts/:post_id/comments(.:format)
        #     new_post_comment GET       /posts/:post_id/comments/new(.:format)
        #     edit_comment     GET       /sekret/comments/:id/edit(.:format)
        #     comment          GET       /sekret/comments/:id(.:format)
1171
        #     comment          PATCH/PUT /sekret/comments/:id(.:format)
1172
        #     comment          DELETE    /sekret/comments/:id(.:format)
1173
        #
1174 1175 1176
        # [:shallow_prefix]
        #   Prefixes nested shallow route names with specified prefix.
        #
V
Vijay Dev 已提交
1177
        #     scope :shallow_prefix => "sekret" do
1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192
        #       resources :posts do
        #         resources :comments, :shallow => true
        #       end
        #     end
        #
        #   The +comments+ resource here will have the following routes generated for it:
        #
        #     post_comments           GET       /posts/:post_id/comments(.:format)
        #     post_comments           POST      /posts/:post_id/comments(.:format)
        #     new_post_comment        GET       /posts/:post_id/comments/new(.:format)
        #     edit_sekret_comment     GET       /comments/:id/edit(.:format)
        #     sekret_comment          GET       /comments/:id(.:format)
        #     sekret_comment          PATCH/PUT /comments/:id(.:format)
        #     sekret_comment          DELETE    /comments/:id(.:format)
        #
1193
        # [:format]
1194 1195
        #   Allows you to specify the default value for optional +format+
        #   segment or disable it if you supply +false+.
1196
        #
1197
        # === Examples
1198
        #
S
Sebastian Martinez 已提交
1199
        #   # routes call <tt>Admin::PostsController</tt>
1200
        #   resources :posts, :module => "admin"
1201
        #
1202
        #   # resource actions are at /admin/posts.
1203
        #   resources :posts, :path => "admin/posts"
J
Joshua Peek 已提交
1204
        def resources(*resources, &block)
J
Joshua Peek 已提交
1205
          options = resources.extract_options!
1206

1207
          if apply_common_behavior_for(:resources, resources, options, &block)
1208 1209 1210
            return self
          end

1211
          resource_scope(:resources, Resource.new(resources.pop, options)) do
1212
            yield if block_given?
J
Joshua Peek 已提交
1213

1214
            collection do
1215 1216
              get  :index if parent_resource.actions.include?(:index)
              post :create if parent_resource.actions.include?(:create)
1217
            end
1218

1219
            new do
1220
              get :new
1221
            end if parent_resource.actions.include?(:new)
1222

1223
            member do
1224 1225
              get :edit if parent_resource.actions.include?(:edit)
              get :show if parent_resource.actions.include?(:show)
1226
              if parent_resource.actions.include?(:update)
1227 1228
                patch :update
                put   :update
1229
              end
1230
              delete :destroy if parent_resource.actions.include?(:destroy)
1231 1232 1233
            end
          end

J
Joshua Peek 已提交
1234
          self
1235 1236
        end

1237 1238 1239 1240 1241 1242 1243 1244 1245
        # To add a route to the collection:
        #
        #   resources :photos do
        #     collection do
        #       get 'search'
        #     end
        #   end
        #
        # This will enable Rails to recognize paths such as <tt>/photos/search</tt>
S
Sebastian Martinez 已提交
1246
        # with GET, and route to the search action of +PhotosController+. It will also
1247 1248
        # create the <tt>search_photos_url</tt> and <tt>search_photos_path</tt>
        # route helpers.
J
Joshua Peek 已提交
1249
        def collection
1250 1251
          unless resource_scope?
            raise ArgumentError, "can't use collection outside resource(s) scope"
1252 1253
          end

1254 1255 1256 1257
          with_scope_level(:collection) do
            scope(parent_resource.collection_scope) do
              yield
            end
J
Joshua Peek 已提交
1258
          end
1259
        end
J
Joshua Peek 已提交
1260

1261 1262 1263 1264 1265 1266 1267 1268 1269
        # To add a member route, add a member block into the resource block:
        #
        #   resources :photos do
        #     member do
        #       get 'preview'
        #     end
        #   end
        #
        # This will recognize <tt>/photos/1/preview</tt> with GET, and route to the
S
Sebastian Martinez 已提交
1270
        # preview action of +PhotosController+. It will also create the
1271
        # <tt>preview_photo_url</tt> and <tt>preview_photo_path</tt> helpers.
J
Joshua Peek 已提交
1272
        def member
1273 1274
          unless resource_scope?
            raise ArgumentError, "can't use member outside resource(s) scope"
J
Joshua Peek 已提交
1275
          end
J
Joshua Peek 已提交
1276

1277 1278 1279 1280
          with_scope_level(:member) do
            scope(parent_resource.member_scope) do
              yield
            end
1281 1282 1283 1284 1285 1286 1287
          end
        end

        def new
          unless resource_scope?
            raise ArgumentError, "can't use new outside resource(s) scope"
          end
1288

1289 1290 1291 1292
          with_scope_level(:new) do
            scope(parent_resource.new_scope(action_path(:new))) do
              yield
            end
J
Joshua Peek 已提交
1293
          end
J
Joshua Peek 已提交
1294 1295
        end

1296
        def nested
1297 1298
          unless resource_scope?
            raise ArgumentError, "can't use nested outside resource(s) scope"
1299 1300 1301
          end

          with_scope_level(:nested) do
1302
            if shallow?
1303
              with_exclusive_scope do
1304
                if @scope[:shallow_path].blank?
1305
                  scope(parent_resource.nested_scope, nested_options) { yield }
1306
                else
1307
                  scope(@scope[:shallow_path], :as => @scope[:shallow_prefix]) do
1308
                    scope(parent_resource.nested_scope, nested_options) { yield }
1309 1310 1311 1312
                  end
                end
              end
            else
1313
              scope(parent_resource.nested_scope, nested_options) { yield }
1314 1315 1316 1317
            end
          end
        end

1318
        # See ActionDispatch::Routing::Mapper::Scoping#namespace
1319
        def namespace(path, options = {})
1320
          if resource_scope?
1321 1322 1323 1324 1325 1326
            nested { super }
          else
            super
          end
        end

1327
        def shallow
1328
          scope(:shallow => true, :shallow_path => @scope[:path]) do
1329 1330 1331 1332
            yield
          end
        end

1333 1334 1335 1336
        def shallow?
          parent_resource.instance_of?(Resource) && @scope[:shallow]
        end

1337
        # match 'path' => 'controller#action'
R
Rafael Mendonça França 已提交
1338
        # match 'path', to: 'controller#action'
1339
        # match 'path', 'otherpath', on: :member, via: :get
1340 1341 1342 1343
        def match(path, *rest)
          if rest.empty? && Hash === path
            options  = path
            path, to = options.find { |name, value| name.is_a?(String) }
1344 1345
            options[:to] = to
            options.delete(path)
1346 1347 1348 1349 1350 1351
            paths = [path]
          else
            options = rest.pop || {}
            paths = [path] + rest
          end

1352 1353
          options[:anchor] = true unless options.key?(:anchor)

A
Aaron Patterson 已提交
1354 1355 1356 1357
          if options[:on] && !VALID_ON_OPTIONS.include?(options[:on])
            raise ArgumentError, "Unknown scope #{on.inspect} given to :on"
          end

1358
          paths.each { |_path| decomposed_match(_path, options.dup) }
1359 1360
          self
        end
1361

1362
        def decomposed_match(path, options) # :nodoc:
A
Aaron Patterson 已提交
1363 1364
          if on = options.delete(:on)
            send(on) { decomposed_match(path, options) }
1365
          else
A
Aaron Patterson 已提交
1366 1367 1368 1369 1370 1371 1372 1373
            case @scope[:scope_level]
            when :resources
              nested { decomposed_match(path, options) }
            when :resource
              member { decomposed_match(path, options) }
            else
              add_route(path, options)
            end
J
Joshua Peek 已提交
1374
          end
1375
        end
J
Joshua Peek 已提交
1376

1377
        def add_route(action, options) # :nodoc:
1378
          path = path_for_action(action, options.delete(:path))
1379

1380 1381 1382
          if action.to_s =~ /^[\w\/]+$/
            options[:action] ||= action unless action.to_s.include?("/")
          else
1383 1384 1385
            action = nil
          end

1386
          if !options.fetch(:as, true)
1387 1388 1389
            options.delete(:as)
          else
            options[:as] = name_for_action(options[:as], action)
J
Joshua Peek 已提交
1390
          end
J
Joshua Peek 已提交
1391

1392
          mapping = Mapping.new(@set, @scope, URI.parser.escape(path), options)
1393 1394
          app, conditions, requirements, defaults, as, anchor = mapping.to_route
          @set.add_route(app, conditions, requirements, defaults, as, anchor)
J
Joshua Peek 已提交
1395 1396
        end

1397
        def root(options={})
1398
          if @scope[:scope_level] == :resources
1399 1400
            with_scope_level(:root) do
              scope(parent_resource.path) do
1401 1402 1403 1404 1405 1406
                super(options)
              end
            end
          else
            super(options)
          end
1407 1408
        end

1409
        protected
1410

1411
          def parent_resource #:nodoc:
1412 1413 1414
            @scope[:scope_level_resource]
          end

J
José Valim 已提交
1415
          def apply_common_behavior_for(method, resources, options, &block) #:nodoc:
1416 1417 1418 1419 1420
            if resources.length > 1
              resources.each { |r| send(method, r, options, &block) }
              return true
            end

1421 1422 1423 1424 1425
            if resource_scope?
              nested { send(method, resources.pop, options, &block) }
              return true
            end

1426
            options.keys.each do |k|
1427 1428 1429
              (options[:constraints] ||= {})[k] = options.delete(k) if options[k].is_a?(Regexp)
            end

1430 1431 1432
            scope_options = options.slice!(*RESOURCE_OPTIONS)
            unless scope_options.empty?
              scope(scope_options) do
1433 1434 1435 1436 1437
                send(method, resources.pop, options, &block)
              end
              return true
            end

1438 1439 1440 1441
            unless action_options?(options)
              options.merge!(scope_action_options) if scope_action_options?
            end

1442 1443 1444
            false
          end

J
José Valim 已提交
1445
          def action_options?(options) #:nodoc:
1446 1447 1448
            options[:only] || options[:except]
          end

J
José Valim 已提交
1449
          def scope_action_options? #:nodoc:
A
Aaron Patterson 已提交
1450
            @scope[:options] && (@scope[:options][:only] || @scope[:options][:except])
1451 1452
          end

J
José Valim 已提交
1453
          def scope_action_options #:nodoc:
1454 1455 1456
            @scope[:options].slice(:only, :except)
          end

J
José Valim 已提交
1457
          def resource_scope? #:nodoc:
1458
            [:resource, :resources].include? @scope[:scope_level]
1459 1460
          end

J
José Valim 已提交
1461
          def resource_method_scope? #:nodoc:
1462
            [:collection, :member, :new].include? @scope[:scope_level]
1463 1464
          end

1465
          def with_exclusive_scope
1466
            begin
1467 1468
              old_name_prefix, old_path = @scope[:as], @scope[:path]
              @scope[:as], @scope[:path] = nil, nil
1469

1470 1471 1472
              with_scope_level(:exclusive) do
                yield
              end
1473
            ensure
1474
              @scope[:as], @scope[:path] = old_name_prefix, old_path
1475 1476 1477
            end
          end

1478
          def with_scope_level(kind, resource = parent_resource)
J
Joshua Peek 已提交
1479
            old, @scope[:scope_level] = @scope[:scope_level], kind
1480
            old_resource, @scope[:scope_level_resource] = @scope[:scope_level_resource], resource
J
Joshua Peek 已提交
1481 1482 1483
            yield
          ensure
            @scope[:scope_level] = old
1484
            @scope[:scope_level_resource] = old_resource
J
Joshua Peek 已提交
1485
          end
1486

1487 1488
          def resource_scope(kind, resource) #:nodoc:
            with_scope_level(kind, resource) do
1489
              scope(parent_resource.resource_scope) do
1490 1491 1492 1493 1494
                yield
              end
            end
          end

J
José Valim 已提交
1495
          def nested_options #:nodoc:
1496 1497
            options = { :as => parent_resource.member_name }
            options[:constraints] = {
1498 1499
              parent_resource.nested_param => param_constraint
            } if param_constraint?
1500 1501

            options
1502 1503
          end

1504 1505
          def param_constraint? #:nodoc:
            @scope[:constraints] && @scope[:constraints][parent_resource.param].is_a?(Regexp)
1506 1507
          end

1508 1509
          def param_constraint #:nodoc:
            @scope[:constraints][parent_resource.param]
1510 1511
          end

J
José Valim 已提交
1512
          def canonical_action?(action, flag) #:nodoc:
1513
            flag && resource_method_scope? && CANONICAL_ACTIONS.include?(action.to_s)
1514 1515
          end

J
José Valim 已提交
1516
          def shallow_scoping? #:nodoc:
1517
            shallow? && @scope[:scope_level] == :member
1518 1519
          end

J
José Valim 已提交
1520
          def path_for_action(action, path) #:nodoc:
1521
            prefix = shallow_scoping? ?
1522
              "#{@scope[:shallow_path]}/#{parent_resource.shallow_scope}" : @scope[:path]
1523

1524
            if canonical_action?(action, path.blank?)
1525
              prefix.to_s
1526
            else
1527
              "#{prefix}/#{action_path(action, path)}"
1528 1529 1530
            end
          end

J
José Valim 已提交
1531
          def action_path(name, path = nil) #:nodoc:
1532
            name = name.to_sym if name.is_a?(String)
1533
            path || @scope[:path_names][name] || name.to_s
1534 1535
          end

J
José Valim 已提交
1536
          def prefix_name_for_action(as, action) #:nodoc:
1537
            if as
1538
              as.to_s
1539
            elsif !canonical_action?(action, @scope[:scope_level])
1540
              action.to_s
1541
            end
1542 1543
          end

J
José Valim 已提交
1544
          def name_for_action(as, action) #:nodoc:
1545
            prefix = prefix_name_for_action(as, action)
1546
            prefix = Mapper.normalize_name(prefix) if prefix
1547 1548 1549
            name_prefix = @scope[:as]

            if parent_resource
1550
              return nil unless as || action
1551

1552 1553
              collection_name = parent_resource.collection_name
              member_name = parent_resource.member_name
1554
            end
1555

1556
            name = case @scope[:scope_level]
1557
            when :nested
1558
              [name_prefix, prefix]
1559
            when :collection
1560
              [prefix, name_prefix, collection_name]
1561
            when :new
1562 1563 1564 1565 1566
              [prefix, :new, name_prefix, member_name]
            when :member
              [prefix, shallow_scoping? ? @scope[:shallow_prefix] : name_prefix, member_name]
            when :root
              [name_prefix, collection_name, prefix]
1567
            else
1568
              [name_prefix, member_name, prefix]
1569
            end
1570

1571 1572 1573 1574 1575 1576 1577 1578 1579 1580
            if candidate = name.select(&:present?).join("_").presence
              # If a name was not explicitly given, we check if it is valid
              # and return nil in case it isn't. Otherwise, we pass the invalid name
              # forward so the underlying router engine treats it and raises an exception.
              if as.nil?
                candidate unless @set.routes.find { |r| r.name == candidate } || candidate !~ /\A[_a-z]/i
              else
                candidate
              end
            end
1581
          end
J
Joshua Peek 已提交
1582
      end
J
Joshua Peek 已提交
1583

1584 1585 1586 1587 1588
      def initialize(set) #:nodoc:
        @set = set
        @scope = { :path_names => @set.resources_path_names }
      end

1589 1590
      include Base
      include HttpHelpers
1591
      include Redirection
1592 1593
      include Scoping
      include Resources
J
Joshua Peek 已提交
1594 1595
    end
  end
J
Joshua Peek 已提交
1596
end