简体   繁体   English

带有 json 和文件数据的法拉第多部分请求

[英]Faraday multipart request with json & file data

When trying to POST multipart data (file & json) with Faraday gem, remote server is not recognizing json data parameter and failing with validation error as it is required parameter.尝试使用 Faraday gem POST 多部分数据(文件和 json)时,远程服务器无法识别 json 数据参数,并且由于验证错误而失败,因为它是必需参数。

connection = Faraday.new(url: "#{uri.scheme}://#{uri.host}:#{uri.port}") do |faraday|
              faraday.request :multipart
              faraday.request :url_encoded

              faraday.adapter :net_http
            end

request_data = { file: Faraday::UploadIO.new('somefile.png', 'image/png'), jsonBody: JSON.dump({ id: 'foo', name: 'bar' }) }

response = connection.post(uri.request_uri) do |request|
  request = Ff::Api::RequestHeaders.set(request, self.api_options)
  request.headers['content-type'] = 'multipart/form-data; boundary=-----------RubyMultipartPost'
  request.body = request_data
end

Working CURL request:工作 CURL 请求:

curl -v -H 'Authorization: bearer 7d70fb31-0eb9-4846-9ea8-933dfb69d8f1' \
-H 'Accept: application/xml,application/xhtml+xml,application/json,text/html,text/plain' \
-H 'Content-Type: multipart/form-data' \
-F 'jsonBody={"id":"foo","name":"bar"};type=application/json' \
'http://localhost:8080/localproject/rest-resource/items'

Curl Log:卷曲日志:

*   Trying 127.0.0.1...
* TCP_NODELAY set
* Connected to localhost (127.0.0.1) port 8080 (#0)
> POST /localproject/rest-resource/items HTTP/1.1
> Host: localhost:8080
> Authorization: bearer 7d70fb31-0eb9-4846-9ea8-933dfb69d8f1
> USER_IP_ADDR: 127.0.0.1
> Accept: application/xml,application/xhtml+xml,application/json,text/html,text/plain
> Content-Length: 555
> Expect: 100-continue
> Content-Type: multipart/form-data; boundary=------------------------0c5f3acb5a4731ff
> 
< HTTP/1.1 100 Continue
< HTTP/1.1 200 OK
< Server: Apache-Coyote/1.1
< Cache-Control: no-cache, no-store, max-age=0, must-revalidate
< Pragma: no-cache
< Expires: 0
< X-XSS-Protection: 1; mode=block
< X-Frame-Options: DENY
< X-Content-Type-Options: nosniff
< Content-Type: application/xml
< Transfer-Encoding: chunked
< Date: Thu, 08 Feb 2018 10:50:06 GMT
< 
* Connection #0 to host localhost left intact
<?xml version="1.0" encoding="UTF-8" standalone="yes"?><updateResponse><id>5ce0b335-4638-474c-8c4a-9adb6c54b057</id><success>true</success></updateResponse>% 

Additional prefix "type=application/json" to request body seems to be making difference with CURL.请求正文的附加前缀“type=application/json”似乎与 CURL 有所不同。 Did not find any way to do the same with Faraday.没有找到任何方法可以对法拉第做同样的事情。

Faraday Version : 0.12.2 & 0.14.0法拉第版本:0.12.2 & 0.14.0

Ruby Version : 2.3.3红宝石版本:2.3.3

Rails : 5.2导轨:5.2

Any help will be greatly appreciated.任何帮助将不胜感激。

Found that Faraday doesn't support mix multipart request (file upload with json/xml data).发现法拉第不支持混合多部分请求(文件上传与 json/xml 数据)。 This is because mixing file and json data is not restful approach and should be avoided entirely.这是因为混合文件和 json 数据不是宁静的方法,应该完全避免。 More discussion is happening here https://github.com/lostisland/faraday/issues/769更多讨论在这里发生https://github.com/lostisland/faraday/issues/769

I have temporarily fixed the issue by patching Faraday::Mulitipart class to allow mixing JSON data,我通过修补Faraday::Mulitipart类以允许混合 JSON 数据暂时解决了这个问题,

class Faraday::Request::Multipart
  def create_multipart(env, params)
    boundary = env.request.boundary
    parts = process_params(params) do |key, value|
      if (JSON.parse(value) rescue false)
        Faraday::Parts::Part.new(boundary, key, value, 'Content-Type' => 'application/json')
      else
        Faraday::Parts::Part.new(boundary, key, value)
      end
    end
    parts << Faraday::Parts::EpiloguePart.new(boundary)

    body = Faraday::CompositeReadIO.new(parts)
    env.request_headers[Faraday::Env::ContentLength] = body.length.to_s
    return body
  end
end

But in long term one should change server implementation to avoid mix multipart requests.但是从长远来看,应该更改服务器实现以避免混合多部分请求。

Not sure if it's an option for others, but the server I was trying to POST to will accept the json as a file part (rather than the message body).不确定它是否适合其他人,但我尝试 POST 的服务器将接受 json 作为文件部分(而不是消息正文)。 In which case something like this will work:在这种情况下,这样的事情会起作用:

require 'faraday'

conn = Faraday.new("http://server:port") do |f|
    f.request:multipart
    f.request:url_encoded
    f.adapter:net_http
end

payload = {
    filePart:Faraday::UploadIO.new("image.jpg", 'image/jpeg'),
    jsonPart:Faraday::UploadIO.new("stamp.json", 'application/json')
}

res = conn.post('/my/url/', payload)
puts res.body

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM