Ruby CGI :: Cookie原始cookie解析

时间:2012-11-15 14:30:28

标签: ruby cookies cgi ruby-1.9.3

我希望能够在ruby 1.9.3p194(2012-04-20修订版35410)[x86_64-darwin10.8.0]中解析原始cookie字符串。

CGI::Cookie图书馆看起来很有希望,然而,它并不像我想象的那样有效。

例如,

CGI::Cookie::parse("ASPSESSIONIDSCDRSRTS=HHALOHOBJGJMLPIANNLDOMCJ; path=/").each_key {|name| p 'Cookie name: ' + name}

将返回:

"Cookie name: ASPSESSIONIDSCDRSRTS"
"Cookie name: path"

我想要的是像CGI::new实例那样的工作,除了你传递一个原始的cookie字符串:

cookie1 = CGI::Cookie::new('name'    => 'name',
                       'value'   => ['value1', 'value2', ...],
                       'path'    => 'path',   # optional
                       'domain'  => 'domain', # optional
                       'expires' => Time.now, # optional
                       'secure'  => true      # optional
                      )

name    = cookie1.name
values  = cookie1.value
path    = cookie1.path
domain  = cookie1.domain
expires = cookie1.expires
secure  = cookie1.secure

我无法弄清楚如何从原始cookie字符串中雄辩地执行此操作。

编辑---

以下代码位于~/.rvm/src/ruby-1.9.3-p194/lib/cgi/cookie.rb文件中。所以它应该返回cookies[name] = Cookie::new(name, *values)下面的评论。我似乎没有得到。

# Parse a raw cookie string into a hash of cookie-name=>Cookie
# pairs.
#
#   cookies = CGI::Cookie::parse("raw_cookie_string")
#     # { "name1" => cookie1, "name2" => cookie2, ... }
#
def Cookie::parse(raw_cookie)
  cookies = Hash.new([])
  return cookies unless raw_cookie

  raw_cookie.split(/[;,]\s?/).each do |pairs|
    name, values = pairs.split('=',2)
    next unless name and values
    name = CGI::unescape(name)
    values ||= ""
    values = values.split('&').collect{|v| CGI::unescape(v,@@accept_charset) }
    if cookies.has_key?(name)
      values = cookies[name].value + values
    end
    cookies[name] = Cookie::new(name, *values)
  end

  cookies
end

编辑---

这似乎是Ruby的CGI::Cookie.parse方法中的一个错误。我已经打开了关于Ruby bug跟踪器的错误报告 - https://bugs.ruby-lang.org/issues/7364

2 个答案:

答案 0 :(得分:2)

我遇到了同样的问题。 CGI::Cookie.new无法消耗CGI::Cookie::parse产生的内容,它们似乎不像我们预期的那样相互补充。

快速搜索宝石,我找到了https://github.com/dwaite/cookiejar。看起来这个宝石尝试做了很多,但至少它有代码来正确解析原始cookie,正如我们所期望的那样。我没有尝试理解这个库,因为它似乎做了很多东西而我实际上只关心消费cookie - > modiyfing it - >生成cookie"。

我想出了这个快速的黑客攻击:

require 'cgi'
require 'cookiejar'

# Some sample
raw_cookie = "somecookie=some%7Cvalue; expires=Wed, 02 Apr 2014 13:36:50 GMT; path=/; domain=.somedomain.com"

parts = CookieJar::CookieValidation.parse_set_cookie raw_cookie
# Needs manual unescape
parts[:value] = CGI::unescape(parts[:value])
# Per spec, this name is different
parts[:expires] = parts[:expires_at]
# Remove old ones
parts.delete :expires_at
# CookieJar adds them always, remove
parts.delete :version
# Convert symbol keys to strings for GGI::Cookie
cookie = parts.inject(Hash.new) do |acc, (k,v)|
  acc[k.to_s] = v 
  acc 
end
puts CGI::Cookie.new(cookie).to_s
#=> somecookie=some%7Cvalue; domain=.somedomain.com; path=/; expires=Wed, 02 Apr 2014 13:36:50 GMT

我想这应该是一个gem的调用,它只对cookie做一件事并做得很好:能够解析并生成原始cookie字符串。但TBH,问题领域可能要比我做的方式更加复杂。

答案 1 :(得分:1)

我还没有使用过这个库,但是根据Ruby文档(http://www.ruby-doc.org/stdlib-1.9.3/libdoc/cgi/rdoc/CGI/Cookie.html)你要调用的是什么?在你的块中实际上是一个Cookie对象。

parse(raw_cookie) 
Parse a raw cookie string into a hash of cookie-name=>Cookie pairs.

您可能只是看到打印出来的名称是隐含的.to_s会在Cookie上为您提供的内容。

尝试打印name.path或name.value(或Cookie对象的任何访问者)。