Ruby: Как превратить хэш в параметры HTTP?
Это довольно легко с простым хешем, например
{:a => "a", :b => "b"}
который переводится в
"a=a&b=b"
Но что вы делаете с чем-то более сложным, чем
{:a => "a", :b => ["c", "d", "e"]}
который должен перевести на
"a=a&b[0]=c&b[1]=d&b[2]=e"
Или, что еще хуже, (что делать) с чем-то вроде:
{:a => "a", :b => [{:c => "c", :d => "d"}, {:e => "e", :f => "f"}]
Спасибо за очень ценную помощь в этом!
Ответы
Ответ 1
Обновление: Эта функция была удалена из драгоценного камня.
Жюльен, ваш собственный ответ хороший, и я бесстыдно заимствован из него, но он не скрывает скрытых символов, и есть несколько других случаев, когда он ломается.
require "addressable/uri"
uri = Addressable::URI.new
uri.query_values = {:a => "a", :b => ["c", "d", "e"]}
uri.query
# => "a=a&b[0]=c&b[1]=d&b[2]=e"
uri.query_values = {:a => "a", :b => [{:c => "c", :d => "d"}, {:e => "e", :f => "f"}]}
uri.query
# => "a=a&b[0][c]=c&b[0][d]=d&b[1][e]=e&b[1][f]=f"
uri.query_values = {:a => "a", :b => {:c => "c", :d => "d"}}
uri.query
# => "a=a&b[c]=c&b[d]=d"
uri.query_values = {:a => "a", :b => {:c => "c", :d => true}}
uri.query
# => "a=a&b[c]=c&b[d]"
uri.query_values = {:a => "a", :b => {:c => "c", :d => true}, :e => []}
uri.query
# => "a=a&b[c]=c&b[d]"
Жемчужина: addressable '
gem install addressable
Ответ 2
Для базовых, не вложенных хэшей, Rails/ActiveSupport имеет Object#to_query
.
>> {:a => "a", :b => ["c", "d", "e"]}.to_query
=> "a=a&b%5B%5D=c&b%5B%5D=d&b%5B%5D=e"
>> CGI.unescape({:a => "a", :b => ["c", "d", "e"]}.to_query)
=> "a=a&b[]=c&b[]=d&b[]=e"
http://api.rubyonrails.org/classes/Object.html#method-i-to_query
Ответ 3
Если вы используете Ruby 1.9.2 или новее, вы можете использовать URI.encode_www_form
, если вам не нужны массивы.
например. (из документов Ruby в 1.9.3):
URI.encode_www_form([["q", "ruby"], ["lang", "en"]])
#=> "q=ruby&lang=en"
URI.encode_www_form("q" => "ruby", "lang" => "en")
#=> "q=ruby&lang=en"
URI.encode_www_form("q" => ["ruby", "perl"], "lang" => "en")
#=> "q=ruby&q=perl&lang=en"
URI.encode_www_form([["q", "ruby"], ["q", "perl"], ["lang", "en"]])
#=> "q=ruby&q=perl&lang=en"
Вы заметите, что значения массива не заданы с именами ключей, содержащими []
, как мы все привыкли в строках запроса. Спецификация, используемая encode_www_form
, соответствует определению HTML5 данных application/x-www-form-urlencoded
.
Ответ 4
Там намного проще сделать это, если вы используете Rails:
http://apidock.com/rails/ActiveSupport/CoreExtensions/Hash/to_query
Итак, вы можете сделать:
{:a => "a", :b => "b"}.to_query
Ответ 5
Не нужно загружать раздутый ActiveSupport или сворачивать свой собственный, вы можете использовать Rack::Utils.build_query
и Rack::Utils.build_nested_query
. Вот сообщение в блоге, которое дает хороший пример:
require 'rack'
Rack::Utils.build_query(
authorization_token: "foo",
access_level: "moderator",
previous: "index"
)
# => "authorization_token=foo&access_level=moderator&previous=index"
Он даже обрабатывает массивы:
Rack::Utils.build_query( {:a => "a", :b => ["c", "d", "e"]} )
# => "a=a&b=c&b=d&b=e"
Rack::Utils.parse_query _
# => {"a"=>"a", "b"=>["c", "d", "e"]}
Или более сложный вложенный материал:
Rack::Utils.build_nested_query( {:a => "a", :b => [{:c => "c", :d => "d"}, {:e => "e", :f => "f"}] } )
# => "a=a&b[][c]=c&b[][d]=d&b[][e]=e&b[][f]=f"
Rack::Utils.parse_nested_query _
# => {"a"=>"a", "b"=>[{"c"=>"c", "d"=>"d", "e"=>"e", "f"=>"f"}]}
Ответ 6
Украсть у Merb:
# File merb/core_ext/hash.rb, line 87
def to_params
params = ''
stack = []
each do |k, v|
if v.is_a?(Hash)
stack << [k,v]
else
params << "#{k}=#{v}&"
end
end
stack.each do |parent, hash|
hash.each do |k, v|
if v.is_a?(Hash)
stack << ["#{parent}[#{k}]", v]
else
params << "#{parent}[#{k}]=#{v}&"
end
end
end
params.chop! # trailing &
params
end
См. http://noobkit.com/show/ruby/gems/development/merb/hash/to_params.html
Ответ 7
Здесь короткий и сладкий лайнер, если вам нужно только поддерживать простые строки запроса/значения ключа ASCII:
hash = {"foo" => "bar", "fooz" => 123}
# => {"foo"=>"bar", "fooz"=>123}
query_string = hash.to_a.map { |x| "#{x[0]}=#{x[1]}" }.join("&")
# => "foo=bar&fooz=123"
Ответ 8
class Hash
def to_params
params = ''
stack = []
each do |k, v|
if v.is_a?(Hash)
stack << [k,v]
elsif v.is_a?(Array)
stack << [k,Hash.from_array(v)]
else
params << "#{k}=#{v}&"
end
end
stack.each do |parent, hash|
hash.each do |k, v|
if v.is_a?(Hash)
stack << ["#{parent}[#{k}]", v]
else
params << "#{parent}[#{k}]=#{v}&"
end
end
end
params.chop!
params
end
def self.from_array(array = [])
h = Hash.new
array.size.times do |t|
h[t] = array[t]
end
h
end
end
Ответ 9
{:a=>"a", :b=>"b", :c=>"c"}.map{ |x,v| "#{x}=#{v}" }.reduce{|x,v| "#{x}&#{v}" }
"a=a&b=b&c=c"
Здесь другой путь. Для простых запросов.
Ответ 10
Я знаю, что это старый вопрос, но я просто хотел опубликовать этот бит кода, поскольку я не мог найти простой камень для выполнения этой задачи для меня.
module QueryParams
def self.encode(value, key = nil)
case value
when Hash then value.map { |k,v| encode(v, append_key(key,k)) }.join('&')
when Array then value.map { |v| encode(v, "#{key}[]") }.join('&')
when nil then ''
else
"#{key}=#{CGI.escape(value.to_s)}"
end
end
private
def self.append_key(root_key, key)
root_key.nil? ? key : "#{root_key}[#{key.to_s}]"
end
end
Подкатил как драгоценный камень здесь: https://github.com/simen/queryparams
Ответ 11
Мне нравится использовать этот драгоценный камень:
https://rubygems.org/gems/php_http_build_query
Использование образца:
puts PHP.http_build_query({"a"=>"b","c"=>"d","e"=>[{"hello"=>"world","bah"=>"black"},{"hello"=>"world","bah"=>"black"}]})
# a=b&c=d&e%5B0%5D%5Bbah%5D=black&e%5B0%5D%5Bhello%5D=world&e%5B1%5D%5Bbah%5D=black&e%5B1%5D%5Bhello%5D=world
Ответ 12
Лучший подход - использовать Hash.to_params, который отлично работает с массивами.
{a: 1, b: [1,2,3]}.to_param
"a=1&b[]=1&b[]=2&b[]=3"
Ответ 13
require 'uri'
class Hash
def to_query_hash(key)
reduce({}) do |h, (k, v)|
new_key = key.nil? ? k : "#{key}[#{k}]"
v = Hash[v.each_with_index.to_a.map(&:reverse)] if v.is_a?(Array)
if v.is_a?(Hash)
h.merge!(v.to_query_hash(new_key))
else
h[new_key] = v
end
h
end
end
def to_query(key = nil)
URI.encode_www_form(to_query_hash(key))
end
end
2.4.2 :019 > {:a => "a", :b => "b"}.to_query_hash(nil)
=> {:a=>"a", :b=>"b"}
2.4.2 :020 > {:a => "a", :b => "b"}.to_query
=> "a=a&b=b"
2.4.2 :021 > {:a => "a", :b => ["c", "d", "e"]}.to_query_hash(nil)
=> {:a=>"a", "b[0]"=>"c", "b[1]"=>"d", "b[2]"=>"e"}
2.4.2 :022 > {:a => "a", :b => ["c", "d", "e"]}.to_query
=> "a=a&b%5B0%5D=c&b%5B1%5D=d&b%5B2%5D=e"
Ответ 14
Еще один ответ, оказывается, есть:
require 'uri'
URI.encode_www_form({"one" => "value with space", "two" => ["v1", "v2"]})
# => => "one=value+with+space&two=v1&two=v2"
Обратите внимание, что в качестве пробела используется знак "+", который может быть технически корректным для URL/URI или нет. Все становится очень запутанным.