mapper.rb 49.3 KB
Newer Older
1
require 'erb'
2
require 'active_support/core_ext/hash/except'
3
require 'active_support/core_ext/object/blank'
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 20
        attr_reader :app

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

52
        def initialize(set, scope, path, options)
53 54
          @set, @scope = set, scope
          @options = (@scope[:options] || {}).merge(options)
55
          @path = normalize_path(path)
56
          normalize_options!
57
        end
J
Joshua Peek 已提交
58

59
        def to_route
60
          [ app, conditions, requirements, defaults, @options[:as], @options[:anchor] ]
61
        end
J
Joshua Peek 已提交
62

63
        private
64 65 66

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

68 69 70
            if using_match_shorthand?(path_without_format, @options)
              to_shorthand    = @options[:to].blank?
              @options[:to] ||= path_without_format[1..-1].sub(%r{/([^/]*)$}, '#\1')
71 72
            end

73
            @options.merge!(default_controller_and_action(to_shorthand))
74 75 76 77 78 79 80 81 82 83 84 85

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

              if requirement.source =~ %r{\A(\\A|\^)|(\\Z|\\z|\$)\Z}
                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
86
          end
87

88
          # match "account/overview"
89
          def using_match_shorthand?(path, options)
90
            path && options.except(:via, :anchor, :to, :as).empty? && path =~ %r{^/[\w\/]+$}
91
          end
92

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

            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' }
              @options.reverse_merge!(:controller => /.+?/)
            end

107 108 109
            if @options[:format] == false
              @options.delete(:format)
              path
110
            elsif path.include?(":format") || path.end_with?('/') || path.match(/^\/?\*/)
111 112 113 114
              path
            else
              "#{path}(.:format)"
            end
115
          end
116

117 118
          def app
            Constraints.new(
119
              to.respond_to?(:call) ? to : Routing::RouteSet::Dispatcher.new(:defaults => defaults),
120 121
              blocks,
              @set.request_class
122
            )
123 124
          end

125 126 127
          def conditions
            { :path_info => @path }.merge(constraints).merge(request_method_condition)
          end
J
Joshua Peek 已提交
128

129
          def requirements
130
            @requirements ||= (@options[:constraints].is_a?(Hash) ? @options[:constraints] : {}).tap do |requirements|
131 132 133 134
              requirements.reverse_merge!(@scope[:constraints]) if @scope[:constraints]
              @options.each { |k, v| requirements[k] = v if v.is_a?(Regexp) }
            end
          end
135

136
          def defaults
137 138 139 140 141 142
            @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

143
          def default_controller_and_action(to_shorthand=nil)
144
            if to.respond_to?(:call)
145 146
              { }
            else
147
              if to.is_a?(String)
148
                controller, action = to.split('#')
149 150
              elsif to.is_a?(Symbol)
                action = to.to_s
151
              end
J
Joshua Peek 已提交
152

153 154
              controller ||= default_controller
              action     ||= default_action
155

156 157 158
              unless controller.is_a?(Regexp) || to_shorthand
                controller = [@scope[:module], controller].compact.join("/").presence
              end
159

160 161 162 163
              if controller.is_a?(String) && controller =~ %r{\A/}
                raise ArgumentError, "controller name should not start with a slash"
              end

164 165
              controller = controller.to_s unless controller.is_a?(Regexp)
              action     = action.to_s     unless action.is_a?(Regexp)
166

167
              if controller.blank? && segment_keys.exclude?("controller")
168 169
                raise ArgumentError, "missing :controller"
              end
J
Joshua Peek 已提交
170

171
              if action.blank? && segment_keys.exclude?("action")
172 173
                raise ArgumentError, "missing :action"
              end
J
Joshua Peek 已提交
174

A
Aaron Patterson 已提交
175
              hash = {}
A
Aaron Patterson 已提交
176 177
              hash[:controller] = controller unless controller.blank?
              hash[:action]     = action unless action.blank?
A
Aaron Patterson 已提交
178
              hash
179 180
            end
          end
181

182
          def blocks
A
Aaron Patterson 已提交
183 184
            block = @scope[:blocks] || []

185
            if @options[:constraints].present? && !@options[:constraints].is_a?(Hash)
A
Aaron Patterson 已提交
186
              block << @options[:constraints]
187
            end
J
Joshua Peek 已提交
188

A
Aaron Patterson 已提交
189
            block
190
          end
J
Joshua Peek 已提交
191

192 193 194
          def constraints
            @constraints ||= requirements.reject { |k, v| segment_keys.include?(k.to_s) || k == :controller }
          end
195

196 197
          def request_method_condition
            if via = @options[:via]
198 199
              list = Array(via).map { |m| m.to_s.dasherize.upcase }
              { :request_method => list }
200 201
            else
              { }
202
            end
203
          end
J
Joshua Peek 已提交
204

205 206
          def segment_keys
            @segment_keys ||= Rack::Mount::RegexpWithNamedGroups.new(
207 208
              Rack::Mount::Strexp.compile(@path, requirements, SEPARATORS)
            ).names
209
          end
210

211 212 213
          def to
            @options[:to]
          end
J
Joshua Peek 已提交
214

215
          def default_controller
216
            if @options[:controller]
217
              @options[:controller]
218
            elsif @scope[:controller]
219
              @scope[:controller]
220
            end
221
          end
222 223 224

          def default_action
            if @options[:action]
225
              @options[:action]
226 227
            elsif @scope[:action]
              @scope[:action]
228 229
            end
          end
230
      end
231

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

241 242 243 244
      def self.normalize_name(name)
        normalize_path(name)[1..-1].gsub("/", "_")
      end

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

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

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

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

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

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

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

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

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

            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
440 441 442
          end

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

            _route = @set.named_routes.routes[name.to_sym]
P
Piotr Sarnacki 已提交
446 447
            _routes = @set
            app.routes.define_mounted_helper(name)
448 449
            app.routes.class_eval do
              define_method :_generate_prefix do |options|
P
Piotr Sarnacki 已提交
450
                prefix_options = options.slice(*_route.segment_keys)
451 452
                # we must actually delete prefix segment keys to avoid passing them to next url_for
                _route.segment_keys.each { |k| options.delete(k) }
P
Piotr Sarnacki 已提交
453
                _routes.url_helpers.send("#{name}_path", prefix_options)
454 455 456
              end
            end
          end
457 458 459
      end

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

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

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

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

        private
          def map_method(method, *args, &block)
            options = args.extract_options!
            options[:via] = method
            args.push(options)
            match(*args, &block)
            self
          end
      end

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

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

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

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

616 617
          recover[:block] = @scope[:blocks]
          @scope[:blocks] = merge_blocks_scope(@scope[:blocks], block)
618

619 620
          recover[:options] = @scope[:options]
          @scope[:options]  = merge_options_scope(@scope[:options], options)
621 622 623 624

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

          @scope[:options] = recover[:options]
          @scope[:blocks]  = recover[:block]
631 632
        end

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

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

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

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

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

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

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

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

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

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

J
José Valim 已提交
783
          def merge_controller_scope(parent, child) #:nodoc:
784
            child
785 786
          end

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

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

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

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

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

J
José Valim 已提交
809
          def merge_shallow_scope(parent, child) #:nodoc:
810 811
            child ? true : false
          end
812

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

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

864
        class Resource #:nodoc:
865
          DEFAULT_ACTIONS = [:index, :create, :new, :show, :update, :destroy, :edit]
866

867
          attr_reader :controller, :path, :options
868 869

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

877
          def default_actions
878
            self.class::DEFAULT_ACTIONS
879 880
          end

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

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

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

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

903
          alias :member_name :singular
904

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

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

915
          alias :collection_scope :path
916 917

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

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

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

929 930 931
        end

        class SingletonResource < Resource #:nodoc:
932
          DEFAULT_ACTIONS = [:show, :create, :update, :destroy, :new, :edit]
933

934
          def initialize(entities, options)
935
            @as         = nil
936
            @name       = entities.to_s
937
            @path       = (options.delete(:path) || @name).to_s
938
            @controller = (options.delete(:controller) || plural).to_s
939 940 941 942
            @as         = options.delete(:as)
            @options    = options
          end

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

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

          alias :member_name :singular
          alias :collection_name :singular

          alias :member_scope :path
          alias :nested_scope :path
956 957
        end

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

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

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

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

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

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

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

J
Joshua Peek 已提交
1009
          self
1010 1011
        end

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

1111
          if apply_common_behavior_for(:resources, resources, options, &block)
1112 1113 1114
            return self
          end

1115
          resource_scope(Resource.new(resources.pop, options)) do
1116
            yield if block_given?
J
Joshua Peek 已提交
1117

1118
            collection do
1119 1120
              get  :index if parent_resource.actions.include?(:index)
              post :create if parent_resource.actions.include?(:create)
1121
            end
1122

1123
            new do
1124
              get :new
1125
            end if parent_resource.actions.include?(:new)
1126

1127
            member do
1128
              get    :edit if parent_resource.actions.include?(:edit)
1129 1130 1131
              get    :show if parent_resource.actions.include?(:show)
              put    :update if parent_resource.actions.include?(:update)
              delete :destroy if parent_resource.actions.include?(:destroy)
1132 1133 1134
            end
          end

J
Joshua Peek 已提交
1135
          self
1136 1137
        end

1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149
        # 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>
        # with GET, and route to the search action of PhotosController. It will also
        # create the <tt>search_photos_url</tt> and <tt>search_photos_path</tt>
        # route helpers.
J
Joshua Peek 已提交
1150
        def collection
1151 1152
          unless resource_scope?
            raise ArgumentError, "can't use collection outside resource(s) scope"
1153 1154
          end

1155 1156 1157 1158
          with_scope_level(:collection) do
            scope(parent_resource.collection_scope) do
              yield
            end
J
Joshua Peek 已提交
1159
          end
1160
        end
J
Joshua Peek 已提交
1161

1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172
        # 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
        # preview action of PhotosController. It will also create the
        # <tt>preview_photo_url</tt> and <tt>preview_photo_path</tt> helpers.
J
Joshua Peek 已提交
1173
        def member
1174 1175
          unless resource_scope?
            raise ArgumentError, "can't use member outside resource(s) scope"
J
Joshua Peek 已提交
1176
          end
J
Joshua Peek 已提交
1177

1178 1179 1180 1181
          with_scope_level(:member) do
            scope(parent_resource.member_scope) do
              yield
            end
1182 1183 1184 1185 1186 1187 1188
          end
        end

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

1190 1191 1192 1193
          with_scope_level(:new) do
            scope(parent_resource.new_scope(action_path(:new))) do
              yield
            end
J
Joshua Peek 已提交
1194
          end
J
Joshua Peek 已提交
1195 1196
        end

1197
        def nested
1198 1199
          unless resource_scope?
            raise ArgumentError, "can't use nested outside resource(s) scope"
1200 1201 1202
          end

          with_scope_level(:nested) do
1203
            if shallow?
1204
              with_exclusive_scope do
1205
                if @scope[:shallow_path].blank?
1206
                  scope(parent_resource.nested_scope, nested_options) { yield }
1207
                else
1208
                  scope(@scope[:shallow_path], :as => @scope[:shallow_prefix]) do
1209
                    scope(parent_resource.nested_scope, nested_options) { yield }
1210 1211 1212 1213
                  end
                end
              end
            else
1214
              scope(parent_resource.nested_scope, nested_options) { yield }
1215 1216 1217 1218
            end
          end
        end

1219
        # See ActionDispatch::Routing::Mapper::Scoping#namespace
1220
        def namespace(path, options = {})
1221
          if resource_scope?
1222 1223 1224 1225 1226 1227
            nested { super }
          else
            super
          end
        end

1228
        def shallow
1229
          scope(:shallow => true, :shallow_path => @scope[:path]) do
1230 1231 1232 1233
            yield
          end
        end

1234 1235 1236 1237
        def shallow?
          parent_resource.instance_of?(Resource) && @scope[:shallow]
        end

J
Joshua Peek 已提交
1238
        def match(*args)
1239
          options = args.extract_options!.dup
1240 1241
          options[:anchor] = true unless options.key?(:anchor)

1242
          if args.length > 1
1243
            args.each { |path| match(path, options.dup) }
1244 1245 1246
            return self
          end

1247 1248
          on = options.delete(:on)
          if VALID_ON_OPTIONS.include?(on)
1249
            args.push(options)
1250 1251 1252
            return send(on){ match(*args) }
          elsif on
            raise ArgumentError, "Unknown scope #{on.inspect} given to :on"
1253 1254
          end

1255 1256 1257 1258
          if @scope[:scope_level] == :resources
            args.push(options)
            return nested { match(*args) }
          elsif @scope[:scope_level] == :resource
1259
            args.push(options)
J
Joshua Peek 已提交
1260 1261
            return member { match(*args) }
          end
J
Joshua Peek 已提交
1262

1263
          action = args.first
1264
          path = path_for_action(action, options.delete(:path))
1265

1266 1267 1268
          if action.to_s =~ /^[\w\/]+$/
            options[:action] ||= action unless action.to_s.include?("/")
          else
1269 1270 1271 1272 1273 1274 1275
            action = nil
          end

          if options.key?(:as) && !options[:as]
            options.delete(:as)
          else
            options[:as] = name_for_action(options[:as], action)
J
Joshua Peek 已提交
1276
          end
J
Joshua Peek 已提交
1277

1278
          super(path, options)
J
Joshua Peek 已提交
1279 1280
        end

1281
        def root(options={})
1282
          if @scope[:scope_level] == :resources
1283 1284
            with_scope_level(:root) do
              scope(parent_resource.path) do
1285 1286 1287 1288 1289 1290
                super(options)
              end
            end
          else
            super(options)
          end
1291 1292
        end

1293
        protected
1294

1295
          def parent_resource #:nodoc:
1296 1297 1298
            @scope[:scope_level_resource]
          end

J
José Valim 已提交
1299
          def apply_common_behavior_for(method, resources, options, &block) #:nodoc:
1300 1301 1302 1303 1304
            if resources.length > 1
              resources.each { |r| send(method, r, options, &block) }
              return true
            end

1305 1306 1307 1308 1309
            if resource_scope?
              nested { send(method, resources.pop, options, &block) }
              return true
            end

1310
            options.keys.each do |k|
1311 1312 1313
              (options[:constraints] ||= {})[k] = options.delete(k) if options[k].is_a?(Regexp)
            end

1314 1315 1316
            scope_options = options.slice!(*RESOURCE_OPTIONS)
            unless scope_options.empty?
              scope(scope_options) do
1317 1318 1319 1320 1321
                send(method, resources.pop, options, &block)
              end
              return true
            end

1322 1323 1324 1325
            unless action_options?(options)
              options.merge!(scope_action_options) if scope_action_options?
            end

1326 1327 1328
            false
          end

J
José Valim 已提交
1329
          def action_options?(options) #:nodoc:
1330 1331 1332
            options[:only] || options[:except]
          end

J
José Valim 已提交
1333
          def scope_action_options? #:nodoc:
1334 1335 1336
            @scope[:options].is_a?(Hash) && (@scope[:options][:only] || @scope[:options][:except])
          end

J
José Valim 已提交
1337
          def scope_action_options #:nodoc:
1338 1339 1340
            @scope[:options].slice(:only, :except)
          end

J
José Valim 已提交
1341
          def resource_scope? #:nodoc:
1342 1343 1344
            [:resource, :resources].include?(@scope[:scope_level])
          end

J
José Valim 已提交
1345
          def resource_method_scope? #:nodoc:
1346 1347 1348
            [:collection, :member, :new].include?(@scope[:scope_level])
          end

1349
          def with_exclusive_scope
1350
            begin
1351 1352
              old_name_prefix, old_path = @scope[:as], @scope[:path]
              @scope[:as], @scope[:path] = nil, nil
1353

1354 1355 1356
              with_scope_level(:exclusive) do
                yield
              end
1357
            ensure
1358
              @scope[:as], @scope[:path] = old_name_prefix, old_path
1359 1360 1361
            end
          end

1362
          def with_scope_level(kind, resource = parent_resource)
J
Joshua Peek 已提交
1363
            old, @scope[:scope_level] = @scope[:scope_level], kind
1364
            old_resource, @scope[:scope_level_resource] = @scope[:scope_level_resource], resource
J
Joshua Peek 已提交
1365 1366 1367
            yield
          ensure
            @scope[:scope_level] = old
1368
            @scope[:scope_level_resource] = old_resource
J
Joshua Peek 已提交
1369
          end
1370

J
José Valim 已提交
1371
          def resource_scope(resource) #:nodoc:
1372
            with_scope_level(resource.is_a?(SingletonResource) ? :resource : :resources, resource) do
1373
              scope(parent_resource.resource_scope) do
1374 1375 1376 1377 1378
                yield
              end
            end
          end

J
José Valim 已提交
1379
          def nested_options #:nodoc:
1380 1381 1382 1383 1384 1385
            {}.tap do |options|
              options[:as] = parent_resource.member_name
              options[:constraints] = { "#{parent_resource.singular}_id".to_sym => id_constraint } if id_constraint?
            end
          end

J
José Valim 已提交
1386
          def id_constraint? #:nodoc:
1387
            @scope[:constraints] && @scope[:constraints][:id].is_a?(Regexp)
1388 1389
          end

J
José Valim 已提交
1390
          def id_constraint #:nodoc:
1391
            @scope[:constraints][:id]
1392 1393
          end

J
José Valim 已提交
1394
          def canonical_action?(action, flag) #:nodoc:
1395
            flag && resource_method_scope? && CANONICAL_ACTIONS.include?(action.to_s)
1396 1397
          end

J
José Valim 已提交
1398
          def shallow_scoping? #:nodoc:
1399
            shallow? && @scope[:scope_level] == :member
1400 1401
          end

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

1406 1407
            path = if canonical_action?(action, path.blank?)
              prefix.to_s
1408
            else
1409
              "#{prefix}/#{action_path(action, path)}"
1410 1411 1412
            end
          end

J
José Valim 已提交
1413
          def action_path(name, path = nil) #:nodoc:
1414
            path || @scope[:path_names][name.to_sym] || name.to_s
1415 1416
          end

J
José Valim 已提交
1417
          def prefix_name_for_action(as, action) #:nodoc:
1418
            if as
1419
              as.to_s
1420
            elsif !canonical_action?(action, @scope[:scope_level])
1421
              action.to_s
1422
            end
1423 1424
          end

J
José Valim 已提交
1425
          def name_for_action(as, action) #:nodoc:
1426
            prefix = prefix_name_for_action(as, action)
1427
            prefix = Mapper.normalize_name(prefix) if prefix
1428 1429 1430
            name_prefix = @scope[:as]

            if parent_resource
1431 1432
              return nil if as.nil? && action.nil?

1433 1434
              collection_name = parent_resource.collection_name
              member_name = parent_resource.member_name
1435
            end
1436

1437
            name = case @scope[:scope_level]
1438
            when :nested
1439
              [name_prefix, prefix]
1440
            when :collection
1441
              [prefix, name_prefix, collection_name]
1442
            when :new
1443 1444 1445 1446 1447
              [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]
1448
            else
1449
              [name_prefix, member_name, prefix]
1450
            end
1451

1452
            candidate = name.select(&:present?).join("_").presence
1453
            candidate unless as.nil? && @set.routes.find { |r| r.name == candidate }
1454
          end
J
Joshua Peek 已提交
1455
      end
J
Joshua Peek 已提交
1456

J
José Valim 已提交
1457
      module Shorthand #:nodoc:
1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469
        def match(*args)
          if args.size == 1 && args.last.is_a?(Hash)
            options  = args.pop
            path, to = options.find { |name, value| name.is_a?(String) }
            options.merge!(:to => to).delete(path)
            super(path, options)
          else
            super
          end
        end
      end

1470 1471 1472 1473 1474
      def initialize(set) #:nodoc:
        @set = set
        @scope = { :path_names => @set.resources_path_names }
      end

1475 1476
      include Base
      include HttpHelpers
1477
      include Redirection
1478 1479
      include Scoping
      include Resources
1480
      include Shorthand
J
Joshua Peek 已提交
1481 1482
    end
  end
J
Joshua Peek 已提交
1483
end