mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
510bbe015d
[ruby-trunk - Feature #6435] * lib/net/http/backward.rb: ditto. * lib/net/http/response.rb: ditto. * lib/net/http/exceptions.rb: ditto. * lib/net/http/responses.rb: ditto. * lib/net/http/generic_request.rb: ditto. * lib/net/http/header.rb: ditto. * lib/net/http/request.rb: ditto. * lib/net/http/proxy_delta.rb: ditto. * lib/net/http/requests.rb: ditto. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@35761 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
332 lines
8.1 KiB
Ruby
332 lines
8.1 KiB
Ruby
# HTTP response class.
|
|
#
|
|
# This class wraps together the response header and the response body (the
|
|
# entity requested).
|
|
#
|
|
# It mixes in the HTTPHeader module, which provides access to response
|
|
# header values both via hash-like methods and via individual readers.
|
|
#
|
|
# Note that each possible HTTP response code defines its own
|
|
# HTTPResponse subclass. These are listed below.
|
|
#
|
|
# All classes are
|
|
# defined under the Net module. Indentation indicates inheritance.
|
|
#
|
|
# xxx HTTPResponse
|
|
#
|
|
# 1xx HTTPInformation
|
|
# 100 HTTPContinue
|
|
# 101 HTTPSwitchProtocol
|
|
#
|
|
# 2xx HTTPSuccess
|
|
# 200 HTTPOK
|
|
# 201 HTTPCreated
|
|
# 202 HTTPAccepted
|
|
# 203 HTTPNonAuthoritativeInformation
|
|
# 204 HTTPNoContent
|
|
# 205 HTTPResetContent
|
|
# 206 HTTPPartialContent
|
|
#
|
|
# 3xx HTTPRedirection
|
|
# 300 HTTPMultipleChoice
|
|
# 301 HTTPMovedPermanently
|
|
# 302 HTTPFound
|
|
# 303 HTTPSeeOther
|
|
# 304 HTTPNotModified
|
|
# 305 HTTPUseProxy
|
|
# 307 HTTPTemporaryRedirect
|
|
#
|
|
# 4xx HTTPClientError
|
|
# 400 HTTPBadRequest
|
|
# 401 HTTPUnauthorized
|
|
# 402 HTTPPaymentRequired
|
|
# 403 HTTPForbidden
|
|
# 404 HTTPNotFound
|
|
# 405 HTTPMethodNotAllowed
|
|
# 406 HTTPNotAcceptable
|
|
# 407 HTTPProxyAuthenticationRequired
|
|
# 408 HTTPRequestTimeOut
|
|
# 409 HTTPConflict
|
|
# 410 HTTPGone
|
|
# 411 HTTPLengthRequired
|
|
# 412 HTTPPreconditionFailed
|
|
# 413 HTTPRequestEntityTooLarge
|
|
# 414 HTTPRequestURITooLong
|
|
# 415 HTTPUnsupportedMediaType
|
|
# 416 HTTPRequestedRangeNotSatisfiable
|
|
# 417 HTTPExpectationFailed
|
|
#
|
|
# 5xx HTTPServerError
|
|
# 500 HTTPInternalServerError
|
|
# 501 HTTPNotImplemented
|
|
# 502 HTTPBadGateway
|
|
# 503 HTTPServiceUnavailable
|
|
# 504 HTTPGatewayTimeOut
|
|
# 505 HTTPVersionNotSupported
|
|
#
|
|
# xxx HTTPUnknownResponse
|
|
#
|
|
class Net::HTTPResponse
|
|
class << self
|
|
# true if the response has a body.
|
|
def body_permitted?
|
|
self::HAS_BODY
|
|
end
|
|
|
|
def exception_type # :nodoc: internal use only
|
|
self::EXCEPTION_TYPE
|
|
end
|
|
|
|
def read_new(sock) #:nodoc: internal use only
|
|
httpv, code, msg = read_status_line(sock)
|
|
res = response_class(code).new(httpv, code, msg)
|
|
each_response_header(sock) do |k,v|
|
|
res.add_field k, v
|
|
end
|
|
res
|
|
end
|
|
|
|
private
|
|
|
|
def read_status_line(sock)
|
|
str = sock.readline
|
|
m = /\AHTTP(?:\/(\d+\.\d+))?\s+(\d\d\d)\s*(.*)\z/in.match(str) or
|
|
raise Net::HTTPBadResponse, "wrong status line: #{str.dump}"
|
|
m.captures
|
|
end
|
|
|
|
def response_class(code)
|
|
CODE_TO_OBJ[code] or
|
|
CODE_CLASS_TO_OBJ[code[0,1]] or
|
|
Net::HTTPUnknownResponse
|
|
end
|
|
|
|
def each_response_header(sock)
|
|
key = value = nil
|
|
while true
|
|
line = sock.readuntil("\n", true).sub(/\s+\z/, '')
|
|
break if line.empty?
|
|
if line[0] == ?\s or line[0] == ?\t and value
|
|
value << ' ' unless value.empty?
|
|
value << line.strip
|
|
else
|
|
yield key, value if key
|
|
key, value = line.strip.split(/\s*:\s*/, 2)
|
|
raise Net::HTTPBadResponse, 'wrong header line format' if value.nil?
|
|
end
|
|
end
|
|
yield key, value if key
|
|
end
|
|
end
|
|
|
|
# next is to fix bug in RDoc, where the private inside class << self
|
|
# spills out.
|
|
public
|
|
|
|
include Net::HTTPHeader
|
|
|
|
def initialize(httpv, code, msg) #:nodoc: internal use only
|
|
@http_version = httpv
|
|
@code = code
|
|
@message = msg
|
|
initialize_http_header nil
|
|
@body = nil
|
|
@read = false
|
|
end
|
|
|
|
# The HTTP version supported by the server.
|
|
attr_reader :http_version
|
|
|
|
# The HTTP result code string. For example, '302'. You can also
|
|
# determine the response type by examining which response subclass
|
|
# the response object is an instance of.
|
|
attr_reader :code
|
|
|
|
# The HTTP result message sent by the server. For example, 'Not Found'.
|
|
attr_reader :message
|
|
alias msg message # :nodoc: obsolete
|
|
|
|
def inspect
|
|
"#<#{self.class} #{@code} #{@message} readbody=#{@read}>"
|
|
end
|
|
|
|
#
|
|
# response <-> exception relationship
|
|
#
|
|
|
|
def code_type #:nodoc:
|
|
self.class
|
|
end
|
|
|
|
def error! #:nodoc:
|
|
raise error_type().new(@code + ' ' + @message.dump, self)
|
|
end
|
|
|
|
def error_type #:nodoc:
|
|
self.class::EXCEPTION_TYPE
|
|
end
|
|
|
|
# Raises an HTTP error if the response is not 2xx (success).
|
|
def value
|
|
error! unless self.kind_of?(Net::HTTPSuccess)
|
|
end
|
|
|
|
#
|
|
# header (for backward compatibility only; DO NOT USE)
|
|
#
|
|
|
|
def response #:nodoc:
|
|
warn "#{caller(1)[0]}: warning: Net::HTTPResponse#response is obsolete" if $VERBOSE
|
|
self
|
|
end
|
|
|
|
def header #:nodoc:
|
|
warn "#{caller(1)[0]}: warning: Net::HTTPResponse#header is obsolete" if $VERBOSE
|
|
self
|
|
end
|
|
|
|
def read_header #:nodoc:
|
|
warn "#{caller(1)[0]}: warning: Net::HTTPResponse#read_header is obsolete" if $VERBOSE
|
|
self
|
|
end
|
|
|
|
#
|
|
# body
|
|
#
|
|
|
|
def reading_body(sock, reqmethodallowbody) #:nodoc: internal use only
|
|
@socket = sock
|
|
@body_exist = reqmethodallowbody && self.class.body_permitted?
|
|
begin
|
|
yield
|
|
self.body # ensure to read body
|
|
ensure
|
|
@socket = nil
|
|
end
|
|
end
|
|
|
|
# Gets the entity body returned by the remote HTTP server.
|
|
#
|
|
# If a block is given, the body is passed to the block, and
|
|
# the body is provided in fragments, as it is read in from the socket.
|
|
#
|
|
# Calling this method a second or subsequent time for the same
|
|
# HTTPResponse object will return the value already read.
|
|
#
|
|
# http.request_get('/index.html') {|res|
|
|
# puts res.read_body
|
|
# }
|
|
#
|
|
# http.request_get('/index.html') {|res|
|
|
# p res.read_body.object_id # 538149362
|
|
# p res.read_body.object_id # 538149362
|
|
# }
|
|
#
|
|
# # using iterator
|
|
# http.request_get('/index.html') {|res|
|
|
# res.read_body do |segment|
|
|
# print segment
|
|
# end
|
|
# }
|
|
#
|
|
def read_body(dest = nil, &block)
|
|
if @read
|
|
raise IOError, "#{self.class}\#read_body called twice" if dest or block
|
|
return @body
|
|
end
|
|
to = procdest(dest, block)
|
|
stream_check
|
|
if @body_exist
|
|
read_body_0 to
|
|
@body = to
|
|
else
|
|
@body = nil
|
|
end
|
|
@read = true
|
|
|
|
@body
|
|
end
|
|
|
|
# Returns the full entity body.
|
|
#
|
|
# Calling this method a second or subsequent time will return the
|
|
# string already read.
|
|
#
|
|
# http.request_get('/index.html') {|res|
|
|
# puts res.body
|
|
# }
|
|
#
|
|
# http.request_get('/index.html') {|res|
|
|
# p res.body.object_id # 538149362
|
|
# p res.body.object_id # 538149362
|
|
# }
|
|
#
|
|
def body
|
|
read_body()
|
|
end
|
|
|
|
# Because it may be necessary to modify the body, Eg, decompression
|
|
# this method facilitates that.
|
|
def body=(value)
|
|
@body = value
|
|
end
|
|
|
|
alias entity body #:nodoc: obsolete
|
|
|
|
private
|
|
|
|
def read_body_0(dest)
|
|
if chunked?
|
|
read_chunked dest
|
|
return
|
|
end
|
|
clen = content_length()
|
|
if clen
|
|
@socket.read clen, dest, true # ignore EOF
|
|
return
|
|
end
|
|
clen = range_length()
|
|
if clen
|
|
@socket.read clen, dest
|
|
return
|
|
end
|
|
@socket.read_all dest
|
|
end
|
|
|
|
def read_chunked(dest)
|
|
len = nil
|
|
total = 0
|
|
while true
|
|
line = @socket.readline
|
|
hexlen = line.slice(/[0-9a-fA-F]+/) or
|
|
raise Net::HTTPBadResponse, "wrong chunk size line: #{line}"
|
|
len = hexlen.hex
|
|
break if len == 0
|
|
begin
|
|
@socket.read len, dest
|
|
ensure
|
|
total += len
|
|
@socket.read 2 # \r\n
|
|
end
|
|
end
|
|
until @socket.readline.empty?
|
|
# none
|
|
end
|
|
end
|
|
|
|
def stream_check
|
|
raise IOError, 'attempt to read body out of block' if @socket.closed?
|
|
end
|
|
|
|
def procdest(dest, block)
|
|
raise ArgumentError, 'both arg and block given for HTTP method' \
|
|
if dest and block
|
|
if block
|
|
Net::ReadAdapter.new(block)
|
|
else
|
|
dest || ''
|
|
end
|
|
end
|
|
|
|
end
|
|
|