|
|
'use strict'
/* global wiki */
const path = require('path') const Promise = require('bluebird') const fs = Promise.promisifyAll(require('fs-extra')) const multer = require('multer') const os = require('os') const _ = require('lodash')
/** * Local Disk Storage */ module.exports = {
_uploadsPath: './repo/uploads', _uploadsThumbsPath: './data/thumbs',
uploadImgHandler: null,
/** * Initialize Local Data Storage model */ init () { this._uploadsPath = path.resolve(wiki.ROOTPATH, wiki.config.paths.repo, 'uploads') this._uploadsThumbsPath = path.resolve(wiki.ROOTPATH, wiki.config.paths.data, 'thumbs')
this.createBaseDirectories() // this.initMulter()
return this },
/** * Init Multer upload handlers */ initMulter () { let maxFileSizes = { // img: wiki.config.uploads.maxImageFileSize * 1024 * 1024,
// file: wiki.config.uploads.maxOtherFileSize * 1024 * 1024
img: 3 * 1024 * 1024, file: 10 * 1024 * 1024 }
// -> IMAGES
this.uploadImgHandler = multer({ storage: multer.diskStorage({ destination: (req, f, cb) => { cb(null, path.resolve(wiki.ROOTPATH, wiki.config.paths.data, 'temp-upload')) } }), fileFilter: (req, f, cb) => { // -> Check filesize
if (f.size > maxFileSizes.img) { return cb(null, false) }
// -> Check MIME type (quick check only)
if (!_.includes(['image/png', 'image/jpeg', 'image/gif', 'image/webp'], f.mimetype)) { return cb(null, false) }
cb(null, true) } }).array('imgfile', 20)
// -> FILES
this.uploadFileHandler = multer({ storage: multer.diskStorage({ destination: (req, f, cb) => { cb(null, path.resolve(wiki.ROOTPATH, wiki.config.paths.data, 'temp-upload')) } }), fileFilter: (req, f, cb) => { // -> Check filesize
if (f.size > maxFileSizes.file) { return cb(null, false) }
cb(null, true) } }).array('binfile', 20)
return true },
/** * Creates a base directories (Synchronous). */ createBaseDirectories () { try { fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data)) fs.emptyDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data)) fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data, './cache')) fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data, './thumbs')) fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data, './temp-upload'))
if (os.type() !== 'Windows_NT') { fs.chmodSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.data, './temp-upload'), '755') }
fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.repo)) fs.ensureDirSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.repo, './uploads'))
if (os.type() !== 'Windows_NT') { fs.chmodSync(path.resolve(wiki.ROOTPATH, wiki.config.paths.repo, './uploads'), '755') } } catch (err) { wiki.logger.error(err) }
wiki.logger.info('Disk Data Paths: [ OK ]') },
/** * Gets the uploads path. * * @return {String} The uploads path. */ getUploadsPath () { return this._uploadsPath },
/** * Gets the thumbnails folder path. * * @return {String} The thumbs path. */ getThumbsPath () { return this._uploadsThumbsPath },
/** * Check if filename is valid and unique * * @param {String} f The filename * @param {String} fld The containing folder * @param {boolean} isImage Indicates if image * @return {Promise<String>} Promise of the accepted filename */ validateUploadsFilename (f, fld, isImage) { let fObj = path.parse(f) let fname = _.chain(fObj.name).trim().toLower().kebabCase().value().replace(new RegExp('[^a-z0-9-' + wiki.data.regex.cjk + wiki.data.regex.arabic + ']', 'g'), '') let fext = _.toLower(fObj.ext)
if (isImage && !_.includes(['.jpg', '.jpeg', '.png', '.gif', '.webp'], fext)) { fext = '.png' }
f = fname + fext let fpath = path.resolve(this._uploadsPath, fld, f)
return fs.statAsync(fpath).then((s) => { throw new Error(wiki.lang.t('errors:fileexists', { path: f })) }).catch((err) => { if (err.code === 'ENOENT') { return f } throw err }) }
}
|