rack 2.2.8 → 3.1.10
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of rack might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/CHANGELOG.md +353 -81
- data/CONTRIBUTING.md +63 -55
- data/MIT-LICENSE +1 -1
- data/README.md +328 -0
- data/SPEC.rdoc +204 -131
- data/lib/rack/auth/abstract/handler.rb +3 -1
- data/lib/rack/auth/abstract/request.rb +3 -1
- data/lib/rack/auth/basic.rb +1 -4
- data/lib/rack/bad_request.rb +8 -0
- data/lib/rack/body_proxy.rb +21 -3
- data/lib/rack/builder.rb +102 -69
- data/lib/rack/cascade.rb +2 -3
- data/lib/rack/common_logger.rb +25 -19
- data/lib/rack/conditional_get.rb +18 -15
- data/lib/rack/constants.rb +67 -0
- data/lib/rack/content_length.rb +12 -16
- data/lib/rack/content_type.rb +8 -5
- data/lib/rack/deflater.rb +40 -26
- data/lib/rack/directory.rb +9 -3
- data/lib/rack/etag.rb +14 -23
- data/lib/rack/events.rb +4 -0
- data/lib/rack/files.rb +15 -17
- data/lib/rack/head.rb +9 -8
- data/lib/rack/headers.rb +238 -0
- data/lib/rack/lint.rb +840 -644
- data/lib/rack/lock.rb +2 -5
- data/lib/rack/logger.rb +3 -0
- data/lib/rack/media_type.rb +17 -7
- data/lib/rack/method_override.rb +5 -1
- data/lib/rack/mime.rb +14 -5
- data/lib/rack/mock.rb +1 -271
- data/lib/rack/mock_request.rb +161 -0
- data/lib/rack/mock_response.rb +124 -0
- data/lib/rack/multipart/generator.rb +7 -5
- data/lib/rack/multipart/parser.rb +214 -90
- data/lib/rack/multipart/uploaded_file.rb +4 -0
- data/lib/rack/multipart.rb +53 -40
- data/lib/rack/null_logger.rb +9 -0
- data/lib/rack/query_parser.rb +81 -102
- data/lib/rack/recursive.rb +2 -0
- data/lib/rack/reloader.rb +0 -2
- data/lib/rack/request.rb +260 -123
- data/lib/rack/response.rb +151 -66
- data/lib/rack/rewindable_input.rb +24 -5
- data/lib/rack/runtime.rb +7 -6
- data/lib/rack/sendfile.rb +30 -25
- data/lib/rack/show_exceptions.rb +21 -4
- data/lib/rack/show_status.rb +17 -7
- data/lib/rack/static.rb +8 -8
- data/lib/rack/tempfile_reaper.rb +15 -4
- data/lib/rack/urlmap.rb +3 -1
- data/lib/rack/utils.rb +240 -237
- data/lib/rack/version.rb +1 -9
- data/lib/rack.rb +13 -89
- metadata +15 -44
- data/README.rdoc +0 -320
- data/Rakefile +0 -130
- data/bin/rackup +0 -5
- data/contrib/rack.png +0 -0
- data/contrib/rack.svg +0 -150
- data/contrib/rack_logo.svg +0 -164
- data/contrib/rdoc.css +0 -412
- data/example/lobster.ru +0 -6
- data/example/protectedlobster.rb +0 -16
- data/example/protectedlobster.ru +0 -10
- data/lib/rack/auth/digest/md5.rb +0 -131
- data/lib/rack/auth/digest/nonce.rb +0 -54
- data/lib/rack/auth/digest/params.rb +0 -54
- data/lib/rack/auth/digest/request.rb +0 -43
- data/lib/rack/chunked.rb +0 -117
- data/lib/rack/core_ext/regexp.rb +0 -14
- data/lib/rack/file.rb +0 -7
- data/lib/rack/handler/cgi.rb +0 -59
- data/lib/rack/handler/fastcgi.rb +0 -100
- data/lib/rack/handler/lsws.rb +0 -61
- data/lib/rack/handler/scgi.rb +0 -71
- data/lib/rack/handler/thin.rb +0 -36
- data/lib/rack/handler/webrick.rb +0 -129
- data/lib/rack/handler.rb +0 -104
- data/lib/rack/lobster.rb +0 -70
- data/lib/rack/server.rb +0 -466
- data/lib/rack/session/abstract/id.rb +0 -523
- data/lib/rack/session/cookie.rb +0 -204
- data/lib/rack/session/memcache.rb +0 -10
- data/lib/rack/session/pool.rb +0 -85
- data/rack.gemspec +0 -46
| @@ -1,43 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require_relative '../abstract/request'
         | 
| 4 | 
            -
            require_relative 'params'
         | 
| 5 | 
            -
            require_relative 'nonce'
         | 
| 6 | 
            -
             | 
| 7 | 
            -
            module Rack
         | 
| 8 | 
            -
              module Auth
         | 
| 9 | 
            -
                module Digest
         | 
| 10 | 
            -
                  class Request < Auth::AbstractRequest
         | 
| 11 | 
            -
                    def method
         | 
| 12 | 
            -
                      @env[RACK_METHODOVERRIDE_ORIGINAL_METHOD] || @env[REQUEST_METHOD]
         | 
| 13 | 
            -
                    end
         | 
| 14 | 
            -
             | 
| 15 | 
            -
                    def digest?
         | 
| 16 | 
            -
                      "digest" == scheme
         | 
| 17 | 
            -
                    end
         | 
| 18 | 
            -
             | 
| 19 | 
            -
                    def correct_uri?
         | 
| 20 | 
            -
                      request.fullpath == uri
         | 
| 21 | 
            -
                    end
         | 
| 22 | 
            -
             | 
| 23 | 
            -
                    def nonce
         | 
| 24 | 
            -
                      @nonce ||= Nonce.parse(params['nonce'])
         | 
| 25 | 
            -
                    end
         | 
| 26 | 
            -
             | 
| 27 | 
            -
                    def params
         | 
| 28 | 
            -
                      @params ||= Params.parse(parts.last)
         | 
| 29 | 
            -
                    end
         | 
| 30 | 
            -
             | 
| 31 | 
            -
                    def respond_to?(sym, *)
         | 
| 32 | 
            -
                      super or params.has_key? sym.to_s
         | 
| 33 | 
            -
                    end
         | 
| 34 | 
            -
             | 
| 35 | 
            -
                    def method_missing(sym, *args)
         | 
| 36 | 
            -
                      return super unless params.has_key?(key = sym.to_s)
         | 
| 37 | 
            -
                      return params[key] if args.size == 0
         | 
| 38 | 
            -
                      raise ArgumentError, "wrong number of arguments (#{args.size} for 0)"
         | 
| 39 | 
            -
                    end
         | 
| 40 | 
            -
                  end
         | 
| 41 | 
            -
                end
         | 
| 42 | 
            -
              end
         | 
| 43 | 
            -
            end
         | 
    
        data/lib/rack/chunked.rb
    DELETED
    
    | @@ -1,117 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            module Rack
         | 
| 4 | 
            -
             | 
| 5 | 
            -
              # Middleware that applies chunked transfer encoding to response bodies
         | 
| 6 | 
            -
              # when the response does not include a Content-Length header.
         | 
| 7 | 
            -
              #
         | 
| 8 | 
            -
              # This supports the Trailer response header to allow the use of trailing
         | 
| 9 | 
            -
              # headers in the chunked encoding.  However, using this requires you manually
         | 
| 10 | 
            -
              # specify a response body that supports a +trailers+ method.  Example:
         | 
| 11 | 
            -
              #
         | 
| 12 | 
            -
              #   [200, { 'Trailer' => 'Expires'}, ["Hello", "World"]]
         | 
| 13 | 
            -
              #   # error raised
         | 
| 14 | 
            -
              #
         | 
| 15 | 
            -
              #   body = ["Hello", "World"]
         | 
| 16 | 
            -
              #   def body.trailers
         | 
| 17 | 
            -
              #     { 'Expires' => Time.now.to_s }
         | 
| 18 | 
            -
              #   end
         | 
| 19 | 
            -
              #   [200, { 'Trailer' => 'Expires'}, body]
         | 
| 20 | 
            -
              #   # No exception raised
         | 
| 21 | 
            -
              class Chunked
         | 
| 22 | 
            -
                include Rack::Utils
         | 
| 23 | 
            -
             | 
| 24 | 
            -
                # A body wrapper that emits chunked responses.
         | 
| 25 | 
            -
                class Body
         | 
| 26 | 
            -
                  TERM = "\r\n"
         | 
| 27 | 
            -
                  TAIL = "0#{TERM}"
         | 
| 28 | 
            -
             | 
| 29 | 
            -
                  # Store the response body to be chunked.
         | 
| 30 | 
            -
                  def initialize(body)
         | 
| 31 | 
            -
                    @body = body
         | 
| 32 | 
            -
                  end
         | 
| 33 | 
            -
             | 
| 34 | 
            -
                  # For each element yielded by the response body, yield
         | 
| 35 | 
            -
                  # the element in chunked encoding.
         | 
| 36 | 
            -
                  def each(&block)
         | 
| 37 | 
            -
                    term = TERM
         | 
| 38 | 
            -
                    @body.each do |chunk|
         | 
| 39 | 
            -
                      size = chunk.bytesize
         | 
| 40 | 
            -
                      next if size == 0
         | 
| 41 | 
            -
             | 
| 42 | 
            -
                      yield [size.to_s(16), term, chunk.b, term].join
         | 
| 43 | 
            -
                    end
         | 
| 44 | 
            -
                    yield TAIL
         | 
| 45 | 
            -
                    yield_trailers(&block)
         | 
| 46 | 
            -
                    yield term
         | 
| 47 | 
            -
                  end
         | 
| 48 | 
            -
             | 
| 49 | 
            -
                  # Close the response body if the response body supports it.
         | 
| 50 | 
            -
                  def close
         | 
| 51 | 
            -
                    @body.close if @body.respond_to?(:close)
         | 
| 52 | 
            -
                  end
         | 
| 53 | 
            -
             | 
| 54 | 
            -
                  private
         | 
| 55 | 
            -
             | 
| 56 | 
            -
                  # Do nothing as this class does not support trailer headers.
         | 
| 57 | 
            -
                  def yield_trailers
         | 
| 58 | 
            -
                  end
         | 
| 59 | 
            -
                end
         | 
| 60 | 
            -
             | 
| 61 | 
            -
                # A body wrapper that emits chunked responses and also supports
         | 
| 62 | 
            -
                # sending Trailer headers.  Note that the response body provided to
         | 
| 63 | 
            -
                # initialize must have a +trailers+ method that returns a hash
         | 
| 64 | 
            -
                # of trailer headers, and the rack response itself should have a
         | 
| 65 | 
            -
                # Trailer header listing the headers that the +trailers+ method
         | 
| 66 | 
            -
                # will return.
         | 
| 67 | 
            -
                class TrailerBody < Body
         | 
| 68 | 
            -
                  private
         | 
| 69 | 
            -
             | 
| 70 | 
            -
                  # Yield strings for each trailer header.
         | 
| 71 | 
            -
                  def yield_trailers
         | 
| 72 | 
            -
                    @body.trailers.each_pair do |k, v|
         | 
| 73 | 
            -
                      yield "#{k}: #{v}\r\n"
         | 
| 74 | 
            -
                    end
         | 
| 75 | 
            -
                  end
         | 
| 76 | 
            -
                end
         | 
| 77 | 
            -
             | 
| 78 | 
            -
                def initialize(app)
         | 
| 79 | 
            -
                  @app = app
         | 
| 80 | 
            -
                end
         | 
| 81 | 
            -
             | 
| 82 | 
            -
                # Whether the HTTP version supports chunked encoding (HTTP 1.1 does).
         | 
| 83 | 
            -
                def chunkable_version?(ver)
         | 
| 84 | 
            -
                  case ver
         | 
| 85 | 
            -
                  # pre-HTTP/1.0 (informally "HTTP/0.9") HTTP requests did not have
         | 
| 86 | 
            -
                  # a version (nor response headers)
         | 
| 87 | 
            -
                  when 'HTTP/1.0', nil, 'HTTP/0.9'
         | 
| 88 | 
            -
                    false
         | 
| 89 | 
            -
                  else
         | 
| 90 | 
            -
                    true
         | 
| 91 | 
            -
                  end
         | 
| 92 | 
            -
                end
         | 
| 93 | 
            -
             | 
| 94 | 
            -
                # If the rack app returns a response that should have a body,
         | 
| 95 | 
            -
                # but does not have Content-Length or Transfer-Encoding headers,
         | 
| 96 | 
            -
                # modify the response to use chunked Transfer-Encoding.
         | 
| 97 | 
            -
                def call(env)
         | 
| 98 | 
            -
                  status, headers, body = @app.call(env)
         | 
| 99 | 
            -
                  headers = HeaderHash[headers]
         | 
| 100 | 
            -
             | 
| 101 | 
            -
                  if chunkable_version?(env[SERVER_PROTOCOL]) &&
         | 
| 102 | 
            -
                     !STATUS_WITH_NO_ENTITY_BODY.key?(status.to_i) &&
         | 
| 103 | 
            -
                     !headers[CONTENT_LENGTH] &&
         | 
| 104 | 
            -
                     !headers[TRANSFER_ENCODING]
         | 
| 105 | 
            -
             | 
| 106 | 
            -
                    headers[TRANSFER_ENCODING] = 'chunked'
         | 
| 107 | 
            -
                    if headers['Trailer']
         | 
| 108 | 
            -
                      body = TrailerBody.new(body)
         | 
| 109 | 
            -
                    else
         | 
| 110 | 
            -
                      body = Body.new(body)
         | 
| 111 | 
            -
                    end
         | 
| 112 | 
            -
                  end
         | 
| 113 | 
            -
             | 
| 114 | 
            -
                  [status, headers, body]
         | 
| 115 | 
            -
                end
         | 
| 116 | 
            -
              end
         | 
| 117 | 
            -
            end
         | 
    
        data/lib/rack/core_ext/regexp.rb
    DELETED
    
    | @@ -1,14 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            # Regexp has `match?` since Ruby 2.4
         | 
| 4 | 
            -
            # so to support Ruby < 2.4 we need to define this method
         | 
| 5 | 
            -
             | 
| 6 | 
            -
            module Rack
         | 
| 7 | 
            -
              module RegexpExtensions
         | 
| 8 | 
            -
                refine Regexp do
         | 
| 9 | 
            -
                  def match?(string, pos = 0)
         | 
| 10 | 
            -
                    !!match(string, pos)
         | 
| 11 | 
            -
                  end
         | 
| 12 | 
            -
                end unless //.respond_to?(:match?)
         | 
| 13 | 
            -
              end
         | 
| 14 | 
            -
            end
         | 
    
        data/lib/rack/file.rb
    DELETED
    
    
    
        data/lib/rack/handler/cgi.rb
    DELETED
    
    | @@ -1,59 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            module Rack
         | 
| 4 | 
            -
              module Handler
         | 
| 5 | 
            -
                class CGI
         | 
| 6 | 
            -
                  def self.run(app, **options)
         | 
| 7 | 
            -
                    $stdin.binmode
         | 
| 8 | 
            -
                    serve app
         | 
| 9 | 
            -
                  end
         | 
| 10 | 
            -
             | 
| 11 | 
            -
                  def self.serve(app)
         | 
| 12 | 
            -
                    env = ENV.to_hash
         | 
| 13 | 
            -
                    env.delete "HTTP_CONTENT_LENGTH"
         | 
| 14 | 
            -
             | 
| 15 | 
            -
                    env[SCRIPT_NAME] = ""  if env[SCRIPT_NAME] == "/"
         | 
| 16 | 
            -
             | 
| 17 | 
            -
                    env.update(
         | 
| 18 | 
            -
                      RACK_VERSION      => Rack::VERSION,
         | 
| 19 | 
            -
                      RACK_INPUT        => Rack::RewindableInput.new($stdin),
         | 
| 20 | 
            -
                      RACK_ERRORS       => $stderr,
         | 
| 21 | 
            -
                      RACK_MULTITHREAD  => false,
         | 
| 22 | 
            -
                      RACK_MULTIPROCESS => true,
         | 
| 23 | 
            -
                      RACK_RUNONCE      => true,
         | 
| 24 | 
            -
                      RACK_URL_SCHEME   => ["yes", "on", "1"].include?(ENV[HTTPS]) ? "https" : "http"
         | 
| 25 | 
            -
                    )
         | 
| 26 | 
            -
             | 
| 27 | 
            -
                    env[QUERY_STRING] ||= ""
         | 
| 28 | 
            -
                    env[HTTP_VERSION] ||= env[SERVER_PROTOCOL]
         | 
| 29 | 
            -
                    env[REQUEST_PATH] ||= "/"
         | 
| 30 | 
            -
             | 
| 31 | 
            -
                    status, headers, body = app.call(env)
         | 
| 32 | 
            -
                    begin
         | 
| 33 | 
            -
                      send_headers status, headers
         | 
| 34 | 
            -
                      send_body body
         | 
| 35 | 
            -
                    ensure
         | 
| 36 | 
            -
                      body.close  if body.respond_to? :close
         | 
| 37 | 
            -
                    end
         | 
| 38 | 
            -
                  end
         | 
| 39 | 
            -
             | 
| 40 | 
            -
                  def self.send_headers(status, headers)
         | 
| 41 | 
            -
                    $stdout.print "Status: #{status}\r\n"
         | 
| 42 | 
            -
                    headers.each { |k, vs|
         | 
| 43 | 
            -
                      vs.split("\n").each { |v|
         | 
| 44 | 
            -
                        $stdout.print "#{k}: #{v}\r\n"
         | 
| 45 | 
            -
                      }
         | 
| 46 | 
            -
                    }
         | 
| 47 | 
            -
                    $stdout.print "\r\n"
         | 
| 48 | 
            -
                    $stdout.flush
         | 
| 49 | 
            -
                  end
         | 
| 50 | 
            -
             | 
| 51 | 
            -
                  def self.send_body(body)
         | 
| 52 | 
            -
                    body.each { |part|
         | 
| 53 | 
            -
                      $stdout.print part
         | 
| 54 | 
            -
                      $stdout.flush
         | 
| 55 | 
            -
                    }
         | 
| 56 | 
            -
                  end
         | 
| 57 | 
            -
                end
         | 
| 58 | 
            -
              end
         | 
| 59 | 
            -
            end
         | 
    
        data/lib/rack/handler/fastcgi.rb
    DELETED
    
    | @@ -1,100 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require 'fcgi'
         | 
| 4 | 
            -
            require 'socket'
         | 
| 5 | 
            -
             | 
| 6 | 
            -
            if defined? FCGI::Stream
         | 
| 7 | 
            -
              class FCGI::Stream
         | 
| 8 | 
            -
                alias _rack_read_without_buffer read
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                def read(n, buffer = nil)
         | 
| 11 | 
            -
                  buf = _rack_read_without_buffer n
         | 
| 12 | 
            -
                  buffer.replace(buf.to_s)  if buffer
         | 
| 13 | 
            -
                  buf
         | 
| 14 | 
            -
                end
         | 
| 15 | 
            -
              end
         | 
| 16 | 
            -
            end
         | 
| 17 | 
            -
             | 
| 18 | 
            -
            module Rack
         | 
| 19 | 
            -
              module Handler
         | 
| 20 | 
            -
                class FastCGI
         | 
| 21 | 
            -
                  def self.run(app, **options)
         | 
| 22 | 
            -
                    if options[:File]
         | 
| 23 | 
            -
                      STDIN.reopen(UNIXServer.new(options[:File]))
         | 
| 24 | 
            -
                    elsif options[:Port]
         | 
| 25 | 
            -
                      STDIN.reopen(TCPServer.new(options[:Host], options[:Port]))
         | 
| 26 | 
            -
                    end
         | 
| 27 | 
            -
                    FCGI.each { |request|
         | 
| 28 | 
            -
                      serve request, app
         | 
| 29 | 
            -
                    }
         | 
| 30 | 
            -
                  end
         | 
| 31 | 
            -
             | 
| 32 | 
            -
                  def self.valid_options
         | 
| 33 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 34 | 
            -
                    default_host = environment == 'development' ? 'localhost' : '0.0.0.0'
         | 
| 35 | 
            -
             | 
| 36 | 
            -
                    {
         | 
| 37 | 
            -
                      "Host=HOST" => "Hostname to listen on (default: #{default_host})",
         | 
| 38 | 
            -
                      "Port=PORT" => "Port to listen on (default: 8080)",
         | 
| 39 | 
            -
                      "File=PATH" => "Creates a Domain socket at PATH instead of a TCP socket. Ignores Host and Port if set.",
         | 
| 40 | 
            -
                    }
         | 
| 41 | 
            -
                  end
         | 
| 42 | 
            -
             | 
| 43 | 
            -
                  def self.serve(request, app)
         | 
| 44 | 
            -
                    env = request.env
         | 
| 45 | 
            -
                    env.delete "HTTP_CONTENT_LENGTH"
         | 
| 46 | 
            -
             | 
| 47 | 
            -
                    env[SCRIPT_NAME] = ""  if env[SCRIPT_NAME] == "/"
         | 
| 48 | 
            -
             | 
| 49 | 
            -
                    rack_input = RewindableInput.new(request.in)
         | 
| 50 | 
            -
             | 
| 51 | 
            -
                    env.update(
         | 
| 52 | 
            -
                      RACK_VERSION      => Rack::VERSION,
         | 
| 53 | 
            -
                      RACK_INPUT        => rack_input,
         | 
| 54 | 
            -
                      RACK_ERRORS       => request.err,
         | 
| 55 | 
            -
                      RACK_MULTITHREAD  => false,
         | 
| 56 | 
            -
                      RACK_MULTIPROCESS => true,
         | 
| 57 | 
            -
                      RACK_RUNONCE      => false,
         | 
| 58 | 
            -
                      RACK_URL_SCHEME   => ["yes", "on", "1"].include?(env[HTTPS]) ? "https" : "http"
         | 
| 59 | 
            -
                    )
         | 
| 60 | 
            -
             | 
| 61 | 
            -
                    env[QUERY_STRING] ||= ""
         | 
| 62 | 
            -
                    env[HTTP_VERSION] ||= env[SERVER_PROTOCOL]
         | 
| 63 | 
            -
                    env[REQUEST_PATH] ||= "/"
         | 
| 64 | 
            -
                    env.delete "CONTENT_TYPE"  if env["CONTENT_TYPE"] == ""
         | 
| 65 | 
            -
                    env.delete "CONTENT_LENGTH"  if env["CONTENT_LENGTH"] == ""
         | 
| 66 | 
            -
             | 
| 67 | 
            -
                    begin
         | 
| 68 | 
            -
                      status, headers, body = app.call(env)
         | 
| 69 | 
            -
                      begin
         | 
| 70 | 
            -
                        send_headers request.out, status, headers
         | 
| 71 | 
            -
                        send_body request.out, body
         | 
| 72 | 
            -
                      ensure
         | 
| 73 | 
            -
                        body.close  if body.respond_to? :close
         | 
| 74 | 
            -
                      end
         | 
| 75 | 
            -
                    ensure
         | 
| 76 | 
            -
                      rack_input.close
         | 
| 77 | 
            -
                      request.finish
         | 
| 78 | 
            -
                    end
         | 
| 79 | 
            -
                  end
         | 
| 80 | 
            -
             | 
| 81 | 
            -
                  def self.send_headers(out, status, headers)
         | 
| 82 | 
            -
                    out.print "Status: #{status}\r\n"
         | 
| 83 | 
            -
                    headers.each { |k, vs|
         | 
| 84 | 
            -
                      vs.split("\n").each { |v|
         | 
| 85 | 
            -
                        out.print "#{k}: #{v}\r\n"
         | 
| 86 | 
            -
                      }
         | 
| 87 | 
            -
                    }
         | 
| 88 | 
            -
                    out.print "\r\n"
         | 
| 89 | 
            -
                    out.flush
         | 
| 90 | 
            -
                  end
         | 
| 91 | 
            -
             | 
| 92 | 
            -
                  def self.send_body(out, body)
         | 
| 93 | 
            -
                    body.each { |part|
         | 
| 94 | 
            -
                      out.print part
         | 
| 95 | 
            -
                      out.flush
         | 
| 96 | 
            -
                    }
         | 
| 97 | 
            -
                  end
         | 
| 98 | 
            -
                end
         | 
| 99 | 
            -
              end
         | 
| 100 | 
            -
            end
         | 
    
        data/lib/rack/handler/lsws.rb
    DELETED
    
    | @@ -1,61 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require 'lsapi'
         | 
| 4 | 
            -
             | 
| 5 | 
            -
            module Rack
         | 
| 6 | 
            -
              module Handler
         | 
| 7 | 
            -
                class LSWS
         | 
| 8 | 
            -
                  def self.run(app, **options)
         | 
| 9 | 
            -
                    while LSAPI.accept != nil
         | 
| 10 | 
            -
                      serve app
         | 
| 11 | 
            -
                    end
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
                  def self.serve(app)
         | 
| 14 | 
            -
                    env = ENV.to_hash
         | 
| 15 | 
            -
                    env.delete "HTTP_CONTENT_LENGTH"
         | 
| 16 | 
            -
                    env[SCRIPT_NAME] = "" if env[SCRIPT_NAME] == "/"
         | 
| 17 | 
            -
             | 
| 18 | 
            -
                    rack_input = RewindableInput.new($stdin.read.to_s)
         | 
| 19 | 
            -
             | 
| 20 | 
            -
                    env.update(
         | 
| 21 | 
            -
                      RACK_VERSION      => Rack::VERSION,
         | 
| 22 | 
            -
                      RACK_INPUT        => rack_input,
         | 
| 23 | 
            -
                      RACK_ERRORS       => $stderr,
         | 
| 24 | 
            -
                      RACK_MULTITHREAD  => false,
         | 
| 25 | 
            -
                      RACK_MULTIPROCESS => true,
         | 
| 26 | 
            -
                      RACK_RUNONCE      => false,
         | 
| 27 | 
            -
                      RACK_URL_SCHEME   => ["yes", "on", "1"].include?(ENV[HTTPS]) ? "https" : "http"
         | 
| 28 | 
            -
                    )
         | 
| 29 | 
            -
             | 
| 30 | 
            -
                    env[QUERY_STRING] ||= ""
         | 
| 31 | 
            -
                    env[HTTP_VERSION] ||= env[SERVER_PROTOCOL]
         | 
| 32 | 
            -
                    env[REQUEST_PATH] ||= "/"
         | 
| 33 | 
            -
                    status, headers, body = app.call(env)
         | 
| 34 | 
            -
                    begin
         | 
| 35 | 
            -
                      send_headers status, headers
         | 
| 36 | 
            -
                      send_body body
         | 
| 37 | 
            -
                    ensure
         | 
| 38 | 
            -
                      body.close if body.respond_to? :close
         | 
| 39 | 
            -
                    end
         | 
| 40 | 
            -
                  ensure
         | 
| 41 | 
            -
                    rack_input.close
         | 
| 42 | 
            -
                  end
         | 
| 43 | 
            -
                  def self.send_headers(status, headers)
         | 
| 44 | 
            -
                    print "Status: #{status}\r\n"
         | 
| 45 | 
            -
                    headers.each { |k, vs|
         | 
| 46 | 
            -
                      vs.split("\n").each { |v|
         | 
| 47 | 
            -
                        print "#{k}: #{v}\r\n"
         | 
| 48 | 
            -
                      }
         | 
| 49 | 
            -
                    }
         | 
| 50 | 
            -
                    print "\r\n"
         | 
| 51 | 
            -
                    STDOUT.flush
         | 
| 52 | 
            -
                  end
         | 
| 53 | 
            -
                  def self.send_body(body)
         | 
| 54 | 
            -
                    body.each { |part|
         | 
| 55 | 
            -
                      print part
         | 
| 56 | 
            -
                      STDOUT.flush
         | 
| 57 | 
            -
                    }
         | 
| 58 | 
            -
                  end
         | 
| 59 | 
            -
                end
         | 
| 60 | 
            -
              end
         | 
| 61 | 
            -
            end
         | 
    
        data/lib/rack/handler/scgi.rb
    DELETED
    
    | @@ -1,71 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require 'scgi'
         | 
| 4 | 
            -
            require 'stringio'
         | 
| 5 | 
            -
             | 
| 6 | 
            -
            module Rack
         | 
| 7 | 
            -
              module Handler
         | 
| 8 | 
            -
                class SCGI < ::SCGI::Processor
         | 
| 9 | 
            -
                  attr_accessor :app
         | 
| 10 | 
            -
             | 
| 11 | 
            -
                  def self.run(app, **options)
         | 
| 12 | 
            -
                    options[:Socket] = UNIXServer.new(options[:File]) if options[:File]
         | 
| 13 | 
            -
                    new(options.merge(app: app,
         | 
| 14 | 
            -
                                      host: options[:Host],
         | 
| 15 | 
            -
                                      port: options[:Port],
         | 
| 16 | 
            -
                                      socket: options[:Socket])).listen
         | 
| 17 | 
            -
                  end
         | 
| 18 | 
            -
             | 
| 19 | 
            -
                  def self.valid_options
         | 
| 20 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 21 | 
            -
                    default_host = environment == 'development' ? 'localhost' : '0.0.0.0'
         | 
| 22 | 
            -
             | 
| 23 | 
            -
                    {
         | 
| 24 | 
            -
                      "Host=HOST" => "Hostname to listen on (default: #{default_host})",
         | 
| 25 | 
            -
                      "Port=PORT" => "Port to listen on (default: 8080)",
         | 
| 26 | 
            -
                    }
         | 
| 27 | 
            -
                  end
         | 
| 28 | 
            -
             | 
| 29 | 
            -
                  def initialize(settings = {})
         | 
| 30 | 
            -
                    @app = settings[:app]
         | 
| 31 | 
            -
                    super(settings)
         | 
| 32 | 
            -
                  end
         | 
| 33 | 
            -
             | 
| 34 | 
            -
                  def process_request(request, input_body, socket)
         | 
| 35 | 
            -
                    env = Hash[request]
         | 
| 36 | 
            -
                    env.delete "HTTP_CONTENT_TYPE"
         | 
| 37 | 
            -
                    env.delete "HTTP_CONTENT_LENGTH"
         | 
| 38 | 
            -
                    env[REQUEST_PATH], env[QUERY_STRING] = env["REQUEST_URI"].split('?', 2)
         | 
| 39 | 
            -
                    env[HTTP_VERSION] ||= env[SERVER_PROTOCOL]
         | 
| 40 | 
            -
                    env[PATH_INFO] = env[REQUEST_PATH]
         | 
| 41 | 
            -
                    env[QUERY_STRING] ||= ""
         | 
| 42 | 
            -
                    env[SCRIPT_NAME] = ""
         | 
| 43 | 
            -
             | 
| 44 | 
            -
                    rack_input = StringIO.new(input_body)
         | 
| 45 | 
            -
                    rack_input.set_encoding(Encoding::BINARY)
         | 
| 46 | 
            -
             | 
| 47 | 
            -
                    env.update(
         | 
| 48 | 
            -
                      RACK_VERSION      => Rack::VERSION,
         | 
| 49 | 
            -
                      RACK_INPUT        => rack_input,
         | 
| 50 | 
            -
                      RACK_ERRORS       => $stderr,
         | 
| 51 | 
            -
                      RACK_MULTITHREAD  => true,
         | 
| 52 | 
            -
                      RACK_MULTIPROCESS => true,
         | 
| 53 | 
            -
                      RACK_RUNONCE      => false,
         | 
| 54 | 
            -
                      RACK_URL_SCHEME   => ["yes", "on", "1"].include?(env[HTTPS]) ? "https" : "http"
         | 
| 55 | 
            -
                    )
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                    status, headers, body = app.call(env)
         | 
| 58 | 
            -
                    begin
         | 
| 59 | 
            -
                      socket.write("Status: #{status}\r\n")
         | 
| 60 | 
            -
                      headers.each do |k, vs|
         | 
| 61 | 
            -
                        vs.split("\n").each { |v| socket.write("#{k}: #{v}\r\n")}
         | 
| 62 | 
            -
                      end
         | 
| 63 | 
            -
                      socket.write("\r\n")
         | 
| 64 | 
            -
                      body.each {|s| socket.write(s)}
         | 
| 65 | 
            -
                    ensure
         | 
| 66 | 
            -
                      body.close if body.respond_to? :close
         | 
| 67 | 
            -
                    end
         | 
| 68 | 
            -
                  end
         | 
| 69 | 
            -
                end
         | 
| 70 | 
            -
              end
         | 
| 71 | 
            -
            end
         | 
    
        data/lib/rack/handler/thin.rb
    DELETED
    
    | @@ -1,36 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require "thin"
         | 
| 4 | 
            -
            require "thin/server"
         | 
| 5 | 
            -
            require "thin/logging"
         | 
| 6 | 
            -
            require "thin/backends/tcp_server"
         | 
| 7 | 
            -
             | 
| 8 | 
            -
            module Rack
         | 
| 9 | 
            -
              module Handler
         | 
| 10 | 
            -
                class Thin
         | 
| 11 | 
            -
                  def self.run(app, **options)
         | 
| 12 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 13 | 
            -
                    default_host = environment == 'development' ? 'localhost' : '0.0.0.0'
         | 
| 14 | 
            -
             | 
| 15 | 
            -
                    host = options.delete(:Host) || default_host
         | 
| 16 | 
            -
                    port = options.delete(:Port) || 8080
         | 
| 17 | 
            -
                    args = [host, port, app, options]
         | 
| 18 | 
            -
                    # Thin versions below 0.8.0 do not support additional options
         | 
| 19 | 
            -
                    args.pop if ::Thin::VERSION::MAJOR < 1 && ::Thin::VERSION::MINOR < 8
         | 
| 20 | 
            -
                    server = ::Thin::Server.new(*args)
         | 
| 21 | 
            -
                    yield server if block_given?
         | 
| 22 | 
            -
                    server.start
         | 
| 23 | 
            -
                  end
         | 
| 24 | 
            -
             | 
| 25 | 
            -
                  def self.valid_options
         | 
| 26 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 27 | 
            -
                    default_host = environment == 'development' ? 'localhost' : '0.0.0.0'
         | 
| 28 | 
            -
             | 
| 29 | 
            -
                    {
         | 
| 30 | 
            -
                      "Host=HOST" => "Hostname to listen on (default: #{default_host})",
         | 
| 31 | 
            -
                      "Port=PORT" => "Port to listen on (default: 8080)",
         | 
| 32 | 
            -
                    }
         | 
| 33 | 
            -
                  end
         | 
| 34 | 
            -
                end
         | 
| 35 | 
            -
              end
         | 
| 36 | 
            -
            end
         | 
    
        data/lib/rack/handler/webrick.rb
    DELETED
    
    | @@ -1,129 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            require 'webrick'
         | 
| 4 | 
            -
            require 'stringio'
         | 
| 5 | 
            -
             | 
| 6 | 
            -
            # This monkey patch allows for applications to perform their own chunking
         | 
| 7 | 
            -
            # through WEBrick::HTTPResponse if rack is set to true.
         | 
| 8 | 
            -
            class WEBrick::HTTPResponse
         | 
| 9 | 
            -
              attr_accessor :rack
         | 
| 10 | 
            -
             | 
| 11 | 
            -
              alias _rack_setup_header setup_header
         | 
| 12 | 
            -
              def setup_header
         | 
| 13 | 
            -
                app_chunking = rack && @header['transfer-encoding'] == 'chunked'
         | 
| 14 | 
            -
             | 
| 15 | 
            -
                @chunked = app_chunking if app_chunking
         | 
| 16 | 
            -
             | 
| 17 | 
            -
                _rack_setup_header
         | 
| 18 | 
            -
             | 
| 19 | 
            -
                @chunked = false if app_chunking
         | 
| 20 | 
            -
              end
         | 
| 21 | 
            -
            end
         | 
| 22 | 
            -
             | 
| 23 | 
            -
            module Rack
         | 
| 24 | 
            -
              module Handler
         | 
| 25 | 
            -
                class WEBrick < ::WEBrick::HTTPServlet::AbstractServlet
         | 
| 26 | 
            -
                  def self.run(app, **options)
         | 
| 27 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 28 | 
            -
                    default_host = environment == 'development' ? 'localhost' : nil
         | 
| 29 | 
            -
             | 
| 30 | 
            -
                    if !options[:BindAddress] || options[:Host]
         | 
| 31 | 
            -
                      options[:BindAddress] = options.delete(:Host) || default_host
         | 
| 32 | 
            -
                    end
         | 
| 33 | 
            -
                    options[:Port] ||= 8080
         | 
| 34 | 
            -
                    if options[:SSLEnable]
         | 
| 35 | 
            -
                      require 'webrick/https'
         | 
| 36 | 
            -
                    end
         | 
| 37 | 
            -
             | 
| 38 | 
            -
                    @server = ::WEBrick::HTTPServer.new(options)
         | 
| 39 | 
            -
                    @server.mount "/", Rack::Handler::WEBrick, app
         | 
| 40 | 
            -
                    yield @server  if block_given?
         | 
| 41 | 
            -
                    @server.start
         | 
| 42 | 
            -
                  end
         | 
| 43 | 
            -
             | 
| 44 | 
            -
                  def self.valid_options
         | 
| 45 | 
            -
                    environment  = ENV['RACK_ENV'] || 'development'
         | 
| 46 | 
            -
                    default_host = environment == 'development' ? 'localhost' : '0.0.0.0'
         | 
| 47 | 
            -
             | 
| 48 | 
            -
                    {
         | 
| 49 | 
            -
                      "Host=HOST" => "Hostname to listen on (default: #{default_host})",
         | 
| 50 | 
            -
                      "Port=PORT" => "Port to listen on (default: 8080)",
         | 
| 51 | 
            -
                    }
         | 
| 52 | 
            -
                  end
         | 
| 53 | 
            -
             | 
| 54 | 
            -
                  def self.shutdown
         | 
| 55 | 
            -
                    if @server
         | 
| 56 | 
            -
                      @server.shutdown
         | 
| 57 | 
            -
                      @server = nil
         | 
| 58 | 
            -
                    end
         | 
| 59 | 
            -
                  end
         | 
| 60 | 
            -
             | 
| 61 | 
            -
                  def initialize(server, app)
         | 
| 62 | 
            -
                    super server
         | 
| 63 | 
            -
                    @app = app
         | 
| 64 | 
            -
                  end
         | 
| 65 | 
            -
             | 
| 66 | 
            -
                  def service(req, res)
         | 
| 67 | 
            -
                    res.rack = true
         | 
| 68 | 
            -
                    env = req.meta_vars
         | 
| 69 | 
            -
                    env.delete_if { |k, v| v.nil? }
         | 
| 70 | 
            -
             | 
| 71 | 
            -
                    rack_input = StringIO.new(req.body.to_s)
         | 
| 72 | 
            -
                    rack_input.set_encoding(Encoding::BINARY)
         | 
| 73 | 
            -
             | 
| 74 | 
            -
                    env.update(
         | 
| 75 | 
            -
                      RACK_VERSION      => Rack::VERSION,
         | 
| 76 | 
            -
                      RACK_INPUT        => rack_input,
         | 
| 77 | 
            -
                      RACK_ERRORS       => $stderr,
         | 
| 78 | 
            -
                      RACK_MULTITHREAD  => true,
         | 
| 79 | 
            -
                      RACK_MULTIPROCESS => false,
         | 
| 80 | 
            -
                      RACK_RUNONCE      => false,
         | 
| 81 | 
            -
                      RACK_URL_SCHEME   => ["yes", "on", "1"].include?(env[HTTPS]) ? "https" : "http",
         | 
| 82 | 
            -
                      RACK_IS_HIJACK    => true,
         | 
| 83 | 
            -
                      RACK_HIJACK       => lambda { raise NotImplementedError, "only partial hijack is supported."},
         | 
| 84 | 
            -
                      RACK_HIJACK_IO    => nil
         | 
| 85 | 
            -
                    )
         | 
| 86 | 
            -
             | 
| 87 | 
            -
                    env[HTTP_VERSION] ||= env[SERVER_PROTOCOL]
         | 
| 88 | 
            -
                    env[QUERY_STRING] ||= ""
         | 
| 89 | 
            -
                    unless env[PATH_INFO] == ""
         | 
| 90 | 
            -
                      path, n = req.request_uri.path, env[SCRIPT_NAME].length
         | 
| 91 | 
            -
                      env[PATH_INFO] = path[n, path.length - n]
         | 
| 92 | 
            -
                    end
         | 
| 93 | 
            -
                    env[REQUEST_PATH] ||= [env[SCRIPT_NAME], env[PATH_INFO]].join
         | 
| 94 | 
            -
             | 
| 95 | 
            -
                    status, headers, body = @app.call(env)
         | 
| 96 | 
            -
                    begin
         | 
| 97 | 
            -
                      res.status = status.to_i
         | 
| 98 | 
            -
                      io_lambda = nil
         | 
| 99 | 
            -
                      headers.each { |k, vs|
         | 
| 100 | 
            -
                        if k == RACK_HIJACK
         | 
| 101 | 
            -
                          io_lambda = vs
         | 
| 102 | 
            -
                        elsif k.downcase == "set-cookie"
         | 
| 103 | 
            -
                          res.cookies.concat vs.split("\n")
         | 
| 104 | 
            -
                        else
         | 
| 105 | 
            -
                          # Since WEBrick won't accept repeated headers,
         | 
| 106 | 
            -
                          # merge the values per RFC 1945 section 4.2.
         | 
| 107 | 
            -
                          res[k] = vs.split("\n").join(", ")
         | 
| 108 | 
            -
                        end
         | 
| 109 | 
            -
                      }
         | 
| 110 | 
            -
             | 
| 111 | 
            -
                      if io_lambda
         | 
| 112 | 
            -
                        rd, wr = IO.pipe
         | 
| 113 | 
            -
                        res.body = rd
         | 
| 114 | 
            -
                        res.chunked = true
         | 
| 115 | 
            -
                        io_lambda.call wr
         | 
| 116 | 
            -
                      elsif body.respond_to?(:to_path)
         | 
| 117 | 
            -
                        res.body = ::File.open(body.to_path, 'rb')
         | 
| 118 | 
            -
                      else
         | 
| 119 | 
            -
                        body.each { |part|
         | 
| 120 | 
            -
                          res.body << part
         | 
| 121 | 
            -
                        }
         | 
| 122 | 
            -
                      end
         | 
| 123 | 
            -
                    ensure
         | 
| 124 | 
            -
                      body.close  if body.respond_to? :close
         | 
| 125 | 
            -
                    end
         | 
| 126 | 
            -
                  end
         | 
| 127 | 
            -
                end
         | 
| 128 | 
            -
              end
         | 
| 129 | 
            -
            end
         |