如何处理multipart / form-data请求而不表达

时间:2017-05-25 23:22:48

标签: javascript node.js forms multipartform-data boundary

所以我有这样的表格:

  <form id="myForm" enctype="multipart/form-data">
      <input id="name" title="name" type="text" placeholder="name" />
      <input id="quantity" title="quantity" type="text" placeholder="quantity" />
      <input id="price" title="price" type="text" placeholder="price" />
      <input id="imageLocation" title="imageLocation" type="text" placeholder="imagelocation" />
      <input id="description" title="description" type="text" placeholder="description" />
  </form>

这是我发送数据的地方:

 function postMultiPartHttpCall() {
    var XHR = new XMLHttpRequest();

    var form = document.getElementById("myForm");

    var FD = new FormData(form);

    XHR.addEventListener("load", function (event) {
        var callingObj = {};
        callingObj.responseText = event.target.responseText;
        console.log(callingObj);
    });

    XHR.open("POST", '/articlePost');
    XHR.send(FD);
}

这是我收到它的地方:

    function _processFormData (request, res, onSuccess,onError) {
    var requestBody = '';
    request.on('data', function (data) {
        requestBody += data;
        if (requestBody.length > 1e7) {
            res.writeHead(413, 'Request length too long', { 'Content-Type': 'text/html' });
            res.end('413 : Request Entity Too Large');
        }
    });

    request.on('end', function () {
        var oFormData = qsLib.parse(requestBody);
        console.log(request.headers);

    });
}

所以当我发送一些数据时,我会收到这个(console.log):

Debugger listening on 127.0.0.1:5858
Server was started
{ host: 'localhost:1337',
connection: 'keep-alive',
'content-length': '44',
origin: 'http://localhost:1337',
'user-agent': 'Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36     (KHTML, like Gecko) Chrome/58.0.3029.110 Safari/537.36',
'content-type': 'multipart/form-data; boundary=----WebKitFormBoundaryyVFw7KZwIaAIQeQ1',
accept: '*/*',
referer: 'http://localhost:1337/CartSPA.html',
'accept-encoding': 'gzip, deflate, br',
'accept-language': 'en-US,en;q=0.8,hr;q=0.6,de-AT;q=0.4,de;q=0.2,de-DE;q=0.2' }

所以我想要的是在不使用expressJS或其他类似的3 rd 派对库的情况下获取具有表单字段的对象。是否可以使用边界编号获取属性,或者在哪里可以看到发送了哪些数据?

3 个答案:

答案 0 :(得分:3)

是的,&#34; 可以使用边界号&#34;来获取属性,但您必须匹配字段并手动解析名称...

此外,为了让表单中的值得到发送,需要在name上设置属性input tags。例如:

<input id="quantity" title="quantity" type="text" placeholder="quantity" />

应更新为包含名称属性,如下所示:

<input name="quantity" id="quantity" title="quantity" type="text" placeholder="quantity" />

在表单字段上设置名称属性后,请求正文应包含表单数据(即requestBody应该在加载数据时具有编码表单的内容)。例如,它可能类似于下面的输出:

-----------------------------150182554619156
Content-Disposition: form-data; name="name"

z4520
-----------------------------150182554619156
Content-Disposition: form-data; name="quantity"

2
-----------------------------150182554619156
Content-Disposition: form-data; name="price"

32.90
-----------------------------150182554619156
Content-Disposition: form-data; name="imagelocation"

/img/z4520.jpg
-----------------------------150182554619156
Content-Disposition: form-data; name="description"

water filter
-----------------------------150182554619156--

解析表单字段

在下面的入门代码中,它会检查边界的请求标头 Content-Type (您可能还需要确保Content-Type实际上是&# 34; multipart / form-data &#34;)使用String.indexOf()然后使用String.split()设置边界并从结果数组中取第2个元素。该边界值可用于将主体数据分成数组(也使用String.split())。

我将把它作为练习让读者解析出值并将它们存储在一个数组中(参见@TODO;)。

提示: Array.reduce()可能非常方便......

request.on('end', function() {
    if (request.headers.hasOwnProperty('content-type') && request.headers['content-type'].indexOf('boundary=') > -1) {
        var parts = request.headers['content-type'].split('boundary=');
        var boundary = parts[1];
        var splitBody = requestBody.split(boundary);
        /**
         @TODO: iterate over elements in splitBody, matching name and value and add to array of fields
        */

        res.writeHead(200, {"Content-Type": "application/json"});
        res.write(JSON.stringify({formFields: splitBody}));
    }
    else {
        //bad request
        res.writeHead(400, {"Content-Type": "application/json"});
        res.write("missing boundary in content-type ));
    }
    res.end();
});

答案 1 :(得分:1)

答案 2 :(得分:0)

我一直在使用npm模块multiparty。这很直接。

在您的请求处理程序中,执行类似的操作。

var form = new multiparty.Form();
form.parse(req, function(err, fields, files) {
  res.writeHead(200, {'content-type': 'text/plain'});
  res.write('received upload:\n\n');
  res.end(util.inspect({fields: fields, files: files}));
});