LaoShui a09f126ab2 fix(upload): 修复文件上传功能中的路径替换和临时文件支持
- 将单个斜杠替换改为全局正则替换以正确处理所有路径分隔符
- 移除未使用的 crypto 模块引入
- 添加对临时文件上传的支持,使用 fs 模块读取临时文件
- 动态设置 Content-Type 头部,优先使用文件的 mimetype 属性
- 重构多处文件上传逻辑以支持流式读取大文件
- 优化分片上传时的文件大小获取方式
2026-02-18 19:09:15 +08:00

202 lines
5.4 KiB
JavaScript

const { default: axios } = require('axios')
const fs = require('fs')
var xml2js = require('xml2js')
const createOption = require('../util/option.js')
var parser = new xml2js.Parser()
function createDupkey() {
var s = []
var hexDigits = '0123456789abcdef'
for (var i = 0; i < 36; i++) {
s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1)
}
s[14] = '4'
s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1)
s[8] = s[13] = s[18] = s[23] = '-'
return s.join('')
}
module.exports = async (query, request) => {
let ext = 'mp3'
if (query.songFile.name.indexOf('flac') > -1) {
ext = 'flac'
}
const filename =
query.songName ||
query.songFile.name
.replace('.' + ext, '')
.replace(/\s/g, '')
.replace(/\./g, '_')
if (!query.songFile) {
return Promise.reject({
status: 500,
body: {
msg: '请上传音频文件',
code: 500,
},
})
}
const tokenRes = await request(
`/api/nos/token/alloc`,
{
bucket: 'ymusic',
ext: ext,
filename: filename,
local: false,
nos_product: 0,
type: 'other',
},
createOption(query, 'weapi'),
)
const objectKey = tokenRes.body.result.objectKey.replace(/\//g, '%2F')
const docId = tokenRes.body.result.docId
const res = await axios({
method: 'post',
url: `https://ymusic.nos-hz.163yun.com/${objectKey}?uploads`,
headers: {
'x-nos-token': tokenRes.body.result.token,
'X-Nos-Meta-Content-Type': query.songFile.mimetype || 'audio/mpeg',
},
data: null,
})
const res2 = await parser.parseStringPromise(res.data)
const useTempFile = !!query.songFile.tempFilePath
let fileSize = query.songFile.size
let fileData
if (useTempFile) {
const stats = await fs.promises.stat(query.songFile.tempFilePath)
fileSize = stats.size
fileData = query.songFile.tempFilePath
} else {
fileSize = query.songFile.data.length
fileData = query.songFile.data
}
const blockSize = 10 * 1024 * 1024
let offset = 0
let blockIndex = 1
let etags = []
while (offset < fileSize) {
let chunk
if (useTempFile) {
const fd = await fs.promises.open(query.songFile.tempFilePath, 'r')
const buffer = Buffer.alloc(Math.min(blockSize, fileSize - offset))
await fd.read(buffer, 0, buffer.length, offset)
await fd.close()
chunk = buffer
} else {
chunk = query.songFile.data.slice(offset, Math.min(offset + blockSize, fileSize))
}
const res3 = await axios({
method: 'put',
url: `https://ymusic.nos-hz.163yun.com/${objectKey}?partNumber=${blockIndex}&uploadId=${res2.InitiateMultipartUploadResult.UploadId[0]}`,
headers: {
'x-nos-token': tokenRes.body.result.token,
'Content-Type': query.songFile.mimetype || 'audio/mpeg',
},
data: chunk,
})
const etag = res3.headers.etag
etags.push(etag)
offset += blockSize
blockIndex++
}
let completeStr = '<CompleteMultipartUpload>'
for (let i = 0; i < etags.length; i++) {
completeStr += `<Part><PartNumber>${i + 1}</PartNumber><ETag>${
etags[i]
}</ETag></Part>`
}
completeStr += '</CompleteMultipartUpload>'
await axios({
method: 'post',
url: `https://ymusic.nos-hz.163yun.com/${objectKey}?uploadId=${res2.InitiateMultipartUploadResult.UploadId[0]}`,
headers: {
'Content-Type': 'text/plain;charset=UTF-8',
'X-Nos-Meta-Content-Type': query.songFile.mimetype || 'audio/mpeg',
'x-nos-token': tokenRes.body.result.token,
},
data: completeStr,
})
await request(
`/api/voice/workbench/voice/batch/upload/preCheck`,
{
dupkey: createDupkey(),
voiceData: JSON.stringify([
{
name: filename,
autoPublish: query.autoPublish == 1 ? true : false,
autoPublishText: query.autoPublishText || '',
description: query.description,
voiceListId: query.voiceListId,
coverImgId: query.coverImgId,
dfsId: docId,
categoryId: query.categoryId,
secondCategoryId: query.secondCategoryId,
composedSongs: query.composedSongs
? query.composedSongs.split(',')
: [],
privacy: query.privacy == 1 ? true : false,
publishTime: query.publishTime || 0,
orderNo: query.orderNo || 1,
},
]),
},
{
...createOption(query),
headers: {
'x-nos-token': tokenRes.body.result.token,
},
},
)
const result = await request(
`/api/voice/workbench/voice/batch/upload/v2`,
{
dupkey: createDupkey(),
voiceData: JSON.stringify([
{
name: filename,
autoPublish: query.autoPublish == 1 ? true : false,
autoPublishText: query.autoPublishText || '',
description: query.description,
voiceListId: query.voiceListId,
coverImgId: query.coverImgId,
dfsId: docId,
categoryId: query.categoryId,
secondCategoryId: query.secondCategoryId,
composedSongs: query.composedSongs
? query.composedSongs.split(',')
: [],
privacy: query.privacy == 1 ? true : false,
publishTime: query.publishTime || 0,
orderNo: query.orderNo || 1,
},
]),
},
{
...createOption(query),
headers: {
'x-nos-token': tokenRes.body.result.token,
},
},
)
return {
status: 200,
body: {
code: 200,
data: result.body.data,
},
}
}