web-dev-qa-db-ja.com

マルチパートフォームデータで投稿を取得する

私はこのようなURLを取得しています:

fetch(url, {
  mode: 'no-cors',
  method: method || null,
  headers: {
    'Accept': 'application/json, application/xml, text/plain, text/html, *.*',
    'Content-Type': 'multipart/form-data'
  },
  body: JSON.stringify(data) || null,
}).then(function(response) {
  console.log(response.status)
  console.log("response");
  console.log(response)
})

私のAPIは、データがmultipart/form-dataであると想定しているため、このタイプのcontent-typeを使用しています...しかし、ステータスコード400の応答が返されます。

私のコードの何が問題になっていますか?

52
aryan

Content-Typemultipart/form-dataに設定していますが、ボディデータでJSON.stringifyを使用すると、application/jsonが返されます。コンテンツタイプが一致していません。

データをjsonではなくmultipart/form-dataとしてエンコードする必要があります。通常、multipart/form-dataはファイルのアップロード時に使用され、application/x-www-form-urlencoded(HTMLフォームのデフォルト)よりも少し複雑です。

multipart/form-dataの仕様は RFC 1867 にあります。

JavaScriptを介してそのようなデータを送信する方法については、 here を参照してください。

基本的な考え方は、 FormData オブジェクトを使用することです(IE <10ではサポートされていません):

function sendData(url, data) {
  var formData  = new FormData();

  for(var name in data) {
    formData.append(name, data[name]);
  }

  fetch(url, {
    method: 'POST',
    body: formData
  }).then(function (response) {
     ...
  });
}
94
rossipedia

私は最近IPFSを使っていて、これを解決しました。ファイルをアップロードするIPFSのcurlの例は次のようになります。

curl -i -H "Content-Type: multipart/form-data; boundary=CUSTOM" -d $'--CUSTOM\r\nContent-Type: multipart/octet-stream\r\nContent-Disposition: file; filename="test"\r\n\r\nHello World!\n--CUSTOM--' "http://localhost:5001/api/v0/add"

基本的な考え方は、各部分(boundaryname__と--の文字列で分割)に独自のヘッダー(たとえば、2番目の部分のContent-Type)があるということです。 FormDataNAME _ オブジェクトはこれらすべてを管理するため、目標を達成するためのより良い方法です。

これは、次のようなフェッチAPIに変換されます。

const formData = new FormData()
formData.append('blob', new Blob(['Hello World!\n']), 'test')

fetch('http://localhost:5001/api/v0/add', {
  method: 'POST',
  body: formData
})
.then(r => r.json())
.then(data => {
  console.log(data)
})
11
konsumer