Apipie/apipie-rails

View on GitHub
lib/apipie/static_dispatcher.rb

Summary

Maintainability
A
0 mins
Test Coverage
module Apipie

  class FileHandler
    def initialize(root)
      @root          = root.chomp('/')
      @compiled_root = /^#{Regexp.escape(root)}/
      @file_server   = if defined?(::Rack::Files)
                         ::Rack::Files.new(@root)
                       else
                         # Deprecated in Rack 3.0, kept
                         # for backward compatibility
                         ::Rack::File.new(@root)
                       end
    end

    def match?(path)
      # Replace all null bytes
      path = ::Rack::Utils.unescape(path || '')
        .encode(Encoding::UTF_8, invalid: :replace, replace: '')
        .gsub("\x0", '')

      full_path = path.empty? ? @root : File.join(@root, path)
      paths = "#{full_path}#{ext}"

      matches = Dir[paths]
      match = matches.detect { |m| File.file?(m) }
      if match
        match.sub!(@compiled_root, '')
        match
      end
    end

    def call(env)
      @file_server.call(env)
    end

    def ext
      @ext ||= begin
        ext = cache_extension
        "{,#{ext},/index#{ext}}"
      end
    end

    def cache_extension
      if ::ActionController::Base.respond_to?(:default_static_extension)
        ::ActionController::Base.default_static_extension
      else
        ::ActionController::Base.page_cache_extension
      end

    end
  end

  class StaticDispatcher
    # Dispatches the static files. Similar to ActionDispatch::Static, but
    # it supports different baseurl configurations
    def initialize(app, path)
      @app = app
      @file_handler = Apipie::FileHandler.new(path)
    end

    def call(env)
      @baseurl ||= Apipie.configuration.doc_base_url
      case env['REQUEST_METHOD']
      when 'GET', 'HEAD'
        path = env['PATH_INFO'].sub("#{@baseurl}/","/apipie/").chomp('/')

        if match = @file_handler.match?(path)
          env["PATH_INFO"] = match
          return @file_handler.call(env)
        end
      end

      @app.call(env)
    end
  end
end