mirror of https://github.com/status-im/codimd.git
Merge pull request #1530 from hackmdio/fix/improve-filesystem-upload
fix: improve image upload to filesystem may caused app crash
This commit is contained in:
commit
7b148457a7
|
@ -1,10 +1,39 @@
|
||||||
'use strict'
|
'use strict'
|
||||||
|
|
||||||
|
const crypto = require('crypto')
|
||||||
|
const fs = require('fs')
|
||||||
const URL = require('url').URL
|
const URL = require('url').URL
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
|
|
||||||
const config = require('../config')
|
const config = require('../config')
|
||||||
const logger = require('../logger')
|
const logger = require('../logger')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* generate a random filename for uploaded image
|
||||||
|
*/
|
||||||
|
function randomFilename () {
|
||||||
|
const buf = crypto.randomBytes(16)
|
||||||
|
return `upload_${buf.toString('hex')}`
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* pick a filename not exist in filesystem
|
||||||
|
* maximum attempt 5 times
|
||||||
|
*/
|
||||||
|
function pickFilename (defaultFilename) {
|
||||||
|
let retryCounter = 5
|
||||||
|
let filename = defaultFilename
|
||||||
|
const extname = path.extname(defaultFilename)
|
||||||
|
while (retryCounter-- > 0) {
|
||||||
|
if (fs.existsSync(path.join(config.uploadsPath, filename))) {
|
||||||
|
filename = `${randomFilename()}${extname}`
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return filename
|
||||||
|
}
|
||||||
|
throw new Error('file exists.')
|
||||||
|
}
|
||||||
|
|
||||||
exports.uploadImage = function (imagePath, callback) {
|
exports.uploadImage = function (imagePath, callback) {
|
||||||
if (!imagePath || typeof imagePath !== 'string') {
|
if (!imagePath || typeof imagePath !== 'string') {
|
||||||
callback(new Error('Image path is missing or wrong'), null)
|
callback(new Error('Image path is missing or wrong'), null)
|
||||||
|
@ -16,11 +45,24 @@ exports.uploadImage = function (imagePath, callback) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let filename = path.basename(imagePath)
|
||||||
|
try {
|
||||||
|
filename = pickFilename(path.basename(imagePath))
|
||||||
|
} catch (e) {
|
||||||
|
return callback(e, null)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
fs.copyFileSync(imagePath, path.join(config.uploadsPath, filename))
|
||||||
|
} catch (e) {
|
||||||
|
return callback(e, null)
|
||||||
|
}
|
||||||
|
|
||||||
let url
|
let url
|
||||||
try {
|
try {
|
||||||
url = (new URL(path.basename(imagePath), config.serverURL + '/uploads/')).href
|
url = (new URL(filename, config.serverURL + '/uploads/')).href
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
url = config.serverURL + '/uploads/' + path.basename(imagePath)
|
url = config.serverURL + '/uploads/' + filename
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(null, url)
|
callback(null, url)
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
'use strict'
|
'use strict'
|
||||||
|
|
||||||
|
const fs = require('fs')
|
||||||
const Router = require('express').Router
|
const Router = require('express').Router
|
||||||
const formidable = require('formidable')
|
const formidable = require('formidable')
|
||||||
|
|
||||||
|
@ -15,10 +16,6 @@ imageRouter.post('/uploadimage', function (req, res) {
|
||||||
|
|
||||||
form.keepExtensions = true
|
form.keepExtensions = true
|
||||||
|
|
||||||
if (config.imageUploadType === 'filesystem') {
|
|
||||||
form.uploadDir = config.uploadsPath
|
|
||||||
}
|
|
||||||
|
|
||||||
form.parse(req, function (err, fields, files) {
|
form.parse(req, function (err, fields, files) {
|
||||||
if (err || !files.image || !files.image.path) {
|
if (err || !files.image || !files.image.path) {
|
||||||
response.errorForbidden(req, res)
|
response.errorForbidden(req, res)
|
||||||
|
@ -29,6 +26,8 @@ imageRouter.post('/uploadimage', function (req, res) {
|
||||||
|
|
||||||
const uploadProvider = require('./' + config.imageUploadType)
|
const uploadProvider = require('./' + config.imageUploadType)
|
||||||
uploadProvider.uploadImage(files.image.path, function (err, url) {
|
uploadProvider.uploadImage(files.image.path, function (err, url) {
|
||||||
|
// remove temporary upload file, and ignore any error
|
||||||
|
fs.unlink(files.image.path, () => {})
|
||||||
if (err !== null) {
|
if (err !== null) {
|
||||||
logger.error(err)
|
logger.error(err)
|
||||||
return res.status(500).end('upload image error')
|
return res.status(500).end('upload image error')
|
||||||
|
|
Loading…
Reference in New Issue