curb
एक महान समाधान की तरह दिखता है, लेकिन अगर यह आपकी आवश्यकताओं को पूरा नहीं करता है, तो आप इसे कर सकते हैं Net::HTTP
। एक मल्टीपार्ट फॉर्म पोस्ट कुछ अतिरिक्त हेडर के साथ एक सावधानी से स्वरूपित स्ट्रिंग है। ऐसा लगता है कि हर रूबी प्रोग्रामर को जो मल्टीपार्ट पोस्ट करने की जरूरत है, वह इसके लिए अपनी छोटी लाइब्रेरी लिखता है, जो मुझे आश्चर्यचकित करता है कि यह कार्यक्षमता अंतर्निहित क्यों नहीं है। शायद यह है ... वैसे भी, आपके पढ़ने की खुशी के लिए, मैं आगे बढ़ूंगा और यहां अपना समाधान दूंगा। यह कोड उन दो उदाहरणों पर आधारित है जो मुझे ब्लॉग के एक जोड़े पर मिले थे, लेकिन मुझे अफसोस है कि मुझे अब लिंक नहीं मिल रहे हैं। इसलिए मुझे लगता है कि मुझे सिर्फ अपने लिए सारा श्रेय लेना होगा ...
मैंने इसके लिए जो मॉड्यूल लिखा था, उसमें एक सार्वजनिक वर्ग शामिल है, प्रपत्र डेटा और हेडर को वस्तुओं String
और File
वस्तुओं के हैश से उत्पन्न करने के लिए । उदाहरण के लिए, यदि आप "शीर्षक" नामक स्ट्रिंग पैरामीटर और "दस्तावेज़" नामक फ़ाइल पैरामीटर के साथ एक फॉर्म पोस्ट करना चाहते हैं, तो आप निम्नलिखित कार्य करेंगे:
#prepare the query
data, headers = Multipart::Post.prepare_query("title" => my_string, "document" => my_file)
तो आप बस के POST
साथ एक सामान्य करते हैं Net::HTTP
:
http = Net::HTTP.new(upload_uri.host, upload_uri.port)
res = http.start {|con| con.post(upload_uri.path, data, headers) }
या फिर आप करना चाहते हैं POST
। मुद्दा यह है कि Multipart
डेटा और हेडर जो आपको भेजने की आवश्यकता है, लौटाता है। और बस! सरल, सही? यहां मल्टीपार्ट मॉड्यूल के लिए कोड है (आपको mime-types
मणि की आवश्यकता है ):
# Takes a hash of string and file parameters and returns a string of text
# formatted to be sent as a multipart form post.
#
# Author:: Cody Brimhall <mailto:brimhall@somuchwit.com>
# Created:: 22 Feb 2008
# License:: Distributed under the terms of the WTFPL (http://www.wtfpl.net/txt/copying/)
require 'rubygems'
require 'mime/types'
require 'cgi'
module Multipart
VERSION = "1.0.0"
# Formats a given hash as a multipart form post
# If a hash value responds to :string or :read messages, then it is
# interpreted as a file and processed accordingly; otherwise, it is assumed
# to be a string
class Post
# We have to pretend we're a web browser...
USERAGENT = "Mozilla/5.0 (Macintosh; U; PPC Mac OS X; en-us) AppleWebKit/523.10.6 (KHTML, like Gecko) Version/3.0.4 Safari/523.10.6"
BOUNDARY = "0123456789ABLEWASIEREISAWELBA9876543210"
CONTENT_TYPE = "multipart/form-data; boundary=#{ BOUNDARY }"
HEADER = { "Content-Type" => CONTENT_TYPE, "User-Agent" => USERAGENT }
def self.prepare_query(params)
fp = []
params.each do |k, v|
# Are we trying to make a file parameter?
if v.respond_to?(:path) and v.respond_to?(:read) then
fp.push(FileParam.new(k, v.path, v.read))
# We must be trying to make a regular parameter
else
fp.push(StringParam.new(k, v))
end
end
# Assemble the request body using the special multipart format
query = fp.collect {|p| "--" + BOUNDARY + "\r\n" + p.to_multipart }.join("") + "--" + BOUNDARY + "--"
return query, HEADER
end
end
private
# Formats a basic string key/value pair for inclusion with a multipart post
class StringParam
attr_accessor :k, :v
def initialize(k, v)
@k = k
@v = v
end
def to_multipart
return "Content-Disposition: form-data; name=\"#{CGI::escape(k)}\"\r\n\r\n#{v}\r\n"
end
end
# Formats the contents of a file or string for inclusion with a multipart
# form post
class FileParam
attr_accessor :k, :filename, :content
def initialize(k, filename, content)
@k = k
@filename = filename
@content = content
end
def to_multipart
# If we can tell the possible mime-type from the filename, use the
# first in the list; otherwise, use "application/octet-stream"
mime_type = MIME::Types.type_for(filename)[0] || MIME::Types["application/octet-stream"][0]
return "Content-Disposition: form-data; name=\"#{CGI::escape(k)}\"; filename=\"#{ filename }\"\r\n" +
"Content-Type: #{ mime_type.simplified }\r\n\r\n#{ content }\r\n"
end
end
end