读取 multipart/form-data 保存不正确
Reading multipart/form-data is saving incorectly
我正在尝试从上传的 http 图像文件中读取数据。好像是读取了文件,但是不知为什么输出的文件大约是原文件大小的两倍。
原文:94.7KB
临时文件:168.8 KB
当我在文本编辑器中打开这两个文件时,它们的行数相同,而且起始字符和结束字符也相同。这让我相信上传的文件被保存为字符串而不是二进制数据。
我认为重要的部分如下
读取的数据:
let body = ''
req.on('data', data => {
body += data.toString()
}).on('end', data => {
if (data) body += data.toString()
// Rest of block
})
数据保存:
// If there is a filename grab the file data
if (result.filename.length > 0) {
// Create a temporary url
let temp = join(os.tmpdir(), (Math.random() * 10000).toString(12).substr(5, 10))
// Get the data between the blocks after the first two newlines
let matches = item.match(/^.+?(\r\n\r\n|\n\n)(.+)/s)
// Write the data to file
fs.createWriteStream(temp).write(matches[2])
}
下面是完整的数据解析:
http.createServer((req, res) => {
let body = ''
req.on('data', data => {
body += data.toString()
}).on('end', data => {
if (data) body += data.toString()
let boundary = req.headers['content-type'].split('boundary=')[1]
// Split all the boundary items and loop over them
body.split(new RegExp(`(--${boundary}|--${boundary}--)`)).forEach(item => {
if (item.trim().toLowerCase().startsWith('content-disposition')) {
item = item.trim()
// Find the name and filename
let result = item.split(':')[1].split(';').map(i => i.trim()).reduce((obj, itm) => {
if (itm.startsWith('name=')) obj.name = itm.match(/^name="(.+)"/)[1]
if (itm.startsWith('filename=')) obj.filename = itm.match(/^filename="(.+)"/)[1]
return obj
}, { name: '', filename: '' })
// If there is a filename grab the file data
if (result.filename.length > 0) {
// Create a temporary url
let temp = join(os.tmpdir(), (Math.random() * 10000).toString(12).substr(5, 10))
// Get the data
let matches = item.match(/^.+?(\r\n\r\n|\n\n)(.+)/s)
// Write the data to file
fs.createWriteStream(temp).write(matches[2])
}
}
})
})
})
所以,我在这两种情况下都有些正确...
首先,我必须像这样将数据读取为二进制数据:
body += data.toString('binary')
接下来保存的时候,我需要像这样保存为二进制数据:
fs.createWriteStream(temp).write(matches[2], 'binary')
这现在保存为正确的文件大小并且图像可读!
我正在尝试从上传的 http 图像文件中读取数据。好像是读取了文件,但是不知为什么输出的文件大约是原文件大小的两倍。
原文:94.7KB
临时文件:168.8 KB
当我在文本编辑器中打开这两个文件时,它们的行数相同,而且起始字符和结束字符也相同。这让我相信上传的文件被保存为字符串而不是二进制数据。
我认为重要的部分如下
读取的数据:
let body = ''
req.on('data', data => {
body += data.toString()
}).on('end', data => {
if (data) body += data.toString()
// Rest of block
})
数据保存:
// If there is a filename grab the file data
if (result.filename.length > 0) {
// Create a temporary url
let temp = join(os.tmpdir(), (Math.random() * 10000).toString(12).substr(5, 10))
// Get the data between the blocks after the first two newlines
let matches = item.match(/^.+?(\r\n\r\n|\n\n)(.+)/s)
// Write the data to file
fs.createWriteStream(temp).write(matches[2])
}
下面是完整的数据解析:
http.createServer((req, res) => {
let body = ''
req.on('data', data => {
body += data.toString()
}).on('end', data => {
if (data) body += data.toString()
let boundary = req.headers['content-type'].split('boundary=')[1]
// Split all the boundary items and loop over them
body.split(new RegExp(`(--${boundary}|--${boundary}--)`)).forEach(item => {
if (item.trim().toLowerCase().startsWith('content-disposition')) {
item = item.trim()
// Find the name and filename
let result = item.split(':')[1].split(';').map(i => i.trim()).reduce((obj, itm) => {
if (itm.startsWith('name=')) obj.name = itm.match(/^name="(.+)"/)[1]
if (itm.startsWith('filename=')) obj.filename = itm.match(/^filename="(.+)"/)[1]
return obj
}, { name: '', filename: '' })
// If there is a filename grab the file data
if (result.filename.length > 0) {
// Create a temporary url
let temp = join(os.tmpdir(), (Math.random() * 10000).toString(12).substr(5, 10))
// Get the data
let matches = item.match(/^.+?(\r\n\r\n|\n\n)(.+)/s)
// Write the data to file
fs.createWriteStream(temp).write(matches[2])
}
}
})
})
})
所以,我在这两种情况下都有些正确...
首先,我必须像这样将数据读取为二进制数据:
body += data.toString('binary')
接下来保存的时候,我需要像这样保存为二进制数据:
fs.createWriteStream(temp).write(matches[2], 'binary')
这现在保存为正确的文件大小并且图像可读!