mapper.rb 50.5 KB
Newer Older
1
require 'active_support/core_ext/hash/except'
2
require 'active_support/core_ext/object/blank'
3
require 'active_support/core_ext/object/inclusion'
4
require 'active_support/inflector'
5
require 'action_dispatch/routing/redirection'
6

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

19
        attr_reader :app, :constraints
20

21 22
        def initialize(app, constraints, request)
          @app, @constraints, @request = app, constraints, request
23 24
        end

25
        def matches?(env)
26
          req = @request.new(env)
27 28 29

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

36 37 38 39 40
          return true
        end

        def call(env)
          matches?(env) ? @app.call(env) : [ 404, {'X-Cascade' => 'pass'}, [] ]
41
        end
42 43 44 45 46

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

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

55
        def initialize(set, scope, path, options)
56
          @set, @scope = set, scope
57
          @segment_keys = nil
58
          @options = (@scope[:options] || {}).merge(options)
59
          @path = normalize_path(path)
60
          normalize_options!
61
        end
J
Joshua Peek 已提交
62

63
        def to_route
64
          [ app, conditions, requirements, defaults, @options[:as], @options[:anchor] ]
65
        end
J
Joshua Peek 已提交
66

67
        private
68 69 70

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

72 73
            if using_match_shorthand?(path_without_format, @options)
              to_shorthand    = @options[:to].blank?
74
              @options[:to] ||= path_without_format.gsub(/\(.*\)/, "")[1..-1].sub(%r{/([^/]*)$}, '#\1')
75 76
            end

77
            @options.merge!(default_controller_and_action(to_shorthand))
78 79 80 81 82

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

83
              if requirement.source =~ ANCHOR_CHARACTERS_REGEX
84 85 86 87 88 89
                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
90
          end
91

92
          # match "account/overview"
93
          def using_match_shorthand?(path, options)
94
            path && (options[:to] || options[:action]).nil? && path =~ SHORTHAND_REGEX
95
          end
96

97
          def normalize_path(path)
98 99
            raise ArgumentError, "path is required" if path.blank?
            path = Mapper.normalize_path(path)
100 101 102 103 104 105 106 107

            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' }
108
              @options[:controller] ||= /.+?/
109 110
            end

111 112
            # Add a constraint for wildcard route to make it non-greedy and match the
            # optional format part of the route by default
113
            if path.match(WILDCARD_PATH) && @options[:format] != false
114
              @options[$1.to_sym] ||= /.+?/
115 116
            end

117 118 119
            if @options[:format] == false
              @options.delete(:format)
              path
120
            elsif path.include?(":format") || path.end_with?('/')
121
              path
122 123
            elsif @options[:format] == true
              "#{path}.:format"
124 125 126
            else
              "#{path}(.:format)"
            end
127
          end
128

129 130
          def app
            Constraints.new(
131
              to.respond_to?(:call) ? to : Routing::RouteSet::Dispatcher.new(:defaults => defaults),
132 133
              blocks,
              @set.request_class
134
            )
135 136
          end

137 138 139
          def conditions
            { :path_info => @path }.merge(constraints).merge(request_method_condition)
          end
J
Joshua Peek 已提交
140

141
          def requirements
142
            @requirements ||= (@options[:constraints].is_a?(Hash) ? @options[:constraints] : {}).tap do |requirements|
143 144 145 146
              requirements.reverse_merge!(@scope[:constraints]) if @scope[:constraints]
              @options.each { |k, v| requirements[k] = v if v.is_a?(Regexp) }
            end
          end
147

148
          def defaults
149 150 151 152 153 154
            @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

155
          def default_controller_and_action(to_shorthand=nil)
156
            if to.respond_to?(:call)
157 158
              { }
            else
159
              if to.is_a?(String)
160
                controller, action = to.split('#')
161 162
              elsif to.is_a?(Symbol)
                action = to.to_s
163
              end
J
Joshua Peek 已提交
164

165 166
              controller ||= default_controller
              action     ||= default_action
167

168 169 170
              unless controller.is_a?(Regexp) || to_shorthand
                controller = [@scope[:module], controller].compact.join("/").presence
              end
171

172 173 174 175
              if controller.is_a?(String) && controller =~ %r{\A/}
                raise ArgumentError, "controller name should not start with a slash"
              end

176 177
              controller = controller.to_s unless controller.is_a?(Regexp)
              action     = action.to_s     unless action.is_a?(Regexp)
178

179
              if controller.blank? && segment_keys.exclude?("controller")
180 181
                raise ArgumentError, "missing :controller"
              end
J
Joshua Peek 已提交
182

183
              if action.blank? && segment_keys.exclude?("action")
184 185
                raise ArgumentError, "missing :action"
              end
J
Joshua Peek 已提交
186

A
Aaron Patterson 已提交
187
              hash = {}
A
Aaron Patterson 已提交
188 189
              hash[:controller] = controller unless controller.blank?
              hash[:action]     = action unless action.blank?
A
Aaron Patterson 已提交
190
              hash
191 192
            end
          end
193

194
          def blocks
195 196 197 198 199
            constraints = @options[:constraints]
            if constraints.present? && !constraints.is_a?(Hash)
              [constraints]
            else
              @scope[:blocks] || []
200 201
            end
          end
J
Joshua Peek 已提交
202

203 204 205
          def constraints
            @constraints ||= requirements.reject { |k, v| segment_keys.include?(k.to_s) || k == :controller }
          end
206

207 208
          def request_method_condition
            if via = @options[:via]
209 210
              list = Array(via).map { |m| m.to_s.dasherize.upcase }
              { :request_method => list }
211 212
            else
              { }
213
            end
214
          end
J
Joshua Peek 已提交
215

216
          def segment_keys
217 218 219
            return @segment_keys if @segment_keys

            @segment_keys = Journey::Path::Pattern.new(
220
              Journey::Router::Strexp.compile(@path, requirements, SEPARATORS)
221
            ).names
222
          end
223

224 225 226
          def to
            @options[:to]
          end
J
Joshua Peek 已提交
227

228
          def default_controller
229
            @options[:controller] || @scope[:controller]
230
          end
231 232

          def default_action
233
            @options[:action] || @scope[:action]
234
          end
235
      end
236

237
      # Invokes Rack::Mount::Utils.normalize path and ensure that
238 239
      # (:locale) becomes (/:locale) instead of /(:locale). Except
      # for root cases, where the latter is the correct one.
240
      def self.normalize_path(path)
241
        path = Journey::Router::Utils.normalize_path(path)
242
        path.gsub!(%r{/(\(+)/?}, '\1/') unless path =~ %r{^/\(+[^/]+\)$}
243 244 245
        path
      end

246 247 248 249
      def self.normalize_name(name)
        normalize_path(name)[1..-1].gsub("/", "_")
      end

250
      module Base
251 252 253 254
        # You can specify what Rails should route "/" to with the root method:
        #
        #   root :to => 'pages#main'
        #
255
        # For options, see +match+, as +root+ uses it internally.
256
        #
257 258 259
        # 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.
260
        def root(options = {})
261
          match '/', { :as => :root }.merge(options)
262
        end
263

264 265 266
        # 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:
267
        #
268
        #   # sets :controller, :action and :id in params
269
        #   match ':controller/:action/:id'
270
        #
271 272 273 274 275 276 277 278 279 280 281 282
        # 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:
283 284 285 286
        #
        #   match 'photos/:id' => 'photos#show'
        #   match 'photos/:id', :to => 'photos#show'
        #   match 'photos/:id', :controller => 'photos', :action => 'show'
287
        #
288 289 290
        # A pattern can also point to a +Rack+ endpoint i.e. anything that
        # responds to +call+:
        #
A
Alexey Vakhov 已提交
291
        #   match 'photos/:id' => lambda {|hash| [200, {}, "Coming soon"] }
292 293 294 295
        #   match 'photos/:id' => PhotoRackApp
        #   # Yes, controller actions are just rack endpoints
        #   match 'photos/:id' => PhotosController.action(:show)
        #
296
        # === Options
297
        #
298
        # Any options not seen here are passed on as params with the url.
299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326
        #
        # [: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]
327 328
        #   Points to a +Rack+ endpoint. Can be an object that responds to
        #   +call+ or a string representing a controller's action.
329
        #
330 331 332
        #      match 'path', :to => 'controller#action'
        #      match 'path', :to => lambda { [200, {}, "Success!"] }
        #      match 'path', :to => RackApp
333 334 335
        #
        # [:on]
        #   Shorthand for wrapping routes in a specific RESTful context. Valid
336
        #   values are +:member+, +:collection+, and +:new+. Only use within
337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352
        #   <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
353
        #   object that responds to <tt>matches?</tt>
354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
        #
        #     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.
372 373
        #
        # [:anchor]
374
        #   Boolean to anchor a <tt>match</tt> pattern. Default is true. When set to
375 376 377 378
        #   false, the pattern matches any request prefixed with the given path.
        #
        #     # Matches any request starting with 'path'
        #     match 'path' => 'c#a', :anchor => false
379
        def match(path, options=nil)
380
        end
381

382 383
        # Mount a Rack-based application to be used within the application.
        #
R
Ryan Bigg 已提交
384
        #   mount SomeRackApp, :at => "some_route"
385 386 387
        #
        # Alternatively:
        #
R
Ryan Bigg 已提交
388
        #   mount(SomeRackApp => "some_route")
389
        #
390 391
        # For options, see +match+, as +mount+ uses it internally.
        #
392 393 394 395 396
        # 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 已提交
397
        #   mount(SomeRackApp => "some_route", :as => "exciting")
398 399 400
        #
        # This will generate the +exciting_path+ and +exciting_url+ helpers
        # which can be used to navigate to this mounted app.
401 402 403 404 405 406 407 408 409 410 411
        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

412 413
          options[:as] ||= app_name(app)

414
          match(path, options.merge(:to => app, :anchor => false, :format => false))
415 416

          define_generate_prefix(app, options[:as])
417 418 419
          self
        end

420 421 422 423
        def default_url_options=(options)
          @set.default_url_options = options
        end
        alias_method :default_url_options, :default_url_options=
424

425 426 427 428 429 430
        def with_default_scope(scope, &block)
          scope(scope) do
            instance_exec(&block)
          end
        end

431 432 433
        private
          def app_name(app)
            return unless app.respond_to?(:routes)
434 435 436 437 438 439 440

            if app.respond_to?(:railtie_name)
              app.railtie_name
            else
              class_name = app.class.is_a?(Class) ? app.name : app.class.name
              ActiveSupport::Inflector.underscore(class_name).gsub("/", "_")
            end
441 442 443
          end

          def define_generate_prefix(app, name)
444
            return unless app.respond_to?(:routes) && app.routes.respond_to?(:define_mounted_helper)
445 446

            _route = @set.named_routes.routes[name.to_sym]
P
Piotr Sarnacki 已提交
447 448
            _routes = @set
            app.routes.define_mounted_helper(name)
449 450
            app.routes.class_eval do
              define_method :_generate_prefix do |options|
P
Piotr Sarnacki 已提交
451
                prefix_options = options.slice(*_route.segment_keys)
452 453
                # 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 已提交
454 455 456
                prefix = _routes.url_helpers.send("#{name}_path", prefix_options)
                prefix = '' if prefix == '/'
                prefix
457 458 459
              end
            end
          end
460 461 462
      end

      module HttpHelpers
463
        # Define a route that only recognizes HTTP GET.
464
        # For supported arguments, see <tt>Base#match</tt>.
465 466 467 468
        #
        # Example:
        #
        # get 'bacon', :to => 'food#bacon'
469
        def get(*args, &block)
470
          map_method(:get, args, &block)
471 472
        end

473
        # Define a route that only recognizes HTTP POST.
474
        # For supported arguments, see <tt>Base#match</tt>.
475 476 477 478
        #
        # Example:
        #
        # post 'bacon', :to => 'food#bacon'
479
        def post(*args, &block)
480
          map_method(:post, args, &block)
481 482
        end

483
        # Define a route that only recognizes HTTP PUT.
484
        # For supported arguments, see <tt>Base#match</tt>.
485 486 487 488
        #
        # Example:
        #
        # put 'bacon', :to => 'food#bacon'
489
        def put(*args, &block)
490
          map_method(:put, args, &block)
491 492
        end

493
        # Define a route that only recognizes HTTP DELETE.
494
        # For supported arguments, see <tt>Base#match</tt>.
495 496 497 498
        #
        # Example:
        #
        # delete 'broccoli', :to => 'food#broccoli'
499
        def delete(*args, &block)
500
          map_method(:delete, args, &block)
501 502 503
        end

        private
504
          def map_method(method, args, &block)
505 506
            options = args.extract_options!
            options[:via] = method
507
            match(*args, options, &block)
508 509 510 511
            self
          end
      end

512 513 514
      # 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 已提交
515 516
      # the <tt>app/controllers/admin</tt> directory, and you can group them
      # together in your router:
517 518 519 520
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
521
      #
522
      # This will create a number of routes for each of the posts and comments
S
Sebastian Martinez 已提交
523
      # controller. For <tt>Admin::PostsController</tt>, Rails will create:
524
      #
525 526 527 528 529 530 531
      #   GET	    /admin/posts
      #   GET	    /admin/posts/new
      #   POST	  /admin/posts
      #   GET	    /admin/posts/1
      #   GET	    /admin/posts/1/edit
      #   PUT	    /admin/posts/1
      #   DELETE  /admin/posts/1
532
      #
533
      # If you want to route /posts (without the prefix /admin) to
S
Sebastian Martinez 已提交
534
      # <tt>Admin::PostsController</tt>, you could use
535
      #
536
      #   scope :module => "admin" do
537
      #     resources :posts
538 539 540
      #   end
      #
      # or, for a single case
541
      #
542
      #   resources :posts, :module => "admin"
543
      #
S
Sebastian Martinez 已提交
544
      # If you want to route /admin/posts to +PostsController+
545
      # (without the Admin:: module prefix), you could use
546
      #
547
      #   scope "/admin" do
548
      #     resources :posts
549 550 551
      #   end
      #
      # or, for a single case
552
      #
553
      #   resources :posts, :path => "/admin/posts"
554 555 556
      #
      # 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 已提交
557
      # +PostsController+:
558
      #
559 560 561 562 563 564 565
      #   GET	    /admin/posts
      #   GET	    /admin/posts/new
      #   POST	  /admin/posts
      #   GET	    /admin/posts/1
      #   GET	    /admin/posts/1/edit
      #   PUT	    /admin/posts/1
      #   DELETE  /admin/posts/1
566
      module Scoping
567
        # Scopes a set of routes to the given default options.
568 569 570 571 572 573 574 575
        #
        # 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.
576 577
        # The difference here being that the routes generated are like /:account_id/projects,
        # rather than /accounts/:account_id/projects.
578
        #
579
        # === Options
580
        #
581
        # Takes same options as <tt>Base#match</tt> and <tt>Resources#resources</tt>.
582
        #
583
        # === Examples
584
        #
S
Sebastian Martinez 已提交
585
        #   # route /posts (without the prefix /admin) to <tt>Admin::PostsController</tt>
586 587 588
        #   scope :module => "admin" do
        #     resources :posts
        #   end
589
        #
590 591 592 593
        #   # prefix the posts resource's requests with '/admin'
        #   scope :path => "/admin" do
        #     resources :posts
        #   end
594
        #
S
Sebastian Martinez 已提交
595
        #   # prefix the routing helper name: +sekret_posts_path+ instead of +posts_path+
596 597 598
        #   scope :as => "sekret" do
        #     resources :posts
        #   end
599 600
        def scope(*args)
          options = args.extract_options!
601
          options = options.dup
602

603
          options[:path] = args.first if args.first.is_a?(String)
604
          recover = {}
605

606 607
          options[:constraints] ||= {}
          unless options[:constraints].is_a?(Hash)
608
            block, options[:constraints] = options[:constraints], {}
609
          end
610

611 612 613 614 615
          scope_options.each do |option|
            if value = options.delete(option)
              recover[option] = @scope[option]
              @scope[option]  = send("merge_#{option}_scope", @scope[option], value)
            end
616 617
          end

618 619 620 621 622 623
          recover[:block] = @scope[:blocks]
          @scope[:blocks] = merge_blocks_scope(@scope[:blocks], block)

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

624 625 626
          yield
          self
        ensure
627 628 629
          scope_options.each do |option|
            @scope[option] = recover[option] if recover.has_key?(option)
          end
630 631 632

          @scope[:options] = recover[:options]
          @scope[:blocks]  = recover[:block]
633 634
        end

635 636 637 638 639 640
        # Scopes routes to a specific controller
        #
        # Example:
        #   controller "food" do
        #     match "bacon", :action => "bacon"
        #   end
641 642 643
        def controller(controller, options={})
          options[:controller] = controller
          scope(options) { yield }
644 645
        end

646 647 648 649 650 651 652 653
        # Scopes routes to a specific namespace. For example:
        #
        #   namespace :admin do
        #     resources :posts
        #   end
        #
        # This generates the following routes:
        #
654 655 656 657 658 659 660
        #       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
        #        admin_post PUT    /admin/posts/:id(.:format)      admin/posts#update
        #        admin_post DELETE /admin/posts/:id(.:format)      admin/posts#destroy
661
        #
662
        # === Options
663
        #
664 665
        # The +:path+, +:as+, +:module+, +:shallow_path+ and +:shallow_prefix+
        # options all default to the name of the namespace.
666
        #
667 668
        # For options, see <tt>Base#match</tt>. For +:shallow_path+ option, see
        # <tt>Resources#resources</tt>.
669
        #
670
        # === Examples
671
        #
672 673 674 675
        #   # accessible through /sekret/posts rather than /admin/posts
        #   namespace :admin, :path => "sekret" do
        #     resources :posts
        #   end
676
        #
S
Sebastian Martinez 已提交
677
        #   # maps to <tt>Sekret::PostsController</tt> rather than <tt>Admin::PostsController</tt>
678 679 680
        #   namespace :admin, :module => "sekret" do
        #     resources :posts
        #   end
681
        #
S
Sebastian Martinez 已提交
682
        #   # generates +sekret_posts_path+ rather than +admin_posts_path+
683 684 685
        #   namespace :admin, :as => "sekret" do
        #     resources :posts
        #   end
686
        def namespace(path, options = {})
687
          path = path.to_s
688 689 690
          options = { :path => path, :as => path, :module => path,
                      :shallow_path => path, :shallow_prefix => path }.merge!(options)
          scope(options) { yield }
691
        end
692

R
Ryan Bigg 已提交
693 694 695 696
        # === 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 已提交
697
        #   constraints(:id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
698 699 700 701 702
        #     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.
703
        #
R
R.T. Lechow 已提交
704
        # You may use this to also restrict other parameters:
R
Ryan Bigg 已提交
705 706
        #
        #   resources :posts do
M
mjy 已提交
707
        #     constraints(:post_id => /\d+\.\d+/) do
R
Ryan Bigg 已提交
708 709
        #       resources :comments
        #     end
J
James Miller 已提交
710
        #   end
R
Ryan Bigg 已提交
711 712 713 714 715 716 717 718 719 720 721 722 723 724
        #
        # === 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 已提交
725
        # Requests to routes can be constrained based on specific criteria:
R
Ryan Bigg 已提交
726 727 728 729 730 731 732 733 734 735
        #
        #    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
736
        #      def self.matches?(request)
R
Ryan Bigg 已提交
737 738 739 740 741 742 743 744 745 746 747
        #        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
748 749 750 751
        def constraints(constraints = {})
          scope(:constraints => constraints) { yield }
        end

R
Ryan Bigg 已提交
752
        # Allows you to set default parameters for a route, such as this:
753 754 755
        #   defaults :id => 'home' do
        #     match 'scoped_pages/(:id)', :to => 'pages#show'
        #   end
R
Ryan Bigg 已提交
756
        # Using this, the +:id+ parameter here will default to 'home'.
757 758 759 760
        def defaults(defaults = {})
          scope(:defaults => defaults) { yield }
        end

761
        private
J
José Valim 已提交
762
          def scope_options #:nodoc:
763 764 765
            @scope_options ||= private_methods.grep(/^merge_(.+)_scope$/) { $1.to_sym }
          end

J
José Valim 已提交
766
          def merge_path_scope(parent, child) #:nodoc:
767
            Mapper.normalize_path("#{parent}/#{child}")
768 769
          end

J
José Valim 已提交
770
          def merge_shallow_path_scope(parent, child) #:nodoc:
771 772 773
            Mapper.normalize_path("#{parent}/#{child}")
          end

J
José Valim 已提交
774
          def merge_as_scope(parent, child) #:nodoc:
775
            parent ? "#{parent}_#{child}" : child
776 777
          end

J
José Valim 已提交
778
          def merge_shallow_prefix_scope(parent, child) #:nodoc:
779 780 781
            parent ? "#{parent}_#{child}" : child
          end

J
José Valim 已提交
782
          def merge_module_scope(parent, child) #:nodoc:
783 784 785
            parent ? "#{parent}/#{child}" : child
          end

J
José Valim 已提交
786
          def merge_controller_scope(parent, child) #:nodoc:
787
            child
788 789
          end

J
José Valim 已提交
790
          def merge_path_names_scope(parent, child) #:nodoc:
791 792 793
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
794
          def merge_constraints_scope(parent, child) #:nodoc:
795 796 797
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
798
          def merge_defaults_scope(parent, child) #:nodoc:
799 800 801
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
802
          def merge_blocks_scope(parent, child) #:nodoc:
803 804 805
            merged = parent ? parent.dup : []
            merged << child if child
            merged
806 807
          end

J
José Valim 已提交
808
          def merge_options_scope(parent, child) #:nodoc:
809
            (parent || {}).except(*override_keys(child)).merge(child)
810
          end
811

J
José Valim 已提交
812
          def merge_shallow_scope(parent, child) #:nodoc:
813 814
            child ? true : false
          end
815

J
José Valim 已提交
816
          def override_keys(child) #:nodoc:
817 818
            child.key?(:only) || child.key?(:except) ? [:only, :except] : []
          end
819 820
      end

821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844
      # 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 已提交
845 846
      # <tt>app/controllers/admin</tt> directory, and you can group them together
      # in your router:
847 848 849 850 851
      #
      #   namespace "admin" do
      #     resources :posts, :comments
      #   end
      #
S
Sebastian Martinez 已提交
852 853
      # 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
854 855 856 857
      # overrides this restriction, e.g:
      #
      #   resources :articles, :id => /[^\/]+/
      #
S
Sebastian Martinez 已提交
858
      # This allows any character other than a slash as part of your +:id+.
859
      #
J
Joshua Peek 已提交
860
      module Resources
861 862
        # CANONICAL_ACTIONS holds all actions that does not need a prefix or
        # a path appended since they fit properly in their scope level.
863 864 865
        VALID_ON_OPTIONS  = [:new, :collection, :member]
        RESOURCE_OPTIONS  = [:as, :controller, :path, :only, :except]
        CANONICAL_ACTIONS = %w(index create new show update destroy)
866

867
        class Resource #:nodoc:
868
          attr_reader :controller, :path, :options
869 870

          def initialize(entities, options = {})
871
            @name       = entities.to_s
872 873 874
            @path       = (options[:path] || @name).to_s
            @controller = (options[:controller] || @name).to_s
            @as         = options[:as]
875
            @options    = options
876 877
          end

878
          def default_actions
879
            [:index, :create, :new, :show, :update, :destroy, :edit]
880 881
          end

882
          def actions
883
            if only = @options[:only]
884
              Array(only).map(&:to_sym)
885
            elsif except = @options[:except]
886 887 888 889 890 891
              default_actions - Array(except).map(&:to_sym)
            else
              default_actions
            end
          end

892
          def name
893
            @as || @name
894 895
          end

896
          def plural
897
            @plural ||= name.to_s
898 899 900
          end

          def singular
901
            @singular ||= name.to_s.singularize
902 903
          end

904
          alias :member_name :singular
905

906
          # Checks for uncountable plurals, and appends "_index" if the plural
907
          # and singular form are the same.
908
          def collection_name
909
            singular == plural ? "#{plural}_index" : plural
910 911
          end

912
          def resource_scope
913
            { :controller => controller }
914 915
          end

916
          alias :collection_scope :path
917 918

          def member_scope
919
            "#{path}/:id"
920 921
          end

922
          def new_scope(new_path)
923
            "#{path}/#{new_path}"
924 925 926
          end

          def nested_scope
927
            "#{path}/:#{singular}_id"
928
          end
929

930 931 932
        end

        class SingletonResource < Resource #:nodoc:
933
          def initialize(entities, options)
934
            super
935
            @as         = nil
936 937
            @controller = (options[:controller] || plural).to_s
            @as         = options[:as]
938 939
          end

940 941 942 943
          def default_actions
            [:show, :create, :update, :destroy, :new, :edit]
          end

944 945
          def plural
            @plural ||= name.to_s.pluralize
946 947
          end

948 949
          def singular
            @singular ||= name.to_s
950
          end
951 952 953 954 955 956

          alias :member_name :singular
          alias :collection_name :singular

          alias :member_scope :path
          alias :nested_scope :path
957 958
        end

959 960 961 962
        def resources_path_names(options)
          @scope[:path_names].merge!(options)
        end

963 964 965 966 967 968 969 970 971
        # 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 已提交
972
        # the +GeoCoders+ controller (note that the controller is named after
973 974 975 976 977 978 979 980
        # the plural):
        #
        #   GET     /geocoder/new
        #   POST    /geocoder
        #   GET     /geocoder
        #   GET     /geocoder/edit
        #   PUT     /geocoder
        #   DELETE  /geocoder
981
        #
982
        # === Options
983
        # Takes same options as +resources+.
J
Joshua Peek 已提交
984
        def resource(*resources, &block)
J
Joshua Peek 已提交
985
          options = resources.extract_options!
J
Joshua Peek 已提交
986

987
          if apply_common_behavior_for(:resource, resources, options, &block)
988 989 990
            return self
          end

991
          resource_scope(:resource, SingletonResource.new(resources.pop, options)) do
992
            yield if block_given?
993

994
            collection do
995
              post :create
996
            end if parent_resource.actions.include?(:create)
997

998
            new do
999
              get :new
1000
            end if parent_resource.actions.include?(:new)
1001

1002
            member do
1003
              get    :edit if parent_resource.actions.include?(:edit)
1004 1005 1006
              get    :show if parent_resource.actions.include?(:show)
              put    :update if parent_resource.actions.include?(:update)
              delete :destroy if parent_resource.actions.include?(:destroy)
1007 1008 1009
            end
          end

J
Joshua Peek 已提交
1010
          self
1011 1012
        end

1013 1014 1015 1016 1017 1018 1019 1020
        # 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 已提交
1021
        # the +Photos+ controller:
1022
        #
A
Alexey Vakhov 已提交
1023
        #   GET     /photos
1024 1025 1026 1027 1028 1029
        #   GET     /photos/new
        #   POST    /photos
        #   GET     /photos/:id
        #   GET     /photos/:id/edit
        #   PUT     /photos/:id
        #   DELETE  /photos/:id
1030
        #
1031 1032 1033 1034 1035 1036 1037 1038
        # Resources can also be nested infinitely by using this block syntax:
        #
        #   resources :photos do
        #     resources :comments
        #   end
        #
        # This generates the following comments routes:
        #
A
Alexey Vakhov 已提交
1039
        #   GET     /photos/:photo_id/comments
1040 1041 1042 1043 1044 1045
        #   GET     /photos/:photo_id/comments/new
        #   POST    /photos/:photo_id/comments
        #   GET     /photos/:photo_id/comments/:id
        #   GET     /photos/:photo_id/comments/:id/edit
        #   PUT     /photos/:photo_id/comments/:id
        #   DELETE  /photos/:photo_id/comments/:id
1046
        #
1047
        # === Options
1048 1049
        # Takes same options as <tt>Base#match</tt> as well as:
        #
1050
        # [:path_names]
A
Aviv Ben-Yosef 已提交
1051 1052
        #   Allows you to change the segment component of the +edit+ and +new+ actions.
        #   Actions not specified are not changed.
1053 1054 1055 1056
        #
        #     resources :posts, :path_names => { :new => "brand_new" }
        #
        #   The above example will now change /posts/new to /posts/brand_new
1057
        #
1058 1059 1060 1061 1062 1063 1064
        # [: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
        #
1065 1066
        # [:only]
        #   Only generate routes for the given actions.
1067
        #
1068 1069
        #     resources :cows, :only => :show
        #     resources :cows, :only => [:show, :index]
1070
        #
1071 1072
        # [:except]
        #   Generate all routes except for the given actions.
1073
        #
1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087
        #     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
1088
        #       resources :comments, :except => [:show, :edit, :update, :destroy]
1089
        #     end
1090 1091 1092 1093 1094
        #     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>.
1095 1096 1097 1098
        #
        # [:shallow_path]
        #   Prefixes nested shallow routes with the specified path.
        #
1099 1100 1101 1102
        #     scope :shallow_path => "sekret" do
        #       resources :posts do
        #         resources :comments, :shallow => true
        #       end
1103 1104 1105 1106
        #     end
        #
        #   The +comments+ resource here will have the following routes generated for it:
        #
G
ganesh 已提交
1107 1108 1109
        #     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)
1110 1111 1112 1113 1114 1115
        #     edit_comment     GET    /sekret/comments/:id/edit(.:format)
        #     comment          GET    /sekret/comments/:id(.:format)
        #     comment          PUT    /sekret/comments/:id(.:format)
        #     comment          DELETE /sekret/comments/:id(.:format)
        #
        # === Examples
1116
        #
S
Sebastian Martinez 已提交
1117
        #   # routes call <tt>Admin::PostsController</tt>
1118
        #   resources :posts, :module => "admin"
1119
        #
1120
        #   # resource actions are at /admin/posts.
1121
        #   resources :posts, :path => "admin/posts"
J
Joshua Peek 已提交
1122
        def resources(*resources, &block)
J
Joshua Peek 已提交
1123
          options = resources.extract_options!
1124

1125
          if apply_common_behavior_for(:resources, resources, options, &block)
1126 1127 1128
            return self
          end

1129
          resource_scope(:resources, Resource.new(resources.pop, options)) do
1130
            yield if block_given?
J
Joshua Peek 已提交
1131

1132
            collection do
1133 1134
              get  :index if parent_resource.actions.include?(:index)
              post :create if parent_resource.actions.include?(:create)
1135
            end
1136

1137
            new do
1138
              get :new
1139
            end if parent_resource.actions.include?(:new)
1140

1141
            member do
1142
              get    :edit if parent_resource.actions.include?(:edit)
1143 1144 1145
              get    :show if parent_resource.actions.include?(:show)
              put    :update if parent_resource.actions.include?(:update)
              delete :destroy if parent_resource.actions.include?(:destroy)
1146 1147 1148
            end
          end

J
Joshua Peek 已提交
1149
          self
1150 1151
        end

1152 1153 1154 1155 1156 1157 1158 1159 1160
        # 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 已提交
1161
        # with GET, and route to the search action of +PhotosController+. It will also
1162 1163
        # create the <tt>search_photos_url</tt> and <tt>search_photos_path</tt>
        # route helpers.
J
Joshua Peek 已提交
1164
        def collection
1165 1166
          unless resource_scope?
            raise ArgumentError, "can't use collection outside resource(s) scope"
1167 1168
          end

1169 1170 1171 1172
          with_scope_level(:collection) do
            scope(parent_resource.collection_scope) do
              yield
            end
J
Joshua Peek 已提交
1173
          end
1174
        end
J
Joshua Peek 已提交
1175

1176 1177 1178 1179 1180 1181 1182 1183 1184
        # 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 已提交
1185
        # preview action of +PhotosController+. It will also create the
1186
        # <tt>preview_photo_url</tt> and <tt>preview_photo_path</tt> helpers.
J
Joshua Peek 已提交
1187
        def member
1188 1189
          unless resource_scope?
            raise ArgumentError, "can't use member outside resource(s) scope"
J
Joshua Peek 已提交
1190
          end
J
Joshua Peek 已提交
1191

1192 1193 1194 1195
          with_scope_level(:member) do
            scope(parent_resource.member_scope) do
              yield
            end
1196 1197 1198 1199 1200 1201 1202
          end
        end

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

1204 1205 1206 1207
          with_scope_level(:new) do
            scope(parent_resource.new_scope(action_path(:new))) do
              yield
            end
J
Joshua Peek 已提交
1208
          end
J
Joshua Peek 已提交
1209 1210
        end

1211
        def nested
1212 1213
          unless resource_scope?
            raise ArgumentError, "can't use nested outside resource(s) scope"
1214 1215 1216
          end

          with_scope_level(:nested) do
1217
            if shallow?
1218
              with_exclusive_scope do
1219
                if @scope[:shallow_path].blank?
1220
                  scope(parent_resource.nested_scope, nested_options) { yield }
1221
                else
1222
                  scope(@scope[:shallow_path], :as => @scope[:shallow_prefix]) do
1223
                    scope(parent_resource.nested_scope, nested_options) { yield }
1224 1225 1226 1227
                  end
                end
              end
            else
1228
              scope(parent_resource.nested_scope, nested_options) { yield }
1229 1230 1231 1232
            end
          end
        end

1233
        # See ActionDispatch::Routing::Mapper::Scoping#namespace
1234
        def namespace(path, options = {})
1235
          if resource_scope?
1236 1237 1238 1239 1240 1241
            nested { super }
          else
            super
          end
        end

1242
        def shallow
1243
          scope(:shallow => true, :shallow_path => @scope[:path]) do
1244 1245 1246 1247
            yield
          end
        end

1248 1249 1250 1251
        def shallow?
          parent_resource.instance_of?(Resource) && @scope[:shallow]
        end

1252 1253 1254 1255
        def match(path, *rest)
          if rest.empty? && Hash === path
            options  = path
            path, to = options.find { |name, value| name.is_a?(String) }
1256 1257
            options[:to] = to
            options.delete(path)
1258 1259 1260 1261 1262 1263
            paths = [path]
          else
            options = rest.pop || {}
            paths = [path] + rest
          end

1264 1265
          options[:anchor] = true unless options.key?(:anchor)

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

1270
          paths.each { |_path| decomposed_match(_path, options.dup) }
1271 1272
          self
        end
1273

1274
        def decomposed_match(path, options) # :nodoc:
A
Aaron Patterson 已提交
1275 1276
          if on = options.delete(:on)
            send(on) { decomposed_match(path, options) }
1277
          else
A
Aaron Patterson 已提交
1278 1279 1280 1281 1282 1283 1284 1285
            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 已提交
1286
          end
1287
        end
J
Joshua Peek 已提交
1288

1289
        def add_route(action, options) # :nodoc:
1290
          path = path_for_action(action, options.delete(:path))
1291

1292 1293 1294
          if action.to_s =~ /^[\w\/]+$/
            options[:action] ||= action unless action.to_s.include?("/")
          else
1295 1296 1297
            action = nil
          end

1298
          if !options.fetch(:as, true)
1299 1300 1301
            options.delete(:as)
          else
            options[:as] = name_for_action(options[:as], action)
J
Joshua Peek 已提交
1302
          end
J
Joshua Peek 已提交
1303

1304 1305 1306
          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 已提交
1307 1308
        end

1309
        def root(options={})
1310
          if @scope[:scope_level] == :resources
1311 1312
            with_scope_level(:root) do
              scope(parent_resource.path) do
1313 1314 1315 1316 1317 1318
                super(options)
              end
            end
          else
            super(options)
          end
1319 1320
        end

1321
        protected
1322

1323
          def parent_resource #:nodoc:
1324 1325 1326
            @scope[:scope_level_resource]
          end

J
José Valim 已提交
1327
          def apply_common_behavior_for(method, resources, options, &block) #:nodoc:
1328 1329 1330 1331 1332
            if resources.length > 1
              resources.each { |r| send(method, r, options, &block) }
              return true
            end

1333 1334 1335 1336 1337
            if resource_scope?
              nested { send(method, resources.pop, options, &block) }
              return true
            end

1338
            options.keys.each do |k|
1339 1340 1341
              (options[:constraints] ||= {})[k] = options.delete(k) if options[k].is_a?(Regexp)
            end

1342 1343 1344
            scope_options = options.slice!(*RESOURCE_OPTIONS)
            unless scope_options.empty?
              scope(scope_options) do
1345 1346 1347 1348 1349
                send(method, resources.pop, options, &block)
              end
              return true
            end

1350 1351 1352 1353
            unless action_options?(options)
              options.merge!(scope_action_options) if scope_action_options?
            end

1354 1355 1356
            false
          end

J
José Valim 已提交
1357
          def action_options?(options) #:nodoc:
1358 1359 1360
            options[:only] || options[:except]
          end

J
José Valim 已提交
1361
          def scope_action_options? #:nodoc:
A
Aaron Patterson 已提交
1362
            @scope[:options] && (@scope[:options][:only] || @scope[:options][:except])
1363 1364
          end

J
José Valim 已提交
1365
          def scope_action_options #:nodoc:
1366 1367 1368
            @scope[:options].slice(:only, :except)
          end

J
José Valim 已提交
1369
          def resource_scope? #:nodoc:
1370
            [:resource, :resources].include? @scope[:scope_level]
1371 1372
          end

J
José Valim 已提交
1373
          def resource_method_scope? #:nodoc:
1374
            [:collection, :member, :new].include? @scope[:scope_level]
1375 1376
          end

1377
          def with_exclusive_scope
1378
            begin
1379 1380
              old_name_prefix, old_path = @scope[:as], @scope[:path]
              @scope[:as], @scope[:path] = nil, nil
1381

1382 1383 1384
              with_scope_level(:exclusive) do
                yield
              end
1385
            ensure
1386
              @scope[:as], @scope[:path] = old_name_prefix, old_path
1387 1388 1389
            end
          end

1390
          def with_scope_level(kind, resource = parent_resource)
J
Joshua Peek 已提交
1391
            old, @scope[:scope_level] = @scope[:scope_level], kind
1392
            old_resource, @scope[:scope_level_resource] = @scope[:scope_level_resource], resource
J
Joshua Peek 已提交
1393 1394 1395
            yield
          ensure
            @scope[:scope_level] = old
1396
            @scope[:scope_level_resource] = old_resource
J
Joshua Peek 已提交
1397
          end
1398

1399 1400
          def resource_scope(kind, resource) #:nodoc:
            with_scope_level(kind, resource) do
1401
              scope(parent_resource.resource_scope) do
1402 1403 1404 1405 1406
                yield
              end
            end
          end

J
José Valim 已提交
1407
          def nested_options #:nodoc:
1408 1409 1410 1411 1412 1413
            options = { :as => parent_resource.member_name }
            options[:constraints] = {
              :"#{parent_resource.singular}_id" => id_constraint
            } if id_constraint?

            options
1414 1415
          end

J
José Valim 已提交
1416
          def id_constraint? #:nodoc:
1417
            @scope[:constraints] && @scope[:constraints][:id].is_a?(Regexp)
1418 1419
          end

J
José Valim 已提交
1420
          def id_constraint #:nodoc:
1421
            @scope[:constraints][:id]
1422 1423
          end

J
José Valim 已提交
1424
          def canonical_action?(action, flag) #:nodoc:
1425
            flag && resource_method_scope? && CANONICAL_ACTIONS.include?(action.to_s)
1426 1427
          end

J
José Valim 已提交
1428
          def shallow_scoping? #:nodoc:
1429
            shallow? && @scope[:scope_level] == :member
1430 1431
          end

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

1436 1437
            path = if canonical_action?(action, path.blank?)
              prefix.to_s
1438
            else
1439
              "#{prefix}/#{action_path(action, path)}"
1440 1441 1442
            end
          end

J
José Valim 已提交
1443
          def action_path(name, path = nil) #:nodoc:
1444
            name = name.to_sym if name.is_a?(String)
1445
            path || @scope[:path_names][name] || name.to_s
1446 1447
          end

J
José Valim 已提交
1448
          def prefix_name_for_action(as, action) #:nodoc:
1449
            if as
1450
              as.to_s
1451
            elsif !canonical_action?(action, @scope[:scope_level])
1452
              action.to_s
1453
            end
1454 1455
          end

J
José Valim 已提交
1456
          def name_for_action(as, action) #:nodoc:
1457
            prefix = prefix_name_for_action(as, action)
1458
            prefix = Mapper.normalize_name(prefix) if prefix
1459 1460 1461
            name_prefix = @scope[:as]

            if parent_resource
1462
              return nil unless as || action
1463

1464 1465
              collection_name = parent_resource.collection_name
              member_name = parent_resource.member_name
1466
            end
1467

1468
            name = case @scope[:scope_level]
1469
            when :nested
1470
              [name_prefix, prefix]
1471
            when :collection
1472
              [prefix, name_prefix, collection_name]
1473
            when :new
1474 1475 1476 1477 1478
              [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]
1479
            else
1480
              [name_prefix, member_name, prefix]
1481
            end
1482

1483 1484 1485 1486 1487 1488 1489 1490 1491 1492
            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
1493
          end
J
Joshua Peek 已提交
1494
      end
J
Joshua Peek 已提交
1495

1496 1497 1498 1499 1500
      def initialize(set) #:nodoc:
        @set = set
        @scope = { :path_names => @set.resources_path_names }
      end

1501 1502
      include Base
      include HttpHelpers
1503
      include Redirection
1504 1505
      include Scoping
      include Resources
J
Joshua Peek 已提交
1506 1507
    end
  end
J
Joshua Peek 已提交
1508
end