Files
NodeBB/src/controllers/admin/uploads.js

289 lines
8.0 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2016-05-16 12:34:47 +03:00
var path = require('path');
var async = require('async');
var nconf = require('nconf');
var mime = require('mime');
2018-01-26 18:56:17 -05:00
var fs = require('fs');
var meta = require('../../meta');
2016-05-16 12:34:47 +03:00
var file = require('../../file');
var image = require('../../image');
var plugins = require('../../plugins');
2018-01-26 18:56:17 -05:00
var pagination = require('../../pagination');
2016-05-06 04:42:59 -04:00
var allowedImageTypes = ['image/png', 'image/jpeg', 'image/pjpeg', 'image/jpg', 'image/gif', 'image/svg+xml'];
2017-05-10 15:34:36 -04:00
var uploadsController = module.exports;
2018-01-26 18:56:17 -05:00
uploadsController.get = function (req, res, next) {
var currentFolder = path.join(nconf.get('upload_path'), req.query.dir || '');
if (!currentFolder.startsWith(nconf.get('upload_path'))) {
return next(new Error('[[error:invalid-path]]'));
}
var itemsPerPage = 20;
var itemCount = 0;
var page = parseInt(req.query.page, 10) || 1;
async.waterfall([
function (next) {
fs.readdir(currentFolder, next);
},
function (files, next) {
files = files.filter(function (filename) {
return filename !== '.gitignore';
});
itemCount = files.length;
var start = Math.max(0, (page - 1) * itemsPerPage);
var stop = start + itemsPerPage;
files = files.slice(start, stop);
filesToData(currentFolder, files, next);
},
function (files) {
files.sort(function (a, b) {
if (a.isDirectory && !b.isDirectory) {
return -1;
} else if (!a.isDirectory && b.isDirectory) {
return 1;
}
return 0;
});
res.render('admin/manage/uploads', {
currentFolder: currentFolder.replace(nconf.get('upload_path'), ''),
files: files,
breadcrumbs: buildBreadcrumbs(currentFolder),
pagination: pagination.create(page, Math.ceil(itemCount / itemsPerPage), req.query),
});
},
], next);
};
function buildBreadcrumbs(currentFolder) {
var crumbs = [];
var parts = currentFolder.replace(nconf.get('upload_path'), '').split(path.sep);
var currentPath = '';
parts.forEach(function (part) {
var dir = path.join(currentPath, part);
crumbs.push({
text: part || 'Uploads',
url: part ? '/admin/manage/uploads?dir=' + dir : '/admin/manage/uploads',
});
currentPath = dir;
});
return crumbs;
}
function filesToData(currentDir, files, callback) {
async.map(files, function (file, next) {
var stat;
async.waterfall([
function (next) {
fs.stat(path.join(currentDir, file), next);
},
function (_stat, next) {
stat = _stat;
if (stat.isDirectory()) {
fs.readdir(path.join(currentDir, file), next);
} else {
next(null, []);
}
},
function (filesInDir, next) {
var url = nconf.get('upload_url') + currentDir.replace(nconf.get('upload_path'), '') + '/' + file;
next(null, {
name: file,
path: path.join(currentDir, file).replace(nconf.get('upload_path'), ''),
url: url,
fileCount: filesInDir.length - 1, // ignore .gitignore
size: stat.size,
sizeHumanReadable: (stat.size / 1024).toFixed(1) + 'KiB',
isDirectory: stat.isDirectory(),
isFile: stat.isFile(),
});
},
], next);
}, callback);
}
uploadsController.uploadCategoryPicture = function (req, res, next) {
2015-01-10 18:59:24 -05:00
var uploadedFile = req.files.files[0];
2016-05-06 04:42:59 -04:00
var params = null;
try {
params = JSON.parse(req.body.params);
} catch (e) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
2017-05-10 15:51:19 -04:00
return next(new Error('[[error:invalid-json]]'));
}
2016-05-06 04:42:59 -04:00
if (validateUpload(req, res, next, uploadedFile, allowedImageTypes)) {
var filename = 'category-' + params.cid + path.extname(uploadedFile.name);
2015-01-12 17:33:11 -05:00
uploadImage(filename, 'category', uploadedFile, req, res, next);
}
};
uploadsController.uploadFavicon = function (req, res, next) {
2015-01-10 18:59:24 -05:00
var uploadedFile = req.files.files[0];
var allowedTypes = ['image/x-icon', 'image/vnd.microsoft.icon'];
2015-02-17 15:12:31 -05:00
if (validateUpload(req, res, next, uploadedFile, allowedTypes)) {
file.saveFileToLocal('favicon.ico', 'system', uploadedFile.path, function (err, image) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
2015-01-12 17:33:11 -05:00
if (err) {
return next(err);
}
2017-02-18 12:30:49 -07:00
res.json([{ name: uploadedFile.name, url: image.url }]);
});
}
};
uploadsController.uploadTouchIcon = function (req, res, next) {
2017-02-17 20:20:42 -07:00
var uploadedFile = req.files.files[0];
var allowedTypes = ['image/png'];
var sizes = [36, 48, 72, 96, 144, 192];
if (validateUpload(req, res, next, uploadedFile, allowedTypes)) {
file.saveFileToLocal('touchicon-orig.png', 'system', uploadedFile.path, function (err, imageObj) {
2016-08-16 19:46:59 +02:00
if (err) {
return next(err);
}
// Resize the image into squares for use as touch icons at various DPIs
async.each(sizes, function (size, next) {
async.series([
async.apply(file.saveFileToLocal, 'touchicon-' + size + '.png', 'system', uploadedFile.path),
async.apply(image.resizeImage, {
path: path.join(nconf.get('upload_path'), 'system', 'touchicon-' + size + '.png'),
extension: 'png',
width: size,
2017-02-17 19:31:21 -07:00
height: size,
}),
], next);
}, function (err) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
if (err) {
return next(err);
}
2017-02-18 12:30:49 -07:00
res.json([{ name: uploadedFile.name, url: imageObj.url }]);
});
});
}
};
uploadsController.uploadLogo = function (req, res, next) {
2014-04-28 01:52:25 -04:00
upload('site-logo', req, res, next);
};
uploadsController.uploadSound = function (req, res, next) {
2016-02-23 16:06:02 -05:00
var uploadedFile = req.files.files[0];
var mimeType = mime.getType(uploadedFile.name);
if (!/^audio\//.test(mimeType)) {
return next(Error('[[error:invalid-data]]'));
}
2017-05-10 15:34:36 -04:00
async.waterfall([
function (next) {
file.saveFileToLocal(uploadedFile.name, 'sounds', uploadedFile.path, next);
},
function (uploadedSound, next) {
meta.sounds.build(next);
},
], function (err) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
2016-02-23 16:06:02 -05:00
if (err) {
return next(err);
}
2017-05-10 15:34:36 -04:00
res.json([{}]);
2016-02-23 16:06:02 -05:00
});
};
2018-01-26 18:56:17 -05:00
uploadsController.uploadFile = function (req, res, next) {
var uploadedFile = req.files.files[0];
var params;
try {
params = JSON.parse(req.body.params);
} catch (e) {
file.delete(uploadedFile.path);
return next(new Error('[[error:invalid-json]]'));
}
file.saveFileToLocal(uploadedFile.name, params.folder, uploadedFile.path, function (err, data) {
file.delete(uploadedFile.path);
if (err) {
return next(err);
}
res.json([{ url: data.url }]);
});
};
uploadsController.uploadDefaultAvatar = function (req, res, next) {
upload('avatar-default', req, res, next);
2014-04-28 01:52:25 -04:00
};
uploadsController.uploadOgImage = function (req, res, next) {
2016-05-16 12:34:47 +03:00
upload('og:image', req, res, next);
};
2014-04-28 01:52:25 -04:00
function upload(name, req, res, next) {
2015-01-10 18:59:24 -05:00
var uploadedFile = req.files.files[0];
2016-05-06 04:42:59 -04:00
if (validateUpload(req, res, next, uploadedFile, allowedImageTypes)) {
2015-01-10 18:59:24 -05:00
var filename = name + path.extname(uploadedFile.name);
2015-03-06 13:05:22 -05:00
uploadImage(filename, 'system', uploadedFile, req, res, next);
2015-01-10 18:59:24 -05:00
}
}
2015-01-12 17:33:11 -05:00
function validateUpload(req, res, next, uploadedFile, allowedTypes) {
2015-01-10 18:59:24 -05:00
if (allowedTypes.indexOf(uploadedFile.type) === -1) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
2017-02-18 12:30:49 -07:00
res.json({ error: '[[error:invalid-image-type, ' + allowedTypes.join(', ') + ']]' });
2015-01-10 18:59:24 -05:00
return false;
}
2015-01-10 18:59:24 -05:00
return true;
}
2015-01-12 17:33:11 -05:00
function uploadImage(filename, folder, uploadedFile, req, res, next) {
2017-05-10 15:34:36 -04:00
async.waterfall([
function (next) {
if (plugins.hasListeners('filter:uploadImage')) {
plugins.fireHook('filter:uploadImage', { image: uploadedFile, uid: req.user.uid }, next);
} else {
file.saveFileToLocal(filename, folder, uploadedFile.path, next);
2015-09-06 00:49:39 -04:00
}
2017-05-10 15:34:36 -04:00
},
function (imageData, next) {
// Post-processing for site-logo
if (path.basename(filename, path.extname(filename)) === 'site-logo' && folder === 'system') {
var uploadPath = path.join(nconf.get('upload_path'), folder, 'site-logo-x50.png');
async.series([
async.apply(image.resizeImage, {
path: uploadedFile.path,
target: uploadPath,
extension: 'png',
height: 50,
}),
2017-07-28 16:50:18 -04:00
async.apply(meta.configs.set, 'brand:emailLogo', path.join(nconf.get('upload_url'), 'system/site-logo-x50.png')),
], function (err) {
next(err, imageData);
});
} else {
setImmediate(next, null, imageData);
}
},
2017-05-10 15:34:36 -04:00
], function (err, image) {
2017-05-24 00:02:30 -04:00
file.delete(uploadedFile.path);
2015-01-12 17:33:11 -05:00
if (err) {
return next(err);
}
2017-02-18 12:30:49 -07:00
res.json([{ name: uploadedFile.name, url: image.url.startsWith('http') ? image.url : nconf.get('relative_path') + image.url }]);
2017-05-10 15:34:36 -04:00
});
2014-04-28 01:52:25 -04:00
}
2018-01-26 18:56:17 -05:00