gpt4 book ai didi

node.js - 流式上传到云存储(busboy,请求)

转载 作者:太空宇宙 更新时间:2023-11-04 00:33:24 25 4
gpt4 key购买 nike

我是 Node.js 新手。我想做的是通过我的node.js 服务器将文件上传从网络浏览器流式传输到云存储。

我正在使用“express”、“request”和“busboy”模块。

var express = require("express");
var request = require("request");
var BusBoy = require("busboy");
var router = express.Router();

router.post("/upload", function(req, res, next) {
var busboy = new BusBoy({ headers: req.headers });
var json = {};

busboy.on("file", function (fieldname, file, filename, encoding, mimetype) {
file.on("data", function(data) {
console.log(`streamed ${data.length}`);
});

file.on("end", function() {
console.log(`finished streaming ${filename}`);
});

var r = request({
url: "http://<my_cloud_storage_api_url>",
method: "POST",
headers: {
"CUSTOM-HEADER": "Hello",
},
formData: {
"upload": file
}
}, function(err, httpResponse, body) {
console.log("uploaded");
json.response = body;
});
});

busboy.on("field", function(name, val) {
console.log(`name: ${name}, value: ${value}`);
});

busboy.on("finish", function() {
res.send(json);
});

req.pipe(busboy);
});

module.exports = router;

但是我在服务器上不断收到以下错误。我在这里做错了什么?如有任何帮助,我们将不胜感激。

Error: Part terminated early due to unexpected end of multipart data
at node_modules\busboy\node_modules\dicer\lib\Dicer.js:65:36
at nextTickCallbackWith0Args (node.js:420:9)
at process._tickCallback (node.js:349:13)

最佳答案

我意识到这个问题已经存在了大约 7 个月了,但我将在这里回答它,以试图帮助其他目前对此有异议的人。

实际上,您有两个选择:添加文件大小,或使用请求以外的其他内容。

注意:我在首次发布后不久对其进行了编辑,希望能提供更多背景信息。

使用其他东西

如果您不需要 Request 所具有的所有内置功能,您可以使用一些替代方案来代替 Request。

  • form-data 在简单情况下可以单独使用,或者可以与 got 一起使用。 。 request在内部使用它。
  • bhttp 宣传 Streams2+ 支持,尽管根据我的经验 Stream2+ 支持对我来说不是问题。没有内置https支持,你必须指定 custom agent
  • got 另一位瘦身了。没有对表单数据进行任何特殊处理,例如 request确实如此,但通常与 form-data 一起使用或form-data2 。不过,我在通过公司代理让它工作时遇到了麻烦,但这可能是因为我是一个网络新手。
  • needle 看起来很轻,但我还没有实际尝试过。

使用请求:添加文件大小

Request (截至撰写本文时)不支持使用 transfer-encoding: chunked因此,要使用它上传文件,您需要将文件的大小与文件一起添加,如果您从网络客户端上传,则意味着除了文件本身之外,客户端还需要将该文件大小发送到您的服务器。

我想出的方法是在文件字段之前在其自己的字段中发送文件元数据。

我用描述我所做的事情的评论修改了你的示例。请注意,我没有对收到的数据进行任何验证,但我建议您添加这一点。

var express = require("express");
var request = require("request");
var BusBoy = require("busboy");
var router = express.Router();

router.post("/upload", function(req, res, next) {
var busboy = new BusBoy({ headers: req.headers });
var json = {};

// Use this to cache any fields which are file metadata.
var fileMetas = {};

busboy.on("file", function (fieldname, file, filename, encoding, mimetype) {
// Be sure to match this prop name here with the pattern you use to detect meta fields.
var meta = fileMetas[fieldname + '.meta'];

if (!meta) {
// Make sure to dump the file.
file.resume();
// Then, do some sort of error handling here, because you cannot upload a file
// without knowing it's length.
return;
}

file.on("data", function(data) {
console.log(`streamed ${data.length}`);
});

file.on("end", function() {
console.log(`finished streaming ${filename}`);
});

var r = request({
url: "http://<my_cloud_storage_api_url>",
method: "POST",
headers: {
"CUSTOM-HEADER": "Hello",
},
formData: {
// value + options form of a formData field.
"upload": {
value: file,
options: {
filename: meta.name,
knownLength: meta.size
}
}
}
}, function(err, httpResponse, body) {
console.log("uploaded");
json.response = body;
});
});

busboy.on("field", function(name, val) {
// Use whatever pattern you want. I used (fileFieldName + ".meta").
// Another good one might be ("meta:" + fileFieldName).
if (/\.meta$/.test(name)) {
// I send an object with { name, size, type, lastModified },
// which are just the public props pulled off a File object.
// Note: Should probably add error handling if val is somehow not parsable.
fileMetas[name] = JSON.parse(val);
console.log(`file metadata: name: ${name}, value: ${value}`);
return;
}

// Otherwise, process field as normal.
console.log(`name: ${name}, value: ${value}`);
});

busboy.on("finish", function() {
res.send(json);
});

req.pipe(busboy);
});

module.exports = router;

在客户端上,您需要在文件本身之前发送所谓字段上的元数据。这可以通过订购 <input type="hidden"> 来完成文件之前的控制并更新其值 onchangeThe order of values sent is guaranteed to follow the order of inputs in appearance 。如果您自己使用 FormData 构建请求正文,您可以通过在附加 File 之前附加适当的元数据来完成此操作.

示例为 <form>

<script>
function extractFileMeta(file) {
return JSON.stringify({
size: file.size,
name: file.name,
type: file.type,
lastUpdated: file.lastUpdated
});
}

function onFileUploadChange(event) {
// change this to use arrays if using the multiple attribute on the file input.
var file = event.target.files[0];
var fileMetaInput = document.querySelector('input[name=fileUpload.meta]');

if (fileMetaInput) {
fileMetaInput.value = extractFileMeta(file);
}
}
</script>
<form action="/upload-to-cloud">
<input type="hidden" name="fileUpload.meta">
<input type="file" name="fileUpload" onchange="onFileUploadChange(event)">
</form>

示例为 FormData :

function onSubmit(event) {
event.preventDefault();

var form = document.getElementById('my-upload-form');
var formData = new FormData();

var fileUpload = form.elements['fileUpload'];
var fileUploadMeta = JSON.stringify({
size: fileUpload.size,
name: fileUpload.name,
type: fileUpload.type,
lastUpdated: fileUpload.lastUpdated
});

// Append fileUploadMeta BEFORE fileUpload.
formData.append('fileUpload.meta', fileUploadMeta);
formData.append('fileUpload', fileUpload);

// Do whatever you do to POST here.
}

关于node.js - 流式上传到云存储(busboy,请求),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40077003/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com