Sha256: 6bab040d873e06894ab507f17307c5e5512ad90f4bc0a2e444d64eaa34951626

Contents?: true

Size: 2 KB

Versions: 2

Compression:

Stored size: 2 KB

Contents

require 'ipaddress'

module Rack
  class Subdomain
    def initialize(app, domain, options = {}, &block)
      # Maintain compatibility with previous rack-subdomain gem
      options = {to: options} if options.is_a? String

      @options = {except: ['', 'www']}.merge(options)

      @app = app
      @domain = domain
      @mappings = {}

      if @options[:to]
        map('*', @options[:to])
      elsif block_given?
        instance_eval &block
      else
        raise ArgumentError, "missing `:to` option or block to define mapping"
      end
    end

    def call(env)
      @env = env
      @subdomain = subdomain

      if @subdomain && !@subdomain.empty? && !@options[:except].include?(@subdomain)
        pattern, route = @mappings.detect do |pattern, route|
          pattern === subdomain
        end

        if route
          remap_with_substituted_path!(route.gsub(/\:subdomain/, @subdomain))
        end
      end

      @app.call(env)
    end

    def map(pattern, route)
      map(pattern, route[:to]) and return if route.kind_of?(Hash)
      raise ArgumentError, "missing route" unless route

      pattern = /.*/ if pattern == '*'
      regexp = Regexp.new(pattern)
      @mappings[regexp] = route
    end

  private

    def domain
      case @domain
      when Proc
        @domain.call(@env['HTTP_HOST'])
      when Regexp
        @domain.match(@env['HTTP_HOST'])
      else
        @domain
      end
    end

    def subdomain
      @env['HTTP_HOST'].sub(/\.?#{domain}.*$/,'') unless @env['HTTP_HOST'].match(/^localhost/) or IPAddress.valid?(@env['SERVER_NAME'])
    end

    def remap_with_substituted_path!(path)
      scheme = @env["rack.url_scheme"]
      host = "#{@subdomain}.#{domain}"
      port = ":#{@env['SERVER_PORT']}" unless @env['SERVER_PORT'] == '80'
      path = @env["PATH_INFO"] = ::File.join(path, @env["PATH_INFO"])
      query_string = "?" + @env["QUERY_STRING"] unless @env["QUERY_STRING"].empty?

      @env["REQUEST_URI"] = "#{scheme}://#{host}#{port}#{path}#{query_string}"
    end
  end
end

Version data entries

2 entries across 2 versions & 1 rubygems

Version Path
rack-subdomain-0.4.0 ./lib/rack/subdomain.rb
rack-subdomain-0.3.1 ./lib/rack/subdomain.rb