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