mapper.rb 50.2 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/core_ext/object/inclusion'
5
require 'active_support/inflector'
6
require 'action_dispatch/routing/redirection'
7

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

20 21
        attr_reader :app

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

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

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

37 38 39 40 41
          return true
        end

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

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

50
      class Mapping #:nodoc:
51
        IGNORE_OPTIONS = [:to, :as, :via, :on, :constraints, :defaults, :only, :except, :anchor, :shallow, :shallow_path, :shallow_prefix]
52

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

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

64
        private
65 66 67

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

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

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

            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
87
          end
88

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

94
          def normalize_path(path)
95 96
            raise ArgumentError, "path is required" if path.blank?
            path = Mapper.normalize_path(path)
97 98 99 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' }
              @options.reverse_merge!(:controller => /.+?/)
            end

108 109 110 111 112 113
            # Add a constraint for wildcard route to make it non-greedy and match the
            # optional format part of the route by default
            if path.match(/\*([^\/]+)$/) && @options[:format] != false
              @options.reverse_merge!(:"#{$1}" => /.+?/)
            end

114 115 116
            if @options[:format] == false
              @options.delete(:format)
              path
117
            elsif path.include?(":format") || path.end_with?('/')
118 119 120 121
              path
            else
              "#{path}(.:format)"
            end
122
          end
123

124 125
          def app
            Constraints.new(
126
              to.respond_to?(:call) ? to : Routing::RouteSet::Dispatcher.new(:defaults => defaults),
127 128
              blocks,
              @set.request_class
129
            )
130 131
          end

132 133 134
          def conditions
            { :path_info => @path }.merge(constraints).merge(request_method_condition)
          end
J
Joshua Peek 已提交
135

136
          def requirements
137
            @requirements ||= (@options[:constraints].is_a?(Hash) ? @options[:constraints] : {}).tap do |requirements|
138 139 140 141
              requirements.reverse_merge!(@scope[:constraints]) if @scope[:constraints]
              @options.each { |k, v| requirements[k] = v if v.is_a?(Regexp) }
            end
          end
142

143
          def defaults
144 145 146 147 148 149
            @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

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

160 161
              controller ||= default_controller
              action     ||= default_action
162

163 164 165
              unless controller.is_a?(Regexp) || to_shorthand
                controller = [@scope[:module], controller].compact.join("/").presence
              end
166

167 168 169 170
              if controller.is_a?(String) && controller =~ %r{\A/}
                raise ArgumentError, "controller name should not start with a slash"
              end

171 172
              controller = controller.to_s unless controller.is_a?(Regexp)
              action     = action.to_s     unless action.is_a?(Regexp)
173

174
              if controller.blank? && segment_keys.exclude?("controller")
175 176
                raise ArgumentError, "missing :controller"
              end
J
Joshua Peek 已提交
177

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

A
Aaron Patterson 已提交
182
              hash = {}
A
Aaron Patterson 已提交
183 184
              hash[:controller] = controller unless controller.blank?
              hash[:action]     = action unless action.blank?
A
Aaron Patterson 已提交
185
              hash
186 187
            end
          end
188

189
          def blocks
A
Aaron Patterson 已提交
190 191
            block = @scope[:blocks] || []

192
            if @options[:constraints].present? && !@options[:constraints].is_a?(Hash)
A
Aaron Patterson 已提交
193
              block << @options[:constraints]
194
            end
J
Joshua Peek 已提交
195

A
Aaron Patterson 已提交
196
            block
197
          end
J
Joshua Peek 已提交
198

199 200 201
          def constraints
            @constraints ||= requirements.reject { |k, v| segment_keys.include?(k.to_s) || k == :controller }
          end
202

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

212 213
          def segment_keys
            @segment_keys ||= Rack::Mount::RegexpWithNamedGroups.new(
214 215
              Rack::Mount::Strexp.compile(@path, requirements, SEPARATORS)
            ).names
216
          end
217

218 219 220
          def to
            @options[:to]
          end
J
Joshua Peek 已提交
221

222
          def default_controller
223
            if @options[:controller]
224
              @options[:controller]
225
            elsif @scope[:controller]
226
              @scope[:controller]
227
            end
228
          end
229 230 231

          def default_action
            if @options[:action]
232
              @options[:action]
233 234
            elsif @scope[:action]
              @scope[:action]
235 236
            end
          end
237
      end
238

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

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

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

266 267 268
        # 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:
269
        #
270
        #   # sets :controller, :action and :id in params
271
        #   match ':controller/:action/:id'
272
        #
273 274 275 276 277 278 279 280 281 282 283 284
        # 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:
285 286 287 288
        #
        #   match 'photos/:id' => 'photos#show'
        #   match 'photos/:id', :to => 'photos#show'
        #   match 'photos/:id', :controller => 'photos', :action => 'show'
289
        #
290 291 292 293 294 295 296 297
        # 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)
        #
298
        # === Options
299
        #
300
        # Any options not seen here are passed on as params with the url.
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 327 328
        #
        # [: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]
329 330
        #   Points to a +Rack+ endpoint. Can be an object that responds to
        #   +call+ or a string representing a controller's action.
331
        #
332 333 334
        #      match 'path', :to => 'controller#action'
        #      match 'path', :to => lambda { [200, {}, "Success!"] }
        #      match 'path', :to => RackApp
335 336 337
        #
        # [:on]
        #   Shorthand for wrapping routes in a specific RESTful context. Valid
338
        #   values are +:member+, +:collection+, and +:new+. Only use within
339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
        #   <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
355
        #   object that responds to <tt>matches?</tt>
356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373
        #
        #     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.
374 375
        #
        # [:anchor]
376
        #   Boolean to anchor a <tt>match</tt> pattern. Default is true. When set to
377 378 379 380
        #   false, the pattern matches any request prefixed with the given path.
        #
        #     # Matches any request starting with 'path'
        #     match 'path' => 'c#a', :anchor => false
381
        def match(path, options=nil)
382 383 384
          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)
385 386
          self
        end
387

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

418 419
          options[:as] ||= app_name(app)

420
          match(path, options.merge(:to => app, :anchor => false, :format => false))
421 422

          define_generate_prefix(app, options[:as])
423 424 425
          self
        end

426 427 428 429
        def default_url_options=(options)
          @set.default_url_options = options
        end
        alias_method :default_url_options, :default_url_options=
430

431 432 433 434 435 436
        def with_default_scope(scope, &block)
          scope(scope) do
            instance_exec(&block)
          end
        end

437 438 439
        private
          def app_name(app)
            return unless app.respond_to?(:routes)
440 441 442 443 444 445 446

            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
447 448 449
          end

          def define_generate_prefix(app, name)
450
            return unless app.respond_to?(:routes) && app.routes.respond_to?(:define_mounted_helper)
451 452

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

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

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

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

497
        # Define a route that only recognizes HTTP PUT.
498
        # For supported arguments, see <tt>Base#match</tt>.
499 500 501 502
        #
        # Example:
        #
        # delete 'broccoli', :to => 'food#broccoli'
503 504 505 506 507 508 509 510 511 512 513 514 515 516
        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

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

608
          options[:path] = args.first if args.first.is_a?(String)
609
          recover = {}
610

611 612 613
          options[:constraints] ||= {}
          unless options[:constraints].is_a?(Hash)
            block, options[:constraints] = options[:constraints], {}
614
          end
615

616 617 618 619 620
          scope_options.each do |option|
            if value = options.delete(option)
              recover[option] = @scope[option]
              @scope[option]  = send("merge_#{option}_scope", @scope[option], value)
            end
621 622
          end

623 624
          recover[:block] = @scope[:blocks]
          @scope[:blocks] = merge_blocks_scope(@scope[:blocks], block)
625

626 627
          recover[:options] = @scope[:options]
          @scope[:options]  = merge_options_scope(@scope[:options], options)
628 629 630 631

          yield
          self
        ensure
632 633 634 635 636 637
          scope_options.each do |option|
            @scope[option] = recover[option] if recover.has_key?(option)
          end

          @scope[:options] = recover[:options]
          @scope[:blocks]  = recover[:block]
638 639
        end

640 641 642 643 644 645
        # Scopes routes to a specific controller
        #
        # Example:
        #   controller "food" do
        #     match "bacon", :action => "bacon"
        #   end
646 647 648
        def controller(controller, options={})
          options[:controller] = controller
          scope(options) { yield }
649 650
        end

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

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

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

766
        private
J
José Valim 已提交
767
          def scope_options #:nodoc:
768 769 770
            @scope_options ||= private_methods.grep(/^merge_(.+)_scope$/) { $1.to_sym }
          end

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

J
José Valim 已提交
775
          def merge_shallow_path_scope(parent, child) #:nodoc:
776 777 778
            Mapper.normalize_path("#{parent}/#{child}")
          end

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

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

J
José Valim 已提交
787
          def merge_module_scope(parent, child) #:nodoc:
788 789 790
            parent ? "#{parent}/#{child}" : child
          end

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

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

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

J
José Valim 已提交
803
          def merge_defaults_scope(parent, child) #:nodoc:
804 805 806
            merge_options_scope(parent, child)
          end

J
José Valim 已提交
807
          def merge_blocks_scope(parent, child) #:nodoc:
808 809 810
            merged = parent ? parent.dup : []
            merged << child if child
            merged
811 812
          end

J
José Valim 已提交
813
          def merge_options_scope(parent, child) #:nodoc:
814
            (parent || {}).except(*override_keys(child)).merge(child)
815
          end
816

J
José Valim 已提交
817
          def merge_shallow_scope(parent, child) #:nodoc:
818 819
            child ? true : false
          end
820

J
José Valim 已提交
821
          def override_keys(child) #:nodoc:
822 823
            child.key?(:only) || child.key?(:except) ? [:only, :except] : []
          end
824 825
      end

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

872
        class Resource #:nodoc:
873
          DEFAULT_ACTIONS = [:index, :create, :new, :show, :update, :destroy, :edit]
874

875
          attr_reader :controller, :path, :options
876 877

          def initialize(entities, options = {})
878
            @name       = entities.to_s
879
            @path       = (options.delete(:path) || @name).to_s
880
            @controller = (options.delete(:controller) || @name).to_s
881
            @as         = options.delete(:as)
882
            @options    = options
883 884
          end

885
          def default_actions
886
            self.class::DEFAULT_ACTIONS
887 888
          end

889
          def actions
890
            if only = @options[:only]
891
              Array(only).map(&:to_sym)
892
            elsif except = @options[:except]
893 894 895 896 897 898
              default_actions - Array(except).map(&:to_sym)
            else
              default_actions
            end
          end

899
          def name
900
            @as || @name
901 902
          end

903
          def plural
904
            @plural ||= name.to_s
905 906 907
          end

          def singular
908
            @singular ||= name.to_s.singularize
909 910
          end

911
          alias :member_name :singular
912

913
          # Checks for uncountable plurals, and appends "_index" if the plural
914
          # and singular form are the same.
915
          def collection_name
916
            singular == plural ? "#{plural}_index" : plural
917 918
          end

919
          def resource_scope
920
            { :controller => controller }
921 922
          end

923
          alias :collection_scope :path
924 925

          def member_scope
926
            "#{path}/:id"
927 928
          end

929
          def new_scope(new_path)
930
            "#{path}/#{new_path}"
931 932 933
          end

          def nested_scope
934
            "#{path}/:#{singular}_id"
935
          end
936

937 938 939
        end

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

942
          def initialize(entities, options)
943
            @as         = nil
944
            @name       = entities.to_s
945
            @path       = (options.delete(:path) || @name).to_s
946
            @controller = (options.delete(:controller) || plural).to_s
947 948 949 950
            @as         = options.delete(:as)
            @options    = options
          end

951 952
          def plural
            @plural ||= name.to_s.pluralize
953 954
          end

955 956
          def singular
            @singular ||= name.to_s
957
          end
958 959 960 961 962 963

          alias :member_name :singular
          alias :collection_name :singular

          alias :member_scope :path
          alias :nested_scope :path
964 965
        end

966 967 968 969
        def resources_path_names(options)
          @scope[:path_names].merge!(options)
        end

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

994
          if apply_common_behavior_for(:resource, resources, options, &block)
995 996 997
            return self
          end

998 999
          resource_scope(SingletonResource.new(resources.pop, options)) do
            yield if block_given?
1000

1001
            collection do
1002
              post :create
1003
            end if parent_resource.actions.include?(:create)
1004

1005
            new do
1006
              get :new
1007
            end if parent_resource.actions.include?(:new)
1008

1009
            member do
1010
              get    :edit if parent_resource.actions.include?(:edit)
1011 1012 1013
              get    :show if parent_resource.actions.include?(:show)
              put    :update if parent_resource.actions.include?(:update)
              delete :destroy if parent_resource.actions.include?(:destroy)
1014 1015 1016
            end
          end

J
Joshua Peek 已提交
1017
          self
1018 1019
        end

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
S
Sebastian Martinez 已提交
1028
        # the +Photos+ controller:
1029 1030 1031 1032 1033 1034 1035
        #
        #   GET     /photos/new
        #   POST    /photos
        #   GET     /photos/:id
        #   GET     /photos/:id/edit
        #   PUT     /photos/:id
        #   DELETE  /photos/:id
1036
        #
1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051
        # 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
        #
1052
        # === Options
1053 1054
        # Takes same options as <tt>Base#match</tt> as well as:
        #
1055 1056 1057 1058 1059 1060 1061
        # [: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
1062
        #
1063 1064
        # [:only]
        #   Only generate routes for the given actions.
1065
        #
1066 1067
        #     resources :cows, :only => :show
        #     resources :cows, :only => [:show, :index]
1068
        #
1069 1070
        # [:except]
        #   Generate all routes except for the given actions.
1071
        #
1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085
        #     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
1086
        #       resources :comments, :except => [:show, :edit, :update, :destroy]
1087
        #     end
1088 1089 1090 1091 1092
        #     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>.
1093 1094 1095 1096
        #
        # [:shallow_path]
        #   Prefixes nested shallow routes with the specified path.
        #
1097 1098 1099 1100
        #     scope :shallow_path => "sekret" do
        #       resources :posts do
        #         resources :comments, :shallow => true
        #       end
1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113
        #     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
1114
        #
S
Sebastian Martinez 已提交
1115
        #   # routes call <tt>Admin::PostsController</tt>
1116
        #   resources :posts, :module => "admin"
1117
        #
1118
        #   # resource actions are at /admin/posts.
1119
        #   resources :posts, :path => "admin/posts"
J
Joshua Peek 已提交
1120
        def resources(*resources, &block)
J
Joshua Peek 已提交
1121
          options = resources.extract_options!
1122

1123
          if apply_common_behavior_for(:resources, resources, options, &block)
1124 1125 1126
            return self
          end

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

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

1135
            new do
1136
              get :new
1137
            end if parent_resource.actions.include?(:new)
1138

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

J
Joshua Peek 已提交
1147
          self
1148 1149
        end

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

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

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

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

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

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

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

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

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

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

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

J
Joshua Peek 已提交
1250
        def match(*args)
1251
          options = args.extract_options!.dup
1252 1253
          options[:anchor] = true unless options.key?(:anchor)

1254
          if args.length > 1
1255
            args.each { |path| match(path, options.dup) }
1256 1257 1258
            return self
          end

1259 1260
          on = options.delete(:on)
          if VALID_ON_OPTIONS.include?(on)
1261
            args.push(options)
1262 1263 1264
            return send(on){ match(*args) }
          elsif on
            raise ArgumentError, "Unknown scope #{on.inspect} given to :on"
1265 1266
          end

1267 1268 1269 1270
          if @scope[:scope_level] == :resources
            args.push(options)
            return nested { match(*args) }
          elsif @scope[:scope_level] == :resource
1271
            args.push(options)
J
Joshua Peek 已提交
1272 1273
            return member { match(*args) }
          end
J
Joshua Peek 已提交
1274

1275
          action = args.first
1276
          path = path_for_action(action, options.delete(:path))
1277

1278 1279 1280
          if action.to_s =~ /^[\w\/]+$/
            options[:action] ||= action unless action.to_s.include?("/")
          else
1281 1282 1283 1284 1285 1286 1287
            action = nil
          end

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

1290
          super(path, options)
J
Joshua Peek 已提交
1291 1292
        end

1293
        def root(options={})
1294
          if @scope[:scope_level] == :resources
1295 1296
            with_scope_level(:root) do
              scope(parent_resource.path) do
1297 1298 1299 1300 1301 1302
                super(options)
              end
            end
          else
            super(options)
          end
1303 1304
        end

1305
        protected
1306

1307
          def parent_resource #:nodoc:
1308 1309 1310
            @scope[:scope_level_resource]
          end

J
José Valim 已提交
1311
          def apply_common_behavior_for(method, resources, options, &block) #:nodoc:
1312 1313 1314 1315 1316
            if resources.length > 1
              resources.each { |r| send(method, r, options, &block) }
              return true
            end

1317 1318 1319 1320 1321
            if resource_scope?
              nested { send(method, resources.pop, options, &block) }
              return true
            end

1322
            options.keys.each do |k|
1323 1324 1325
              (options[:constraints] ||= {})[k] = options.delete(k) if options[k].is_a?(Regexp)
            end

1326 1327 1328
            scope_options = options.slice!(*RESOURCE_OPTIONS)
            unless scope_options.empty?
              scope(scope_options) do
1329 1330 1331 1332 1333
                send(method, resources.pop, options, &block)
              end
              return true
            end

1334 1335 1336 1337
            unless action_options?(options)
              options.merge!(scope_action_options) if scope_action_options?
            end

1338 1339 1340
            false
          end

J
José Valim 已提交
1341
          def action_options?(options) #:nodoc:
1342 1343 1344
            options[:only] || options[:except]
          end

J
José Valim 已提交
1345
          def scope_action_options? #:nodoc:
1346 1347 1348
            @scope[:options].is_a?(Hash) && (@scope[:options][:only] || @scope[:options][:except])
          end

J
José Valim 已提交
1349
          def scope_action_options #:nodoc:
1350 1351 1352
            @scope[:options].slice(:only, :except)
          end

J
José Valim 已提交
1353
          def resource_scope? #:nodoc:
1354
            @scope[:scope_level].in?([:resource, :resources])
1355 1356
          end

J
José Valim 已提交
1357
          def resource_method_scope? #:nodoc:
1358
            @scope[:scope_level].in?([:collection, :member, :new])
1359 1360
          end

1361
          def with_exclusive_scope
1362
            begin
1363 1364
              old_name_prefix, old_path = @scope[:as], @scope[:path]
              @scope[:as], @scope[:path] = nil, nil
1365

1366 1367 1368
              with_scope_level(:exclusive) do
                yield
              end
1369
            ensure
1370
              @scope[:as], @scope[:path] = old_name_prefix, old_path
1371 1372 1373
            end
          end

1374
          def with_scope_level(kind, resource = parent_resource)
J
Joshua Peek 已提交
1375
            old, @scope[:scope_level] = @scope[:scope_level], kind
1376
            old_resource, @scope[:scope_level_resource] = @scope[:scope_level_resource], resource
J
Joshua Peek 已提交
1377 1378 1379
            yield
          ensure
            @scope[:scope_level] = old
1380
            @scope[:scope_level_resource] = old_resource
J
Joshua Peek 已提交
1381
          end
1382

J
José Valim 已提交
1383
          def resource_scope(resource) #:nodoc:
1384
            with_scope_level(resource.is_a?(SingletonResource) ? :resource : :resources, resource) do
1385
              scope(parent_resource.resource_scope) do
1386 1387 1388 1389 1390
                yield
              end
            end
          end

J
José Valim 已提交
1391
          def nested_options #:nodoc:
1392 1393 1394 1395 1396 1397
            {}.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 已提交
1398
          def id_constraint? #:nodoc:
1399
            @scope[:constraints] && @scope[:constraints][:id].is_a?(Regexp)
1400 1401
          end

J
José Valim 已提交
1402
          def id_constraint #:nodoc:
1403
            @scope[:constraints][:id]
1404 1405
          end

J
José Valim 已提交
1406
          def canonical_action?(action, flag) #:nodoc:
1407
            flag && resource_method_scope? && CANONICAL_ACTIONS.include?(action.to_s)
1408 1409
          end

J
José Valim 已提交
1410
          def shallow_scoping? #:nodoc:
1411
            shallow? && @scope[:scope_level] == :member
1412 1413
          end

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

1418 1419
            path = if canonical_action?(action, path.blank?)
              prefix.to_s
1420
            else
1421
              "#{prefix}/#{action_path(action, path)}"
1422 1423 1424
            end
          end

J
José Valim 已提交
1425
          def action_path(name, path = nil) #:nodoc:
1426 1427 1428
            # Ruby 1.8 can't transform empty strings to symbols
            name = name.to_sym if name.is_a?(String) && !name.empty?
            path || @scope[:path_names][name] || name.to_s
1429 1430
          end

J
José Valim 已提交
1431
          def prefix_name_for_action(as, action) #:nodoc:
1432
            if as
1433
              as.to_s
1434
            elsif !canonical_action?(action, @scope[:scope_level])
1435
              action.to_s
1436
            end
1437 1438
          end

J
José Valim 已提交
1439
          def name_for_action(as, action) #:nodoc:
1440
            prefix = prefix_name_for_action(as, action)
1441
            prefix = Mapper.normalize_name(prefix) if prefix
1442 1443 1444
            name_prefix = @scope[:as]

            if parent_resource
1445 1446
              return nil if as.nil? && action.nil?

1447 1448
              collection_name = parent_resource.collection_name
              member_name = parent_resource.member_name
1449
            end
1450

1451
            name = case @scope[:scope_level]
1452
            when :nested
1453
              [name_prefix, prefix]
1454
            when :collection
1455
              [prefix, name_prefix, collection_name]
1456
            when :new
1457 1458 1459 1460 1461
              [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]
1462
            else
1463
              [name_prefix, member_name, prefix]
1464
            end
1465

1466
            candidate = name.select(&:present?).join("_").presence
1467
            candidate unless as.nil? && @set.routes.find { |r| r.name == candidate }
1468
          end
J
Joshua Peek 已提交
1469
      end
J
Joshua Peek 已提交
1470

J
José Valim 已提交
1471
      module Shorthand #:nodoc:
1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483
        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

1484 1485 1486 1487 1488
      def initialize(set) #:nodoc:
        @set = set
        @scope = { :path_names => @set.resources_path_names }
      end

1489 1490
      include Base
      include HttpHelpers
1491
      include Redirection
1492 1493
      include Scoping
      include Resources
1494
      include Shorthand
J
Joshua Peek 已提交
1495 1496
    end
  end
J
Joshua Peek 已提交
1497
end