Net::HTTPHeader
module Net::HTTPHeader
HTTPHeader 模块定义了读写HTTP头的方法。
它被其他类用作mixin,以提供对HTTP头值的散列式访问。与原始哈希访问不同,HTTPHeader通过不区分大小写的密钥提供访问。它还提供了以更方便的格式访问常用HTTP头值的方法。
公共实例方法
Show source
返回对应于不区分大小写的键的标题字段。例如,“Content-Type”键可能会返回“text / html”
# File lib/net/http/header.rb, line 33
def [](key)
a = @header[key.downcase] or return nil
a.join(', ')
end
[]=(key, val) Show source
设置与不区分大小写的键相对应的标题字段。
# File lib/net/http/header.rb, line 39
def []=(key, val)
unless val
@header.delete key.downcase
return val
end
@header[key.downcase] = [val]
end
add_field(key, val) Show source
Ruby 1.8.3
将值添加到指定的标题字段,而不是替换其值。第二个参数val
必须是一个String。另见[] =,[]和get_fields。
request.add_field 'X-My-Header', 'a'
p request['X-My-Header'] #=> "a"
p request.get_fields('X-My-Header') #=> ["a"]
request.add_field 'X-My-Header', 'b'
p request['X-My-Header'] #=> "a, b"
p request.get_fields('X-My-Header') #=> ["a", "b"]
request.add_field 'X-My-Header', 'c'
p request['X-My-Header'] #=> "a, b, c"
p request.get_fields('X-My-Header') #=> ["a", "b", "c"]
# File lib/net/http/header.rb, line 62
def add_field(key, val)
if @header.key?(key.downcase)
@header[key.downcase].push val
else
@header[key.downcase] = [val]
end
end
basic_auth(account, password) Show source
为“基本”授权设置授权:标头。
# File lib/net/http/header.rb, line 433
def basic_auth(account, password)
@header['authorization'] = [basic_encode(account, password)]
end
canonical_each()
Alias for: each_capitalized
chunked?() Show source
如果“传输编码”标题存在并设置为“分块”,则返回“true”。这是一项HTTP / 1.1功能,允许内容以“块”形式发送,而不必一开始就说明整个内容的长度。
# File lib/net/http/header.rb, line 294
def chunked?
return false unless @header['transfer-encoding']
field = self['Transfer-Encoding']
(/(?:\A|[^\-\w])chunked(?![\-\w])/i =~ field) ? true : false
end
connection_close?() Show source
# File lib/net/http/header.rb, line 447
def connection_close?
token = /(?:\A|,)\s*close\s*(?:\z|,)/i
@header['connection']&.grep(token) {return true}
@header['proxy-connection']&.grep(token) {return true}
false
end
connection_keep_alive?() Show source
# File lib/net/http/header.rb, line 454
def connection_keep_alive?
token = /(?:\A|,)\s*keep-alive\s*(?:\z|,)/i
@header['connection']&.grep(token) {return true}
@header['proxy-connection']&.grep(token) {return true}
false
end
content_length() Show source
返回表示HTTP Content-Length:头字段的Integer对象,或者nil
未提供该字段。
# File lib/net/http/header.rb, line 275
def content_length
return nil unless key?('Content-Length')
len = self['Content-Length'].slice(/\d+/) or
raise Net::HTTPHeaderSyntaxError, 'wrong Content-Length format'
len.to_i
end
content_length=(len) Show source
# File lib/net/http/header.rb, line 282
def content_length=(len)
unless len
@header.delete 'content-length'
return nil
end
@header['content-length'] = [len.to_i.to_s]
end
content_range() Show source
返回表示Content-Range:标题字段值的Range对象。对于部分实体主体,这表示这个片段在整个实体主体内的位置,作为字节偏移的范围。
# File lib/net/http/header.rb, line 304
def content_range
return nil unless @header['content-range']
m = %r<bytes\s+(\d+)-(\d+)/(\d+|\*)>i.match(self['Content-Range']) or
raise Net::HTTPHeaderSyntaxError, 'wrong Content-Range format'
m[1].to_i .. m[2].to_i
end
content_type() Show source
返回一个内容类型字符串,如“text / html”。如果Content-Type:头字段不存在,则此方法返回nil。
# File lib/net/http/header.rb, line 319
def content_type
return nil unless main_type()
if sub_type()
then "#{main_type()}/#{sub_type()}"
else main_type()
end
end
content_type=(type, params = {})
Alias for: set_content_type
delete(key) Show source
删除由不区分大小写的键指定的标题字段。
# File lib/net/http/header.rb, line 151
def delete(key)
@header.delete(key.downcase)
end
each()
别名为:each_header
each_capitalized() { |capitalize(k), join(', ')| ... } Show source
至于each_header,除了以大写形式提供密钥外。
请注意,标题名称是系统化的大写; 大写可能与远程HTTP服务器在响应中使用的大小不匹配。
如果没有给出块,则返回一个枚举器。
# File lib/net/http/header.rb, line 176
def each_capitalized
block_given? or return enum_for(__method__) { @header.size }
@header.each do |k,v|
yield capitalize(k), v.join(', ')
end
end
另外别名为:canonical_each
each_capitalized_name() { |key| ... } Show source
迭代头部中的头部名称,将大写的头部名称传递给代码块。
请注意,标题名称是系统化的大写; 大写可能与远程HTTP服务器在响应中使用的大小不匹配。
如果没有给出块,则返回一个枚举器。
# File lib/net/http/header.rb, line 132
def each_capitalized_name #:yield: +key+
block_given? or return enum_for(__method__) { @header.size }
@header.each_key do |k|
yield capitalize(k)
end
end
each_header() { |key| ... } Show source
迭代头部名称和值,将名称和值传递给所提供的代码块。
如果没有给出块,则返回一个枚举器。
例:
response.header.each_header {|key,value| puts "#{key} = #{value}" }
# File lib/net/http/header.rb, line 104
def each_header #:yield: +key+, +value+
block_given? or return enum_for(__method__) { @header.size }
@header.each do |k,va|
yield k, va.join(', ')
end
end
还有别名:每个
each_key()
Alias for: each_name
each_name() { |key| ... } Show source
遍历头部中的头部名称,将每个头部名称传递给代码块。
如果没有给出块,则返回一个枚举器。
# File lib/net/http/header.rb, line 117
def each_name(&block) #:yield: +key+
block_given? or return enum_for(__method__) { @header.size }
@header.each_key(&block)
end
Also aliased as: each_key
each_value() { |value| ... } Show source
迭代头部值,将每个值传递给代码块。
如果没有给出块,则返回一个枚举器。
# File lib/net/http/header.rb, line 143
def each_value #:yield: +value+
block_given? or return enum_for(__method__) { @header.size }
@header.each_value do |va|
yield va.join(', ')
end
end
fetch(key, *args) { |key| ... } Show source
返回对应于不区分大小写的键的标题字段。返回默认值args
或块的结果,或者如果没有名为key
See Hash#fetch的头字段,则引发IndexError
# File lib/net/http/header.rb, line 90
def fetch(key, *args, &block) #:yield: +key+
a = @header.fetch(key.downcase, *args, &block)
a.kind_of?(Array) ? a.join(', ') : a
end
form_data=(params, sep = '&')
Alias for: set_form_data
get_fields(key) Show source
Ruby 1.8.3
返回对应于大小写不敏感的头字段字符串数组key
。这种方法可以让你无需任何处理就可以得到重复的标题字段。也可以看看 []。
p response.get_fields('Set-Cookie')
#=> ["session=al98axx; expires=Fri, 31-Dec-1999 23:58:23",
"query=rubyscript; expires=Fri, 31-Dec-1999 23:58:23"]
p response['Set-Cookie']
#=> "session=al98axx; expires=Fri, 31-Dec-1999 23:58:23, query=rubyscript; expires=Fri, 31-Dec-1999 23:58:23"
# File lib/net/http/header.rb, line 81
def get_fields(key)
return nil unless @header[key.downcase]
@header[key.downcase].dup
end
initialize_http_header(initheader) Show source
# File lib/net/http/header.rb, line 12
def initialize_http_header(initheader)
@header = {}
return unless initheader
initheader.each do |key, value|
warn "net/http: warning: duplicated HTTP header: #{key}" if key?(key) and $VERBOSE
if value.nil?
warn "net/http: warning: nil HTTP header: #{key}" if $VERBOSE
else
@header[key.downcase] = [value.strip]
end
end
end
key?(key) Show source
如果key
头存在,则为true 。
# File lib/net/http/header.rb, line 156
def key?(key)
@header.key?(key.downcase)
end
main_type() Show source
返回一个内容类型字符串,如“text”。如果Content-Type:头字段不存在,则此方法返回nil。
# File lib/net/http/header.rb, line 329
def main_type
return nil unless @header['content-type']
self['Content-Type'].split(';').first.to_s.split('/')[0].to_s.strip
end
proxy_basic_auth(account, password) Show source
为“基本”授权设置代理授权:标头。
# File lib/net/http/header.rb, line 438
def proxy_basic_auth(account, password)
@header['proxy-authorization'] = [basic_encode(account, password)]
end
range() Show source
返回表示范围:HTTP标头字段的Range对象数组,或者nil
如果没有这样的标头。
# File lib/net/http/header.rb, line 192
def range
return nil unless @header['range']
value = self['Range']
# byte-range-set = *( "," OWS ) ( byte-range-spec / suffix-byte-range-spec )
# *( OWS "," [ OWS ( byte-range-spec / suffix-byte-range-spec ) ] )
# corrected collected ABNF
# http://tools.ietf.org/html/draft-ietf-httpbis-p5-range-19#section-5.4.1
# http://tools.ietf.org/html/draft-ietf-httpbis-p5-range-19#appendix-C
# http://tools.ietf.org/html/draft-ietf-httpbis-p1-messaging-19#section-3.2.5
unless /\Abytes=((?:,[ \t]*)*(?:\d+-\d*|-\d+)(?:[ \t]*,(?:[ \t]*\d+-\d*|-\d+)?)*)\z/ =~ value
raise Net::HTTPHeaderSyntaxError, "invalid syntax for byte-ranges-specifier: '#{value}'"
end
byte_range_set = $1
result = byte_range_set.split(/,/).map {|spec|
m = /(\d+)?\s*-\s*(\d+)?/i.match(spec) or
raise Net::HTTPHeaderSyntaxError, "invalid byte-range-spec: '#{spec}'"
d1 = m[1].to_i
d2 = m[2].to_i
if m[1] and m[2]
if d1 > d2
raise Net::HTTPHeaderSyntaxError, "last-byte-pos MUST greater than or equal to first-byte-pos but '#{spec}'"
end
d1..d2
elsif m[1]
d1..-1
elsif m[2]
-d2..-1
else
raise Net::HTTPHeaderSyntaxError, 'range is not specified'
end
}
# if result.empty?
# byte-range-set must include at least one byte-range-spec or suffix-byte-range-spec
# but above regexp already denies it.
if result.size == 1 && result[0].begin == 0 && result[0].end == -1
raise Net::HTTPHeaderSyntaxError, 'only one suffix-byte-range-spec with zero suffix-length'
end
result
end
range=(r, e = nil)
Alias for: set_range
range_length() Show source
Content-Range:表示的范围的长度。
# File lib/net/http/header.rb, line 312
def range_length
r = content_range() or return nil
r.end - r.begin + 1
end
set_content_type(type, params = {}) Show source
在HTTP标头中设置内容类型。本type应该是一个完整的HTTP内容类型,如“text / html的”。这params是在内容类型之后添加的可选散列参数,例如{'charset'=>'iso-8859-1'}
# File lib/net/http/header.rb, line 362
def set_content_type(type, params = {})
@header['content-type'] = [type + params.map{|k,v|"; #{k}=#{v}"}.join('')]
end
Also aliased as: content_type=
set_form(params, enctype='application/x-www-form-urlencoded', formopt={}) Show source
设置一个HTML表单数据集。params
是表单数据集; 它是数组或数组的散列+ enctype是用于对表单数据集进行编码的类型。它是application / x-www-form-urlencoded或multipart / form-data。formopt
是一个可选的哈希来指定细节。
边界
多部分消息的边界
charset
消息的字符集。所有名称和非文件字段的值都被编码为字符集。
每个参数项都是一个数组,并包含以下项目:
name
该字段的名称
value
字段的值,它应该是一个字符串或文件
opt
一个可选的哈希来指定附加信息
每个项目是文件字段或普通字段。如果value
是一个File对象或opt
具有一个文件名键,则该项目被视为一个文件字段。
如果Transfer-Encoding设置为分块,则以分块编码方式发送请求。由于分块编码是HTTP / 1.1功能,因此在发送之前必须确认服务器支持HTTP / 1.1。
例:
http.set_form([["q", "ruby"], ["lang", "en"]])
See also RFC 2388, RFC 2616, HTML 4.01, and HTML5
# File lib/net/http/header.rb, line 418
def set_form(params, enctype='application/x-www-form-urlencoded', formopt={})
@body_data = params
@body = nil
@body_stream = nil
@form_option = formopt
case enctype
when /\Aapplication\/x-www-form-urlencoded\z/i,
/\Amultipart\/form-data\z/i
self.content_type = enctype
else
raise ArgumentError, "invalid enctype: #{enctype}"
end
end
set_form_data(params, sep = '&') Show source
从HTML表单数据设置标题字段和正文。params
应该是包含HTML表单数据的数组或数组。可选参数sep
表示数据记录分隔符。
数值根据需要进行了URL编码,内容类型设置为application / x-www-form-urlencoded
例:
http.form_data = {"q" => "ruby", "lang" => "en"}
http.form_data = {"q" => ["ruby", "perl"], "lang" => "en"}
http.set_form_data{"q" => "ruby", "lang" => "en"}, ';')
# File lib/net/http/header.rb, line 381
def set_form_data(params, sep = '&')
query = URI.encode_www_form(params)
query.gsub!(/&/, sep) if sep != '&'
self.body = query
self.content_type = 'application/x-www-form-urlencoded'
end
Also aliased as: form_data=
set_range(r, e = nil) Show source
设置HTTP范围:标题。接受Range对象作为单个参数,或者从该索引开始的索引和长度。例:
req.range = (0..1023)
req.set_range 0, 1023
# File lib/net/http/header.rb, line 242
def set_range(r, e = nil)
unless r
@header.delete 'range'
return r
end
r = (r...r+e) if e
case r
when Numeric
n = r.to_i
rangestr = (n > 0 ? "0-#{n-1}" : "-#{-n}")
when Range
first = r.first
last = r.end
last -= 1 if r.exclude_end?
if last == -1
rangestr = (first > 0 ? "#{first}-" : "-#{-first}")
else
raise Net::HTTPHeaderSyntaxError, 'range.first is negative' if first < 0
raise Net::HTTPHeaderSyntaxError, 'range.last is negative' if last < 0
raise Net::HTTPHeaderSyntaxError, 'must be .first < .last' if first > last
rangestr = "#{first}-#{last}"
end
else
raise TypeError, 'Range/Integer is required'
end
@header['range'] = ["bytes=#{rangestr}"]
r
end
Also aliased as: range=
sub_type() Show source
返回一个内容类型字符串,如“html”。如果Content-Type:头字段不存在或未提供子类型(例如“Content-Type:text”),则此方法返回nil。
# File lib/net/http/header.rb, line 337
def sub_type
return nil unless @header['content-type']
_, sub = *self['Content-Type'].split(';').first.to_s.split('/')
return nil unless sub
sub.strip
end
to_hash() Show source
返回由标题名称和值数组组成的哈希。例如{“cache-control”=>“private”,
"content-type" => ["text/html"],
"date" => ["Wed, 22 Jun 2005 22:11:50 GMT"]}
# File lib/net/http/header.rb, line 165
def to_hash
@header.dup
end
type_params() Show source
为内容类型指定的任何参数,作为散列返回。例如,Content-Type:text / html; charset = EUC-JP会导致#type_params返回{'charset'=>'EUC-JP'}
# File lib/net/http/header.rb, line 347
def type_params
result = {}
list = self['Content-Type'].to_s.split(';')
list.shift
list.each do |param|
k, v = *param.split('=', 2)
result[k.strip] = v.strip
end
result
end
私有实例方法
basic_encode(account, password) Show source
# File lib/net/http/header.rb, line 442
def basic_encode(account, password)
'Basic ' + ["#{account}:#{password}"].pack('m0')
end
capitalize(name) Show source
# File lib/net/http/header.rb, line 185
def capitalize(name)
name.to_s.split(/-/).map {|s| s.capitalize }.join('-')
end