url_rewriter.rb 3.4 KB
Newer Older
D
Initial  
David Heinemeier Hansson 已提交
1
module ActionController
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
  
  # Write URLs from arbitrary places in your codebase, such as your mailers.
  # 
  # Example:
  # 
  #   class MyMailer
  #     include ActionController::UrlWriter
  #     default_url_options[:host] = 'www.basecamphq.com'
  #     
  #     def signup_url(token)
  #       url_for(:controller => 'signup', action => 'index', :token => token)
  #     end
  #  end
  # 
  # In addition to providing +url_for+, named routes are also accessible after
  # including UrlWriter.
  # 
  module UrlWriter
    
    # The default options for urls written by this writer. Typically a :host pair
    # is provided.
    mattr_accessor :default_url_options
    self.default_url_options = {}
    
    def self.included(base) #:nodoc:
      ActionController::Routing::Routes.named_routes.install base
      base.mattr_accessor :default_url_options
      base.default_url_options ||= default_url_options
    end
    
    # Generate a url with the provided options. The following special options may
    # effect the constructed url:
    # 
    #   * :host Specifies the host the link should be targetted at. This option
    #     must be provided either explicitly, or via default_url_options.
    #   * :protocol The protocol to connect to. Defaults to 'http'
    #   * :port Optionally specify the port to connect to.
    # 
    def url_for(options)
      options = self.class.default_url_options.merge(options)
      
      raise "Missing host to link to! Please provide :host parameter or set default_url_options[:host]" unless options[:host]
      
      url = ''
      url << (options.delete(:protocol) || 'http')
      url << '://'
      url << options.delete(:host)
      url << ":#{options.delete(:port)}" if options.key?(:port)
      url << Routing::Routes.generate(options, {})
      return url
    end
    
  end
  
56
  # Rewrites URLs for Base.redirect_to and Base.url_for in the controller.
D
Initial  
David Heinemeier Hansson 已提交
57
  class UrlRewriter #:nodoc:
58
    RESERVED_OPTIONS = [:anchor, :params, :only_path, :host, :protocol, :trailing_slash, :skip_relative_url_root]
59 60
    def initialize(request, parameters)
      @request, @parameters = request, parameters
D
Initial  
David Heinemeier Hansson 已提交
61 62
    end
    
63
    def rewrite(options = {})
64
      rewrite_url(rewrite_path(options), options)
D
Initial  
David Heinemeier Hansson 已提交
65 66 67
    end

    def to_str
68
      "#{@request.protocol}, #{@request.host_with_port}, #{@request.path}, #{@parameters[:controller]}, #{@parameters[:action]}, #{@request.parameters.inspect}"
D
Initial  
David Heinemeier Hansson 已提交
69 70
    end

71 72
    alias_method :to_s, :to_str

D
Initial  
David Heinemeier Hansson 已提交
73
    private
74
      def rewrite_url(path, options)
D
Initial  
David Heinemeier Hansson 已提交
75
        rewritten_url = ""
76 77 78 79
        unless options[:only_path]
          rewritten_url << (options[:protocol] || @request.protocol)
          rewritten_url << (options[:host] || @request.host_with_port)
        end
D
Initial  
David Heinemeier Hansson 已提交
80

81
        rewritten_url << @request.relative_url_root.to_s unless options[:skip_relative_url_root]
D
Initial  
David Heinemeier Hansson 已提交
82
        rewritten_url << path
83
        rewritten_url << '/' if options[:trailing_slash]
D
Initial  
David Heinemeier Hansson 已提交
84
        rewritten_url << "##{options[:anchor]}" if options[:anchor]
85

86
        rewritten_url
D
Initial  
David Heinemeier Hansson 已提交
87 88
      end

89 90 91
      def rewrite_path(options)
        options = options.symbolize_keys
        options.update(options[:params].symbolize_keys) if options[:params]
92
        if (overwrite = options.delete(:overwrite_params))
93
          options.update(@parameters.symbolize_keys)
94 95
          options.update(overwrite)
        end
96
        RESERVED_OPTIONS.each {|k| options.delete k}
97

98
        # Generates the query string, too
99
        Routing::Routes.generate(options, @request.symbolized_path_parameters)
D
Initial  
David Heinemeier Hansson 已提交
100 101
      end
  end
102
  
103
end