mapper.rb 54.3 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
        def match(path, options=nil)
408
        end
409

410 411
        # Mount a Rack-based application to be used within the application.
        #
R
Ryan Bigg 已提交
412
        #   mount SomeRackApp, :at => "some_route"
413 414 415
        #
        # Alternatively:
        #
R
Ryan Bigg 已提交
416
        #   mount(SomeRackApp => "some_route")
417
        #
418 419
        # For options, see +match+, as +mount+ uses it internally.
        #
420 421 422 423 424
        # 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 已提交
425
        #   mount(SomeRackApp => "some_route", :as => "exciting")
426 427 428
        #
        # This will generate the +exciting_path+ and +exciting_url+ helpers
        # which can be used to navigate to this mounted app.
429 430 431 432 433 434 435 436 437 438 439
        def mount(app, options = nil)
          if options
            path = options.delete(:at)
          else
            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

440 441
          options[:as] ||= app_name(app)

442
          match(path, options.merge(:to => app, :anchor => false, :format => false, :via => :all))
443 444

          define_generate_prefix(app, options[:as])
445 446 447
          self
        end

448 449 450 451
        def default_url_options=(options)
          @set.default_url_options = options
        end
        alias_method :default_url_options, :default_url_options=
452

453 454 455 456 457 458
        def with_default_scope(scope, &block)
          scope(scope) do
            instance_exec(&block)
          end
        end

459 460 461
        private
          def app_name(app)
            return unless app.respond_to?(:routes)
462 463 464 465 466

            if app.respond_to?(:railtie_name)
              app.railtie_name
            else
              class_name = app.class.is_a?(Class) ? app.name : app.class.name
467
              ActiveSupport::Inflector.underscore(class_name).tr("/", "_")
468
            end
469 470 471
          end

          def define_generate_prefix(app, name)
472
            return unless app.respond_to?(:routes) && app.routes.respond_to?(:define_mounted_helper)
473 474

            _route = @set.named_routes.routes[name.to_sym]
P
Piotr Sarnacki 已提交
475 476
            _routes = @set
            app.routes.define_mounted_helper(name)
J
José Valim 已提交
477 478 479 480 481
            app.routes.singleton_class.class_eval do
              define_method :mounted? do
                true
              end

482
              define_method :_generate_prefix do |options|
P
Piotr Sarnacki 已提交
483
                prefix_options = options.slice(*_route.segment_keys)
484 485
                # 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 已提交
486 487 488
                prefix = _routes.url_helpers.send("#{name}_path", prefix_options)
                prefix = '' if prefix == '/'
                prefix
489 490 491
              end
            end
          end
492 493 494
      end

      module HttpHelpers
495
        # Define a route that only recognizes HTTP GET.
496
        # For supported arguments, see <tt>Base#match</tt>.
497
        #
498
        #   get 'bacon', :to => 'food#bacon'
499
        def get(*args, &block)
500
          map_method(:get, args, &block)
501 502
        end

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

511 512 513 514 515 516 517 518
        # 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

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

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

        private
536
          def map_method(method, args, &block)
537
            options = args.extract_options!
538 539
            options[:via]    = method
            options[:path] ||= args.first if args.first.is_a?(String)
540
            match(*args, options, &block)
541 542 543 544
            self
          end
      end

545 546 547
      # 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 已提交
548 549
      # the <tt>app/controllers/admin</tt> directory, and you can group them
      # together in your router:
550 551 552 553
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
554
      #
555
      # This will create a number of routes for each of the posts and comments
S
Sebastian Martinez 已提交
556
      # controller. For <tt>Admin::PostsController</tt>, Rails will create:
557
      #
558 559 560 561 562
      #   GET       /admin/posts
      #   GET       /admin/posts/new
      #   POST      /admin/posts
      #   GET       /admin/posts/1
      #   GET       /admin/posts/1/edit
563
      #   PATCH/PUT /admin/posts/1
564
      #   DELETE    /admin/posts/1
565
      #
566
      # If you want to route /posts (without the prefix /admin) to
S
Sebastian Martinez 已提交
567
      # <tt>Admin::PostsController</tt>, you could use
568
      #
569
      #   scope :module => "admin" do
570
      #     resources :posts
571 572 573
      #   end
      #
      # or, for a single case
574
      #
575
      #   resources :posts, :module => "admin"
576
      #
S
Sebastian Martinez 已提交
577
      # If you want to route /admin/posts to +PostsController+
578
      # (without the Admin:: module prefix), you could use
579
      #
580
      #   scope "/admin" do
581
      #     resources :posts
582 583 584
      #   end
      #
      # or, for a single case
585
      #
586
      #   resources :posts, :path => "/admin/posts"
587 588 589
      #
      # 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 已提交
590
      # +PostsController+:
591
      #
592 593 594 595 596
      #   GET       /admin/posts
      #   GET       /admin/posts/new
      #   POST      /admin/posts
      #   GET       /admin/posts/1
      #   GET       /admin/posts/1/edit
597
      #   PATCH/PUT /admin/posts/1
598
      #   DELETE    /admin/posts/1
599
      module Scoping
600
        # Scopes a set of routes to the given default options.
601 602 603 604 605 606 607 608
        #
        # 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.
609 610
        # The difference here being that the routes generated are like /:account_id/projects,
        # rather than /accounts/:account_id/projects.
611
        #
612
        # === Options
613
        #
614
        # Takes same options as <tt>Base#match</tt> and <tt>Resources#resources</tt>.
615
        #
616
        # === Examples
617
        #
S
Sebastian Martinez 已提交
618
        #   # route /posts (without the prefix /admin) to <tt>Admin::PostsController</tt>
619 620 621
        #   scope :module => "admin" do
        #     resources :posts
        #   end
622
        #
623 624 625 626
        #   # prefix the posts resource's requests with '/admin'
        #   scope :path => "/admin" do
        #     resources :posts
        #   end
627
        #
S
Sebastian Martinez 已提交
628
        #   # prefix the routing helper name: +sekret_posts_path+ instead of +posts_path+
629 630 631
        #   scope :as => "sekret" do
        #     resources :posts
        #   end
632 633
        def scope(*args)
          options = args.extract_options!
634
          options = options.dup
635

636
          options[:path] = args.first if args.first.is_a?(String)
637
          recover = {}
638

639 640
          options[:constraints] ||= {}
          unless options[:constraints].is_a?(Hash)
641
            block, options[:constraints] = options[:constraints], {}
642
          end
643

644 645 646 647
          if options[:constraints].is_a?(Hash)
            (options[:defaults] ||= {}).reverse_merge!(defaults_from_constraints(options[:constraints]))
          end

648 649 650 651 652
          scope_options.each do |option|
            if value = options.delete(option)
              recover[option] = @scope[option]
              @scope[option]  = send("merge_#{option}_scope", @scope[option], value)
            end
653 654
          end

655 656 657 658 659 660
          recover[:block] = @scope[:blocks]
          @scope[:blocks] = merge_blocks_scope(@scope[:blocks], block)

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

661 662 663
          yield
          self
        ensure
664 665 666
          scope_options.each do |option|
            @scope[option] = recover[option] if recover.has_key?(option)
          end
667 668 669

          @scope[:options] = recover[:options]
          @scope[:blocks]  = recover[:block]
670 671
        end

672 673 674 675 676
        # Scopes routes to a specific controller
        #
        #   controller "food" do
        #     match "bacon", :action => "bacon"
        #   end
677 678 679
        def controller(controller, options={})
          options[:controller] = controller
          scope(options) { yield }
680 681
        end

682 683 684 685 686 687 688 689
        # Scopes routes to a specific namespace. For example:
        #
        #   namespace :admin do
        #     resources :posts
        #   end
        #
        # This generates the following routes:
        #
690 691 692 693 694
        #       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
695
        #        admin_post PATCH/PUT /admin/posts/:id(.:format)      admin/posts#update
696
        #        admin_post DELETE    /admin/posts/:id(.:format)      admin/posts#destroy
697
        #
698
        # === Options
699
        #
700 701
        # The +:path+, +:as+, +:module+, +:shallow_path+ and +:shallow_prefix+
        # options all default to the name of the namespace.
702
        #
703 704
        # For options, see <tt>Base#match</tt>. For +:shallow_path+ option, see
        # <tt>Resources#resources</tt>.
705
        #
706
        # === Examples
707
        #
708 709 710 711
        #   # accessible through /sekret/posts rather than /admin/posts
        #   namespace :admin, :path => "sekret" do
        #     resources :posts
        #   end
712
        #
S
Sebastian Martinez 已提交
713
        #   # maps to <tt>Sekret::PostsController</tt> rather than <tt>Admin::PostsController</tt>
714 715 716
        #   namespace :admin, :module => "sekret" do
        #     resources :posts
        #   end
717
        #
S
Sebastian Martinez 已提交
718
        #   # generates +sekret_posts_path+ rather than +admin_posts_path+
719 720 721
        #   namespace :admin, :as => "sekret" do
        #     resources :posts
        #   end
722
        def namespace(path, options = {})
723
          path = path.to_s
724 725 726
          options = { :path => path, :as => path, :module => path,
                      :shallow_path => path, :shallow_prefix => path }.merge!(options)
          scope(options) { yield }
727
        end
728

R
Ryan Bigg 已提交
729 730 731 732
        # === 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 已提交
733
        #   constraints(:id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
734 735 736 737 738
        #     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.
739
        #
R
R.T. Lechow 已提交
740
        # You may use this to also restrict other parameters:
R
Ryan Bigg 已提交
741 742
        #
        #   resources :posts do
M
mjy 已提交
743
        #     constraints(:post_id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
744 745
        #       resources :comments
        #     end
J
James Miller 已提交
746
        #   end
R
Ryan Bigg 已提交
747 748 749 750 751 752 753 754 755 756 757 758 759 760
        #
        # === 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 已提交
761
        # Requests to routes can be constrained based on specific criteria:
R
Ryan Bigg 已提交
762 763 764 765 766 767 768 769 770 771
        #
        #    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
772
        #      def self.matches?(request)
R
Ryan Bigg 已提交
773 774 775 776 777 778 779 780 781 782 783
        #        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
784 785 786 787
        def constraints(constraints = {})
          scope(:constraints => constraints) { yield }
        end

R
Ryan Bigg 已提交
788
        # Allows you to set default parameters for a route, such as this:
789 790 791
        #   defaults :id => 'home' do
        #     match 'scoped_pages/(:id)', :to => 'pages#show'
        #   end
R
Ryan Bigg 已提交
792
        # Using this, the +:id+ parameter here will default to 'home'.
793 794 795 796
        def defaults(defaults = {})
          scope(:defaults => defaults) { yield }
        end

797
        private
J
José Valim 已提交
798
          def scope_options #:nodoc:
799 800 801
            @scope_options ||= private_methods.grep(/^merge_(.+)_scope$/) { $1.to_sym }
          end

J
José Valim 已提交
802
          def merge_path_scope(parent, child) #:nodoc:
803
            Mapper.normalize_path("#{parent}/#{child}")
804 805
          end

J
José Valim 已提交
806
          def merge_shallow_path_scope(parent, child) #:nodoc:
807 808 809
            Mapper.normalize_path("#{parent}/#{child}")
          end

J
José Valim 已提交
810
          def merge_as_scope(parent, child) #:nodoc:
811
            parent ? "#{parent}_#{child}" : child
812 813
          end

J
José Valim 已提交
814
          def merge_shallow_prefix_scope(parent, child) #:nodoc:
815 816 817
            parent ? "#{parent}_#{child}" : child
          end

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

J
José Valim 已提交
822
          def merge_controller_scope(parent, child) #:nodoc:
823
            child
824 825
          end

J
José Valim 已提交
826
          def merge_path_names_scope(parent, child) #:nodoc:
827 828 829
            merge_options_scope(parent, child)
          end

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

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

J
José Valim 已提交
838
          def merge_blocks_scope(parent, child) #:nodoc:
839 840 841
            merged = parent ? parent.dup : []
            merged << child if child
            merged
842 843
          end

J
José Valim 已提交
844
          def merge_options_scope(parent, child) #:nodoc:
845
            (parent || {}).except(*override_keys(child)).merge(child)
846
          end
847

J
José Valim 已提交
848
          def merge_shallow_scope(parent, child) #:nodoc:
849 850
            child ? true : false
          end
851

J
José Valim 已提交
852
          def override_keys(child) #:nodoc:
853 854
            child.key?(:only) || child.key?(:except) ? [:only, :except] : []
          end
855 856 857 858 859

          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
860 861
      end

862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885
      # 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 已提交
886 887
      # <tt>app/controllers/admin</tt> directory, and you can group them together
      # in your router:
888 889 890 891 892
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
      #
S
Sebastian Martinez 已提交
893 894
      # 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
895 896 897 898
      # overrides this restriction, e.g:
      #
      #   resources :articles, :id => /[^\/]+/
      #
S
Sebastian Martinez 已提交
899
      # This allows any character other than a slash as part of your +:id+.
900
      #
J
Joshua Peek 已提交
901
      module Resources
902 903
        # CANONICAL_ACTIONS holds all actions that does not need a prefix or
        # a path appended since they fit properly in their scope level.
904
        VALID_ON_OPTIONS  = [:new, :collection, :member]
905
        RESOURCE_OPTIONS  = [:as, :controller, :path, :only, :except, :param]
906
        CANONICAL_ACTIONS = %w(index create new show update destroy)
907

908
        class Resource #:nodoc:
909
          attr_reader :controller, :path, :options, :param
910 911

          def initialize(entities, options = {})
912
            @name       = entities.to_s
913 914 915
            @path       = (options[:path] || @name).to_s
            @controller = (options[:controller] || @name).to_s
            @as         = options[:as]
916
            @param      = options[:param] || :id
917
            @options    = options
918 919
          end

920
          def default_actions
921
            [:index, :create, :new, :show, :update, :destroy, :edit]
922 923
          end

924
          def actions
925
            if only = @options[:only]
926
              Array(only).map(&:to_sym)
927
            elsif except = @options[:except]
928 929 930 931 932 933
              default_actions - Array(except).map(&:to_sym)
            else
              default_actions
            end
          end

934
          def name
935
            @as || @name
936 937
          end

938
          def plural
939
            @plural ||= name.to_s
940 941 942
          end

          def singular
943
            @singular ||= name.to_s.singularize
944 945
          end

946
          alias :member_name :singular
947

948
          # Checks for uncountable plurals, and appends "_index" if the plural
949
          # and singular form are the same.
950
          def collection_name
951
            singular == plural ? "#{plural}_index" : plural
952 953
          end

954
          def resource_scope
955
            { :controller => controller }
956 957
          end

958
          alias :collection_scope :path
959 960

          def member_scope
961
            "#{path}/:#{param}"
962 963
          end

964
          def new_scope(new_path)
965
            "#{path}/#{new_path}"
966 967 968
          end

          def nested_scope
969
            "#{path}/:#{singular}_#{param}"
970
          end
971

972 973 974
        end

        class SingletonResource < Resource #:nodoc:
975
          def initialize(entities, options)
976
            super
977
            @as         = nil
978 979
            @controller = (options[:controller] || plural).to_s
            @as         = options[:as]
980 981
          end

982 983 984 985
          def default_actions
            [:show, :create, :update, :destroy, :new, :edit]
          end

986 987
          def plural
            @plural ||= name.to_s.pluralize
988 989
          end

990 991
          def singular
            @singular ||= name.to_s
992
          end
993 994 995 996 997 998

          alias :member_name :singular
          alias :collection_name :singular

          alias :member_scope :path
          alias :nested_scope :path
999 1000
        end

1001 1002 1003 1004
        def resources_path_names(options)
          @scope[:path_names].merge!(options)
        end

1005 1006 1007 1008 1009 1010 1011 1012 1013
        # 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 已提交
1014
        # the +GeoCoders+ controller (note that the controller is named after
1015 1016
        # the plural):
        #
1017 1018 1019 1020
        #   GET       /geocoder/new
        #   POST      /geocoder
        #   GET       /geocoder
        #   GET       /geocoder/edit
1021
        #   PATCH/PUT /geocoder
1022
        #   DELETE    /geocoder
1023
        #
1024
        # === Options
1025
        # Takes same options as +resources+.
J
Joshua Peek 已提交
1026
        def resource(*resources, &block)
J
Joshua Peek 已提交
1027
          options = resources.extract_options!
J
Joshua Peek 已提交
1028

1029
          if apply_common_behavior_for(:resource, resources, options, &block)
1030 1031 1032
            return self
          end

1033
          resource_scope(:resource, SingletonResource.new(resources.pop, options)) do
1034
            yield if block_given?
1035

1036
            collection do
1037
              post :create
1038
            end if parent_resource.actions.include?(:create)
1039

1040
            new do
1041
              get :new
1042
            end if parent_resource.actions.include?(:new)
1043

1044
            member do
1045 1046
              get :edit if parent_resource.actions.include?(:edit)
              get :show if parent_resource.actions.include?(:show)
1047
              if parent_resource.actions.include?(:update)
1048 1049
                patch :update
                put   :update
1050
              end
1051
              delete :destroy if parent_resource.actions.include?(:destroy)
1052 1053 1054
            end
          end

J
Joshua Peek 已提交
1055
          self
1056 1057
        end

1058 1059 1060 1061 1062 1063 1064 1065
        # 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 已提交
1066
        # the +Photos+ controller:
1067
        #
1068 1069 1070 1071 1072
        #   GET       /photos
        #   GET       /photos/new
        #   POST      /photos
        #   GET       /photos/:id
        #   GET       /photos/:id/edit
1073
        #   PATCH/PUT /photos/:id
1074
        #   DELETE    /photos/:id
1075
        #
1076 1077 1078 1079 1080 1081 1082 1083
        # Resources can also be nested infinitely by using this block syntax:
        #
        #   resources :photos do
        #     resources :comments
        #   end
        #
        # This generates the following comments routes:
        #
1084 1085 1086 1087 1088
        #   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
1089
        #   PATCH/PUT /photos/:photo_id/comments/:id
1090
        #   DELETE    /photos/:photo_id/comments/:id
1091
        #
1092
        # === Options
1093 1094
        # Takes same options as <tt>Base#match</tt> as well as:
        #
1095
        # [:path_names]
A
Aviv Ben-Yosef 已提交
1096 1097
        #   Allows you to change the segment component of the +edit+ and +new+ actions.
        #   Actions not specified are not changed.
1098 1099 1100 1101
        #
        #     resources :posts, :path_names => { :new => "brand_new" }
        #
        #   The above example will now change /posts/new to /posts/brand_new
1102
        #
1103 1104 1105 1106 1107 1108 1109
        # [: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
        #
1110 1111
        # [:only]
        #   Only generate routes for the given actions.
1112
        #
1113 1114
        #     resources :cows, :only => :show
        #     resources :cows, :only => [:show, :index]
1115
        #
1116 1117
        # [:except]
        #   Generate all routes except for the given actions.
1118
        #
1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132
        #     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
1133
        #       resources :comments, :except => [:show, :edit, :update, :destroy]
1134
        #     end
1135 1136 1137 1138 1139
        #     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>.
1140 1141 1142 1143
        #
        # [:shallow_path]
        #   Prefixes nested shallow routes with the specified path.
        #
1144 1145 1146 1147
        #     scope :shallow_path => "sekret" do
        #       resources :posts do
        #         resources :comments, :shallow => true
        #       end
1148 1149 1150 1151
        #     end
        #
        #   The +comments+ resource here will have the following routes generated for it:
        #
1152 1153 1154 1155 1156
        #     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)
1157
        #     comment          PATCH/PUT /sekret/comments/:id(.:format)
1158
        #     comment          DELETE    /sekret/comments/:id(.:format)
1159
        #
1160 1161 1162
        # [:shallow_prefix]
        #   Prefixes nested shallow route names with specified prefix.
        #
V
Vijay Dev 已提交
1163
        #     scope :shallow_prefix => "sekret" do
1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178
        #       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)
        #
1179
        # === Examples
1180
        #
S
Sebastian Martinez 已提交
1181
        #   # routes call <tt>Admin::PostsController</tt>
1182
        #   resources :posts, :module => "admin"
1183
        #
1184
        #   # resource actions are at /admin/posts.
1185
        #   resources :posts, :path => "admin/posts"
J
Joshua Peek 已提交
1186
        def resources(*resources, &block)
J
Joshua Peek 已提交
1187
          options = resources.extract_options!
1188

1189
          if apply_common_behavior_for(:resources, resources, options, &block)
1190 1191 1192
            return self
          end

1193
          resource_scope(:resources, Resource.new(resources.pop, options)) do
1194
            yield if block_given?
J
Joshua Peek 已提交
1195

1196
            collection do
1197 1198
              get  :index if parent_resource.actions.include?(:index)
              post :create if parent_resource.actions.include?(:create)
1199
            end
1200

1201
            new do
1202
              get :new
1203
            end if parent_resource.actions.include?(:new)
1204

1205
            member do
1206 1207
              get :edit if parent_resource.actions.include?(:edit)
              get :show if parent_resource.actions.include?(:show)
1208
              if parent_resource.actions.include?(:update)
1209 1210
                patch :update
                put   :update
1211
              end
1212
              delete :destroy if parent_resource.actions.include?(:destroy)
1213 1214 1215
            end
          end

J
Joshua Peek 已提交
1216
          self
1217 1218
        end

1219 1220 1221 1222 1223 1224 1225 1226 1227
        # 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 已提交
1228
        # with GET, and route to the search action of +PhotosController+. It will also
1229 1230
        # create the <tt>search_photos_url</tt> and <tt>search_photos_path</tt>
        # route helpers.
J
Joshua Peek 已提交
1231
        def collection
1232 1233
          unless resource_scope?
            raise ArgumentError, "can't use collection outside resource(s) scope"
1234 1235
          end

1236 1237 1238 1239
          with_scope_level(:collection) do
            scope(parent_resource.collection_scope) do
              yield
            end
J
Joshua Peek 已提交
1240
          end
1241
        end
J
Joshua Peek 已提交
1242

1243 1244 1245 1246 1247 1248 1249 1250 1251
        # 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 已提交
1252
        # preview action of +PhotosController+. It will also create the
1253
        # <tt>preview_photo_url</tt> and <tt>preview_photo_path</tt> helpers.
J
Joshua Peek 已提交
1254
        def member
1255 1256
          unless resource_scope?
            raise ArgumentError, "can't use member outside resource(s) scope"
J
Joshua Peek 已提交
1257
          end
J
Joshua Peek 已提交
1258

1259 1260 1261 1262
          with_scope_level(:member) do
            scope(parent_resource.member_scope) do
              yield
            end
1263 1264 1265 1266 1267 1268 1269
          end
        end

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

1271 1272 1273 1274
          with_scope_level(:new) do
            scope(parent_resource.new_scope(action_path(:new))) do
              yield
            end
J
Joshua Peek 已提交
1275
          end
J
Joshua Peek 已提交
1276 1277
        end

1278
        def nested
1279 1280
          unless resource_scope?
            raise ArgumentError, "can't use nested outside resource(s) scope"
1281 1282 1283
          end

          with_scope_level(:nested) do
1284
            if shallow?
1285
              with_exclusive_scope do
1286
                if @scope[:shallow_path].blank?
1287
                  scope(parent_resource.nested_scope, nested_options) { yield }
1288
                else
1289
                  scope(@scope[:shallow_path], :as => @scope[:shallow_prefix]) do
1290
                    scope(parent_resource.nested_scope, nested_options) { yield }
1291 1292 1293 1294
                  end
                end
              end
            else
1295
              scope(parent_resource.nested_scope, nested_options) { yield }
1296 1297 1298 1299
            end
          end
        end

1300
        # See ActionDispatch::Routing::Mapper::Scoping#namespace
1301
        def namespace(path, options = {})
1302
          if resource_scope?
1303 1304 1305 1306 1307 1308
            nested { super }
          else
            super
          end
        end

1309
        def shallow
1310
          scope(:shallow => true, :shallow_path => @scope[:path]) do
1311 1312 1313 1314
            yield
          end
        end

1315 1316 1317 1318
        def shallow?
          parent_resource.instance_of?(Resource) && @scope[:shallow]
        end

1319
        def draw(name)
1320
          path = @draw_paths.find do |_path|
1321
            File.exists? "#{_path}/#{name}.rb"
1322 1323 1324 1325 1326
          end

          unless path
            msg  = "Your router tried to #draw the external file #{name}.rb,\n" \
                   "but the file was not found in:\n\n"
1327
            msg += @draw_paths.map { |_path| " * #{_path}" }.join("\n")
1328
            raise ArgumentError, msg
1329
          end
1330

1331 1332
          route_path = "#{path}/#{name}.rb"
          instance_eval(File.read(route_path), route_path.to_s)
1333 1334
        end

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

1350 1351
          options[:anchor] = true unless options.key?(:anchor)

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

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

1360
        def decomposed_match(path, options) # :nodoc:
A
Aaron Patterson 已提交
1361 1362
          if on = options.delete(:on)
            send(on) { decomposed_match(path, options) }
1363
          else
A
Aaron Patterson 已提交
1364 1365 1366 1367 1368 1369 1370 1371
            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 已提交
1372
          end
1373
        end
J
Joshua Peek 已提交
1374

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

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

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

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

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

1407
        protected
1408

1409
          def parent_resource #:nodoc:
1410 1411 1412
            @scope[:scope_level_resource]
          end

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

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

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

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

1436 1437 1438 1439
            unless action_options?(options)
              options.merge!(scope_action_options) if scope_action_options?
            end

1440 1441 1442
            false
          end

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

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

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

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

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

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

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

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

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

J
José Valim 已提交
1493
          def nested_options #:nodoc:
1494 1495 1496 1497 1498 1499
            options = { :as => parent_resource.member_name }
            options[:constraints] = {
              :"#{parent_resource.singular}_id" => id_constraint
            } if id_constraint?

            options
1500 1501
          end

J
José Valim 已提交
1502
          def id_constraint? #:nodoc:
1503
            @scope[:constraints] && @scope[:constraints][:id].is_a?(Regexp)
1504 1505
          end

J
José Valim 已提交
1506
          def id_constraint #:nodoc:
1507
            @scope[:constraints][:id]
1508 1509
          end

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

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

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

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

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

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

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

            if parent_resource
1548
              return nil unless as || action
1549

1550 1551
              collection_name = parent_resource.collection_name
              member_name = parent_resource.member_name
1552
            end
1553

1554
            name = case @scope[:scope_level]
1555
            when :nested
1556
              [name_prefix, prefix]
1557
            when :collection
1558
              [prefix, name_prefix, collection_name]
1559
            when :new
1560 1561 1562 1563 1564
              [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]
1565
            else
1566
              [name_prefix, member_name, prefix]
1567
            end
1568

1569 1570 1571 1572 1573 1574 1575 1576 1577 1578
            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
1579
          end
J
Joshua Peek 已提交
1580
      end
J
Joshua Peek 已提交
1581

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

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